objecttolerance="10"
inkscape:pageopacity="0.0"
inkscape:pageshadow="2"
- inkscape:zoom="1.8955697"
- inkscape:cx="560.42433"
- inkscape:cy="704.00469"
+ inkscape:zoom="7.5822788"
+ inkscape:cx="532.83563"
+ inkscape:cy="896.76979"
inkscape:document-units="px"
inkscape:current-layer="layer1"
showgrid="false"
sodipodi:nodetypes="csssccccc"
inkscape:export-ydpi="138.63892"
inkscape:export-xdpi="138.63892"
- inkscape:export-filename="/home/gburri/projets/euphorik/img/css1/extraction.png"
+ inkscape:export-filename="/home/gburri/projets/euphorik/branches/1.1/styles/1/img/extraction.png"
id="path3233"
d="M 531.65729,145.35293 C 529.36216,145.35293 527.50104,147.21405 527.50104,149.50918 C 527.50104,151.80431 529.36216,153.66543 531.65729,153.66543 C 533.95242,153.66543 535.81354,151.80431 535.81354,149.50918 C 535.81354,147.21405 533.95242,145.35293 531.65729,145.35293 z M 530.04764,146.86648 L 534.5789,149.44579 L 530.16777,152.15985 L 530.04764,146.86648 z"
- style="opacity:1;fill:#841919;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.2;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
+ style="opacity:0.5;fill:#841919;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.20000000000000001;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
<path
sodipodi:nodetypes="csssccccc"
inkscape:export-ydpi="138.63892"
inkscape:export-xdpi="138.63892"
- inkscape:export-filename="/home/gburri/projets/euphorik/img/css1/extraction_hover.png"
+ inkscape:export-filename="/home/gburri/projets/euphorik/branches/1.1/styles/1/img/extraction_hover.png"
id="path3235"
d="M 531.65729,158.13031 C 529.36216,158.13031 527.50104,159.99143 527.50104,162.28656 C 527.50104,164.58169 529.36216,166.44281 531.65729,166.44281 C 533.95242,166.44281 535.81354,164.58169 535.81354,162.28656 C 535.81354,159.99143 533.95242,158.13031 531.65729,158.13031 z M 530.04764,159.64386 L 534.5789,162.22317 L 530.16777,164.93723 L 530.04764,159.64386 z"
- style="opacity:1;fill:#cb2626;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.2;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
+ style="opacity:0.5;fill:#ff6565;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.20000000000000001;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
<path
inkscape:export-ydpi="140.75188"
inkscape:export-xdpi="140.75188"
<path
inkscape:export-ydpi="138.63892"
inkscape:export-xdpi="138.63892"
- inkscape:export-filename="/home/gburri/projets/euphorik/trunk/img/css1/extraction_complete.png"
+ inkscape:export-filename="/home/gburri/projets/euphorik/branches/1.1/styles/1/img/extraction_complete.png"
id="path2721"
d="M 519.27037,145.35293 C 516.97525,145.35292 515.11412,147.21405 515.11412,149.50918 C 515.11413,151.8043 516.97524,153.66543 519.27037,153.66543 C 521.56551,153.66542 523.42662,151.80431 523.42662,149.50918 C 523.42659,147.21404 521.5655,145.35293 519.27037,145.35293 z M 519.27037,146.57168 L 522.23912,149.50918 L 519.27037,152.47793 L 516.33287,149.50918 L 519.27037,146.57168 z"
- style="opacity:1;fill:#841919;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.2;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
+ style="opacity:0.5;fill:#841919;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.20000000000000001;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
<path
inkscape:export-ydpi="138.63892"
inkscape:export-xdpi="138.63892"
- inkscape:export-filename="/home/gburri/projets/euphorik/trunk/img/css1/extraction_complete_hover.png"
+ inkscape:export-filename="/home/gburri/projets/euphorik/branches/1.1/styles/1/img/extraction_complete_hover.png"
id="path3502"
d="M 519.27037,158.13031 C 516.97525,158.1303 515.11412,159.99143 515.11412,162.28656 C 515.11413,164.58168 516.97524,166.44281 519.27037,166.44281 C 521.56551,166.4428 523.42662,164.58169 523.42662,162.28656 C 523.42659,159.99142 521.5655,158.13031 519.27037,158.13031 z M 519.27037,159.34906 L 522.23912,162.28656 L 519.27037,165.25531 L 516.33287,162.28656 L 519.27037,159.34906 z"
- style="opacity:1;fill:#cb2626;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.2;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
+ style="opacity:0.5;fill:#ff6565;fill-opacity:1;fill-rule:evenodd;stroke:none;stroke-width:0.20000000000000001;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1;display:inline" />
<path
inkscape:export-ydpi="138.63892"
inkscape:export-xdpi="138.63892"
</erl>
<div id="info" style="display:none" ><div id="icone"></div><div class="fermer" ></div><div class="message" ></div><div class="boutons"></div></div>
+ <div id="waitbar" style="display:none" ><div class="image"></div></div>
<ul id="menu">
<li class="minichat">chat</li><li class="admin" style="display:none">admin</li><li class="profile"></li><li class="register">register</li><li class="logout">logout</li><li class="about">about</li>
</ul>
// Regroupe la partie communication JSON client -> serveur de euphorik.\r
// Voir : http://dev.euphorik.ch/wiki/euk/Protocole\r
\r
-/**\r
- * @param funError un fonction executé lors d'un réponse 'error' de la part du serveur, peut être redéfinit pour une requête.\r
+/**
+ * Les fonctions debutReq et finReq servent, par exemple, à afficher à l'utilisateur
+ * qu'une communication est en cours.\r
+ * @param funError un fonction executée lors d'un réponse 'error' de la part du serveur, peut être redéfinit pour une requête.
+ * @param funDebutReq fonction appelée au début d'une requête (facultatif)
+ * @param funFinReq fonction appelée à la fin d'une requête (facultatif)\r
*/\r
-euphorik.Communication = function(funError) {\r
- this.funError = funError;\r
+euphorik.Communication = function(funError, funDebutReq, funFinReq) {\r
+ this.funError = funError;
+ this.funDebutReq = funDebutReq;
+ this.funFinReq = funFinReq;\r
+};
+
+/**
+ * Charge un fichier depuis une url et retourne son contenu.
+ */
+euphorik.Communication.prototype.load = function(url) {
+ if (this.funDebutReq) {
+ this.funDebutReq();
+ }
+ var contenu = "";
+ $.ajax({async: false, url: url, success : function(page) { contenu += page; }});
+ if (this.funFinReq) {
+ this.funFinReq();
+ }
+ return contenu;
};\r
-\r
+
+/**
+ * Effectue une requête JSON auprès du serveur.
+ * @param action une chaine spécifiant l'action, par exemple "put_message"
+ * @param json les données à envoyer associé à l'action, par exemple {"cookie" : "LKJDLAKSJBFLKASN", "nick" : "Paul", "content" : "Bonjour", "answer_to" : [] }
+ * @param funOk la fonction exécuté après réception des données du serveur
+ * @param funError la fonction exécuté si une erreur arrive (facultatif)
+ * @param asynchrone true pour une communication asychrone (facultatif, truepar défaut)
+ * @param paramsSupp un objet contenant des paramètres supplémentaire pour la fonction ajax de jQuery (facultatif)
+ */\r
euphorik.Communication.prototype.requete = function(action, json, funOk, funError, asynchrone, paramsSupp) {\r
var thisCommunication = this;\r
if (asynchrone === undefined) {\r
var mess = this.getBase(action);\r
objectEach(json, function(nom, val) {\r
mess[nom] = val;\r
- });\r
+ });
+
+ if (this.funDebutReq) {
+ this.funDebutReq();
+ }\r
\r
paramsAjax = {\r
async: asynchrone,\r
dataType: "json",\r
data: { action : JSON.stringify(mess) },\r
success:\r
- function(data) {\r
+ function(data) {
+ if (thisCommunication.funFinReq) {
+ thisCommunication.funFinReq();
+ }\r
if (data.reply === "error") {\r
if (funError) {\r
funError(data);\r
} else if (funOk) {\r
funOk(data);\r
}\r
+ },
+ error:
+ function(data) {
+ if (thisCommunication.funFinReq) {
+ thisCommunication.funFinReq();
+ }
}\r
};\r
- \r
+ \r
if (paramsSupp) {\r
objectEach(paramsSupp, function(nom, val) {\r
paramsAjax[nom] = val;\r
var fragment = new Fragment();
var formateur = new euphorik.Formateur();
var util = new euphorik.Util(formateur);
- var communication = new euphorik.Communication(function(data) { util.messageDialogue(data.error_message); });
+ var communication = new euphorik.Communication(
+ function(data) { util.messageDialogue(data.error_message); },
+ function() { $("#waitbar").show(); },
+ function() { $("#waitbar").hide(); }
+ );
var client = new euphorik.Client(util, communication);
- var pages = new euphorik.Pages(fragment);
+ var pages = new euphorik.Pages(fragment, communication);
// connexion vers le serveur (utilise un cookie qui traine)
client.connexionCookie();
$("#footer .conditions").click(function(){ pages.afficherPage("conditions_utilisation"); });
pages.ajouterPage(new euphorik.PageMinichat(client, formateur, util, communication), true);
- pages.ajouterPage(new euphorik.PageAdmin(client, formateur, util));
+ pages.ajouterPage(new euphorik.PageAdmin(client, formateur, util, communication));
pages.ajouterPage(new euphorik.PageProfile(client, formateur, util));
pages.ajouterPage(new euphorik.PageRegister(client, formateur, util));
- pages.ajouterPage(new euphorik.PageAbout(client, formateur, util));
+ pages.ajouterPage(new euphorik.PageAbout(client, formateur, util, communication));
pages.ajouterPage("conditions_utilisation");
pages.afficherPage();
// You should have received a copy of the GNU General Public License\r
// along with Euphorik. If not, see <http://www.gnu.org/licenses/>.\r
\r
-euphorik.PageAbout = function(client, formateur, util) {\r
+euphorik.PageAbout = function(client, formateur, util, communication) {\r
this.nom = "about";\r
\r
this.client = client;\r
this.formateur = formateur;\r
- this.util = util;\r
+ this.util = util;
+ this.communication = communication;\r
};\r
\r
euphorik.PageAbout.prototype.contenu = function() {\r
- var contenu = "";\r
- $.ajax({async: false, url: "pages/about.html", success : function(page) { contenu += page; }});\r
- \r
+ var contenu = this.communication.load("pages/about.html"); \r
var email = this.util.rot13("znvygb:tert.oheev@tznvy.pbz");\r
return contenu.replace("{EMAIL}", "<a href=\"" + email+ "\">" + email + "</a>").replace("{EMAIL_LIEN}", email);\r
};\r
/*jslint laxbreak:true */
-euphorik.PageAdmin = function(client, formateur, util) {
+euphorik.PageAdmin = function(client, formateur, util, communication) {
this.nom = "admin";
this.client = client;
this.formateur = formateur;
this.util = util;
+ this.communication = communication;
this.comet = new Comet("admin", euphorik.conf.versionProtocole);
var thisPage = this;
// la liste des trolls proposés par les ekMasters
- this.trolls = new euphorik.Trolls(this.client, this.util, this.formateur);
+ this.trolls = new euphorik.Trolls(this.client, this.util, this.formateur, this.communication);
this.waitEvent();
return;
}
- var dataToSend = {
- "header" : { "action" : "put_troll", "version" : euphorik.conf.versionProtocole },
- "cookie" : this.client.cookie,
- "content" : content
- };
-
- jQuery.ajax({
- type: "POST",
- url: "request",
- dataType: "json",
- data: this.util.jsonVersAction(dataToSend),
- success:
- function(data){
- if (data.reply === "ok") {
- $("#page form#nouveauTroll input.troll").val("");
- } else if (data.reply === "error") {
- thisPageAdmin.util.messageDialogue(data.error_message);
- }
- }
- });
+ this.communication.requete(
+ "put_troll",
+ {"cookie" : this.client.cookie, "content" : content},
+ function(data) {
+ $("#page form#nouveauTroll input.troll").val("");
+ }
+ );
};
/**
}
var thisPageAdmin = this;
-
- var dataToSend = {
- "header" : { "action" : "list_banned_ips", "version" : euphorik.conf.versionProtocole },
- "cookie" : this.client.cookie
- };
-
- jQuery.ajax({
- type: "POST",
- url: "request",
- dataType: "json",
- data: this.util.jsonVersAction(dataToSend),
- success:
- function(data) {
- if (data.reply === "list_banned_ips") {
- var XHTML = "";
- data.list.each(function(i, ip) {
- XHTML += '<div class="ban"><span class="ip">' + ip.ip + '</span>|' +
- '<span class="temps">' +
- ip.remaining_time +
- '</span>|';
- ip.users.each(function(j, user) {
- XHTML += (j > 0 ? ", " : "") +
- '<span class="pseudo">' + thisPageAdmin.formateur.traitementComplet(user.nick) + '</span>' +
- (user.login === "" ? "" : '<span class="login">(' + thisPageAdmin.formateur.traitementComplet(user.login) + ')</span>');
- });
- XHTML += '<span class="deban">débannir</span></div>';
- });
-
- if (data.list.length === 0) {
- XHTML += '<p>Aucune IP bannie</p>';
- }
-
- $("#ips").html(XHTML);
-
- $(".ban").each(function() {
- var ip = $(".ip", this).html();
- $(".deban", this).click(
- function() {
- thisPageAdmin.util.messageDialogue("Êtes-vous sur de vouloir débannir l'IP ''" + ip + "'' ?", euphorik.Util.messageType.question,
- {"Oui" : function() {
- thisPageAdmin.deban(ip);
- },
- "Non" : function(){}
- }
- );
+
+ this.communication.requete(
+ "list_banned_ips",
+ {"cookie" : this.client.cookie},
+ function(data) {
+ var XHTML = "";
+ data.list.each(function(i, ip) {
+ XHTML += '<div class="ban"><span class="ip">' + ip.ip + '</span>|' +
+ '<span class="temps">' +
+ ip.remaining_time +
+ '</span>|';
+ ip.users.each(function(j, user) {
+ XHTML += (j > 0 ? ", " : "") +
+ '<span class="pseudo">' + thisPageAdmin.formateur.traitementComplet(user.nick) + '</span>' +
+ (user.login === "" ? "" : '<span class="login">(' + thisPageAdmin.formateur.traitementComplet(user.login) + ')</span>');
+ });
+ XHTML += '<span class="deban">débannir</span></div>';
+ });
+
+ if (data.list.length === 0) {
+ XHTML += '<p>Aucune IP bannie</p>';
+ }
+
+ $("#ips").html(XHTML);
+
+ $(".ban").each(function() {
+ var ip = $(".ip", this).html();
+ $(".deban", this).click(
+ function() {
+ thisPageAdmin.util.messageDialogue("Êtes-vous sur de vouloir débannir l'IP ''" + ip + "'' ?", euphorik.Util.messageType.question,
+ {"Oui" : function() {
+ thisPageAdmin.deban(ip);
+ },
+ "Non" : function(){}
}
);
- });
- } else if (data.reply === "error") {
- thisPageAdmin.util.messageDialogue(data.error_message);
- }
+ }
+ );
+ });
- // rafraichissement toutes les minutes (je sais c'est mal)
- // le problème est le rafraichissement des temps restant de bannissement qui doit être fait du coté client
- thisPageAdmin.timeoutIDmajIPs = setTimeout(function(){ thisPageAdmin.majIPs(); }, 60 * 1000);
- }
- });
+ // rafraichissement toutes les minutes (je sais c'est mal)
+ // le problème est le rafraichissement des temps restant de bannissement qui doit être fait du coté client
+ thisPageAdmin.timeoutIDmajIPs = setTimeout(function(){ thisPageAdmin.majIPs(); }, 60 * 1000);
+ }
+ );
};
/**
*/
euphorik.PageAdmin.prototype.deban = function(ip) {
var thisPageAdmin = this;
-
- var dataToSend = {
- "header" : { "action" : "unban", "version" : euphorik.conf.versionProtocole },
- "cookie" : this.client.cookie,
- "ip" : ip
- };
-
- jQuery.ajax({
- type: "POST",
- url: "request",
- dataType: "json",
- data: this.util.jsonVersAction(dataToSend),
- success:
- function(data){
- if(data.reply === "error") {
- thisPageAdmin.util.messageDialogue(data.error_message);
- }
- }
- });
+
+ this.communication.requete(
+ "unban",
+ {"cookie" : this.client.cookie, "ip" : ip}
+ );
};
/**
///////////////////////////////////////////////////////////////////////////////////////////////////
-euphorik.Trolls = function(client, util, formateur) {
+euphorik.Trolls = function(client, util, formateur, communication) {
this.client = client;
this.util = util;
this.formateur = formateur;
+ this.communication = communication;
this.dernierTroll = 0;
this.trolls = {};
};
euphorik.Trolls.prototype.modifier = function(id, content) {
- var thisTrolls = this;
-
- var dataToSend = {
- "header" : { "action" : "mod_troll", "version" : euphorik.conf.versionProtocole },
- "cookie" : this.client.cookie,
- "troll_id" : id,
- "content" : content
- };
-
- jQuery.ajax({
- type: "POST",
- url: "request",
- dataType: "json",
- data: this.util.jsonVersAction(dataToSend),
- success:
- function(data) {
- if (data.reply === "error") {
- thisTrolls.util.messageDialogue(data.error_message);
- }
- }
- });
+ this.communication.requete(
+ "mod_troll",
+ {"cookie" : this.client.cookie, "troll_id" : id, "content" : content}
+ );
};
/**
* Supprime un troll en fonction de son id.
*/
euphorik.Trolls.prototype.supprimer = function(id) {
- var thisTrolls = this;
-
- var dataToSend = {
- "header" : { "action" : "del_troll", "version" : euphorik.conf.versionProtocole },
- "cookie" : this.client.cookie,
- "troll_id" : id
- };
-
- jQuery.ajax({
- type: "POST",
- url: "request",
- dataType: "json",
- data: this.util.jsonVersAction(dataToSend),
- success:
- function(data) {
- if (data.reply === "error") {
- thisTrolls.util.messageDialogue(data.error_message);
- }
- }
- });
+ this.communication.requete(
+ "del_troll",
+ {"cookie" : this.client.cookie, "troll_id" : id}
+ );
};
pageMinichat.envoyerMessage("C'est pas faux");
return [euphorik.Commandes.statut.ok, ''];
}
+ },
+ "osef" : {
+ description : "Envoie le message \"On s'en fout !\"",
+ usage : "/osef",
+ exec : function(args, client, pageMinichat) {
+ pageMinichat.envoyerMessage("On s'en fout !");
+ return [euphorik.Commandes.statut.ok, ''];
+ }
}
};
var mess = thisConversation.messagesParId[messId];
if (mess) {
mess.clientARepondu = true;
- $("#conversations #" + mess.getId(thisConversation.getId())).addClass("repondu")
+ $("#conversations #" + mess.getId(thisConversation.getId())).addClass("repondu");
}
});
}
- });
+ });\r
\r
if (this.messages.length > 0) {\r
this.idDernierMessageAffiche = this.messages[this.messages.length-1].id;\r
/**
* Correspond à une page html statique se nommant "<nom>.html" et se trouvant dans "/pages/".
*/
-euphorik.PageStatique = function(nom) {
+euphorik.PageStatique = function(nom, communication) {
this.nom = nom;
+ this.communication = communication;
};
euphorik.PageStatique.prototype.contenu = function() {
- var contenu = "";
- $.ajax({async: false, url: "pages/" + this.nom + ".html", success : function(page) { contenu += page; }});
- return contenu;
+ return this.communication.load("pages/" + this.nom + ".html");
};
euphorik.PageStatique.prototype.charger = function() {
/**\r
* Gestion des pages.\r
*/\r
-euphorik.Pages = function(fragment) {\r
- this.fragment = fragment;\r
+euphorik.Pages = function(fragment, communication) {\r
+ this.fragment = fragment;
+ this.communication = communication;\r
this.pageCourante = undefined;\r
this.pageDefaut = undefined;\r
this.pages = {};\r
*/\r
euphorik.Pages.prototype.ajouterPage = function(page, defaut) {\r
if (typeof page === "string") {\r
- page = new euphorik.PageStatique(page);\r
+ page = new euphorik.PageStatique(page, this.communication);\r
}\r
\r
page.pages = this; // la magie des langages dynamiques : le foutoire\r
* Utilisé pour l'envoie de données avec la méthode ajax de jQuery.\r
* Obsolète : à virer\r
*/\r
-euphorik.Util.prototype.jsonVersAction = function(json) {\r
+/*euphorik.Util.prototype.jsonVersAction = function(json) {\r
return { action : JSON.stringify(json) };\r
-};\r
+};*/\r
\r
/**\r
* Retourne un hash md5 d'une chaine, dépend de md5.js.\r
elire_troll/0,\r
\r
% utiles :\r
- resultat_transaction/1\r
+ resultat_transaction/1,
+ get_tuples/3 % must be in a transaction\r
]).\r
-import(qlc, [e/2, q/1, cursor/2]).\r
-include("../include/euphorik_bd.hrl").\r
messages(N, P) ->\r
F = fun() ->
% % #minichat{contenu = contenu_message(E)}
- get_tuples_avant(minichat, reculer(minichat, mnesia:last(minichat), N * (P - 1)), N) \r
+ get_tuples(minichat, mnesia:table_info(minichat, size) - N * P + 1, N)\r
end,\r
resultat_transaction(mnesia:transaction(F)).
-get_tuples_avant(Table, Id, N) ->
- get_tuples_avant(Table, Id, N, []).
-get_tuples_avant(_, '$end_of_table', _, Tuples) -> Tuples;
-get_tuples_avant(_, _, 0, Tuples) ->
- Tuples;
-get_tuples_avant(Table, Id, N, Tuples) ->
- [T] = mnesia:read({Table, Id}),
- get_tuples_avant(Table, mnesia:prev(Table, Id), N - 1, [T | Tuples]).
-reculer(_, '$end_of_table' = Fin, _) -> Fin;
-reculer(_, Id, 0) -> Id;
-reculer(Table, Id, N) ->
- reculer(Table, mnesia:prev(Table, Id), N - 1).
+get_tuples(Table, First_id, N) ->
+ lists:foldr(
+ fun(Id, Acc) ->
+ case mnesia:read({Table, Id}) of
+ [T] -> [T | Acc];
+ _ -> Acc
+ end
+ end,
+ [],
+ lists:seq(First_id, First_id + N - 1)
+ ).
\r
\r
% Renvoie les messages manquants pour la page P en sachant qu'il y a N message\r
Nb_troll_poste_total >= ?NB_MAX_TROLL_WAITING ->\r
max_troll_reached;\r
true ->\r
- Id = nouvel_id(minichat),\r
+ Id = nouvel_id(troll),\r
mnesia:write(#troll{id = Id, id_user = User_id, date_create = now(), content = Content}),\r
Id\r
end;\r
% crée un backup avant l'application du patch
% dans BD/backups nommé "backup<num>" où <num> et le numéro de la version.
% 1 -> 2
-patch(1) ->
+patch(1) ->\r
+ % Prend un chemin vers la feuille de style de type "css/1/euphorik.css"\r
+ % et renvoie "styles/1/euphorik.css"\r
+ Transforme_css = fun("css" ++ Reste) ->
+ "styles" ++ Reste;\r
+ (F) -> F\r
+ end,\r
+ Traiter_message = fun(M, Racine) ->\r
+ F = fun(F, M2) -> % seul moyen à ma connaissance pour faire de la récursion dans une lambda fonction, voir : http://www.nabble.com/Auto-generated-functions-td15279499.html\r
+ % met à jour la racine de chaque message qui répond à M\r
+ lists:foreach(\r
+ fun(M3) ->\r
+ mnesia:write(M2#minichat{racine_id = Racine}),\r
+ F(F, M3)\r
+ end,\r
+ euphorik_bd:enfants(M#minichat.id)\r
+ )\r
+ end,\r
+ F(F, M, Racine)\r
+ end,
% Prend un chemin vers la feuille de style de type "css/1/euphorik.css"\r
% et renvoie "styles/1/euphorik.css"\r
Transforme_css = fun("css" ++ Reste) ->
{atomic, _} -> ok
end;\r
% 2 -> 3\r
-patch(2) ->
- % première étape : changer le type de la table minichat de set à ordered_set
- % TODO : trouver un meilleur moyen que de passer par un backup
- backup("tmp"),
- create(),
- restore("tmp"),
- file:delete(dossier_backups() ++ "tmp"),\r
+patch(2) ->\r
mnesia:transform_table(\r
troll,\r
fun({troll, Id_troll, Id_user, Date_create, Date_post, Content}) ->\r
mnesia:transform_table(\r
minichat,\r
fun({minichat, Id, Auteur_id, Date, Pseudo, Contenu, _Troll_id, Racine_id}) ->\r
- {minichat, Id, Auteur_id, Date, Pseudo, Contenu, Racine_id}\r
+ {minichat, Id, Auteur_id, Date, Pseudo, Contenu, Racine_id, normal}\r
end,\r
record_info(fields, minichat)\r
- ),\r
- creer_indexes(). % uniquement pour l'indice sur id_minichat de la table troll\r
+ ),
+ mnesia:transaction(fun() ->
+ % comble les trous entre les id non-contigues
+ lists:foreach(fun(Id) ->
+ case mnesia:read({minichat, Id}) of
+ [] ->
+ {ok, #user{profile = Profile}} = euphorik_bd:user_by_id(0),
+ mnesia:write(#minichat{id = Id, auteur_id = 0, date = undefined, pseudo = Profile#profile.pseudo, contenu = "Comblement...", racine_id = Id, status = deleted});
+ _ -> rien
+ end
+ end,
+ lists:seq(1, mnesia:table_info(minichat, size))
+ ),
+ % la table troll utilise maintenant son index et pas celui de la table minichat (correction d'un vieux bug)
+ mnesia:write(#counter{key = troll, value = mnesia:table_info(minichat, size)})
+ end),\r
+ creer_indexes(). % uniquement pour l'indice sur id_minichat de la table troll
% Renvoie le dossier dans lequel les backups sont effectué, ce dossier doit être en écriture.
conversations_detailees(Racines, N, D, P) -> \r
Conversations = map(fun({Racine, P_conv, Dernier}) -> conversation(Racine, N, Dernier, P_conv) end, Racines),\r
Conversation_principale = resultat_transaction(transaction(fun() ->
- Dernier_id = mnesia:last(minichat),\r
+ Dernier_id = mnesia:table_info(minichat, size),\r
{CP, Plus} = conversation_principale(Dernier_id, Conversations, N, P),\r
{[M || M <- CP, M > D], Plus} % filtre en fonction de D\r
end)),\r
% @spec conversation_principale2(integer(), [integer()], integer(), integer()) -> [integer()]\r
conversation_principale2(_, _, 0, _) ->\r
[];
-conversation_principale2('$end_of_table', _, _, _) ->
+conversation_principale2(0, _, _, _) ->\r
[];\r
conversation_principale2(Id, Messages, N, S) ->
- % traitement message par message (pas des plus performant :/)
- Id_prev = mnesia:prev(minichat, Id),\r
+ % traitement message par message (pas des plus performant :/)\r
Doit_etre_saute = any(fun(E) -> E == Id end, Messages),\r
if Doit_etre_saute -> \r
- conversation_principale2(Id_prev, Messages, N, S); % le message ne fait pas partie de la conversation\r
+ conversation_principale2(Id - 1, Messages, N, S); % le message ne fait pas partie de la conversation\r
S =:= 0 ->\r
- [Id | conversation_principale2(Id_prev, Messages, N - 1, S)]; % ok : le message fait partie de la conversation\r
+ [Id | conversation_principale2(Id - 1, Messages, N - 1, S)]; % ok : le message fait partie de la conversation\r
true ->\r
- conversation_principale2(Id_prev, Messages, N, S - 1) % on n'a pas encore atteint le début de la page\r
+ conversation_principale2(Id - 1, Messages, N, S - 1) % on n'a pas encore atteint le début de la page\r
end.\r
\r
\r
{Racine_id, _, _} = Racine,\r
case euphorik_bd:message_by_id(Racine_id) of\r
{ok, Mess} ->\r
- json_message(Mess, euphorik_bd:parents(Racine), User);\r
+ json_message(Mess, euphorik_bd:parents(Racine_id), User);\r
_ ->\r
null\r
end\r
{struct, [\r
{id, Mess#minichat.id},\r
{user_id, User_mess#user.id},\r
- {date, format_date(Mess#minichat.date)},\r
+ {date, case Mess#minichat.date of undefined -> "?"; _ -> format_date(Mess#minichat.date) end},\r
{system, Mess#minichat.auteur_id =:= 0},\r
{owner, Est_proprietaire},\r
{answered, A_repondu_a_message},\r
\r
\r
% Point d'entrée pour les requêtes AJAX sur http://www.euphorik.ch/request.\r
-out(A) ->\r
+out(A) -> \r
IP = case inet:peername(A#arg.clisock) of\r
{ok, {Adresse, _Port}} -> Adresse;\r
_ -> inconnue\r
pseudo, % chaine de caractère
contenu, % chaine de caractère
racine_id = undefined % la racine, par défaut correspond à l'id du message
+ status = normal % can be equal to normal, censored or deleted
}).
font-weight: bold
}
+/***** Une barre de progression s'affichant lors des communications entre le client et le serveur *****/
+#waitbar {
+ text-align: center;
+ height: 16px;
+ width:100%;
+ position: fixed;
+ left: 0px;
+ top: 0px;
+ z-index: 500;
+}
+#waitbar .image {
+ background-image: url(../../img/waitbar.gif);
+ margin-top: 3px;
+ margin-right: 3px;
+ float: right;
+ height: 13px;
+ width: 105px;
+}
+
+/***** Textile *****/
+em.leger {
+ font-style: italic
+}
+em.fort {
+ font-style: normal;
+ font-weight: bold
+}
+
/***** Menu *****/
ul#menu {
background-image: url(img/logo_fond.png);
padding-left: 10px;
}
+#page.minichat form#posterMessage p > input,
+#page.minichat form#posterMessage p > span,
+#page.minichat form#posterMessage p > button{
+ vertical-align: middle;
+}
+
#page.minichat.orderReverse form#posterMessage {
padding-bottom: 10px;
}
background-image: url(img/return.png);
background-repeat: no-repeat;
background-position: 5px 2px;
- vertical-align: top;
margin-right: 5px;
}
#page.minichat form#posterMessage .smiles {
background-image: url(../../img/bouton_smiles.png);
background-repeat: no-repeat;
background-position: 2px 2px;
- vertical-align: top;
margin-right: 5px;
}
/* La boite montrant les messages auquels l'utilisateur répond */
#!/bin/bash
# coding: utf-8
# screen est utilisé par exemple pour lancé une version de preproduction et permettre de la faire tourner après un délog
-yaws --conf ./yaws.conf --sname yaws_dev --mnesiadir "../var/BD/" -I debian_yaws_dev\r
+yaws --conf ./yaws.conf --sname yaws_dev --mnesiadir "../var/BD/" -I debian_yaws_dev
# Effectue la mise en préproduction.
def miseEnPreProd
copierFichiers()
- copierVAR()
lancerYaws()
maj('yaws_dev')
end
set_droits_fichiers()
end
- # TODO
- def copierVar
- #TODO
- end
-
def lancerYaws
creer_rep("tools")
system("rsync tools/yaws.conf #{@uri}:#{@rep}/tools")
Dir.chdir('..')
end
- def creer_repertoire_var
+ def creer_repertoire_bd
+ log "création du répertoire de la base de données"
# création du repertoire BD
creer_rep('var')
creer_rep('var/images')