decidim-elections 0.24.0 → 0.25.0.rc1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +13 -0
- data/app/cells/decidim/elections/election_m/tags.erb +3 -0
- data/app/cells/decidim/elections/election_m_cell.rb +1 -8
- data/app/cells/decidim/elections/election_preview/show.erb +28 -0
- data/app/cells/decidim/elections/election_preview_cell.rb +13 -0
- data/app/cells/decidim/elections/election_results/blank_votes.erb +7 -0
- data/app/cells/decidim/elections/election_results/progress_bar.erb +16 -0
- data/app/cells/decidim/elections/election_results/show.erb +40 -0
- data/app/cells/decidim/elections/election_results_cell.rb +13 -0
- data/app/cells/decidim/elections/election_vote_cta/show.erb +35 -0
- data/app/cells/decidim/elections/election_vote_cta_cell.rb +83 -0
- data/app/cells/decidim/elections/remaining_time_callout/show.erb +5 -0
- data/app/cells/decidim/elections/remaining_time_callout_cell.rb +28 -0
- data/app/cells/decidim/elections/voting_step_navigation/show.erb +1 -1
- data/app/cells/decidim/elections/voting_step_navigation_cell.rb +1 -1
- data/app/cells/decidim/votings/content_blocks/highlighted_votings/show.erb +1 -1
- data/app/cells/decidim/votings/content_blocks/landing_page/attachments_and_folders_cell.rb +8 -1
- data/app/cells/decidim/votings/content_blocks/landing_page/description/show.erb +25 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/description_cell.rb +20 -2
- data/app/cells/decidim/votings/content_blocks/landing_page/elections/show.erb +15 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/elections/single.erb +11 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/elections_cell.rb +33 -1
- data/app/cells/decidim/votings/content_blocks/landing_page/header/show.erb +29 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/header_cell.rb +95 -2
- data/app/cells/decidim/votings/content_blocks/landing_page/header_settings_form/show.erb +4 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/{stats_cell.rb → header_settings_form_cell.rb} +5 -3
- data/app/cells/decidim/votings/content_blocks/landing_page/metrics/show.erb +12 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/metrics_cell.rb +6 -2
- data/app/cells/decidim/votings/content_blocks/landing_page/polling_stations/address.erb +14 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/polling_stations/map.erb +18 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/polling_stations/show.erb +25 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/polling_stations_cell.rb +42 -1
- data/app/cells/decidim/votings/content_blocks/landing_page/statistics_cell.rb +24 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/timeline/show.erb +7 -0
- data/app/cells/decidim/votings/content_blocks/landing_page/timeline_cell.rb +4 -2
- data/app/cells/decidim/votings/polling_officers/polling_officers_picker/show.erb +1 -1
- data/app/cells/decidim/votings/polling_station_closure_certificate/show.erb +9 -0
- data/app/cells/decidim/votings/polling_station_closure_certificate_cell.rb +12 -0
- data/app/cells/decidim/votings/polling_station_closure_recount/show.erb +74 -0
- data/app/cells/decidim/votings/polling_station_closure_recount_cell.rb +9 -0
- data/app/cells/decidim/votings/voting_m_cell.rb +2 -2
- data/app/commands/decidim/elections/admin/add_user_as_trustee.rb +2 -1
- data/app/commands/decidim/elections/admin/create_election.rb +2 -1
- data/app/commands/decidim/elections/admin/publish_results.rb +10 -6
- data/app/commands/decidim/elections/admin/setup_election.rb +73 -67
- data/app/commands/decidim/elections/admin/update_answer_selection.rb +1 -1
- data/app/commands/decidim/elections/trustee_zone/update_election_bulletin_board_status.rb +61 -2
- data/app/commands/decidim/elections/trustee_zone/update_trustee.rb +3 -0
- data/app/commands/decidim/elections/voter/cast_vote.rb +4 -7
- data/app/commands/decidim/elections/voter/update_vote_status.rb +23 -7
- data/app/commands/decidim/votings/admin/create_ballot_style.rb +58 -0
- data/app/commands/decidim/votings/admin/create_voting.rb +2 -1
- data/app/commands/decidim/votings/admin/destroy_ballot_style.rb +33 -0
- data/app/commands/decidim/votings/admin/monitoring_committee_validate_polling_station_closure.rb +37 -0
- data/app/commands/decidim/votings/admin/update_ballot_style.rb +70 -0
- data/app/commands/decidim/votings/admin/update_voting.rb +5 -14
- data/app/commands/decidim/votings/census/admin/create_dataset.rb +82 -0
- data/app/commands/decidim/votings/census/admin/create_datum.rb +55 -0
- data/app/commands/decidim/votings/census/admin/destroy_dataset.rb +45 -0
- data/app/commands/decidim/votings/census/admin/increment_dataset_processed_rows.rb +43 -0
- data/app/commands/decidim/votings/census/admin/launch_access_codes_export.rb +38 -0
- data/app/commands/decidim/votings/census/admin/launch_access_codes_generation.rb +44 -0
- data/app/commands/decidim/votings/census/admin/update_dataset.rb +51 -0
- data/app/commands/decidim/votings/certify_polling_station_closure.rb +46 -0
- data/app/commands/decidim/votings/check_census.rb +35 -0
- data/app/commands/decidim/votings/create_polling_station_closure.rb +53 -0
- data/app/commands/decidim/votings/create_polling_station_results.rb +83 -0
- data/app/commands/decidim/votings/send_access_code.rb +48 -0
- data/app/commands/decidim/votings/sign_polling_station_closure.rb +41 -0
- data/app/commands/decidim/votings/voter/in_person_vote.rb +55 -0
- data/app/commands/decidim/votings/voter/update_in_person_vote_status.rb +52 -0
- data/app/controllers/concerns/decidim/elections/has_vote_flow.rb +43 -0
- data/app/controllers/concerns/decidim/monitoring_committee_polling_station_closures/admin/filterable.rb +72 -0
- data/app/controllers/concerns/decidim/votings/needs_voting.rb +5 -5
- data/app/controllers/decidim/elections/admin/steps_controller.rb +5 -2
- data/app/controllers/decidim/elections/elections_controller.rb +18 -5
- data/app/controllers/decidim/elections/feedbacks_controller.rb +37 -3
- data/app/controllers/decidim/elections/trustee_zone/elections_controller.rb +15 -1
- data/app/controllers/decidim/elections/votes_controller.rb +101 -41
- data/app/controllers/decidim/votings/admin/ballot_styles_controller.rb +93 -0
- data/app/controllers/decidim/votings/admin/monitoring_committee_election_results_controller.rb +69 -0
- data/app/controllers/decidim/votings/admin/monitoring_committee_members_controller.rb +4 -0
- data/app/controllers/decidim/votings/admin/monitoring_committee_polling_station_closures_controller.rb +73 -0
- data/app/controllers/decidim/votings/admin/monitoring_committee_verify_elections_controller.rb +24 -0
- data/app/controllers/decidim/votings/admin/polling_stations_controller.rb +4 -0
- data/app/controllers/decidim/votings/admin/votings_controller.rb +1 -1
- data/app/controllers/decidim/votings/census/admin/application_controller.rb +19 -0
- data/app/controllers/decidim/votings/census/admin/census_controller.rb +191 -0
- data/app/controllers/decidim/votings/polling_officer_zone/closures_controller.rb +131 -0
- data/app/controllers/decidim/votings/polling_officer_zone/in_person_votes_controller.rb +149 -0
- data/app/controllers/decidim/votings/polling_officer_zone/polling_officers_controller.rb +10 -2
- data/app/controllers/decidim/votings/votings_controller.rb +91 -4
- data/app/events/decidim/elections/votes/vote_accepted_event.rb +1 -1
- data/app/forms/decidim/elections/admin/action_form.rb +5 -0
- data/app/forms/decidim/elections/admin/question_form.rb +1 -1
- data/app/forms/decidim/elections/admin/trustees_participatory_space_form.rb +1 -1
- data/app/forms/decidim/elections/{ballot → voter}/verify_vote_form.rb +1 -5
- data/app/forms/decidim/elections/voter/vote_form.rb +50 -0
- data/app/forms/decidim/votings/admin/ballot_style_form.rb +19 -0
- data/app/forms/decidim/votings/admin/monitoring_committee_polling_station_closure_form.rb +11 -0
- data/app/forms/decidim/votings/admin/publish_results_form.rb +64 -0
- data/app/forms/decidim/votings/admin/voting_form.rb +3 -2
- data/app/forms/decidim/votings/answer_result_form.rb +24 -0
- data/app/forms/decidim/votings/ballot_result_form.rb +28 -0
- data/app/forms/decidim/votings/census/admin/dataset_form.rb +25 -0
- data/app/forms/decidim/votings/census/admin/datum_form.rb +30 -0
- data/app/forms/decidim/votings/census/check_fields.rb +28 -0
- data/app/forms/decidim/votings/census/check_form.rb +13 -0
- data/app/forms/decidim/votings/census/frontend_fields.rb +42 -0
- data/app/forms/decidim/votings/census/in_person_fields.rb +48 -0
- data/app/forms/decidim/votings/census/in_person_form.rb +13 -0
- data/app/forms/decidim/votings/census/login_form.rb +13 -0
- data/app/forms/decidim/votings/census/online_fields.rb +28 -0
- data/app/forms/decidim/votings/closure_certify_form.rb +20 -0
- data/app/forms/decidim/votings/closure_result_form.rb +37 -0
- data/app/forms/decidim/votings/closure_sign_form.rb +11 -0
- data/app/forms/decidim/votings/envelopes_result_form.rb +34 -0
- data/app/forms/decidim/votings/question_result_form.rb +24 -0
- data/app/forms/decidim/votings/voter/in_person_vote_form.rb +40 -0
- data/app/jobs/decidim/votings/census/admin/create_datum_job.rb +41 -0
- data/app/jobs/decidim/votings/census/admin/export_access_codes_job.rb +39 -0
- data/app/jobs/decidim/votings/census/admin/generate_access_codes_job.rb +44 -0
- data/app/mailers/decidim/elections/vote_accepted_mailer.rb +29 -0
- data/app/mailers/decidim/votings/access_code_mailer.rb +34 -0
- data/app/mailers/decidim/votings/census/export_mailer.rb +44 -0
- data/app/models/decidim/elections/action.rb +1 -1
- data/app/models/decidim/elections/answer.rb +21 -0
- data/app/models/decidim/elections/bulletin_board_closure.rb +19 -0
- data/app/models/decidim/elections/election.rb +5 -4
- data/app/models/decidim/elections/question.rb +24 -13
- data/app/models/decidim/elections/result.rb +22 -0
- data/app/models/decidim/elections/trustee.rb +10 -1
- data/app/models/decidim/elections/vote.rb +2 -2
- data/app/models/decidim/votings/ballot_style.rb +29 -0
- data/app/models/decidim/votings/ballot_style_question.rb +11 -0
- data/app/models/decidim/votings/census/dataset.rb +32 -0
- data/app/models/decidim/votings/census/datum.rb +34 -0
- data/app/models/decidim/votings/in_person_vote.rb +22 -0
- data/app/models/decidim/votings/polling_station.rb +18 -1
- data/app/models/decidim/votings/polling_station_closure.rb +43 -0
- data/app/models/decidim/votings/voting.rb +38 -15
- data/app/packs/entrypoints/decidim_elections.js +2 -0
- data/app/packs/entrypoints/decidim_elections_admin_pending_action.js +1 -0
- data/app/packs/entrypoints/decidim_elections_admin_vote_statistics.js +1 -0
- data/app/packs/entrypoints/decidim_elections_election_log.js +1 -0
- data/app/packs/entrypoints/decidim_elections_onboarding.js +1 -0
- data/app/packs/entrypoints/decidim_elections_trustee_key_ceremony.js +1 -0
- data/app/packs/entrypoints/decidim_elections_trustee_tally.js +1 -0
- data/app/packs/entrypoints/decidim_elections_trustee_trustee_zone.js +1 -0
- data/app/packs/entrypoints/decidim_elections_trustee_zone.js +1 -0
- data/app/packs/entrypoints/decidim_elections_trustee_zone.scss +1 -0
- data/app/packs/entrypoints/decidim_elections_voter_casting-vote.js +1 -0
- data/app/packs/entrypoints/decidim_elections_voter_new-vote.js +1 -0
- data/app/packs/entrypoints/decidim_elections_voter_setup-preview.js +1 -0
- data/app/packs/entrypoints/decidim_elections_voter_setup-vote.js +1 -0
- data/app/packs/entrypoints/decidim_elections_voter_verify-vote.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_monitoring_committee_members_form.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_polling_officers_form.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_polling_officers_picker.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_polling_stations_form.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_update_census_dataset_status.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_votings.js +1 -0
- data/app/packs/entrypoints/decidim_votings_admin_votings.scss +1 -0
- data/app/packs/entrypoints/decidim_votings_in-person-vote.js +1 -0
- data/app/packs/entrypoints/decidim_votings_voting-description-cell.js +1 -0
- data/app/packs/entrypoints/decidim_votings_voting_polling_officer_zone-edit-closure.js +1 -0
- data/app/packs/entrypoints/decidim_votings_voting_polling_officer_zone-in-person-vote.js +1 -0
- data/app/packs/entrypoints/decidim_votings_voting_polling_officer_zone-new-closure.js +1 -0
- data/app/packs/entrypoints/decidim_votings_voting_polling_officer_zone-sign-closure.js +1 -0
- data/app/{assets/images/decidim/elections/icon.svg → packs/images/decidim/elections/decidim_elections.svg} +0 -0
- data/app/{assets/images/decidim/votings/icon.svg → packs/images/decidim/votings/decidim_votings.svg} +0 -0
- data/app/{assets/javascripts/decidim/elections/admin/pending_action.js.es6 → packs/src/decidim/elections/admin/pending_action.js} +4 -6
- data/app/{assets/javascripts/decidim/elections/admin/vote_statistics.js.es6 → packs/src/decidim/elections/admin/vote_statistics.js} +0 -0
- data/app/packs/src/decidim/elections/election_log.js +145 -0
- data/app/packs/src/decidim/elections/onboarding.js +5 -0
- data/app/{assets/javascripts/decidim/elections/trustee_zone/key_ceremony.js.es6 → packs/src/decidim/elections/trustee/key_ceremony.js} +59 -22
- data/app/{assets/javascripts/decidim/elections/trustee_zone/tally.js.es6 → packs/src/decidim/elections/trustee/tally.js} +48 -12
- data/app/{assets/javascripts/decidim/elections/trustee_zone.js.es6 → packs/src/decidim/elections/trustee/trustee_zone.js} +3 -6
- data/app/packs/src/decidim/elections/voter/casting-vote.js +14 -0
- data/app/packs/src/decidim/elections/voter/new-vote.js +83 -0
- data/app/packs/src/decidim/elections/voter/setup-preview.js +79 -0
- data/app/packs/src/decidim/elections/voter/setup-vote.js +50 -0
- data/app/{assets/javascripts/decidim/elections/vote_verify.js.es6 → packs/src/decidim/elections/voter/verify-vote.js} +22 -21
- data/app/packs/src/decidim/elections/voter/vote_questions.component.js +124 -0
- data/app/{assets/javascripts/decidim/votings/admin/monitoring_committee_members_form.js.es6 → packs/src/decidim/votings/admin/monitoring_committee_members_form.js} +3 -3
- data/app/{assets/javascripts/decidim/votings/admin/polling_officers_form.js.es6 → packs/src/decidim/votings/admin/polling_officers_form.js} +3 -3
- data/app/{assets/javascripts/decidim/votings/admin/polling_officers_picker.js.es6 → packs/src/decidim/votings/admin/polling_officers_picker.js} +0 -0
- data/app/{assets/javascripts/decidim/votings/admin/polling_stations_form.js.es6 → packs/src/decidim/votings/admin/polling_stations_form.js} +4 -3
- data/app/packs/src/decidim/votings/admin/update_census_dataset_status.js +13 -0
- data/app/packs/src/decidim/votings/in-person-vote.js +20 -0
- data/app/packs/src/decidim/votings/polling_officer_zone/edit-closure.js +26 -0
- data/app/packs/src/decidim/votings/polling_officer_zone/in-person-vote.js +5 -0
- data/app/packs/src/decidim/votings/polling_officer_zone/new-closure.js +35 -0
- data/app/packs/src/decidim/votings/polling_officer_zone/sign-closure.js +13 -0
- data/app/packs/src/decidim/votings/voting-description-cell.js +41 -0
- data/app/packs/stylesheets/decidim/elections/elections.scss +8 -0
- data/app/{assets → packs}/stylesheets/decidim/elections/focus/_accordion.scss +0 -0
- data/app/{assets → packs}/stylesheets/decidim/elections/focus/_evote.scss +8 -1
- data/app/{assets → packs}/stylesheets/decidim/elections/focus/_focus.scss +0 -0
- data/app/{assets → packs}/stylesheets/decidim/elections/trustee_zone.scss +0 -0
- data/app/packs/stylesheets/decidim/votings/admin/votings.scss +3 -0
- data/app/packs/stylesheets/decidim/votings/admin/votings/_ballot-styles.scss +16 -0
- data/app/packs/stylesheets/decidim/votings/admin/votings/_monitoring-committee-polling-station-closures.scss +9 -0
- data/app/packs/stylesheets/decidim/votings/admin/votings/_results.scss +3 -0
- data/app/packs/stylesheets/decidim/votings/votings.scss +2 -0
- data/app/packs/stylesheets/decidim/votings/votings/_polling-stations-cell.scss +6 -0
- data/app/packs/stylesheets/decidim/votings/votings/_voting-description-cell.scss +29 -0
- data/app/permissions/decidim/elections/admin/permissions.rb +5 -0
- data/app/permissions/decidim/elections/permissions.rb +18 -33
- data/app/permissions/decidim/votings/admin/permissions.rb +87 -27
- data/app/permissions/decidim/votings/permissions.rb +5 -1
- data/app/permissions/decidim/votings/polling_officer_zone/permissions.rb +8 -2
- data/app/presenters/decidim/votings/census/admin_log/dataset_presenter.rb +30 -0
- data/app/presenters/decidim/votings/voting_presenter.rb +2 -11
- data/app/presenters/decidim/votings/voting_stats_presenter.rb +52 -0
- data/app/queries/decidim/elections/admin/votes_for_statistics.rb +1 -1
- data/app/queries/decidim/elections/votes/last_vote_for_voter.rb +29 -0
- data/app/queries/decidim/votings/admin/admin_users.rb +15 -10
- data/app/queries/decidim/votings/admin/ballot_style_by_voting_code.rb +31 -0
- data/app/queries/decidim/votings/votes/in_person_vote_for_voter.rb +27 -0
- data/app/queries/decidim/votings/votes/pending_in_person_votes.rb +20 -0
- data/app/serializers/decidim/votings/census/datum_serializer.rb +29 -0
- data/app/services/decidim/elections/current_user_vote_flow.rb +68 -0
- data/app/services/decidim/elections/vote_flow.rb +109 -0
- data/app/services/decidim/votings/census/access_codes_exporter.rb +50 -0
- data/app/services/decidim/votings/census_vote_flow.rb +106 -0
- data/app/uploaders/decidim/cw/votings/census/voting_census_uploader.rb +19 -0
- data/app/views/decidim/elections/admin/answers/index.html.erb +23 -4
- data/app/views/decidim/elections/admin/elections/index.html.erb +5 -5
- data/app/views/decidim/elections/admin/questions/index.html.erb +2 -2
- data/app/views/decidim/elections/admin/steps/_results_published.html.erb +62 -0
- data/app/views/decidim/elections/admin/steps/_tally_ended.html.erb +49 -3
- data/app/views/decidim/elections/admin/steps/_vote.html.erb +1 -1
- data/app/views/decidim/elections/admin/steps/_vote_ended.html.erb +4 -0
- data/app/views/decidim/elections/admin/steps/index.html.erb +2 -2
- data/app/views/decidim/elections/admin/trustees_participatory_spaces/index.html.erb +7 -3
- data/app/views/decidim/elections/elections/_filters.html.erb +1 -0
- data/app/views/decidim/elections/elections/election_log.html.erb +182 -0
- data/app/views/decidim/elections/elections/index.html.erb +0 -2
- data/app/views/decidim/elections/elections/show.html.erb +11 -37
- data/app/views/decidim/elections/trustee_zone/elections/_key_ceremony_steps.html.erb +41 -39
- data/app/views/decidim/elections/trustee_zone/elections/show.html.erb +11 -27
- data/app/views/decidim/elections/trustee_zone/elections/update.js.erb +1 -1
- data/app/views/decidim/elections/trustee_zone/trustees/show.html.erb +7 -7
- data/app/views/decidim/elections/vote_accepted_mailer/notification.html.erb +3 -0
- data/app/views/decidim/elections/votes/{_election_votes_header.html.erb → _focus_header.html.erb} +0 -0
- data/app/views/decidim/elections/votes/_new_ballot_decision_step.html.erb +30 -0
- data/app/views/decidim/elections/votes/{_election_votes_confirm.html.erb → _new_confirm_step.html.erb} +5 -5
- data/app/views/decidim/elections/votes/{_election_votes_confirm_footer.html.erb → _new_confirm_step_footer.html.erb} +2 -3
- data/app/views/decidim/elections/votes/{_election_votes_encrypting.html.erb → _new_encrypting_step.html.erb} +1 -1
- data/app/views/decidim/elections/votes/{_election_votes_question.html.erb → _new_question.html.erb} +7 -7
- data/app/views/decidim/elections/votes/{_election_votes_modal.html.erb → _new_question_modal.html.erb} +1 -1
- data/app/views/decidim/elections/votes/_onboarding_modal.html.erb +23 -0
- data/app/views/decidim/elections/votes/_show_casted.html.erb +36 -0
- data/app/views/decidim/elections/votes/_show_casting.html.erb +25 -0
- data/app/views/decidim/elections/votes/{_election_votes_failed.html.erb → _show_failed.html.erb} +6 -4
- data/app/views/decidim/elections/votes/new.html.erb +29 -16
- data/app/views/decidim/elections/votes/show.html.erb +11 -0
- data/app/views/decidim/elections/votes/verify.html.erb +6 -9
- data/app/views/decidim/votings/access_code_mailer/send_access_code.html.erb +5 -0
- data/app/views/decidim/votings/admin/ballot_styles/_form.html.erb +24 -0
- data/app/views/decidim/votings/admin/ballot_styles/edit.html.erb +7 -0
- data/app/views/decidim/votings/admin/ballot_styles/index.html.erb +68 -0
- data/app/views/decidim/votings/admin/ballot_styles/new.html.erb +7 -0
- data/app/views/decidim/votings/admin/monitoring_committee_election_results/_results.html.erb +53 -0
- data/app/views/decidim/votings/admin/monitoring_committee_election_results/index.html.erb +32 -0
- data/app/views/decidim/votings/admin/monitoring_committee_election_results/show.html.erb +24 -0
- data/app/views/decidim/votings/admin/monitoring_committee_members/_form.html.erb +1 -1
- data/app/views/decidim/votings/admin/monitoring_committee_members/index.html.erb +1 -1
- data/app/views/decidim/votings/admin/monitoring_committee_polling_station_closures/_closure_certificate_results.erb +9 -0
- data/app/views/decidim/votings/admin/monitoring_committee_polling_station_closures/_closures.html.erb +71 -0
- data/app/views/decidim/votings/admin/monitoring_committee_polling_station_closures/_elections.html.erb +32 -0
- data/app/views/decidim/votings/admin/monitoring_committee_polling_station_closures/edit.html.erb +29 -0
- data/app/views/decidim/votings/admin/monitoring_committee_polling_station_closures/index.html.erb +5 -0
- data/app/views/decidim/votings/admin/monitoring_committee_polling_station_closures/show.html.erb +10 -0
- data/app/views/decidim/votings/admin/monitoring_committee_verify_elections/index.html.erb +59 -0
- data/app/views/decidim/votings/admin/polling_officers/_form.html.erb +1 -1
- data/app/views/decidim/votings/admin/polling_officers/index.html.erb +1 -1
- data/app/views/decidim/votings/admin/polling_stations/_form.html.erb +1 -1
- data/app/views/decidim/votings/admin/polling_stations/index.html.erb +2 -2
- data/app/views/decidim/votings/admin/votings/_form.html.erb +9 -2
- data/app/views/decidim/votings/admin/votings/edit.html.erb +1 -1
- data/app/views/decidim/votings/admin/votings/index.html.erb +3 -2
- data/app/views/decidim/votings/census/admin/census/_creating_data.html.erb +10 -0
- data/app/views/decidim/votings/census/admin/census/_export_codes.html.erb +8 -0
- data/app/views/decidim/votings/census/admin/census/_exporting_codes.html.erb +3 -0
- data/app/views/decidim/votings/census/admin/census/_freeze.html.erb +3 -0
- data/app/views/decidim/votings/census/admin/census/_generate_codes.html.erb +25 -0
- data/app/views/decidim/votings/census/admin/census/_generating_codes.html.erb +3 -0
- data/app/views/decidim/votings/census/admin/census/_new_census.html.erb +22 -0
- data/app/views/decidim/votings/census/admin/census/_upload_info.html.erb +17 -0
- data/app/views/decidim/votings/census/admin/census/show.html.erb +23 -0
- data/app/views/decidim/votings/census/admin/census/status.js.erb +1 -0
- data/app/views/decidim/votings/census/export_mailer/access_codes_export.erb +7 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_answer_results_form_fields.html.erb +12 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_ballot_results_form_fields.html.erb +40 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_certify_form.html.erb +15 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_modal_ballots_count_error.html.erb +60 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_modal_ballots_results_count_error.html.erb +19 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_question_results_form_fields.html.erb +22 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/_sign_form.html.erb +55 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/edit.html.erb +48 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/new.html.erb +62 -0
- data/app/views/decidim/votings/polling_officer_zone/closures/show.html.erb +35 -0
- data/app/views/decidim/votings/polling_officer_zone/in_person_votes/_complete_voting.html.erb +59 -0
- data/app/views/decidim/votings/polling_officer_zone/in_person_votes/_in_person_form.html.erb +50 -0
- data/app/views/decidim/votings/polling_officer_zone/in_person_votes/_verify_document.html.erb +23 -0
- data/app/views/decidim/votings/polling_officer_zone/in_person_votes/new.html.erb +21 -0
- data/app/views/decidim/votings/polling_officer_zone/in_person_votes/show.html.erb +25 -0
- data/app/views/decidim/votings/polling_officer_zone/polling_officers/_polling_station.html.erb +60 -0
- data/app/views/decidim/votings/polling_officer_zone/polling_officers/index.html.erb +15 -0
- data/app/views/decidim/votings/votings/_access_code_modal.html.erb +33 -0
- data/app/views/decidim/votings/votings/_check_fields.html.erb +31 -0
- data/app/views/decidim/votings/votings/_filters.html.erb +1 -0
- data/app/views/decidim/votings/votings/_promoted_voting.html.erb +1 -1
- data/app/views/decidim/votings/votings/check_census.html.erb +64 -0
- data/app/views/decidim/votings/votings/elections_log.html.erb +34 -0
- data/app/views/decidim/votings/votings/index.html.erb +0 -2
- data/app/views/decidim/votings/votings/login.html.erb +50 -0
- data/app/views/decidim/votings/votings/show.html.erb +11 -11
- data/app/views/layouts/decidim/_election_votes_header.html.erb +3 -3
- data/app/views/layouts/decidim/admin/voting.html.erb +5 -83
- data/app/views/layouts/decidim/election_votes.html.erb +1 -0
- data/app/views/layouts/decidim/voting_landing.html.erb +13 -0
- data/app/views/layouts/decidim/{voting.html.erb → votings.html.erb} +3 -1
- data/config/assets.rb +35 -0
- data/config/initializers/decidim_bulletin_board.rb +17 -11
- data/config/initializers/rack_attack.rb +15 -0
- data/config/locales/ca.yml +519 -71
- data/config/locales/cs.yml +523 -59
- data/config/locales/de.yml +471 -45
- data/config/locales/el.yml +0 -7
- data/config/locales/en.yml +491 -42
- data/config/locales/es-MX.yml +350 -53
- data/config/locales/es-PY.yml +350 -53
- data/config/locales/es.yml +367 -70
- data/config/locales/fi-plain.yml +491 -43
- data/config/locales/fi.yml +509 -61
- data/config/locales/fr-CA.yml +246 -31
- data/config/locales/fr-LU.yml +890 -0
- data/config/locales/fr.yml +246 -31
- data/config/locales/gl.yml +427 -18
- data/config/locales/hu.yml +29 -0
- data/config/locales/it.yml +1030 -12
- data/config/locales/ja.yml +581 -13
- data/config/locales/lb-LU.yml +1 -0
- data/config/locales/lv.yml +0 -7
- data/config/locales/nl.yml +308 -37
- data/config/locales/no.yml +21 -5
- data/config/locales/pl.yml +674 -220
- data/config/locales/pt-BR.yml +1338 -0
- data/config/locales/pt.yml +0 -12
- data/config/locales/ro-RO.yml +100 -13
- data/config/locales/sv.yml +290 -12
- data/config/locales/tr-TR.yml +3 -16
- data/config/locales/zh-CN.yml +0 -12
- data/db/migrate/20210308104024_add_decidim_votings_census_datasets.rb +17 -0
- data/db/migrate/20210308104154_add_decidim_votings_census_data.rb +23 -0
- data/db/migrate/20210326090435_create_elections_results.rb +16 -0
- data/db/migrate/20210330102348_remove_votes_count_from_answer.rb +7 -0
- data/db/migrate/20210330123606_add_voting_ballot_style.rb +19 -0
- data/db/migrate/20210330183204_add_email_to_votes.rb +10 -0
- data/db/migrate/20210331152729_add_census_contact_information_to_votings.rb +7 -0
- data/db/migrate/20210401095507_add_organization_to_decidim_elections_trustee.rb +11 -0
- data/db/migrate/20210402102215_add_ballot_style_to_decidim_votings_census_data.rb +7 -0
- data/db/migrate/20210402140402_add_salt_to_elections.rb +22 -0
- data/db/migrate/20210412144721_change_elections_results.rb +24 -0
- data/db/migrate/20210412144740_create_elections_bulletin_board_closures.rb +13 -0
- data/db/migrate/20210412144741_create_votings_polling_station_closures.rb +22 -0
- data/db/migrate/20210420112721_create_decidim_votings_in_person_votes.rb +21 -0
- data/db/migrate/20210422124826_add_verifiable_results_to_decidim_elections_election.rb +8 -0
- data/db/migrate/20210426072845_add_signed_at_to_polling_station_closure.rb +7 -0
- data/db/migrate/20210427131742_add_validated_at_to_votings_polling_station_closures.rb +8 -0
- data/db/seeds/Exampledocument.pdf +0 -0
- data/db/seeds/city2.jpeg +0 -0
- data/lib/decidim/api/bulletin_board_closure_type.rb +16 -0
- data/lib/decidim/api/election_answer_type.rb +3 -2
- data/lib/decidim/api/election_result_type.rb +19 -0
- data/lib/decidim/api/polling_station_closure_type.rb +19 -0
- data/lib/decidim/api/polling_station_type.rb +24 -0
- data/lib/decidim/elections.rb +8 -0
- data/lib/decidim/elections/admin_engine.rb +14 -15
- data/lib/decidim/elections/answer_serializer.rb +2 -1
- data/lib/decidim/elections/api.rb +2 -0
- data/lib/decidim/elections/component.rb +85 -19
- data/lib/decidim/elections/engine.rb +4 -6
- data/lib/decidim/elections/test/factories.rb +76 -11
- data/lib/decidim/elections/trustee_zone_engine.rb +5 -4
- data/lib/decidim/elections/version.rb +1 -1
- data/lib/decidim/votings.rb +17 -0
- data/lib/decidim/votings/admin_engine.rb +134 -8
- data/lib/decidim/votings/api.rb +2 -0
- data/lib/decidim/votings/census.rb +16 -0
- data/lib/decidim/votings/census_admin.rb +12 -0
- data/lib/decidim/votings/census_admin_engine.rb +22 -0
- data/lib/decidim/votings/census_engine.rb +19 -0
- data/lib/decidim/votings/engine.rb +25 -20
- data/lib/decidim/votings/participatory_space.rb +149 -2
- data/lib/decidim/votings/polling_officer_zone_engine.rb +15 -6
- data/lib/decidim/votings/test/factories.rb +163 -0
- data/lib/decidim/votings/voting_serializer.rb +1 -0
- data/lib/tasks/decidim_voting_census.rake +26 -0
- metadata +326 -72
- data/app/assets/config/admin/decidim_votings_manifest.css +0 -0
- data/app/assets/config/admin/decidim_votings_manifest.js +0 -4
- data/app/assets/config/decidim_elections_manifest.css +0 -3
- data/app/assets/config/decidim_elections_manifest.js +0 -8
- data/app/assets/config/decidim_votings_manifest.css +0 -3
- data/app/assets/config/decidim_votings_manifest.js +0 -0
- data/app/assets/javascripts/decidim/elections/vote.js.es6 +0 -117
- data/app/assets/javascripts/decidim/elections/vote_questions.component.js.es6 +0 -129
- data/app/assets/stylesheets/decidim/elections/elections.scss +0 -6
- data/app/assets/stylesheets/decidim/votings/votings.scss +0 -1
- data/app/assets/stylesheets/decidim/votings/votings/_votings-header.scss +0 -99
- data/app/assets/stylesheets/decidim/votings/votings/_votings-home-banner.scss +0 -11
- data/app/controllers/decidim/votings/polling_officer_zone/polling_stations_controller.rb +0 -22
- data/app/forms/decidim/elections/voter/encrypted_vote_form.rb +0 -49
- data/app/helpers/decidim/elections/elections_helper.rb +0 -29
- data/app/queries/decidim/elections/votes/election_votes.rb +0 -19
- data/app/queries/decidim/elections/votes/user_election_last_vote.rb +0 -26
- data/app/queries/decidim/elections/votes/user_votes.rb +0 -19
- data/app/views/decidim/elections/elections/_preview.html.erb +0 -26
- data/app/views/decidim/elections/elections/_results.html.erb +0 -47
- data/app/views/decidim/elections/votes/_election_votes_confirmed.html.erb +0 -47
- data/app/views/decidim/elections/votes/_election_votes_steps_header.html.erb +0 -12
- data/app/views/decidim/elections/votes/cast_failed.js.erb +0 -1
- data/app/views/decidim/elections/votes/cast_success.js.erb +0 -1
- data/app/views/decidim/votings/polling_officer_zone/polling_officers/show.html.erb +0 -42
- data/app/views/decidim/votings/polling_officer_zone/polling_stations/show.html.erb +0 -1
- data/app/views/decidim/votings/votings/_voting_details.html.erb +0 -7
- data/app/views/layouts/decidim/_voting_header.html.erb +0 -16
- data/config/locales/ja-JP.yml +0 -216
@@ -8,7 +8,7 @@ module Decidim
|
|
8
8
|
|
9
9
|
return permission_action unless user
|
10
10
|
|
11
|
-
return Decidim::Votings::Admin::Permissions.new(user, permission_action, context).permissions if
|
11
|
+
return Decidim::Votings::Admin::Permissions.new(user, permission_action, context).permissions if admin_scope?
|
12
12
|
|
13
13
|
# Delegate the polling_officer_zone permission checks to the polling officer zone permissions class
|
14
14
|
return Decidim::Votings::PollingOfficerZone::Permissions.new(user, permission_action, context).permissions if permission_action.scope == :polling_officer_zone
|
@@ -35,6 +35,10 @@ module Decidim
|
|
35
35
|
allow!
|
36
36
|
end
|
37
37
|
end
|
38
|
+
|
39
|
+
def admin_scope?
|
40
|
+
permission_action.scope == :admin || permission_action.subject == :admin_dashboard
|
41
|
+
end
|
38
42
|
end
|
39
43
|
end
|
40
44
|
end
|
@@ -8,8 +8,10 @@ module Decidim
|
|
8
8
|
return permission_action unless permission_action.scope == :polling_officer_zone
|
9
9
|
|
10
10
|
case permission_action.subject
|
11
|
-
when :polling_officers
|
12
|
-
toggle_allow(polling_officers_for_user?) if
|
11
|
+
when :polling_officers
|
12
|
+
toggle_allow(polling_officers_for_user?) if permission_action.action == :view
|
13
|
+
when :polling_station_results, :in_person_vote
|
14
|
+
toggle_allow(polling_officer.user == user) if permission_action.action == :manage
|
13
15
|
when :user
|
14
16
|
allow! if permission_action.action == :update_profile
|
15
17
|
end
|
@@ -26,6 +28,10 @@ module Decidim
|
|
26
28
|
def polling_officers
|
27
29
|
@polling_officers ||= context.fetch(:polling_officers, [])
|
28
30
|
end
|
31
|
+
|
32
|
+
def polling_officer
|
33
|
+
@polling_officer ||= context.fetch(:polling_officer, [])
|
34
|
+
end
|
29
35
|
end
|
30
36
|
end
|
31
37
|
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Votings
|
5
|
+
module Census
|
6
|
+
module AdminLog
|
7
|
+
# This class holds the logic to present a `Decidim::Votings::Census::Dataset`
|
8
|
+
# for the `AdminLog` log.
|
9
|
+
#
|
10
|
+
# Usage should be automatic and you shouldn't need to call this class
|
11
|
+
# directly, but here's an example:
|
12
|
+
#
|
13
|
+
# action_log = Decidim::ActionLog.last
|
14
|
+
# view_helpers # => this comes from the views
|
15
|
+
# DatasetPresenter.new(action_log, view_helpers).present
|
16
|
+
class DatasetPresenter < Decidim::Log::BasePresenter
|
17
|
+
private
|
18
|
+
|
19
|
+
def i18n_params
|
20
|
+
{
|
21
|
+
user_name: user_presenter.present,
|
22
|
+
resource_name: Dataset.model_name.human,
|
23
|
+
space_name: space_presenter.present
|
24
|
+
}
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
@@ -3,29 +3,20 @@
|
|
3
3
|
module Decidim
|
4
4
|
module Votings
|
5
5
|
class VotingPresenter < SimpleDelegator
|
6
|
-
include Rails.application.routes.mounted_helpers
|
7
|
-
include ActionView::Helpers::UrlHelper
|
8
6
|
include Decidim::SanitizeHelper
|
9
7
|
include Decidim::TranslatableAttributes
|
10
8
|
|
11
|
-
delegate :url, to: :introductory_image, prefix: true
|
12
|
-
delegate :url, to: :banner_image, prefix: true
|
13
|
-
|
14
9
|
def title
|
15
10
|
content = translated_attribute(voting.title)
|
16
11
|
decidim_html_escape(content)
|
17
12
|
end
|
18
13
|
|
19
14
|
def introductory_image_url
|
20
|
-
|
21
|
-
|
22
|
-
URI.join(decidim.root_url(host: voting.organization.host), voting.introductory_image_url).to_s
|
15
|
+
voting.attached_uploader(:introductory_image).url(host: voting.organization.host)
|
23
16
|
end
|
24
17
|
|
25
18
|
def banner_image_url
|
26
|
-
|
27
|
-
|
28
|
-
URI.join(decidim.root_url(host: voting.organization.host), voting.banner_image_url).to_s
|
19
|
+
voting.attached_uploader(:banner_image).url(host: voting.organization.host)
|
29
20
|
end
|
30
21
|
|
31
22
|
def voting
|
@@ -0,0 +1,52 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Votings
|
5
|
+
# A presenter to render statistics in the homepage.
|
6
|
+
class VotingStatsPresenter < Decidim::StatsPresenter
|
7
|
+
attribute :voting, Decidim::Votings::Voting
|
8
|
+
include Decidim::IconHelper
|
9
|
+
|
10
|
+
# Public: returns a collection of stats (Hash) for the Voting Landing.
|
11
|
+
def collection
|
12
|
+
highlighted_stats = voting_participants_stats
|
13
|
+
highlighted_stats.concat(voting_followers_stats(priority: StatsRegistry::HIGH_PRIORITY))
|
14
|
+
highlighted_stats.concat(component_stats(priority: StatsRegistry::HIGH_PRIORITY))
|
15
|
+
highlighted_stats.concat(component_stats(priority: StatsRegistry::MEDIUM_PRIORITY))
|
16
|
+
highlighted_stats.concat(comments_stats(:votings))
|
17
|
+
highlighted_stats = highlighted_stats.reject(&:empty?)
|
18
|
+
highlighted_stats = highlighted_stats.reject { |_stat_manifest, _stat_title, stat_number| stat_number.zero? }
|
19
|
+
grouped_highlighted_stats = highlighted_stats.group_by(&:first)
|
20
|
+
|
21
|
+
statistics(grouped_highlighted_stats)
|
22
|
+
end
|
23
|
+
|
24
|
+
private
|
25
|
+
|
26
|
+
def voting_participants_stats
|
27
|
+
Decidim.stats.only([:participants_count]).with_context(voting)
|
28
|
+
.map { |stat_title, stat_number| [voting.manifest.name, stat_title, stat_number] }
|
29
|
+
end
|
30
|
+
|
31
|
+
def component_stats(conditions)
|
32
|
+
Decidim.component_manifests.map do |component_manifest|
|
33
|
+
component_manifest.stats.except([:proposals_accepted])
|
34
|
+
.filter(conditions)
|
35
|
+
.with_context(published_components)
|
36
|
+
.map { |stat_title, stat_number| [component_manifest.name, stat_title, stat_number] }.flatten
|
37
|
+
end
|
38
|
+
end
|
39
|
+
|
40
|
+
def voting_followers_stats(conditions)
|
41
|
+
Decidim.stats.only([:followers_count])
|
42
|
+
.filter(conditions)
|
43
|
+
.with_context(voting)
|
44
|
+
.map { |stat_title, stat_number| [voting.manifest.name, stat_title, stat_number] }
|
45
|
+
end
|
46
|
+
|
47
|
+
def published_components
|
48
|
+
@published_components ||= Component.where(participatory_space: voting).published
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
52
|
+
end
|
@@ -16,7 +16,7 @@ module Decidim
|
|
16
16
|
|
17
17
|
# Finds the votes for an election which get count for the statistics
|
18
18
|
def query
|
19
|
-
@election.votes.accepted.
|
19
|
+
@election.votes.accepted.pick(Arel.sql("COUNT(id)"), Arel.sql("COUNT(distinct voter_id)"))
|
20
20
|
end
|
21
21
|
end
|
22
22
|
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Elections
|
5
|
+
module Votes
|
6
|
+
# A class used to find the last vote casted by a voter in an election
|
7
|
+
class LastVoteForVoter < Rectify::Query
|
8
|
+
# Syntactic sugar to initialize the class and return the queried objects.
|
9
|
+
#
|
10
|
+
# election - the election where the vote was casted
|
11
|
+
# voter_id - the identifier of the voter
|
12
|
+
def self.for(election, voter_id)
|
13
|
+
new(election, voter_id).query
|
14
|
+
end
|
15
|
+
|
16
|
+
def initialize(election, voter_id)
|
17
|
+
@voter_id = voter_id
|
18
|
+
@election = election
|
19
|
+
end
|
20
|
+
|
21
|
+
def query
|
22
|
+
Decidim::Elections::Vote.where(election: @election, voter_id: @voter_id)
|
23
|
+
.order("created_at")
|
24
|
+
.last
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -3,7 +3,7 @@
|
|
3
3
|
module Decidim
|
4
4
|
module Votings
|
5
5
|
module Admin
|
6
|
-
# A class used to find the admins for a voting
|
6
|
+
# A class used to find the admins for a voting or an organization votings.
|
7
7
|
class AdminUsers < Rectify::Query
|
8
8
|
# Syntactic sugar to initialize the class and return the queried objects.
|
9
9
|
#
|
@@ -12,27 +12,32 @@ module Decidim
|
|
12
12
|
new(voting).query
|
13
13
|
end
|
14
14
|
|
15
|
+
# Syntactic sugar to initialize the class and return the queried objects.
|
16
|
+
#
|
17
|
+
# organization - an organization that needs to find its voting admins
|
18
|
+
def self.for_organization(organization)
|
19
|
+
new(nil, organization).query
|
20
|
+
end
|
21
|
+
|
15
22
|
# Initializes the class.
|
16
23
|
#
|
17
|
-
# voting - a voting that needs to find its
|
18
|
-
|
24
|
+
# voting - a voting that needs to find its voting admins
|
25
|
+
# organization - an organization that needs to find its voting admins
|
26
|
+
def initialize(voting, organization = nil)
|
19
27
|
@voting = voting
|
28
|
+
@organization = voting&.organization || organization
|
20
29
|
end
|
21
30
|
|
22
|
-
# Finds organization admins and the users with role admin for the given
|
31
|
+
# Finds organization admins and the users with role admin for the given voting.
|
23
32
|
#
|
24
33
|
# Returns an ActiveRecord::Relation.
|
25
34
|
def query
|
26
|
-
|
35
|
+
organization.admins
|
27
36
|
end
|
28
37
|
|
29
38
|
private
|
30
39
|
|
31
|
-
attr_reader :voting
|
32
|
-
|
33
|
-
def organization_admins
|
34
|
-
voting.organization.admins
|
35
|
-
end
|
40
|
+
attr_reader :voting, :organization
|
36
41
|
end
|
37
42
|
end
|
38
43
|
end
|
@@ -0,0 +1,31 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Votings
|
5
|
+
module Admin
|
6
|
+
# A class used to find trustees by participatory space.
|
7
|
+
class BallotStyleByVotingCode < Rectify::Query
|
8
|
+
# Syntactic sugar to initialize the class and return the queried objects.
|
9
|
+
#
|
10
|
+
# voting - the voting of the Ballot Style
|
11
|
+
# code - the code of the Ballot Style
|
12
|
+
def self.for(voting, code)
|
13
|
+
new(voting, code).query
|
14
|
+
end
|
15
|
+
|
16
|
+
# Initializes the class.
|
17
|
+
def initialize(voting, code)
|
18
|
+
@voting = voting
|
19
|
+
@code = code
|
20
|
+
end
|
21
|
+
|
22
|
+
# Gets the ballot style with the specified code in this voting
|
23
|
+
def query
|
24
|
+
Decidim::Votings::BallotStyle
|
25
|
+
.where(voting: @voting)
|
26
|
+
.find_by(code: @code)
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
@@ -0,0 +1,27 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Votings
|
5
|
+
module Votes
|
6
|
+
# A class used to find a non-rejected in person vote registered for a voter in an election
|
7
|
+
class InPersonVoteForVoter < Rectify::Query
|
8
|
+
# Syntactic sugar to initialize the class and return the queried objects.
|
9
|
+
#
|
10
|
+
# election - the election where the vote was casted
|
11
|
+
# voter_id - the identifier of the voter
|
12
|
+
def self.for(election, voter_id)
|
13
|
+
new(election, voter_id).query
|
14
|
+
end
|
15
|
+
|
16
|
+
def initialize(election, voter_id)
|
17
|
+
@voter_id = voter_id
|
18
|
+
@election = election
|
19
|
+
end
|
20
|
+
|
21
|
+
def query
|
22
|
+
Decidim::Votings::InPersonVote.not_rejected.find_by(election: @election, voter_id: @voter_id)
|
23
|
+
end
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
@@ -0,0 +1,20 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Votings
|
5
|
+
module Votes
|
6
|
+
# A class used to find in person votes with a pending status
|
7
|
+
class PendingInPersonVotes < Rectify::Query
|
8
|
+
# Syntactic sugar to initialize the class and return the queried objects.
|
9
|
+
def self.for
|
10
|
+
new.query
|
11
|
+
end
|
12
|
+
|
13
|
+
# Finds the in person votes with pending status
|
14
|
+
def query
|
15
|
+
Decidim::Votings::InPersonVote.pending
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Votings
|
5
|
+
module Census
|
6
|
+
# This class serializes a Voting::Census::Datum
|
7
|
+
class DatumSerializer < Decidim::Exporters::Serializer
|
8
|
+
include Decidim::ApplicationHelper
|
9
|
+
|
10
|
+
# Public: Initializes the serializer with a Voting::Census::Datum.
|
11
|
+
def initialize(datum)
|
12
|
+
@datum = datum
|
13
|
+
end
|
14
|
+
|
15
|
+
# Public: Exports a hash with the serialized data for this datum.
|
16
|
+
def serialize
|
17
|
+
{
|
18
|
+
full_name: datum.full_name,
|
19
|
+
full_address: datum.full_address,
|
20
|
+
postal_code: datum.postal_code,
|
21
|
+
access_code: datum.access_code
|
22
|
+
}
|
23
|
+
end
|
24
|
+
|
25
|
+
attr_reader :datum
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Elections
|
5
|
+
# Service that encapsulates the vote flow used for elections for registered users.
|
6
|
+
class CurrentUserVoteFlow < VoteFlow
|
7
|
+
def initialize(election, current_user, &can_vote_block)
|
8
|
+
super(election)
|
9
|
+
|
10
|
+
@current_user = current_user
|
11
|
+
@can_vote_block = can_vote_block
|
12
|
+
end
|
13
|
+
|
14
|
+
def voter_login(params)
|
15
|
+
# There is no previous login page for this vote flow
|
16
|
+
end
|
17
|
+
|
18
|
+
def has_voter?
|
19
|
+
current_user.present?
|
20
|
+
end
|
21
|
+
|
22
|
+
delegate :name, to: :current_user, prefix: :voter, allow_nil: true
|
23
|
+
delegate :email, to: :current_user, allow_nil: true
|
24
|
+
|
25
|
+
def user
|
26
|
+
current_user
|
27
|
+
end
|
28
|
+
|
29
|
+
def voter_data
|
30
|
+
return nil unless current_user
|
31
|
+
|
32
|
+
{
|
33
|
+
id: current_user.id,
|
34
|
+
created: current_user.created_at.to_i
|
35
|
+
}
|
36
|
+
end
|
37
|
+
|
38
|
+
def vote_check(*)
|
39
|
+
VoteCheckResult.new(
|
40
|
+
allowed: current_user && (received_voter_token || can_vote_block.call),
|
41
|
+
error_message: I18n.t("votes.messages.not_allowed", scope: "decidim.elections")
|
42
|
+
)
|
43
|
+
end
|
44
|
+
|
45
|
+
def login_path(online_vote_path); end
|
46
|
+
|
47
|
+
def questions_for(election)
|
48
|
+
election.questions
|
49
|
+
end
|
50
|
+
|
51
|
+
def ballot_style_id; end
|
52
|
+
|
53
|
+
private
|
54
|
+
|
55
|
+
attr_accessor :current_user, :can_vote_block
|
56
|
+
|
57
|
+
def valid_token_flow_data?
|
58
|
+
return @valid_token_flow_data if defined?(@valid_token_flow_data)
|
59
|
+
|
60
|
+
@valid_token_flow_data = received_voter_token && received_voter_token_user_id && current_user.id == received_voter_token_user_id
|
61
|
+
end
|
62
|
+
|
63
|
+
def received_voter_token_user_id
|
64
|
+
@received_voter_token_user_id ||= received_voter_token_data.dig(:flow, :id)&.to_i
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
@@ -0,0 +1,109 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Decidim
|
4
|
+
module Elections
|
5
|
+
# Service that encapsulates the vote flow used for elections
|
6
|
+
class VoteFlow
|
7
|
+
def initialize(election)
|
8
|
+
@election = election
|
9
|
+
end
|
10
|
+
|
11
|
+
def voter_from_token(params)
|
12
|
+
@received_voter_token = params[:voter_token]
|
13
|
+
@received_voter_id = params[:voter_id]
|
14
|
+
|
15
|
+
received_voter_token.present? && received_voter_id.present?
|
16
|
+
end
|
17
|
+
|
18
|
+
def voter_id
|
19
|
+
@voter_id ||= calculate_voter_id(voter_token_data)
|
20
|
+
end
|
21
|
+
|
22
|
+
def voter_id_token(a_voter_id = nil)
|
23
|
+
@voter_id_token ||= tokenizer.hex_digest(a_voter_id || voter_id)
|
24
|
+
end
|
25
|
+
|
26
|
+
def voter_token
|
27
|
+
@voter_token ||= received_voter_token ||
|
28
|
+
message_encryptor.encrypt_and_sign(
|
29
|
+
voter_token_data.to_json,
|
30
|
+
expires_at: Decidim::Elections.voter_token_expiration_minutes.minutes.from_now
|
31
|
+
)
|
32
|
+
end
|
33
|
+
|
34
|
+
def valid_received_data?
|
35
|
+
valid_token_common_data? && valid_token_flow_data? && valid_voter_id?
|
36
|
+
end
|
37
|
+
|
38
|
+
private
|
39
|
+
|
40
|
+
attr_accessor :election, :context, :received_voter_token, :received_voter_id
|
41
|
+
|
42
|
+
def calculate_voter_id(data)
|
43
|
+
Digest::SHA256.hexdigest(data.to_json)
|
44
|
+
end
|
45
|
+
|
46
|
+
def valid_voter_id?
|
47
|
+
received_voter_id && received_voter_id == calculate_voter_id(received_voter_token_data)
|
48
|
+
end
|
49
|
+
|
50
|
+
def valid_token_common_data?
|
51
|
+
received_voter_token && received_voter_token_data[:common] == voter_common_data.as_json
|
52
|
+
end
|
53
|
+
|
54
|
+
def voter_token_data
|
55
|
+
@voter_token_data = {
|
56
|
+
common: voter_common_data,
|
57
|
+
flow: voter_data
|
58
|
+
}
|
59
|
+
end
|
60
|
+
|
61
|
+
def voter_common_data
|
62
|
+
@voter_common_data = {
|
63
|
+
salt: election.salt,
|
64
|
+
slug: Decidim::Elections.bulletin_board.authority_slug,
|
65
|
+
created: election.created_at.to_i,
|
66
|
+
election: election.id
|
67
|
+
}
|
68
|
+
end
|
69
|
+
|
70
|
+
def received_voter_token_data
|
71
|
+
return {} unless verified_received_voter_token
|
72
|
+
|
73
|
+
@received_voter_token_data ||= JSON.parse(verified_received_voter_token).with_indifferent_access
|
74
|
+
end
|
75
|
+
|
76
|
+
def verified_received_voter_token
|
77
|
+
return @verified_received_voter_token if defined?(@verified_received_voter_token)
|
78
|
+
|
79
|
+
@verified_received_voter_token = begin
|
80
|
+
message_encryptor.decrypt_and_verify(received_voter_token)
|
81
|
+
rescue ActiveSupport::MessageEncryptor::InvalidMessage
|
82
|
+
nil
|
83
|
+
end
|
84
|
+
end
|
85
|
+
|
86
|
+
def message_encryptor
|
87
|
+
@message_encryptor ||= ActiveSupport::MessageEncryptor.new([election.salt].pack("H*"))
|
88
|
+
end
|
89
|
+
|
90
|
+
def tokenizer
|
91
|
+
@tokenizer ||= Decidim::Tokenizer.new(salt: election.salt, length: 10)
|
92
|
+
end
|
93
|
+
|
94
|
+
class VoteCheckResult
|
95
|
+
def initialize(allowed:, error_message:, exit_path: nil)
|
96
|
+
@allowed = allowed
|
97
|
+
@error_message = error_message unless allowed
|
98
|
+
@exit_path = exit_path
|
99
|
+
end
|
100
|
+
|
101
|
+
attr_reader :error_message, :exit_path
|
102
|
+
|
103
|
+
def allowed?
|
104
|
+
@allowed
|
105
|
+
end
|
106
|
+
end
|
107
|
+
end
|
108
|
+
end
|
109
|
+
end
|