decidim-proposals 0.24.0 → 0.25.0.rc1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (108) hide show
  1. checksums.yaml +4 -4
  2. data/app/cells/decidim/proposals/cost_report_cell.rb +1 -1
  3. data/app/cells/decidim/proposals/proposal_m_cell.rb +4 -0
  4. data/app/cells/decidim/proposals/proposals_picker/show.erb +1 -1
  5. data/app/cells/decidim/proposals/proposals_picker_cell.rb +1 -1
  6. data/app/commands/decidim/proposals/admin/import_proposals.rb +21 -4
  7. data/app/controllers/decidim/proposals/admin/proposals_controller.rb +1 -1
  8. data/app/controllers/decidim/proposals/collaborative_drafts_controller.rb +2 -1
  9. data/app/controllers/decidim/proposals/proposals_controller.rb +1 -0
  10. data/app/forms/decidim/proposals/admin/import_participatory_text_form.rb +1 -0
  11. data/app/forms/decidim/proposals/admin/proposals_import_form.rb +6 -0
  12. data/app/forms/decidim/proposals/proposal_form.rb +7 -4
  13. data/app/helpers/decidim/proposals/map_helper.rb +5 -5
  14. data/app/helpers/decidim/proposals/participatory_texts_helper.rb +1 -1
  15. data/app/helpers/decidim/proposals/proposal_wizard_helper.rb +0 -16
  16. data/app/models/decidim/proposals/proposal.rb +11 -1
  17. data/app/{assets → packs}/documents/decidim/proposals/participatory_texts/participatory_text.md +0 -0
  18. data/app/{assets → packs}/documents/decidim/proposals/participatory_texts/participatory_text.odt +0 -0
  19. data/app/packs/entrypoints/decidim_proposals.js +8 -0
  20. data/app/packs/entrypoints/decidim_proposals_admin.js +3 -0
  21. data/app/{assets/images/decidim/gamification/badges/accepted_proposals.svg → packs/images/decidim/gamification/badges/decidim_gamification_badges_accepted_proposals.svg} +0 -0
  22. data/app/{assets/images/decidim/gamification/badges/proposal_votes.svg → packs/images/decidim/gamification/badges/decidim_gamification_badges_proposal_votes.svg} +0 -0
  23. data/app/{assets/images/decidim/gamification/badges/proposals.svg → packs/images/decidim/gamification/badges/decidim_gamification_badges_proposals.svg} +0 -0
  24. data/app/{assets/images/decidim/proposals/icon.svg → packs/images/decidim/proposals/decidim_proposals.svg} +0 -0
  25. data/app/packs/src/decidim/proposals/add_proposal.js +49 -0
  26. data/app/packs/src/decidim/proposals/admin/proposals.js +143 -0
  27. data/app/{assets/javascripts/decidim/proposals/admin/proposals_form.js.es6 → packs/src/decidim/proposals/admin/proposals_form.js} +6 -3
  28. data/app/{assets/javascripts/decidim/proposals/admin/proposals_picker.js.es6 → packs/src/decidim/proposals/admin/proposals_picker.js} +0 -0
  29. data/app/{assets/javascripts/decidim/proposals/utils.js.es6 → packs/src/decidim/proposals/utils.js} +2 -2
  30. data/app/packs/stylesheets/decidim/proposals/_proposals.scss +1 -0
  31. data/app/{assets/stylesheets/decidim/proposals/proposals/_preview.css.scss → packs/stylesheets/decidim/proposals/proposals/_preview.scss} +0 -0
  32. data/app/queries/decidim/proposals/metrics/accepted_proposals_metric_manage.rb +1 -1
  33. data/app/queries/decidim/proposals/metrics/proposals_metric_manage.rb +1 -1
  34. data/app/queries/decidim/proposals/metrics/votes_metric_manage.rb +1 -1
  35. data/app/queries/decidim/proposals/similar_proposals.rb +1 -0
  36. data/app/services/decidim/proposals/collaborative_draft_search.rb +1 -1
  37. data/app/services/decidim/proposals/proposal_builder.rb +4 -4
  38. data/app/views/decidim/proposals/admin/proposals/_form.html.erb +1 -1
  39. data/app/views/decidim/proposals/admin/proposals/index.html.erb +1 -1
  40. data/app/views/decidim/proposals/admin/proposals_imports/new.html.erb +6 -0
  41. data/app/views/decidim/proposals/collaborative_drafts/_filters.html.erb +1 -1
  42. data/app/views/decidim/proposals/collaborative_drafts/edit.html.erb +1 -1
  43. data/app/views/decidim/proposals/collaborative_drafts/index.html.erb +0 -2
  44. data/app/views/decidim/proposals/collaborative_drafts/new.html.erb +1 -1
  45. data/app/views/decidim/proposals/proposals/_dynamic_map_instructions.html.erb +2 -1
  46. data/app/views/decidim/proposals/proposals/_edit_form_fields.html.erb +5 -1
  47. data/app/views/decidim/proposals/proposals/_filters.html.erb +32 -34
  48. data/app/views/decidim/proposals/proposals/_filters_small_view.html.erb +1 -1
  49. data/app/views/decidim/proposals/proposals/complete.html.erb +1 -1
  50. data/app/views/decidim/proposals/proposals/edit.html.erb +1 -1
  51. data/app/views/decidim/proposals/proposals/edit_draft.html.erb +1 -1
  52. data/app/views/decidim/proposals/proposals/index.html.erb +4 -6
  53. data/app/views/decidim/proposals/proposals/new.html.erb +1 -1
  54. data/app/views/decidim/proposals/proposals/preview.html.erb +7 -31
  55. data/config/assets.rb +10 -0
  56. data/config/locales/ar.yml +3 -1
  57. data/config/locales/bg.yml +3 -1
  58. data/config/locales/ca.yml +11 -5
  59. data/config/locales/cs.yml +38 -29
  60. data/config/locales/de.yml +10 -4
  61. data/config/locales/el.yml +3 -1
  62. data/config/locales/en.yml +12 -3
  63. data/config/locales/es-MX.yml +11 -4
  64. data/config/locales/es-PY.yml +11 -4
  65. data/config/locales/es.yml +14 -7
  66. data/config/locales/eu.yml +3 -1
  67. data/config/locales/fi-plain.yml +12 -3
  68. data/config/locales/fi.yml +12 -3
  69. data/config/locales/fr-CA.yml +8 -3
  70. data/config/locales/fr-LU.yml +903 -0
  71. data/config/locales/fr.yml +8 -3
  72. data/config/locales/gl.yml +11 -3
  73. data/config/locales/hu.yml +3 -1
  74. data/config/locales/id-ID.yml +3 -1
  75. data/config/locales/it.yml +55 -1
  76. data/config/locales/ja.yml +38 -1
  77. data/config/locales/lb-LU.yml +1 -0
  78. data/config/locales/lv.yml +3 -1
  79. data/config/locales/nl.yml +13 -2
  80. data/config/locales/no.yml +3 -1
  81. data/config/locales/pl.yml +34 -30
  82. data/config/locales/pt-BR.yml +199 -1
  83. data/config/locales/pt.yml +3 -1
  84. data/config/locales/ro-RO.yml +7 -1
  85. data/config/locales/ru.yml +3 -1
  86. data/config/locales/sk.yml +3 -1
  87. data/config/locales/sv.yml +38 -4
  88. data/config/locales/tr-TR.yml +3 -4
  89. data/config/locales/uk.yml +3 -1
  90. data/config/locales/zh-CN.yml +3 -1
  91. data/db/migrate/20201002085508_fix_proposals_data.rb +7 -8
  92. data/db/migrate/20210318082934_fix_counters_for_copied_proposals.rb +6 -5
  93. data/lib/decidim/proposals.rb +0 -1
  94. data/lib/decidim/proposals/admin_engine.rb +0 -4
  95. data/lib/decidim/proposals/commentable_proposal.rb +5 -13
  96. data/lib/decidim/proposals/component.rb +4 -4
  97. data/lib/decidim/proposals/engine.rb +14 -19
  98. data/lib/decidim/proposals/proposal_serializer.rb +1 -0
  99. data/lib/decidim/proposals/test/capybara_proposals_picker.rb +10 -0
  100. data/lib/decidim/proposals/test/factories.rb +3 -0
  101. data/lib/decidim/proposals/version.rb +1 -1
  102. metadata +39 -113
  103. data/app/assets/config/admin/decidim_proposals_manifest.js +0 -2
  104. data/app/assets/config/decidim_proposals_manifest.js +0 -4
  105. data/app/assets/javascripts/decidim/proposals/add_proposal.js.es6 +0 -31
  106. data/app/assets/javascripts/decidim/proposals/admin/proposals.es6 +0 -130
  107. data/app/assets/stylesheets/decidim/proposals/_proposals.css.scss +0 -1
  108. data/config/locales/ja-JP.yml +0 -865
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: bc19261b23f64f8508319ad79afa60c5268b6071d397a024750ffffa9d708676
4
- data.tar.gz: e9014329c51b1a0d4611f161dc84e2849c14e02646d92d50aa2e340e08c87c1d
3
+ metadata.gz: 47fa5c2b39c2888c4cd18246303276f31bc9814aea6f2a95fbeed90801018cd9
4
+ data.tar.gz: 66b35526e1ba74dd1e4c58e6b81bf5d5785638ade13ac182f040746dabe2e97f
5
5
  SHA512:
6
- metadata.gz: 866c986caaf10b21ba47b4a7e6a5f63cdaae8aa48277b99d4cbceb3e34a7fdb5b2824ad7a7c66122f32c86fa2d4b439fdf82083beba025b119ec572c94207cb8
7
- data.tar.gz: be429c52ecfd6db0ac027291d53d5e8446fd325acaabee3515a4724020bc80a9b5a379e8e918c2b1ad4cd73475812e2388b52a45a80739d82eef3cfa6a4b23a1
6
+ metadata.gz: 4791d3eac05ea12713c702f479edbe8044aa1aa42eefe84f9def5f66c70e621253341cc3988423b7152ac321bd8bb0c65514eb5a5877a9ff1c944a0f01f34b1e
7
+ data.tar.gz: 561231c064de2ccd37f32cce1839f64fcba927de4b9f1490145a747d57214788f36010c1a7464e23d20b35af398ccfc897fcaf514ccfb3eff19856d150f2fa8d
@@ -14,7 +14,7 @@ module Decidim
14
14
  private
15
15
 
16
16
  def cost
17
- number_to_currency(model.cost)
17
+ number_to_currency(model.cost, unit: Decidim.currency_unit)
18
18
  end
19
19
 
20
20
  def cost_report
@@ -132,14 +132,18 @@ module Decidim
132
132
  hash << model.cache_key_with_version
133
133
  hash << model.proposal_votes_count
134
134
  hash << model.endorsements_count
135
+ hash << model.comments_count
135
136
  hash << Digest::MD5.hexdigest(model.component.cache_key_with_version)
136
137
  hash << Digest::MD5.hexdigest(resource_image_path) if resource_image_path
138
+ hash << render_space? ? 1 : 0
137
139
  if current_user
138
140
  hash << current_user.cache_key_with_version
139
141
  hash << current_user.follows?(model) ? 1 : 0
140
142
  end
141
143
  hash << model.follows_count
142
144
  hash << Digest::MD5.hexdigest(model.authors.map(&:cache_key_with_version).to_s)
145
+ hash << (model.must_render_translation?(model.organization) ? 1 : 0) if model.respond_to?(:must_render_translation?)
146
+ hash << model.component.participatory_space.active_step.id if model.component.participatory_space.has_steps?
143
147
 
144
148
  hash.join("/")
145
149
  end
@@ -11,4 +11,4 @@
11
11
  </div>
12
12
  </div>
13
13
 
14
- <%= javascript_include_tag "decidim/proposals/admin/proposals_picker" %>
14
+ <%= javascript_pack_tag "decidim_proposals_admin" %>
@@ -50,7 +50,7 @@ module Decidim
50
50
 
51
51
  def filtered_proposals
52
52
  @filtered_proposals ||= if filtered?
53
- proposals.where("title ILIKE ?", "%#{search_text}%")
53
+ proposals.where("title::text ILIKE ?", "%#{search_text}%")
54
54
  .or(proposals.where("reference ILIKE ?", "%#{search_text}%"))
55
55
  .or(proposals.where("id::text ILIKE ?", "%#{search_text}%"))
56
56
  else
@@ -39,15 +39,17 @@ module Decidim
39
39
  action_user: form.current_user,
40
40
  extra_attributes: {
41
41
  "component" => target_component
42
- }
42
+ }.merge(proposal_answer_attributes(original_proposal))
43
43
  )
44
44
  end.compact
45
45
  end
46
46
 
47
47
  def proposals
48
- Decidim::Proposals::Proposal
49
- .where(component: origin_component)
50
- .where(state: proposal_states)
48
+ @proposals = Decidim::Proposals::Proposal
49
+ .where(component: origin_component)
50
+ .where(state: proposal_states)
51
+ @proposals = @proposals.where(scope: proposal_scopes) unless proposal_scopes.empty?
52
+ @proposals
51
53
  end
52
54
 
53
55
  def proposal_states
@@ -61,6 +63,10 @@ module Decidim
61
63
  @proposal_states
62
64
  end
63
65
 
66
+ def proposal_scopes
67
+ @form.scopes
68
+ end
69
+
64
70
  def origin_component
65
71
  @form.origin_component
66
72
  end
@@ -78,6 +84,17 @@ module Decidim
78
84
  def proposal_author
79
85
  form.keep_authors ? nil : @form.current_organization
80
86
  end
87
+
88
+ def proposal_answer_attributes(original_proposal)
89
+ return {} unless form.keep_answers
90
+
91
+ {
92
+ answer: original_proposal.answer,
93
+ answered_at: original_proposal.answered_at,
94
+ state: original_proposal.state,
95
+ state_published_at: original_proposal.state_published_at
96
+ }
97
+ end
81
98
  end
82
99
  end
83
100
  end
@@ -147,7 +147,7 @@ module Decidim
147
147
  private
148
148
 
149
149
  def collection
150
- @collection ||= Proposal.where(component: current_component).published
150
+ @collection ||= Proposal.where(component: current_component).not_hidden.published
151
151
  end
152
152
 
153
153
  def proposals
@@ -7,6 +7,7 @@ module Decidim
7
7
  helper Decidim::WidgetUrlsHelper
8
8
  helper ProposalWizardHelper
9
9
  helper TooltipHelper
10
+ helper UserGroupHelper
10
11
 
11
12
  include Decidim::ApplicationHelper
12
13
  include FormFactory
@@ -133,7 +134,7 @@ module Decidim
133
134
  end
134
135
 
135
136
  def geocoded_collaborative_draft
136
- @geocoded_collaborative_draft ||= search.results.not_hidden.select(&:geocoded?)
137
+ @geocoded_collaborative_draft ||= search.results.not_hidden.select(&:geocoded_and_valid?)
137
138
  end
138
139
 
139
140
  def search_klass
@@ -7,6 +7,7 @@ module Decidim
7
7
  helper Decidim::WidgetUrlsHelper
8
8
  helper ProposalWizardHelper
9
9
  helper ParticipatoryTextsHelper
10
+ helper UserGroupHelper
10
11
  include Decidim::ApplicationHelper
11
12
  include Flaggable
12
13
  include Withdrawable
@@ -7,6 +7,7 @@ module Decidim
7
7
  # from a participatory text.
8
8
  class ImportParticipatoryTextForm < Decidim::Form
9
9
  include TranslatableAttributes
10
+ include Decidim::HasUploadValidations
10
11
 
11
12
  # WARNING: consider adding/removing the relative translation key at
12
13
  # decidim.assemblies.admin.new_import.accepted_types when modifying this hash
@@ -10,8 +10,10 @@ module Decidim
10
10
 
11
11
  attribute :origin_component_id, Integer
12
12
  attribute :import_proposals, Boolean
13
+ attribute :keep_answers, Boolean
13
14
  attribute :keep_authors, Boolean
14
15
  attribute :states, Array
16
+ attribute :scope_ids, Array
15
17
 
16
18
  validates :origin_component_id, :origin_component, :states, :current_component, presence: true
17
19
  validates :import_proposals, allow_nil: false, acceptance: true
@@ -32,6 +34,10 @@ module Decidim
32
34
  super.reject(&:blank?)
33
35
  end
34
36
 
37
+ def scopes
38
+ Decidim::Scope.where(organization: current_organization, id: scope_ids)
39
+ end
40
+
35
41
  def origin_component
36
42
  @origin_component ||= origin_components.find_by(id: origin_component_id)
37
43
  end
@@ -107,12 +107,15 @@ module Decidim
107
107
 
108
108
  private
109
109
 
110
- # This method will add an error to the `attachment` field only if there's
111
- # any error in any other field. This is needed because when the form has
112
- # an error, the attachment is lost, so we need a way to inform the user of
110
+ # This method will add an error to the `add_photos` and/or "add_documents" fields
111
+ # only if there's any error in any other field. This is needed because when the
112
+ # form has an error, the attachment is lost, so we need a way to inform the user of
113
113
  # this problem.
114
114
  def notify_missing_attachment_if_errored
115
- errors.add(:attachment, :needs_to_be_reattached) if errors.any? && attachment.present?
115
+ if errors.any?
116
+ errors.add(:add_photos, :needs_to_be_reattached) if add_photos.present?
117
+ errors.add(:add_documents, :needs_to_be_reattached) if add_documents.present?
118
+ end
116
119
  end
117
120
 
118
121
  def ordered_hashtag_list(string)
@@ -26,16 +26,16 @@ module Decidim
26
26
  end
27
27
 
28
28
  def proposal_preview_data_for_map(proposal)
29
- [
30
- proposal.slice(
29
+ {
30
+ type: "drag-marker",
31
+ marker: proposal.slice(
31
32
  :latitude,
32
33
  :longitude,
33
34
  :address
34
35
  ).merge(
35
- icon: icon("proposals", width: 40, height: 70, remove_icon_class: true),
36
- draggable: true
36
+ icon: icon("proposals", width: 40, height: 70, remove_icon_class: true)
37
37
  )
38
- ]
38
+ }
39
39
  end
40
40
 
41
41
  def has_position?(proposal)
@@ -30,7 +30,7 @@ module Decidim
30
30
  accepted_mime_types = Decidim::Proposals::DocToMarkdown::ACCEPTED_MIME_TYPES.keys
31
31
  accepted_mime_types.each_with_index do |mime_type, index|
32
32
  links += link_to t(".accepted_mime_types.#{mime_type}"),
33
- asset_path("decidim/proposals/participatory_texts/participatory_text.#{mime_type}"),
33
+ asset_pack_path("media/documents/participatory_text.#{mime_type}"),
34
34
  download: "participatory_text.#{mime_type}"
35
35
  links += ", " unless accepted_mime_types.length == index + 1
36
36
  end
@@ -107,22 +107,6 @@ module Decidim
107
107
  translated_attribute(component_settings.try("proposal_wizard_#{step}_help_text")).present?
108
108
  end
109
109
 
110
- # Renders a user_group select field in a form.
111
- # form - FormBuilder object
112
- # name - attribute user_group_id
113
- #
114
- # Returns nothing.
115
- def user_group_select_field(form, name)
116
- selected = @form.user_group_id.presence
117
- user_groups = Decidim::UserGroups::ManageableUserGroups.for(current_user).verified
118
- form.select(
119
- name,
120
- user_groups.map { |g| [g.name, g.id] },
121
- selected: selected,
122
- include_blank: current_user.name
123
- )
124
- end
125
-
126
110
  private
127
111
 
128
112
  def total_steps
@@ -337,8 +337,18 @@ module Decidim
337
337
  ")
338
338
  end
339
339
 
340
+ def self.sort_by_translated_title_asc
341
+ field = Arel::Nodes::InfixOperation.new("->>", arel_table[:title], Arel::Nodes.build_quoted(I18n.locale))
342
+ order(Arel::Nodes::InfixOperation.new("", field, Arel.sql("ASC")))
343
+ end
344
+
345
+ def self.sort_by_translated_title_desc
346
+ field = Arel::Nodes::InfixOperation.new("->>", arel_table[:title], Arel::Nodes.build_quoted(I18n.locale))
347
+ order(Arel::Nodes::InfixOperation.new("", field, Arel.sql("DESC")))
348
+ end
349
+
340
350
  ransacker :title do
341
- Arel.sql(%{("decidim_proposals_proposals"."title")::text})
351
+ Arel.sql(%{cast("decidim_proposals_proposals"."title" as text)})
342
352
  end
343
353
 
344
354
  ransacker :id_string do
@@ -0,0 +1,8 @@
1
+ import "src/decidim/proposals/utils"
2
+ import "src/decidim/proposals/add_proposal"
3
+
4
+ // Images
5
+ require.context("../images", true)
6
+
7
+ // Documents
8
+ require.context("../documents", true)
@@ -0,0 +1,3 @@
1
+ import "src/decidim/proposals/admin/proposals"
2
+ import "src/decidim/proposals/admin/proposals_form"
3
+ import "src/decidim/proposals/admin/proposals_picker"
@@ -0,0 +1,49 @@
1
+ import attachGeocoding from "src/decidim/geocoding/attach_input"
2
+ import getCoordinateInputName from "src/decidim/geocoding/coordinate_input";
3
+
4
+ $(() => {
5
+ const $checkbox = $("input:checkbox[name$='[has_address]']");
6
+ const $addressInput = $("#address_input");
7
+ const $addressInputField = $("input", $addressInput);
8
+ const $map = $("#address_map");
9
+ const latFieldName = getCoordinateInputName("latitude", $addressInputField, {})
10
+ const longFieldName = getCoordinateInputName("longitude", $addressInputField, {})
11
+ $map.hide();
12
+
13
+ if ($checkbox.length > 0) {
14
+ const toggleInput = () => {
15
+ if ($checkbox[0].checked) {
16
+ $addressInput.show();
17
+ $addressInputField.prop("disabled", false);
18
+ } else {
19
+ $addressInput.hide();
20
+ $addressInputField.prop("disabled", true);
21
+ }
22
+ }
23
+ toggleInput();
24
+ $checkbox.on("change", toggleInput);
25
+ }
26
+
27
+ if ($addressInput.length > 0) {
28
+ if ($checkbox[0].checked) {
29
+ $map.show();
30
+ }
31
+
32
+ const ctrl = $("[data-decidim-map]").data("map-controller");
33
+ ctrl.setEventHandler("coordinates", (ev) => {
34
+ $(`input[name='${latFieldName}']`).val(ev.lat);
35
+ $(`input[name='${longFieldName}']`).val(ev.lng);
36
+ });
37
+
38
+ attachGeocoding($addressInputField, null, (coordinates) => {
39
+ $map.show();
40
+ // Remove previous marker when user updates address in address field
41
+ ctrl.removeMarker();
42
+ ctrl.addMarker({
43
+ latitude: coordinates[0],
44
+ longitude: coordinates[1],
45
+ address: $addressInput.val()
46
+ });
47
+ });
48
+ }
49
+ });
@@ -0,0 +1,143 @@
1
+ /* eslint-disable no-invalid-this */
2
+ /* eslint no-unused-vars: 0 */
3
+ /* eslint id-length: ["error", { "exceptions": ["e"] }] */
4
+
5
+ $(() => {
6
+ const selectedProposalsCount = function() {
7
+ return $(".table-list .js-check-all-proposal:checked").length
8
+ }
9
+
10
+ const selectedProposalsNotPublishedAnswerCount = function() {
11
+ return $(".table-list [data-published-state=false] .js-check-all-proposal:checked").length
12
+ }
13
+
14
+ const selectedProposalsCountUpdate = function() {
15
+ const selectedProposals = selectedProposalsCount();
16
+ const selectedProposalsNotPublishedAnswer = selectedProposalsNotPublishedAnswerCount();
17
+ if (selectedProposals === 0) {
18
+ $("#js-selected-proposals-count").text("")
19
+ } else {
20
+ $("#js-selected-proposals-count").text(selectedProposals);
21
+ }
22
+
23
+ if (selectedProposals >= 2) {
24
+ $('button[data-action="merge-proposals"]').parent().show();
25
+ } else {
26
+ $('button[data-action="merge-proposals"]').parent().hide();
27
+ }
28
+
29
+ if (selectedProposalsNotPublishedAnswer > 0) {
30
+ $('button[data-action="publish-answers"]').parent().show();
31
+ $("#js-form-publish-answers-number").text(selectedProposalsNotPublishedAnswer);
32
+ } else {
33
+ $('button[data-action="publish-answers"]').parent().hide();
34
+ }
35
+ }
36
+
37
+ const showBulkActionsButton = function() {
38
+ if (selectedProposalsCount() > 0) {
39
+ $("#js-bulk-actions-button").removeClass("hide");
40
+ }
41
+ }
42
+
43
+ const hideBulkActionsButton = function(force = false) {
44
+ if (selectedProposalsCount() === 0 || force === true) {
45
+ $("#js-bulk-actions-button").addClass("hide");
46
+ $("#js-bulk-actions-dropdown").removeClass("is-open");
47
+ }
48
+ }
49
+
50
+ const showOtherActionsButtons = function() {
51
+ $("#js-other-actions-wrapper").removeClass("hide");
52
+ }
53
+
54
+ const hideOtherActionsButtons = function() {
55
+ $("#js-other-actions-wrapper").addClass("hide");
56
+ }
57
+
58
+ const hideBulkActionForms = function() {
59
+ $(".js-bulk-action-form").addClass("hide");
60
+ }
61
+
62
+ // Expose functions to make them avaialble in .js.erb templates
63
+ window.selectedProposalsCount = selectedProposalsCount;
64
+ window.selectedProposalsNotPublishedAnswerCount = selectedProposalsNotPublishedAnswerCount;
65
+ window.selectedProposalsCountUpdate = selectedProposalsCountUpdate;
66
+ window.showBulkActionsButton = showBulkActionsButton;
67
+ window.hideBulkActionsButton = hideBulkActionsButton;
68
+ window.showOtherActionsButtons = showOtherActionsButtons;
69
+ window.hideOtherActionsButtons = hideOtherActionsButtons;
70
+ window.hideBulkActionForms = hideBulkActionForms;
71
+
72
+ if ($(".js-bulk-action-form").length) {
73
+ hideBulkActionForms();
74
+ $("#js-bulk-actions-button").addClass("hide");
75
+
76
+ $("#js-bulk-actions-dropdown ul li button").click(function(e) {
77
+ e.preventDefault();
78
+ let action = $(e.target).data("action");
79
+
80
+ if (action) {
81
+ $(`#js-form-${action}`).submit(function() {
82
+ $(".layout-content > .callout-wrapper").html("");
83
+ })
84
+
85
+ $(`#js-${action}-actions`).removeClass("hide");
86
+ hideBulkActionsButton(true);
87
+ hideOtherActionsButtons();
88
+ }
89
+ })
90
+
91
+ // select all checkboxes
92
+ $(".js-check-all").change(function() {
93
+ $(".js-check-all-proposal").prop("checked", $(this).prop("checked"));
94
+
95
+ if ($(this).prop("checked")) {
96
+ $(".js-check-all-proposal").closest("tr").addClass("selected");
97
+ showBulkActionsButton();
98
+ } else {
99
+ $(".js-check-all-proposal").closest("tr").removeClass("selected");
100
+ hideBulkActionsButton();
101
+ }
102
+
103
+ selectedProposalsCountUpdate();
104
+ });
105
+
106
+ // proposal checkbox change
107
+ $(".table-list").on("change", ".js-check-all-proposal", function (e) {
108
+ let proposalId = $(this).val()
109
+ let checked = $(this).prop("checked")
110
+
111
+ // uncheck "select all", if one of the listed checkbox item is unchecked
112
+ if ($(this).prop("checked") === false) {
113
+ $(".js-check-all").prop("checked", false);
114
+ }
115
+ // check "select all" if all checkbox proposals are checked
116
+ if ($(".js-check-all-proposal:checked").length === $(".js-check-all-proposal").length) {
117
+ $(".js-check-all").prop("checked", true);
118
+ showBulkActionsButton();
119
+ }
120
+
121
+ if ($(this).prop("checked")) {
122
+ showBulkActionsButton();
123
+ $(this).closest("tr").addClass("selected");
124
+ } else {
125
+ hideBulkActionsButton();
126
+ $(this).closest("tr").removeClass("selected");
127
+ }
128
+
129
+ if ($(".js-check-all-proposal:checked").length === 0) {
130
+ hideBulkActionsButton();
131
+ }
132
+
133
+ $(".js-bulk-action-form").find(`.js-proposal-id-${proposalId}`).prop("checked", checked);
134
+ selectedProposalsCountUpdate();
135
+ });
136
+
137
+ $(".js-cancel-bulk-action").on("click", function (e) {
138
+ hideBulkActionForms()
139
+ showBulkActionsButton();
140
+ showOtherActionsButtons();
141
+ });
142
+ }
143
+ });