liquid_feedback_frontend

annotate app/main/initiative/_show.lua @ 907:319161d5a904

Some text phrases updated
author bsw
date Sun Sep 23 16:36:44 2012 +0200 (2012-09-23)
parents ae9ab3edff89
children c7215c18168b
rev   line source
bsw@725 1 local initiative = param.get("initiative", "table")
bsw@725 2
bsw@725 3 local show_as_head = param.get("show_as_head", atom.boolean)
bsw@725 4
bsw@725 5 initiative:load_everything_for_member_id(app.session.member_id)
bsw@725 6
bsw@725 7 local issue = initiative.issue
bsw@725 8
bsw@725 9 -- TODO performance
bsw@725 10 local initiator
bsw@725 11 if app.session.member_id then
bsw@725 12 initiator = Initiator:by_pk(initiative.id, app.session.member.id)
bsw@725 13 end
bsw@725 14
bsw@725 15 if app.session.member_id then
bsw@725 16 issue:load_everything_for_member_id(app.session.member_id)
bsw@725 17 end
bsw@725 18
bsw@725 19 app.html_title.title = initiative.name
bsw@725 20 app.html_title.subtitle = _("Initiative ##{id}", { id = initiative.id })
bsw@725 21
bsw@725 22 slot.select("head", function()
bsw@725 23 execute.view{
bsw@725 24 module = "issue", view = "_head",
bsw@725 25 params = { issue = issue, initiative = initiative }
bsw@725 26 }
bsw@725 27 end)
bsw@725 28
bsw@725 29 local initiators_members_selector = initiative:get_reference_selector("initiating_members")
bsw@725 30 :add_field("initiator.accepted", "accepted")
bsw@725 31 :add_order_by("member.name")
bsw@725 32 if initiator and initiator.accepted then
bsw@725 33 initiators_members_selector:add_where("initiator.accepted ISNULL OR initiator.accepted")
bsw@725 34 else
bsw@725 35 initiators_members_selector:add_where("initiator.accepted")
bsw@725 36 end
bsw@725 37
bsw@725 38 local initiators = initiators_members_selector:exec()
bsw@725 39
bsw@725 40
bsw@725 41 local initiatives_selector = initiative.issue:get_reference_selector("initiatives")
bsw@725 42 slot.select("head", function()
bsw@725 43 execute.view{
bsw@725 44 module = "issue",
bsw@725 45 view = "_show",
bsw@725 46 params = {
bsw@725 47 issue = initiative.issue,
bsw@725 48 initiative_limit = 3,
bsw@725 49 for_initiative = initiative
bsw@725 50 }
bsw@725 51 }
bsw@725 52 end)
bsw@725 53
bsw@725 54 util.help("initiative.show")
bsw@725 55
bsw@898 56 local class = "initiative_head"
bsw@898 57
bsw@898 58 if initiative.polling then
bsw@898 59 class = class .. " polling"
bsw@898 60 end
bsw@898 61
bsw@898 62 ui.container{ attr = { class = class }, content = function()
bsw@725 63
bsw@725 64 local text = _("Initiative i#{id}: #{name}", { id = initiative.id, name = initiative.name })
bsw@725 65 if show_as_head then
bsw@725 66 ui.link{
bsw@725 67 attr = { class = "title" }, text = text,
bsw@725 68 module = "initiative", view = "show", id = initiative.id
bsw@725 69 }
bsw@725 70 else
bsw@725 71 ui.container{ attr = { class = "title" }, content = text }
bsw@725 72 end
bsw@813 73 if app.session:has_access("authors_pseudonymous") then
bsw@727 74 ui.container{ attr = { class = "content" }, content = function()
bsw@725 75 ui.tag{
bsw@725 76 attr = { class = "initiator_names" },
bsw@725 77 content = function()
bsw@725 78 for i, initiator in ipairs(initiators) do
bsw@725 79 slot.put(" ")
bsw@813 80 if app.session:has_access("all_pseudonymous") then
bsw@725 81 ui.link{
bsw@725 82 content = function ()
bsw@725 83 execute.view{
bsw@725 84 module = "member_image",
bsw@725 85 view = "_show",
bsw@725 86 params = {
bsw@725 87 member = initiator,
bsw@725 88 image_type = "avatar",
bsw@725 89 show_dummy = true,
bsw@725 90 class = "micro_avatar",
bsw@725 91 popup_text = text
bsw@725 92 }
bsw@725 93 }
bsw@725 94 end,
bsw@725 95 module = "member", view = "show", id = initiator.id
bsw@725 96 }
bsw@725 97 slot.put(" ")
bsw@725 98 end
bsw@725 99 ui.link{
bsw@725 100 text = initiator.name,
bsw@725 101 module = "member", view = "show", id = initiator.id
bsw@725 102 }
bsw@725 103 if not initiator.accepted then
bsw@725 104 ui.tag{ attr = { title = _"Not accepted yet" }, content = "?" }
bsw@725 105 end
bsw@725 106 end
bsw@725 107 if initiator and initiator.accepted and not initiative.issue.fully_frozen and not initiative.issue.closed and not initiative.revoked then
bsw@725 108 slot.put(" · ")
bsw@725 109 ui.link{
bsw@725 110 attr = { class = "action" },
bsw@725 111 content = function()
bsw@725 112 slot.put(_"Invite initiator")
bsw@725 113 end,
bsw@725 114 module = "initiative",
bsw@725 115 view = "add_initiator",
bsw@725 116 params = { initiative_id = initiative.id }
bsw@725 117 }
bsw@725 118 if #initiators > 1 then
bsw@725 119 slot.put(" · ")
bsw@725 120 ui.link{
bsw@725 121 content = function()
bsw@725 122 slot.put(_"Remove initiator")
bsw@725 123 end,
bsw@725 124 module = "initiative",
bsw@725 125 view = "remove_initiator",
bsw@725 126 params = { initiative_id = initiative.id }
bsw@725 127 }
bsw@725 128 end
bsw@725 129 end
bsw@725 130 if initiator and initiator.accepted == false then
bsw@725 131 slot.put(" · ")
bsw@725 132 ui.link{
bsw@725 133 text = _"Cancel refuse of invitation",
bsw@725 134 module = "initiative",
bsw@725 135 action = "remove_initiator",
bsw@725 136 params = {
bsw@725 137 initiative_id = initiative.id,
bsw@725 138 member_id = app.session.member.id
bsw@725 139 },
bsw@725 140 routing = {
bsw@725 141 ok = {
bsw@725 142 mode = "redirect",
bsw@725 143 module = "initiative",
bsw@725 144 view = "show",
bsw@725 145 id = initiative.id
bsw@725 146 }
bsw@725 147 }
bsw@725 148 }
bsw@725 149 end
bsw@725 150 if (initiative.discussion_url and #initiative.discussion_url > 0) then
bsw@725 151 slot.put(" · ")
bsw@725 152 if initiative.discussion_url:find("^https?://") then
bsw@725 153 if initiative.discussion_url and #initiative.discussion_url > 0 then
bsw@725 154 ui.link{
bsw@725 155 attr = {
bsw@725 156 target = "_blank",
bsw@725 157 title = _"Discussion with initiators"
bsw@725 158 },
bsw@725 159 text = _"Discuss with initiators",
bsw@725 160 external = initiative.discussion_url
bsw@725 161 }
bsw@725 162 end
bsw@725 163 else
bsw@725 164 slot.put(encode.html(initiative.discussion_url))
bsw@725 165 end
bsw@725 166 end
bsw@725 167 if initiator and initiator.accepted and not initiative.issue.half_frozen and not initiative.issue.closed and not initiative.revoked then
bsw@725 168 slot.put(" · ")
bsw@725 169 ui.link{
bsw@725 170 text = _"change discussion URL",
bsw@725 171 module = "initiative",
bsw@725 172 view = "edit",
bsw@725 173 id = initiative.id
bsw@725 174 }
bsw@725 175 slot.put(" ")
bsw@725 176 end
bsw@725 177 end
bsw@725 178 }
bsw@727 179 end }
bsw@727 180 end
bsw@725 181
bsw@727 182 if app.session.member_id then
bsw@727 183 ui.container{ attr = { class = "content" }, content = function()
bsw@725 184 execute.view{
bsw@725 185 module = "supporter",
bsw@725 186 view = "_show_box",
bsw@725 187 params = {
bsw@725 188 initiative = initiative
bsw@725 189 }
bsw@725 190 }
bsw@727 191 end }
bsw@727 192 end
bsw@725 193
bsw@725 194
bsw@725 195 -- voting results
bsw@725 196 if initiative.issue.ranks_available and initiative.admitted then
bsw@725 197 local class = initiative.winner and "admitted_info" or "not_admitted_info"
bsw@725 198 ui.container{
bsw@725 199 attr = { class = class },
bsw@725 200 content = function()
bsw@725 201 local max_value = initiative.issue.voter_count
bsw@725 202 slot.put(" ")
bsw@725 203 local positive_votes = initiative.positive_votes
bsw@725 204 local negative_votes = initiative.negative_votes
bsw@725 205 local sum_votes = initiative.positive_votes + initiative.negative_votes
bsw@725 206 local function perc(votes, sum)
bsw@725 207 if sum > 0 and votes > 0 then return " (" .. string.format( "%.f", votes * 100 / sum ) .. "%)" end
bsw@725 208 return ""
bsw@725 209 end
bsw@725 210 slot.put(_"Yes" .. ": <b>" .. tostring(positive_votes) .. perc(positive_votes, sum_votes) .. "</b>")
bsw@725 211 slot.put(" &middot; ")
bsw@725 212 slot.put(_"Abstention" .. ": <b>" .. tostring(max_value - initiative.negative_votes - initiative.positive_votes) .. "</b>")
bsw@725 213 slot.put(" &middot; ")
bsw@725 214 slot.put(_"No" .. ": <b>" .. tostring(initiative.negative_votes) .. perc(negative_votes, sum_votes) .. "</b>")
bsw@725 215 slot.put(" &middot; ")
bsw@725 216 slot.put("<b>")
bsw@725 217 if initiative.winner then
bsw@725 218 slot.put(_"Approved")
bsw@725 219 elseif initiative.rank then
bsw@725 220 slot.put(_("Not approved (rank #{rank})", { rank = initiative.rank }))
bsw@725 221 else
bsw@725 222 slot.put(_"Not approved")
bsw@725 223 end
bsw@725 224 slot.put("</b>")
bsw@725 225 end
bsw@725 226 }
bsw@725 227 end
bsw@725 228
bsw@725 229 ui.container{ attr = { class = "content" }, content = function()
bsw@725 230 execute.view{
bsw@725 231 module = "initiative",
bsw@725 232 view = "_battles",
bsw@725 233 params = { initiative = initiative }
bsw@725 234 }
bsw@725 235 end }
bsw@725 236
bsw@725 237 -- initiative not admitted info
bsw@725 238 if initiative.admitted == false then
bsw@725 239 local policy = initiative.issue.policy
bsw@725 240 ui.container{
bsw@725 241 attr = { class = "not_admitted_info" },
bsw@725 242 content = _("This initiative has not been admitted! It failed the quorum of #{quorum}.", { quorum = format.percentage(policy.initiative_quorum_num / policy.initiative_quorum_den) })
bsw@725 243 }
bsw@725 244 end
bsw@725 245
bsw@725 246 -- initiative revoked info
bsw@725 247 if initiative.revoked then
bsw@725 248 ui.container{
bsw@725 249 attr = { class = "revoked_info" },
bsw@725 250 content = function()
bsw@725 251 slot.put(_("This initiative has been revoked at #{revoked}", { revoked = format.timestamp(initiative.revoked) }))
bsw@725 252 local suggested_initiative = initiative.suggested_initiative
bsw@725 253 if suggested_initiative then
bsw@725 254 slot.put("<br /><br />")
bsw@725 255 slot.put(_("The initiators suggest to support the following initiative:"))
bsw@725 256 slot.put(" ")
bsw@725 257 ui.link{
bsw@725 258 content = _("Issue ##{id}", { id = suggested_initiative.issue.id } ) .. ": " .. encode.html(suggested_initiative.name),
bsw@725 259 module = "initiative",
bsw@725 260 view = "show",
bsw@725 261 id = suggested_initiative.id
bsw@725 262 }
bsw@725 263 end
bsw@725 264 end
bsw@725 265 }
bsw@725 266 end
bsw@725 267
bsw@725 268
bsw@725 269 -- invited as initiator
bsw@725 270 if initiator and initiator.accepted == nil and not initiative.issue.half_frozen and not initiative.issue.closed then
bsw@725 271 ui.container{
bsw@725 272 attr = { class = "initiator_invite_info" },
bsw@725 273 content = function()
bsw@725 274 slot.put(_"You are invited to become initiator of this initiative.")
bsw@725 275 slot.put(" ")
bsw@725 276 ui.link{
bsw@725 277 image = { static = "icons/16/tick.png" },
bsw@725 278 text = _"Accept invitation",
bsw@725 279 module = "initiative",
bsw@725 280 action = "accept_invitation",
bsw@725 281 id = initiative.id,
bsw@725 282 routing = {
bsw@725 283 default = {
bsw@725 284 mode = "redirect",
bsw@725 285 module = request.get_module(),
bsw@725 286 view = request.get_view(),
bsw@725 287 id = param.get_id_cgi(),
bsw@725 288 params = param.get_all_cgi()
bsw@725 289 }
bsw@725 290 }
bsw@725 291 }
bsw@725 292 slot.put(" ")
bsw@725 293 ui.link{
bsw@725 294 image = { static = "icons/16/cross.png" },
bsw@725 295 text = _"Refuse invitation",
bsw@725 296 module = "initiative",
bsw@725 297 action = "reject_initiator_invitation",
bsw@725 298 params = {
bsw@725 299 initiative_id = initiative.id,
bsw@725 300 member_id = app.session.member.id
bsw@725 301 },
bsw@725 302 routing = {
bsw@725 303 default = {
bsw@725 304 mode = "redirect",
bsw@725 305 module = request.get_module(),
bsw@725 306 view = request.get_view(),
bsw@725 307 id = param.get_id_cgi(),
bsw@725 308 params = param.get_all_cgi()
bsw@725 309 }
bsw@725 310 }
bsw@725 311 }
bsw@725 312 end
bsw@725 313 }
bsw@725 314 end
bsw@725 315
bsw@725 316 -- draft updated
bsw@725 317 local supporter
bsw@725 318
bsw@725 319 if app.session.member_id then
bsw@725 320 supporter = app.session.member:get_reference_selector("supporters")
bsw@725 321 :add_where{ "initiative_id = ?", initiative.id }
bsw@725 322 :optional_object_mode()
bsw@725 323 :exec()
bsw@725 324 end
bsw@725 325
bsw@725 326 if supporter and not initiative.issue.closed then
bsw@725 327 local old_draft_id = supporter.draft_id
bsw@725 328 local new_draft_id = initiative.current_draft.id
bsw@725 329 if old_draft_id ~= new_draft_id then
bsw@725 330 ui.container{
bsw@725 331 attr = { class = "draft_updated_info" },
bsw@725 332 content = function()
bsw@725 333 slot.put(_"The draft of this initiative has been updated!")
bsw@725 334 slot.put(" ")
bsw@725 335 ui.link{
bsw@725 336 content = _"Show diff",
bsw@725 337 module = "draft",
bsw@725 338 view = "diff",
bsw@725 339 params = {
bsw@725 340 old_draft_id = old_draft_id,
bsw@725 341 new_draft_id = new_draft_id
bsw@725 342 }
bsw@725 343 }
bsw@725 344 if not initiative.revoked then
bsw@725 345 slot.put(" ")
bsw@725 346 ui.link{
bsw@725 347 text = _"Refresh support to current draft",
bsw@725 348 module = "initiative",
bsw@725 349 action = "add_support",
bsw@725 350 id = initiative.id,
bsw@725 351 routing = {
bsw@725 352 default = {
bsw@725 353 mode = "redirect",
bsw@725 354 module = "initiative",
bsw@725 355 view = "show",
bsw@725 356 id = initiative.id
bsw@725 357 }
bsw@725 358 }
bsw@725 359 }
bsw@725 360 end
bsw@725 361 end
bsw@725 362 }
bsw@725 363 end
bsw@725 364 end
bsw@725 365
bsw@725 366 if not show_as_head then
bsw@725 367 local drafts_count = initiative:get_reference_selector("drafts"):count()
bsw@725 368
bsw@725 369 ui.container{ attr = { class = "content" }, content = function()
bsw@725 370
bsw@725 371 if initiator and initiator.accepted and not initiative.issue.half_frozen and not initiative.issue.closed and not initiative.revoked then
bsw@725 372 ui.link{
bsw@725 373 content = function()
bsw@725 374 slot.put(_"Edit draft")
bsw@725 375 end,
bsw@725 376 module = "draft",
bsw@725 377 view = "new",
bsw@725 378 params = { initiative_id = initiative.id }
bsw@725 379 }
bsw@725 380 slot.put(" &middot; ")
bsw@725 381 ui.link{
bsw@725 382 content = function()
bsw@725 383 slot.put(_"Revoke initiative")
bsw@725 384 end,
bsw@725 385 module = "initiative",
bsw@725 386 view = "revoke",
bsw@725 387 id = initiative.id
bsw@725 388 }
bsw@725 389 slot.put(" &middot; ")
bsw@725 390 end
bsw@725 391
bsw@725 392 ui.tag{
bsw@725 393 attr = { class = "draft_version" },
bsw@725 394 content = _("Latest draft created at #{date} #{time}", {
bsw@725 395 date = format.date(initiative.current_draft.created),
bsw@725 396 time = format.time(initiative.current_draft.created)
bsw@725 397 })
bsw@725 398 }
bsw@725 399 if drafts_count > 1 then
bsw@725 400 slot.put(" &middot; ")
bsw@725 401 ui.link{
bsw@725 402 module = "draft", view = "list", params = { initiative_id = initiative.id },
bsw@725 403 text = _("List all revisions (#{count})", { count = drafts_count })
bsw@725 404 }
bsw@725 405 end
bsw@725 406 end }
bsw@725 407
bsw@725 408 execute.view{
bsw@725 409 module = "draft",
bsw@725 410 view = "_show",
bsw@725 411 params = {
bsw@725 412 draft = initiative.current_draft
bsw@725 413 }
bsw@725 414 }
bsw@725 415 end
bsw@725 416 end }
bsw@725 417
bsw@725 418 if not show_as_head then
bsw@725 419 execute.view{
bsw@725 420 module = "suggestion",
bsw@725 421 view = "_list",
bsw@725 422 params = {
bsw@725 423 initiative = initiative,
bsw@725 424 suggestions_selector = initiative:get_reference_selector("suggestions"),
bsw@725 425 tab_id = param.get("tab_id")
bsw@725 426 }
bsw@725 427 }
bsw@725 428
bsw@725 429
bsw@813 430 if app.session:has_access("all_pseudonymous") then
bsw@725 431 if initiative.issue.ranks_available then
bsw@725 432 local members_selector = initiative.issue:get_reference_selector("direct_voters")
bsw@725 433 :left_join("vote", nil, { "vote.initiative_id = ? AND vote.member_id = member.id", initiative.id })
bsw@725 434 :add_field("direct_voter.weight as voter_weight")
bsw@725 435 :add_field("coalesce(vote.grade, 0) as grade")
bsw@879 436 :add_field("direct_voter.comment as voter_comment")
bsw@725 437 :left_join("initiative", nil, "initiative.id = vote.initiative_id")
bsw@725 438 :left_join("issue", nil, "issue.id = initiative.issue_id")
bsw@725 439
bsw@907 440 ui.anchor{ name = "voter", attr = { class = "heading" }, content = _"Voters" }
bsw@725 441
bsw@725 442 execute.view{
bsw@725 443 module = "member",
bsw@725 444 view = "_list",
bsw@725 445 params = {
bsw@725 446 initiative = initiative,
bsw@725 447 for_votes = true,
bsw@725 448 members_selector = members_selector,
bsw@725 449 paginator_name = "voter"
bsw@725 450 }
bsw@725 451 }
bsw@725 452 end
bsw@725 453
bsw@725 454 local members_selector = initiative:get_reference_selector("supporting_members_snapshot")
bsw@725 455 :join("issue", nil, "issue.id = direct_supporter_snapshot.issue_id")
bsw@725 456 :join("direct_interest_snapshot", nil, "direct_interest_snapshot.event = issue.latest_snapshot_event AND direct_interest_snapshot.issue_id = issue.id AND direct_interest_snapshot.member_id = member.id")
bsw@725 457 :add_field("direct_interest_snapshot.weight")
bsw@725 458 :add_where("direct_supporter_snapshot.event = issue.latest_snapshot_event")
bsw@725 459 :add_where("direct_supporter_snapshot.satisfied")
bsw@725 460 :add_field("direct_supporter_snapshot.informed", "is_informed")
bsw@725 461
bsw@725 462 if members_selector:count() > 0 then
bsw@725 463 if issue.fully_frozen then
bsw@725 464 ui.anchor{ name = "supporters", attr = { class = "heading" }, content = _"Supporters (before begin of voting)" }
bsw@725 465 else
bsw@725 466 ui.anchor{ name = "supporters", attr = { class = "heading" }, content = _"Supporters" }
bsw@725 467 end
bsw@725 468
bsw@725 469 execute.view{
bsw@725 470 module = "member",
bsw@725 471 view = "_list",
bsw@725 472 params = {
bsw@725 473 initiative = initiative,
bsw@725 474 members_selector = members_selector,
bsw@725 475 paginator_name = "supporters"
bsw@725 476 }
bsw@725 477 }
bsw@725 478 else
bsw@725 479 if issue.fully_frozen then
bsw@725 480 ui.anchor{ name = "supporters", attr = { class = "heading" }, content = _"No supporters (before begin of voting)" }
bsw@725 481 else
bsw@725 482 ui.anchor{ name = "supporters", attr = { class = "heading" }, content = _"No supporters" }
bsw@725 483 end
bsw@725 484 slot.put("<br />")
bsw@725 485 end
bsw@725 486
bsw@725 487 local members_selector = initiative:get_reference_selector("supporting_members_snapshot")
bsw@725 488 :join("issue", nil, "issue.id = direct_supporter_snapshot.issue_id")
bsw@725 489 :join("direct_interest_snapshot", nil, "direct_interest_snapshot.event = issue.latest_snapshot_event AND direct_interest_snapshot.issue_id = issue.id AND direct_interest_snapshot.member_id = member.id")
bsw@725 490 :add_field("direct_interest_snapshot.weight")
bsw@725 491 :add_where("direct_supporter_snapshot.event = issue.latest_snapshot_event")
bsw@725 492 :add_where("NOT direct_supporter_snapshot.satisfied")
bsw@725 493 :add_field("direct_supporter_snapshot.informed", "is_informed")
bsw@725 494
bsw@725 495 if members_selector:count() > 0 then
bsw@725 496 if issue.fully_frozen then
bsw@725 497 ui.anchor{ name = "potential_supporters", attr = { class = "heading" }, content = _"Potential supporters (before begin of voting)" }
bsw@725 498 else
bsw@725 499 ui.anchor{ name = "potential_supporters", attr = { class = "heading" }, content = _"Potential supporters" }
bsw@725 500 end
bsw@725 501
bsw@725 502 execute.view{
bsw@725 503 module = "member",
bsw@725 504 view = "_list",
bsw@725 505 params = {
bsw@725 506 initiative = initiative,
bsw@725 507 members_selector = members_selector,
bsw@725 508 paginator_name = "potential_supporters"
bsw@725 509 }
bsw@725 510 }
bsw@725 511 else
bsw@725 512 if issue.fully_frozen then
bsw@725 513 ui.anchor{ name = "potential_supporters", attr = { class = "heading" }, content = _"No potential supporters (before begin of voting)" }
bsw@725 514 else
bsw@725 515 ui.anchor{ name = "potential_supporters", attr = { class = "heading" }, content = _"No potential supporters" }
bsw@725 516 end
bsw@725 517 slot.put("<br />")
bsw@725 518 end
bsw@725 519
bsw@725 520 ui.container{ attr = { class = "heading" }, content = _"Details" }
bsw@725 521 execute.view {
bsw@725 522 module = "initiative",
bsw@725 523 view = "_details",
bsw@725 524 params = {
bsw@725 525 initiative = initiative,
bsw@725 526 members_selector = members_selector
bsw@725 527 }
bsw@725 528 }
bsw@725 529
bsw@725 530 end
bsw@725 531 end

Impressum / About Us