Inhaltsverzeichnis
keine Gliederung
// SimplePoll // originally by neilw, 2009 // translated by carles.coll, 2009 // // Version history: // 1.00 20-August-2009 neilw First published version // 1.01 26-August-2009 neilw Each poll uses a different channel and __request arg so // multiple polls can coexist on the same page // Changed display style of poll results a bit // Added "bar" argument // 1.02 21-October-2009 charles.coll Get the result detail (who) // Added Fixed Behaviour parameter // Now the parameter path works // Multilanguage support // 1.03 23-October-2009 neilw Cleaned up "View details" display and other minor code tweaks // 1.04 15-December-2009 neilw Fixed a timezone-processing bug // 1.05 16-December-2009 neilw Added Italian translation, thanks Giulio! // 1.06 13-January-2010 neilw Fixed language processing problem // 1.07 18-March-2010 varosi Added option to align poll differently // 1.08 29-August-2010 neilw Fixed HTML entity syntax for 10.0 // // Usage: SimplePoll(question:str, answers:list of str, name:str?, path:str?, closed:str? or bool?, color_bar:str?, fixed_behavior: str?, language: str?) // question: The poll question // answers: List of possible answers // name: (optional) Unique name for the poll, in case you want to store multiple poll results // on the same page properties // path: (optional) Path to page where properties will be stored. Must be readable/writable // by voters! // closed: (optional) Either "true" to close the poll, or a valid date (GMT!!!!) string indicating // when the poll should close // color_bar: (optional) color of results bars (default: "#B00000") // fixed_behavior: (optional) default -> Automatic. // edit -> Edit Poll // view -> View Poll Statistics // view_details -> View who answers what // language: (optional) default -> // 1st -> Page language // 2nd -> Site language // 3rd -> 'en-us' // halign: (optional) Horizontal poll alignment // default -> center // // // get initialized // var question = $0 ?? $question ?? ""; var answers = $1 ?? $answers ?? []; var fixed_behavior = $fixed_behavior ?? ""; var language = __request.args.language ?? args.language ?? ( page.language..'' != '' ? page.language : (site.language..'' != '' ? site.language : 'en-us') ); var halign = $halign ?? "center"; // -- START LANGUAGE STRINGS var LANGUAGE_ES = { warning_question: "CUIDADO: la pregunta tiene que ser una cadena de carecteres o un XML", warning_no_answers: "CUIDADO: No has pasado respuestas. Así no podremos hacer un encuesta!", error_no_data_page: "ERROR: no puedo encontrar la pagina con los datos de la encuesta", error_invalid_close_date: "ERROR: 'cerrada' no es una fecha valida; asumo que la encuesta esta abierta", error_invalid_data: "ERROR: los datos de la encuesta existen pero tienen un formato erroneo (tiene que ser de tipo 'map')", error_updating_vote: "ERROR: actualizando el voto ", error_creating_vote: "ERROR: creando el voto ", error_reading_vote: "ERROR: leyendo el voto ", txt_your_answer: "Tu respuesta: ", txt_change_my_vote: "cambiar mi voto", txt_vote: "votar en esta encuesta", txt_view_results: "ver resultados", txt_poll_closes_in: "La encuesta se cierra en ", txt_votes: " votos contados", txt_view_details: "ver detalles", txt_poll_closed: "esta encuesta esta cerrada", msg_poll_closed: "Ahora la encuesta esta cerrada, lo siento!", button_submit: "guardar" }; var LANGUAGE_EN = { warning_question: "WARNING: question must be string or xml", warning_no_answers: "WARNING: You have provided no answers. Not much of a poll!", error_no_data_page: "ERROR: can't find page with data store", error_invalid_close_date: "ERROR: 'closed' is not a valid date; assuming poll is open", error_invalid_data: "ERROR: poll data store exists but has the wrong type (instead of 'map')", error_updating_vote: "ERROR: updating poll ", error_creating_vote: "ERROR: creating poll ", error_reading_vote: "ERROR: reding poll ", txt_your_answer: "Your answer: ", txt_change_my_vote: "change my vote", txt_vote: "vote in this poll", txt_view_results: "view results", txt_poll_closes_in: "Poll closes on ", txt_votes: " votes counted", txt_view_details: "view details", txt_poll_closed: "this poll is closed", msg_poll_closed: "Poll is now closed, sorry!", button_submit: "submit" }; var LANGUAGE_IT = { warning_question: "ATTENZIONE: la domanda deve essere una stringa o xml", warning_no_answers: "ATTENZIONE: non hai risposto. Così non riusciamo a fare un sondaggio!!", error_no_data_page: "ERRORE: non trovo la pagina con i dati del sondaggio", error_invalid_close_date: "ERRORE: 'chiuso' non è una data valida; si presume che il sondaggio sia aperto", error_invalid_data: "ERRORE: lo store per i dati del sondaggio esiste ma ha un formato errato (deve essere di tipo 'map')", error_updating_vote: "ERRORE: aggiornamento sondaggio ", error_creating_vote: "ERRORE: creazione sondaggio ", error_reading_vote: "ERRORE: lettura sondaggio ", txt_your_answer: "La tua risposta: ", txt_change_my_vote: "Cambia il mio voto", txt_vote: "Vota il sondaggio", txt_view_results: "Risultati", txt_poll_closes_in: "Il sondaggio chiude il ", txt_votes: " voti", txt_view_details: "Dettaglio dei voti", txt_poll_closed: "Questo sondaggio è finito", msg_poll_closed: "Il sondaggio adesso è chiuso, spiacenti!", button_submit: "submit" }; var TXTS = { en: LANGUAGE_EN, 'en-us': LANGUAGE_EN, es: LANGUAGE_ES, 'es-es': LANGUAGE_ES, it: LANGUAGE_IT, 'it-it': LANGUAGE_IT }; var lg = language; // -- END LANGUAGE STRINGS if (question is not str && question is not xml) <p>TXTS[lg].warning_question;</p>; if (#answers == 0) <p>TXTS[lg].warning_no_answers;</p>; var CONST_POLLDATA_NAME = "poll_default_name"; var poll_name = $2 ?? $name?? CONST_POLLDATA_NAME; if (poll_name is not str) { /* <p>"ERROR: el nombre tiene que ser una cadena de caracteres. Le assignamos automáticamente el nombre '"..CONST_POLLDATA_NAME.."'"</p>; */ let poll_name = CONST_POLLDATA_NAME; } var poll_arg = "poll_"..poll_name; var path = $3 ?? $path; var p = (path == nil ? page : wiki.getpage(path)); var p_api = null; if (p == nil) { <p>TXTS[lg].error_no_data_page;</p>;} else { let p_api = p.api; } var closed = $4 ?? $closed ?? false; var closing_time = false; if (closed is not bool) { if (!date.isvalid(closed)) { <p>TXTS[lg].error_invalid_close_date;</p>; let closed = false; } else { // convert to local time let closing_time = date.format(closed,"r"); let closed = date.compare(date.now, closing_time) > 0; } } var bar = $5 ?? $color_bar ?? "#B00000"; var viewURI = page.uri & { (poll_arg):"view" }; var viewDetailsURI = page.uri & { (poll_arg):"view_details" }; var editURI = page.uri & { (poll_arg):"edit" }; // Fetch the data store var data = json.parse(p.properties[poll_name].text ?? '{}'); if (data is not map) <p>TXTS[lg].error_invalid_data</p>; // Now figure out what to show var vote = (data[user.name] ?? {}).poll; var can_vote = !closed && !user.anonymous && wiki.pagepermissions(path).update; var has_voted = (vote != nil); var showform = (fixed_behavior=='edit') || ( (fixed_behavior=='') && (can_vote && (!has_voted || __request.args[poll_arg] == "edit") && __request.args[poll_arg] != "view" && __request.args[poll_arg] != "view_details" ) ); <table align=(halign) cellpadding="5" style="background-color:#F4F4F4; border:2px solid #808080"> <tr><td align="center" style="font-weight:bold; padding-bottom:10px"> question </td></tr>; if (showform) { // Allow user to enter or edit poll response(s) <tr><td align="center"><form id=(@form)> <ul style="text-align:left"> foreach (var opt in answers) { if ((fixed_behavior=='edit') && (__request.args[poll_arg]!='edit') && (has_voted)) { if(has_voted && vote==__index) { <B>TXTS[lg].txt_your_answer;</B>opt; } } else { <li style="list-style:none"> <input type="radio" name="poll" value=(__index) checked=(has_voted && vote==__index ? 'checked' : nil)> " "..opt.." " </input> </li>; } } </ul> <span style="text-align:center; padding-top:10px"> if ((fixed_behavior=='edit')&& (__request.args[poll_arg]!='edit') && (has_voted)) { if (can_vote) <a href=(editURI)> has_voted ? TXTS[lg].txt_change_my_vote : TXTS[lg].txt_vote </a>; } else { <input type="button" value=(TXTS[lg].button_submit) ctor="when($this.click) { var m = { }; Deki.$('form#' + {{@form}} + ' input').each(function() { if ($(this).attr('name') == 'poll' && $(this).attr('checked')) m['poll'] = Deki.$(this).val(); }); Deki.publish({{@channel}}, { {{user.name}}:m }); }"/>; " "; } if(fixed_behavior=='') { <a href=(viewURI)> <span style="font-size:smaller">TXTS[lg].txt_view_results</span> </a> } </span> if (closing_time) { <br />; var ct = date.changetimezone(closing_time, user.timezone); let ct = date.format(string.substr(ct,0,string.lastindexof(ct," ")),"MMM d, yyyy, h:mm tt"); <span style="font-size:smaller"> TXTS[lg].txt_poll_closes_in .. ct </span>; } </form></td></tr>; } else { // Display poll results if ((fixed_behavior=='view') || ( (fixed_behavior=='') && (__request.args[poll_arg] != "view_details") )) { // Calculate results var total = #data; <tr><td align="center"> <span style="font-size:smaller"> total .. TXTS[lg].txt_votes </span>; <table> foreach (var i in num.series(0,#answers-1)) { var result = #map.select(data, "$.value.poll == "..i); var pct = num.round(100*result/num.max(total,1),1); var width = num.round(2 * pct, 0); <tr> <td> answers[i] </td> <td> <img src="/skins/common/icons/icon-trans.gif" style=("background-color:"..bar) height="10" width=(width) /> <img src="/skins/common/icons/icon-trans.gif" style="background-color:#C0C0C0" height="10" width=(200-width) /> " " .. pct .. "% (" .. result .. ")" </td> </tr>; } </table> if (fixed_behavior=='') { <span style="text-align:center; padding-top:10px; font-size:smaller"> <a href=(viewDetailsURI)>TXTS[lg].txt_view_details</a>; ; ; if (can_vote) <a href=(editURI)> has_voted ? TXTS[lg].txt_change_my_vote : TXTS[lg].txt_vote </a>; else if (closed) TXTS[lg].txt_poll_closed; </span>; } </td></tr>; } else { // Show details <tr><td align="center"> <table> <tr> foreach (var i in num.series(0,#answers-1)) { <th style="text-align:center; border:1px solid #606060"> answers[i] </th> } </tr>; <tr> foreach (var i in num.series(0,#answers-1)) { var result = Map.Keys(map.select(data, "$.value.poll == "..i)); <td valign="top" style="text-align:center"> if (#result) { <span style="font-size:.8em">"("..#result.." votes)" </span>; <br />; } foreach (var who in result) { if (__index) <br />; who; } </td> } </tr>; </table> if(fixed_behavior=='') { <span style="text-align:center; padding-top:10px; font-size:smaller"> <a href=(viewURI)> <span style="font-size:smaller">TXTS[lg].txt_view_results</span> </a> ; ; if (can_vote) <a href=(editURI)> has_voted ? TXTS[lg].txt_change_my_vote : TXTS[lg].txt_vote </a>; else if (closed) "esta encuesta esta cerrada"; </span>; } </td></tr>; } } </table>; // Code to update the page properties, largely cribbed from SteveB's "UpdateStore" template dekiapi(); var store = poll_name; <script type="text/javascript"> " Deki.subscribe('"..@channel.."', null, function(c, m, d) { var closed = " .. json.emit(closed) .. "; var closing_time = " .. json.emit(closing_time) .. "; var d = new Date(); if (closed || (closing_time && d.getTime() > Date.parse(closing_time))) { alert('"..TXTS[lg].msg_poll_closed.."'); window.location.href = '" .. page.uri .. "'; return; } var prop = 'urn:custom.mindtouch.com#' + '"..store.."'; Deki.Api.ReadPageProperty('"..p_api.."', prop, function(result) { var data = eval('(' + (result.value || '{}') + ')'); for (var k in m) data[k] = m[k]; if(result.etag) Deki.Api.UpdatePageProperty(result.href, YAHOO.lang.JSON.stringify(data), result.etag, function() { window.location.href = '" .. page.uri .. "'; }, function(result) { alert('"..TXTS[lg].error_updating_vote.." (esatdo: ' + result.status + ' - ' + result.text + ')'); } ); else Deki.Api.CreatePageProperty('"..p_api.."', prop, YAHOO.lang.JSON.stringify(data), function() { window.location.href = '" .. page.uri .. "'; }, function(result) { alert('"..TXTS[lg].error_creating_vote.." (esatdo: ' + result.status + ' - ' + result.text + ')'); } ); }, function(result) { alert('"..TXTS[lg].error_reading_vote.." (esatdo: ' + result.status + ' - ' + result.text + ')'); } ); }, null); " </script>