decidim-core 0.26.5 → 0.27.0.rc1

Sign up to get free protection for your applications and to get access to all the features.

Potentially problematic release.


This version of decidim-core might be problematic. Click here for more details.

Files changed (588) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +1 -1
  3. data/app/cells/decidim/amendable/announcement_cell.rb +4 -9
  4. data/app/cells/decidim/announcement_cell.rb +1 -1
  5. data/app/cells/decidim/author_cell.rb +1 -1
  6. data/app/cells/decidim/card_m_cell.rb +1 -1
  7. data/app/cells/decidim/content_blocks/cta/show.erb +1 -1
  8. data/app/cells/decidim/content_blocks/footer_sub_hero/show.erb +1 -1
  9. data/app/cells/decidim/content_blocks/hero/show.erb +1 -1
  10. data/app/cells/decidim/content_blocks/highlighted_content_banner/show.erb +4 -4
  11. data/app/cells/decidim/content_blocks/highlighted_elements_cell.rb +1 -1
  12. data/app/cells/decidim/content_blocks/how_to_participate/show.erb +1 -1
  13. data/app/cells/decidim/content_blocks/last_activity/show.erb +1 -1
  14. data/app/cells/decidim/content_blocks/last_activity_cell.rb +3 -3
  15. data/app/cells/decidim/content_blocks/stats_cell.rb +4 -5
  16. data/app/cells/decidim/content_blocks/sub_hero/show.erb +1 -1
  17. data/app/cells/decidim/data_consent/category.erb +64 -0
  18. data/app/cells/decidim/data_consent/dialog.erb +29 -0
  19. data/app/cells/decidim/data_consent/modal.erb +39 -0
  20. data/app/cells/decidim/data_consent/show.erb +3 -0
  21. data/app/cells/decidim/data_consent_cell.rb +21 -0
  22. data/app/cells/decidim/follow_button_cell.rb +5 -7
  23. data/app/cells/decidim/newsletter_templates/base_cell.rb +0 -8
  24. data/app/cells/decidim/newsletter_templates/basic_only_text/show.erb +4 -4
  25. data/app/cells/decidim/newsletter_templates/image_text_cta/show.erb +4 -4
  26. data/app/cells/decidim/profile_sidebar_cell.rb +1 -1
  27. data/app/cells/decidim/search_results_cell.rb +1 -1
  28. data/app/cells/decidim/tos_page/refuse_btn_modal.erb +1 -1
  29. data/app/cells/decidim/upload_modal/files.erb +52 -0
  30. data/app/cells/decidim/upload_modal/modal.erb +49 -0
  31. data/app/cells/decidim/upload_modal/show.erb +3 -0
  32. data/app/cells/decidim/upload_modal_cell.rb +205 -0
  33. data/app/cells/decidim/user_conversation/reply.erb +1 -1
  34. data/app/cells/decidim/user_conversations/add_conversation_users.erb +5 -3
  35. data/app/cells/decidim/version_cell.rb +2 -13
  36. data/app/cells/decidim/versions_list_cell.rb +2 -13
  37. data/app/cells/decidim/versions_list_item_cell.rb +2 -13
  38. data/app/commands/decidim/accept_group_invitation.rb +1 -1
  39. data/app/commands/decidim/accept_user_group_join_request.rb +1 -1
  40. data/app/commands/decidim/amendable/accept.rb +1 -1
  41. data/app/commands/decidim/amendable/create_draft.rb +1 -1
  42. data/app/commands/decidim/amendable/destroy_draft.rb +1 -1
  43. data/app/commands/decidim/amendable/promote.rb +1 -1
  44. data/app/commands/decidim/amendable/publish_draft.rb +1 -1
  45. data/app/commands/decidim/amendable/reject.rb +1 -1
  46. data/app/commands/decidim/amendable/update_draft.rb +1 -1
  47. data/app/commands/decidim/amendable/withdraw.rb +1 -1
  48. data/app/commands/decidim/attachment_methods.rb +3 -2
  49. data/app/commands/decidim/create_editor_image.rb +1 -1
  50. data/app/commands/decidim/create_follow.rb +1 -1
  51. data/app/commands/decidim/create_omniauth_registration.rb +2 -3
  52. data/app/commands/decidim/create_registration.rb +1 -2
  53. data/app/commands/decidim/create_report.rb +1 -1
  54. data/app/commands/decidim/create_user_group.rb +2 -2
  55. data/app/commands/decidim/create_user_report.rb +5 -3
  56. data/app/commands/decidim/delete_follow.rb +1 -1
  57. data/app/commands/decidim/demote_membership.rb +1 -1
  58. data/app/commands/decidim/destroy_account.rb +1 -1
  59. data/app/commands/decidim/endorse_resource.rb +1 -1
  60. data/app/commands/decidim/gallery_methods.rb +45 -10
  61. data/app/commands/decidim/invite_user.rb +1 -3
  62. data/app/commands/decidim/invite_user_again.rb +1 -1
  63. data/app/commands/decidim/invite_user_to_group.rb +1 -1
  64. data/app/commands/decidim/join_user_group.rb +1 -1
  65. data/app/commands/decidim/leave_user_group.rb +8 -2
  66. data/app/commands/decidim/messaging/reply_to_conversation.rb +2 -2
  67. data/app/commands/decidim/messaging/start_conversation.rb +2 -2
  68. data/app/commands/decidim/multiple_attachments_methods.rb +44 -5
  69. data/app/commands/decidim/promote_membership.rb +1 -1
  70. data/app/commands/decidim/reject_group_invitation.rb +1 -1
  71. data/app/commands/decidim/reject_user_group_join_request.rb +1 -1
  72. data/app/commands/decidim/remove_user_from_group.rb +1 -1
  73. data/app/commands/decidim/resend_confirmation_instructions.rb +18 -0
  74. data/app/commands/decidim/search.rb +10 -9
  75. data/app/commands/decidim/unendorse_resource.rb +2 -2
  76. data/app/commands/decidim/unsubscribe_settings.rb +1 -1
  77. data/app/commands/decidim/update_account.rb +2 -2
  78. data/app/commands/decidim/update_notifications_settings.rb +3 -2
  79. data/app/commands/decidim/update_password.rb +31 -0
  80. data/app/commands/decidim/update_user_group.rb +2 -2
  81. data/app/commands/decidim/update_user_interests.rb +1 -1
  82. data/app/commands/decidim/validate_upload.rb +15 -0
  83. data/app/controllers/concerns/decidim/devise_controllers.rb +0 -1
  84. data/app/controllers/concerns/decidim/filter_resource.rb +23 -24
  85. data/app/controllers/concerns/decidim/force_authentication.rb +1 -2
  86. data/app/controllers/concerns/decidim/http_caching_disabler.rb +1 -1
  87. data/app/controllers/concerns/decidim/needs_password_change.rb +43 -0
  88. data/app/controllers/concerns/decidim/needs_tos_accepted.rb +3 -3
  89. data/app/controllers/concerns/decidim/orderable.rb +1 -1
  90. data/app/controllers/concerns/decidim/participatory_space_context.rb +2 -2
  91. data/app/controllers/concerns/decidim/resource_versions_concern.rb +0 -4
  92. data/app/controllers/decidim/account_controller.rb +45 -0
  93. data/app/controllers/decidim/application_controller.rb +1 -1
  94. data/app/controllers/decidim/devise/invitations_controller.rb +4 -9
  95. data/app/controllers/decidim/devise/passwords_controller.rb +33 -0
  96. data/app/controllers/decidim/devise/registrations_controller.rb +3 -5
  97. data/app/controllers/decidim/devise/sessions_controller.rb +2 -0
  98. data/app/controllers/decidim/{data_portability_controller.rb → download_your_data_controller.rb} +8 -8
  99. data/app/controllers/decidim/groups_controller.rb +5 -5
  100. data/app/controllers/decidim/last_activities_controller.rb +7 -9
  101. data/app/controllers/decidim/links_controller.rb +2 -4
  102. data/app/controllers/decidim/manifests_controller.rb +11 -0
  103. data/app/controllers/decidim/newsletters_controller.rb +1 -1
  104. data/app/controllers/decidim/notifications_subscriptions_controller.rb +16 -0
  105. data/app/controllers/decidim/offline_controller.rb +7 -0
  106. data/app/controllers/decidim/profiles_controller.rb +2 -2
  107. data/app/controllers/decidim/searches_controller.rb +4 -3
  108. data/app/controllers/decidim/short_links_controller.rb +35 -0
  109. data/app/controllers/decidim/upload_validations_controller.rb +44 -0
  110. data/app/controllers/decidim/user_activities_controller.rb +32 -16
  111. data/app/controllers/decidim/user_interests_controller.rb +0 -2
  112. data/app/controllers/decidim/user_timeline_controller.rb +6 -8
  113. data/app/events/decidim/change_nickname_event.rb +21 -0
  114. data/app/events/decidim/resource_endorsed_event.rb +1 -2
  115. data/app/forms/decidim/account_form.rb +3 -3
  116. data/app/forms/decidim/amendable/form.rb +1 -2
  117. data/app/forms/decidim/form.rb +1 -1
  118. data/app/forms/decidim/messaging/conversation_form.rb +1 -1
  119. data/app/forms/decidim/notifications_settings_form.rb +8 -2
  120. data/app/forms/decidim/password_form.rb +10 -0
  121. data/app/forms/decidim/registration_form.rb +2 -2
  122. data/app/forms/decidim/upload_validation_form.rb +54 -0
  123. data/app/forms/decidim/user_group_form.rb +1 -1
  124. data/app/forms/decidim/user_interest_scope_form.rb +1 -1
  125. data/app/forms/decidim/user_interests_form.rb +1 -1
  126. data/app/forms/translatable_presence_validator.rb +2 -2
  127. data/app/forms/url_validator.rb +1 -1
  128. data/app/helpers/decidim/application_helper.rb +2 -2
  129. data/app/helpers/decidim/attachments_helper.rb +1 -1
  130. data/app/helpers/decidim/cells_paginate_helper.rb +1 -1
  131. data/app/helpers/decidim/datalist_select_helper.rb +40 -0
  132. data/app/helpers/decidim/decidim_form_helper.rb +1 -1
  133. data/app/helpers/decidim/filters_helper.rb +1 -5
  134. data/app/helpers/decidim/followable_helper.rb +1 -1
  135. data/app/helpers/decidim/icon_helper.rb +4 -4
  136. data/app/helpers/decidim/layout_helper.rb +15 -1
  137. data/app/helpers/decidim/map_helper.rb +12 -8
  138. data/app/helpers/decidim/meta_tags_helper.rb +0 -2
  139. data/app/helpers/decidim/newsletters_helper.rb +0 -1
  140. data/app/helpers/decidim/notifications_settings_helper.rb +15 -0
  141. data/app/helpers/decidim/passwords_helper.rb +41 -0
  142. data/app/helpers/decidim/sanitize_helper.rb +1 -0
  143. data/app/helpers/decidim/searches_helper.rb +6 -6
  144. data/app/helpers/decidim/short_link_helper.rb +28 -0
  145. data/app/helpers/decidim/translations_helper.rb +2 -2
  146. data/app/jobs/decidim/{data_portability_export_job.rb → download_your_data_export_job.rb} +5 -5
  147. data/app/jobs/decidim/email_notifications_digest_generator_job.rb +21 -0
  148. data/app/jobs/decidim/notification_generator_for_recipient_job.rb +5 -3
  149. data/app/jobs/decidim/reminder_generator_job.rb +12 -0
  150. data/app/jobs/decidim/resend_confirmation_instructions_job.rb +11 -0
  151. data/app/jobs/decidim/user_report_job.rb +2 -2
  152. data/app/mailers/concerns/decidim/multitenant_asset_host.rb +11 -3
  153. data/app/mailers/decidim/application_mailer.rb +1 -1
  154. data/app/mailers/decidim/block_user_mailer.rb +8 -7
  155. data/app/mailers/decidim/export_mailer.rb +2 -2
  156. data/app/mailers/decidim/newsletter_mailer.rb +5 -12
  157. data/app/mailers/decidim/newsletters_opt_in_mailer.rb +7 -5
  158. data/app/mailers/decidim/notification_mailer.rb +0 -1
  159. data/app/mailers/decidim/notifications_digest_mailer.rb +23 -0
  160. data/app/mailers/decidim/user_report_mailer.rb +10 -11
  161. data/app/models/decidim/action_log.rb +95 -11
  162. data/app/models/decidim/amendment.rb +0 -1
  163. data/app/models/decidim/area.rb +1 -2
  164. data/app/models/decidim/area_type.rb +5 -0
  165. data/app/models/decidim/attachment.rb +6 -0
  166. data/app/models/decidim/attachment_collection.rb +5 -0
  167. data/app/models/decidim/category.rb +9 -4
  168. data/app/models/decidim/coauthorship.rb +0 -2
  169. data/app/models/decidim/contextual_help_section.rb +5 -1
  170. data/app/models/decidim/follow.rb +2 -2
  171. data/app/models/decidim/gamification/badge_score.rb +0 -1
  172. data/app/models/decidim/identity.rb +2 -2
  173. data/app/models/decidim/messaging/conversation.rb +4 -4
  174. data/app/models/decidim/messaging/message.rb +1 -2
  175. data/app/models/decidim/messaging/participation.rb +0 -2
  176. data/app/models/decidim/messaging/receipt.rb +0 -2
  177. data/app/models/decidim/newsletter.rb +0 -28
  178. data/app/models/decidim/notification.rb +5 -2
  179. data/app/models/decidim/oauth_application.rb +12 -1
  180. data/app/models/decidim/organization.rb +7 -7
  181. data/app/models/decidim/participatory_space_private_user.rb +4 -2
  182. data/app/models/decidim/permission_action.rb +1 -0
  183. data/app/models/decidim/reminder.rb +10 -0
  184. data/app/models/decidim/reminder_delivery.rb +7 -0
  185. data/app/models/decidim/reminder_record.rb +29 -0
  186. data/app/models/decidim/report.rb +2 -2
  187. data/app/models/decidim/scope.rb +3 -4
  188. data/app/models/decidim/scope_type.rb +5 -0
  189. data/app/models/decidim/share_token.rb +0 -2
  190. data/app/models/decidim/short_link.rb +162 -0
  191. data/app/models/decidim/user.rb +36 -6
  192. data/app/models/decidim/user_base_entity.rb +1 -4
  193. data/app/models/decidim/user_block.rb +2 -2
  194. data/app/models/decidim/user_group.rb +4 -4
  195. data/app/models/decidim/user_moderation.rb +0 -2
  196. data/app/models/decidim/user_report.rb +2 -2
  197. data/app/packs/entrypoints/decidim_core.js +2 -2
  198. data/app/packs/entrypoints/decidim_sw.js +1 -0
  199. data/app/packs/images/decidim/gamification/badges/decidim_gamification_badges_invitations.svg +1 -0
  200. data/app/packs/src/decidim/autocomplete.js +298 -0
  201. data/app/packs/src/decidim/cookie_consent/consent_manager.js +133 -0
  202. data/app/packs/src/decidim/cookie_consent/cookie_consent.js +103 -0
  203. data/app/packs/src/decidim/datalist_select.js +36 -0
  204. data/app/packs/src/decidim/direct_uploads/upload_field.js +117 -0
  205. data/app/packs/src/decidim/direct_uploads/upload_modal.js +264 -0
  206. data/app/packs/src/decidim/direct_uploads/upload_utility.js +49 -0
  207. data/app/packs/src/decidim/direct_uploads/uploader.js +89 -0
  208. data/app/packs/src/decidim/editor/linebreak_module.js +8 -0
  209. data/app/packs/src/decidim/editor.js +2 -9
  210. data/app/packs/src/decidim/external_domain_warning.js +1 -1
  211. data/app/packs/src/decidim/external_link.js +14 -3
  212. data/app/packs/src/decidim/external_link.test.js +64 -0
  213. data/app/packs/src/decidim/form_filter.component.test.js +5 -148
  214. data/app/packs/src/decidim/form_filter.js +4 -26
  215. data/app/packs/src/decidim/geocoding.js +10 -35
  216. data/app/packs/src/decidim/identity_selector_dialog.js +1 -1
  217. data/app/packs/src/decidim/impersonation.js +3 -3
  218. data/app/packs/src/decidim/input_character_counter.js +177 -7
  219. data/app/packs/src/decidim/input_multiple_mentions.js +87 -187
  220. data/app/packs/src/decidim/map/icon.js +2 -1
  221. data/app/packs/src/decidim/map/svg-icon.js +248 -0
  222. data/app/packs/src/decidim/security/selfxss_warning.js +5 -1
  223. data/app/packs/src/decidim/session_timeouter.js +15 -15
  224. data/app/packs/src/decidim/sw/a2hs.js +28 -0
  225. data/app/packs/src/decidim/sw/index.js +3 -0
  226. data/app/packs/src/decidim/sw/loader.js +18 -0
  227. data/app/packs/src/decidim/sw/push-permissions.js +87 -0
  228. data/app/packs/src/decidim/sw/sw.js +80 -0
  229. data/app/packs/src/decidim/vizzs/areachart.js +33 -28
  230. data/app/packs/src/decidim/vizzs/linechart.js +23 -18
  231. data/app/packs/src/decidim/vizzs/rowchart.js +15 -11
  232. data/app/packs/stylesheets/decidim/email.scss +66 -7
  233. data/app/packs/stylesheets/decidim/layouts/_user.scss +8 -0
  234. data/app/packs/stylesheets/decidim/modules/_autocomplete.scss +99 -0
  235. data/app/packs/stylesheets/decidim/modules/_cards.scss +0 -2
  236. data/app/packs/stylesheets/decidim/modules/_comments.scss +0 -2
  237. data/app/packs/stylesheets/decidim/modules/_cookie-consent.scss +115 -0
  238. data/app/packs/stylesheets/decidim/modules/_footer.scss +4 -0
  239. data/app/packs/stylesheets/decidim/modules/_forms.scss +0 -5
  240. data/app/packs/stylesheets/decidim/modules/_input-multiple-mentions.scss +29 -2
  241. data/app/packs/stylesheets/decidim/modules/_modules.scss +3 -1
  242. data/app/packs/stylesheets/decidim/modules/_reveal.scss +15 -0
  243. data/app/packs/stylesheets/decidim/modules/_upload_modal.scss +143 -0
  244. data/app/packs/stylesheets/decidim/utils/_fontface.scss +3 -0
  245. data/app/permissions/decidim/permissions.rb +4 -8
  246. data/app/presenters/decidim/admin_log/area_type_presenter.rb +34 -0
  247. data/app/presenters/decidim/admin_log/attachment_collection_presenter.rb +35 -0
  248. data/app/presenters/decidim/admin_log/attachment_presenter.rb +27 -0
  249. data/app/presenters/decidim/admin_log/category_presenter.rb +36 -0
  250. data/app/presenters/decidim/admin_log/component_presenter.rb +35 -3
  251. data/app/presenters/decidim/admin_log/contextual_help_section_presenter.rb +33 -0
  252. data/app/presenters/decidim/admin_log/organization_presenter.rb +9 -4
  253. data/app/presenters/decidim/admin_log/scope_type_presenter.rb +34 -0
  254. data/app/presenters/decidim/admin_log/user_group_presenter.rb +1 -1
  255. data/app/presenters/decidim/admin_log/user_moderation_presenter.rb +1 -1
  256. data/app/presenters/decidim/home_stats_presenter.rb +8 -13
  257. data/app/presenters/decidim/log/base_presenter.rb +1 -1
  258. data/app/presenters/decidim/menu_item_presenter.rb +1 -1
  259. data/app/presenters/decidim/metric_charts_presenter.rb +7 -1
  260. data/app/presenters/decidim/nil_presenter.rb +3 -1
  261. data/app/presenters/decidim/notification_to_mailer_presenter.rb +39 -0
  262. data/app/presenters/decidim/notifications_digest_presenter.rb +42 -0
  263. data/app/presenters/decidim/official_author_presenter.rb +4 -0
  264. data/app/presenters/decidim/organization_presenter.rb +22 -0
  265. data/app/presenters/decidim/push_notification_presenter.rb +24 -0
  266. data/app/presenters/decidim/stats_presenter.rb +9 -8
  267. data/app/presenters/decidim/user_presenter.rb +4 -12
  268. data/app/presenters/decidim/validation_errors_presenter.rb +1 -1
  269. data/app/queries/decidim/messaging/user_conversations.rb +1 -1
  270. data/app/queries/decidim/metric_manage.rb +1 -1
  271. data/app/queries/decidim/own_activities.rb +11 -0
  272. data/app/queries/decidim/participatory_processes_with_user_role.rb +1 -1
  273. data/app/{services/decidim/activity_search.rb → queries/decidim/public_activities.rb} +35 -78
  274. data/app/queries/decidim/public_components.rb +1 -1
  275. data/app/queries/decidim/similar_emendations.rb +1 -1
  276. data/app/queries/decidim/stats_users_count.rb +1 -1
  277. data/app/queries/decidim/user_groups/accepted_memberships.rb +1 -1
  278. data/app/queries/decidim/user_groups/accepted_user_groups.rb +1 -1
  279. data/app/queries/decidim/user_groups/accepted_users.rb +1 -1
  280. data/app/queries/decidim/user_groups/admin_memberships.rb +1 -1
  281. data/app/queries/decidim/user_groups/invited_memberships.rb +1 -1
  282. data/app/queries/decidim/user_groups/manageable_user_groups.rb +1 -1
  283. data/app/queries/decidim/user_groups/member_memberships.rb +1 -1
  284. data/app/services/decidim/{data_portability_exporter.rb → download_your_data_exporter.rb} +6 -6
  285. data/app/services/decidim/email_notification_generator.rb +2 -2
  286. data/app/services/decidim/engine_resolver.rb +66 -0
  287. data/app/services/decidim/events_manager.rb +1 -1
  288. data/app/services/decidim/iframe_disabler.rb +31 -0
  289. data/app/services/decidim/notification_generator_for_recipient.rb +1 -1
  290. data/app/services/decidim/notifications_digest_sending_decider.rb +18 -0
  291. data/app/services/decidim/notifications_subscriptions_persistor.rb +37 -0
  292. data/app/services/decidim/open_data_exporter.rb +1 -1
  293. data/app/services/decidim/resource_search.rb +17 -164
  294. data/app/services/decidim/send_push_notification.rb +59 -0
  295. data/app/services/decidim/traceability.rb +1 -1
  296. data/app/uploaders/decidim/application_uploader.rb +2 -12
  297. data/app/uploaders/decidim/cw/attachment_uploader.rb +1 -1
  298. data/app/uploaders/decidim/cw/{data_portability_uploader.rb → download_your_data_uploader.rb} +3 -3
  299. data/app/uploaders/decidim/cw/image_uploader.rb +2 -2
  300. data/app/uploaders/decidim/organization_favicon_uploader.rb +3 -2
  301. data/app/validators/etiquette_validator.rb +3 -7
  302. data/app/validators/passthru_validator.rb +29 -21
  303. data/app/validators/password_validator.rb +25 -4
  304. data/app/validators/uploader_image_dimensions_validator.rb +1 -1
  305. data/app/views/decidim/account/_account_notification.js.erb +20 -0
  306. data/app/views/decidim/account/_password_fields.html.erb +2 -2
  307. data/app/views/decidim/account/cancel_email_change.js.erb +6 -0
  308. data/app/views/decidim/account/resend_confirmation_instructions.erb +1 -0
  309. data/app/views/decidim/account/show.html.erb +19 -6
  310. data/app/views/decidim/application/{_attachments.html.erb → _attachments.erb} +2 -2
  311. data/app/views/decidim/application/_collection.html.erb +1 -1
  312. data/app/views/decidim/application/{_documents.html.erb → _documents.erb} +2 -2
  313. data/app/views/decidim/devise/confirmations/new.html.erb +1 -1
  314. data/app/views/decidim/devise/invitations/edit.html.erb +3 -4
  315. data/app/views/decidim/devise/omniauth_registrations/new.html.erb +3 -3
  316. data/app/views/decidim/devise/passwords/edit.html.erb +7 -5
  317. data/app/views/decidim/devise/passwords/new.html.erb +1 -1
  318. data/app/views/decidim/devise/registrations/edit.html.erb +41 -0
  319. data/app/views/decidim/devise/registrations/new.html.erb +5 -5
  320. data/app/views/decidim/devise/sessions/new.html.erb +2 -2
  321. data/app/views/decidim/{data_portability → download_your_data}/show.html.erb +2 -2
  322. data/app/views/decidim/export_mailer/download_your_data_export.html.erb +7 -0
  323. data/app/views/decidim/last_activities/index.html.erb +1 -1
  324. data/app/views/decidim/manifests/show.json.erb +31 -0
  325. data/app/views/decidim/messaging/conversations/_add_conversation_users.html.erb +5 -3
  326. data/app/views/decidim/messaging/conversations/_reply.html.erb +1 -1
  327. data/app/views/decidim/messaging/conversations/_start.html.erb +1 -3
  328. data/app/views/decidim/messaging/conversations/create.js.erb +1 -0
  329. data/app/views/decidim/newsletter_mailer/newsletter.html.erb +3 -3
  330. data/app/views/decidim/newsletters/show.html.erb +1 -1
  331. data/app/views/decidim/notification_mailer/event_received.html.erb +1 -1
  332. data/app/views/decidim/notifications_digest_mailer/_email_content.html.erb +11 -0
  333. data/app/views/decidim/notifications_digest_mailer/digest_mail.erb +27 -0
  334. data/app/views/decidim/notifications_settings/show.html.erb +79 -52
  335. data/app/views/decidim/offline/show.html.erb +9 -0
  336. data/app/views/decidim/pages/_standalone.html.erb +1 -1
  337. data/app/views/decidim/reported_mailer/hide.html.erb +1 -1
  338. data/app/views/decidim/reported_mailer/report.html.erb +1 -1
  339. data/app/views/decidim/scopes/picker.html.erb +1 -1
  340. data/app/views/decidim/searches/_filters.html.erb +2 -2
  341. data/app/views/decidim/shared/participatory_space_filters/_filters.html.erb +3 -3
  342. data/app/views/decidim/user_interests/show.html.erb +13 -11
  343. data/app/views/decidim/user_report_mailer/notify.html.erb +10 -3
  344. data/app/views/devise/mailer/confirmation_instructions.html.erb +3 -1
  345. data/app/views/layouts/decidim/_application.html.erb +6 -3
  346. data/app/views/layouts/decidim/_cookie_warning.html.erb +8 -8
  347. data/app/views/layouts/decidim/_decidim_javascript.html.erb +12 -0
  348. data/app/views/layouts/decidim/_head.html.erb +4 -2
  349. data/app/views/layouts/decidim/_js_configuration.html.erb +5 -1
  350. data/app/views/layouts/decidim/_mailer_logo.html.erb +2 -2
  351. data/app/views/layouts/decidim/_main_footer.html.erb +1 -0
  352. data/app/views/layouts/decidim/_offline_banner.html.erb +11 -0
  353. data/app/views/layouts/decidim/newsletter_base.html.erb +2 -2
  354. data/app/views/layouts/decidim/widget.html.erb +2 -2
  355. data/config/assets.rb +1 -0
  356. data/config/locales/am-ET.yml +1 -0
  357. data/config/locales/ar.yml +76 -41
  358. data/config/locales/bg.yml +42 -36
  359. data/config/locales/ca.yml +222 -55
  360. data/config/locales/cs.yml +221 -51
  361. data/config/locales/da.yml +1 -0
  362. data/config/locales/de.yml +53 -100
  363. data/config/locales/el.yml +42 -34
  364. data/config/locales/en.yml +207 -40
  365. data/config/locales/eo.yml +4 -6
  366. data/config/locales/es-MX.yml +212 -45
  367. data/config/locales/es-PY.yml +212 -45
  368. data/config/locales/es.yml +222 -55
  369. data/config/locales/et.yml +1 -0
  370. data/config/locales/eu.yml +63 -62
  371. data/config/locales/fi-plain.yml +208 -41
  372. data/config/locales/fi.yml +209 -42
  373. data/config/locales/fr-CA.yml +199 -43
  374. data/config/locales/fr.yml +206 -50
  375. data/config/locales/ga-IE.yml +10 -8
  376. data/config/locales/gl.yml +75 -32
  377. data/config/locales/hr.yml +1 -0
  378. data/config/locales/hu.yml +90 -209
  379. data/config/locales/id-ID.yml +43 -34
  380. data/config/locales/is-IS.yml +8 -17
  381. data/config/locales/it.yml +52 -42
  382. data/config/locales/ja.yml +213 -58
  383. data/config/locales/ko.yml +1 -0
  384. data/config/locales/lb.yml +42 -32
  385. data/config/locales/lt.yml +1 -1776
  386. data/config/locales/lv.yml +41 -33
  387. data/config/locales/mt.yml +1 -0
  388. data/config/locales/nl.yml +54 -36
  389. data/config/locales/no.yml +53 -38
  390. data/config/locales/om-ET.yml +1 -0
  391. data/config/locales/pl.yml +94 -37
  392. data/config/locales/pt-BR.yml +46 -35
  393. data/config/locales/pt.yml +43 -33
  394. data/config/locales/ro-RO.yml +45 -41
  395. data/config/locales/ru.yml +20 -31
  396. data/config/locales/si-LK.yml +1 -0
  397. data/config/locales/sk.yml +44 -35
  398. data/config/locales/sl.yml +1 -0
  399. data/config/locales/so-SO.yml +1 -0
  400. data/config/locales/sr-CS.yml +1 -0
  401. data/config/locales/sv.yml +95 -36
  402. data/config/locales/sw-KE.yml +1 -0
  403. data/config/locales/ti-ER.yml +1 -0
  404. data/config/locales/tr-TR.yml +44 -35
  405. data/config/locales/uk.yml +18 -32
  406. data/config/locales/val-ES.yml +1 -0
  407. data/config/locales/vi.yml +1 -0
  408. data/config/locales/zh-CN.yml +43 -33
  409. data/config/locales/zh-TW.yml +1 -0
  410. data/config/routes.rb +19 -22
  411. data/db/migrate/20180508111710_add_accepted_tos_version_field_to_users.rb +1 -0
  412. data/db/migrate/20181030090144_destroy_deleted_users_follows.rb +1 -0
  413. data/db/migrate/20190412131728_fix_user_names.rb +4 -4
  414. data/db/migrate/20210831181634_add_service_name_to_active_storage_blobs.active_storage.rb +22 -0
  415. data/db/migrate/20210831181635_create_active_storage_variant_records.active_storage.rb +14 -0
  416. data/db/migrate/20211208155453_create_decidim_reminders.rb +11 -0
  417. data/db/migrate/20211209121025_create_decidim_reminder_records.rb +11 -0
  418. data/db/migrate/20211209121040_create_decidim_reminder_deliveries.rb +10 -0
  419. data/db/migrate/20220118121921_change_required_description_categories.rb +7 -0
  420. data/db/migrate/20220127113419_add_notification_settings_to_users.rb +7 -0
  421. data/db/migrate/20220203121137_add_notifications_sending_frequency_to_users.rb +7 -0
  422. data/db/migrate/20220215172439_add_digest_sent_at_to_users.rb +7 -0
  423. data/db/migrate/20220323195258_add_index_to_decidim_users_notifications_sending_frequency.rb +7 -0
  424. data/db/migrate/20220427142214_drop_emails_on_notifications_flag_from_user.rb +15 -0
  425. data/db/migrate/20220518094535_add_previous_passwords_to_users.rb +20 -0
  426. data/db/migrate/20220524195530_create_decidim_short_links.rb +23 -0
  427. data/db/seeds.rb +17 -18
  428. data/lib/decidim/acts_as_tree.rb +47 -0
  429. data/lib/decidim/api/input_sorts/component_input_sort.rb +2 -1
  430. data/lib/decidim/api/interfaces/authorable_interface.rb +6 -5
  431. data/lib/decidim/api/types/localized_string_type.rb +0 -9
  432. data/lib/decidim/api/types/translated_field_type.rb +5 -20
  433. data/lib/decidim/asset_router.rb +80 -0
  434. data/lib/decidim/attachment_attributes.rb +8 -8
  435. data/lib/decidim/attribute_object/form.rb +174 -0
  436. data/lib/decidim/attribute_object/model.rb +148 -0
  437. data/lib/decidim/attribute_object/nested_validator.rb +22 -0
  438. data/lib/decidim/attribute_object/type_map.rb +14 -0
  439. data/lib/decidim/attribute_object/type_resolver.rb +91 -0
  440. data/lib/decidim/attribute_object.rb +26 -0
  441. data/lib/decidim/attributes/array.rb +62 -0
  442. data/lib/decidim/attributes/blob.rb +20 -0
  443. data/lib/decidim/attributes/clean_string.rb +12 -6
  444. data/lib/decidim/attributes/hash.rb +36 -0
  445. data/lib/decidim/attributes/integer.rb +17 -0
  446. data/lib/decidim/attributes/localized_date.rb +8 -14
  447. data/lib/decidim/attributes/model.rb +24 -0
  448. data/lib/decidim/attributes/object.rb +28 -0
  449. data/lib/decidim/attributes/symbol.rb +23 -0
  450. data/lib/decidim/attributes/time_with_zone.rb +7 -17
  451. data/lib/decidim/attributes.rb +29 -0
  452. data/lib/decidim/authorable.rb +21 -6
  453. data/lib/decidim/authorization_form_builder.rb +9 -11
  454. data/lib/decidim/carrier_wave_migrator_service.rb +15 -15
  455. data/lib/decidim/coauthorable.rb +22 -4
  456. data/lib/decidim/command.rb +44 -0
  457. data/lib/decidim/common_passwords.rb +1 -1
  458. data/lib/decidim/component_manifest.rb +5 -5
  459. data/lib/decidim/component_validator.rb +2 -2
  460. data/lib/decidim/content_block_manifest.rb +1 -1
  461. data/lib/decidim/content_parsers/hashtag_parser.rb +2 -2
  462. data/lib/decidim/content_parsers/resource_parser.rb +1 -1
  463. data/lib/decidim/content_parsers/user_group_parser.rb +1 -1
  464. data/lib/decidim/content_parsers/user_parser.rb +4 -4
  465. data/lib/decidim/content_renderers/hashtag_renderer.rb +1 -1
  466. data/lib/decidim/content_renderers/link_renderer.rb +2 -2
  467. data/lib/decidim/content_renderers/user_group_renderer.rb +1 -1
  468. data/lib/decidim/content_renderers/user_renderer.rb +1 -1
  469. data/lib/decidim/controller_helpers.rb +41 -0
  470. data/lib/decidim/core/engine.rb +35 -19
  471. data/lib/decidim/core/test/factories.rb +54 -21
  472. data/lib/decidim/core/test/shared_examples/admin_resource_gallery_examples.rb +16 -4
  473. data/lib/decidim/core/test/shared_examples/amendable/amendment_form_examples.rb +1 -1
  474. data/lib/decidim/core/test/shared_examples/authorable_interface_examples.rb +15 -2
  475. data/lib/decidim/core/test/shared_examples/coauthorable_interface_examples.rb +2 -2
  476. data/lib/decidim/core/test/shared_examples/comments_examples.rb +203 -0
  477. data/lib/decidim/core/test/shared_examples/localised_email.rb +1 -1
  478. data/lib/decidim/core/test/shared_examples/logo_email.rb +1 -1
  479. data/lib/decidim/core/test/shared_examples/map_examples.rb +2 -2
  480. data/lib/decidim/core/test/shared_examples/preview_component_with_share_token_examples.rb +0 -6
  481. data/lib/decidim/core/test/shared_examples/resource_search_examples.rb +120 -40
  482. data/lib/decidim/core/test/shared_examples/resourceable.rb +0 -6
  483. data/lib/decidim/core/test/shared_examples/searchable_participatory_space_examples.rb +1 -1
  484. data/lib/decidim/core/test/shared_examples/searchable_resources_shared_context.rb +1 -1
  485. data/lib/decidim/core/test/shared_examples/system_endorse_resource_examples.rb +3 -3
  486. data/lib/decidim/core/test/shared_examples/translated_event_examples.rb +2 -2
  487. data/lib/decidim/core/test/shared_examples/with_endorsable_permissions_examples.rb +1 -1
  488. data/lib/decidim/core/test.rb +0 -3
  489. data/lib/decidim/core/version.rb +1 -1
  490. data/lib/decidim/core.rb +108 -14
  491. data/lib/decidim/{data_portability.rb → download_your_data.rb} +3 -3
  492. data/lib/decidim/{data_portability_serializers/data_portability_conversation_serializer.rb → download_your_data_serializers/download_your_data_conversation_serializer.rb} +2 -2
  493. data/lib/decidim/{data_portability_serializers/data_portability_follow_serializer.rb → download_your_data_serializers/download_your_data_follow_serializer.rb} +2 -2
  494. data/lib/decidim/{data_portability_serializers/data_portability_identity_serializer.rb → download_your_data_serializers/download_your_data_identity_serializer.rb} +2 -2
  495. data/lib/decidim/{data_portability_serializers/data_portability_notification_serializer.rb → download_your_data_serializers/download_your_data_notification_serializer.rb} +2 -2
  496. data/lib/decidim/{data_portability_serializers/data_portability_participatory_space_private_user_serializer.rb → download_your_data_serializers/download_your_data_participatory_space_private_user_serializer.rb} +2 -2
  497. data/lib/decidim/{data_portability_serializers/data_portability_report_serializer.rb → download_your_data_serializers/download_your_data_report_serializer.rb} +2 -2
  498. data/lib/decidim/{data_portability_serializers/data_portability_user_group_serializer.rb → download_your_data_serializers/download_your_data_user_group_serializer.rb} +2 -2
  499. data/lib/decidim/{data_portability_serializers/data_portability_user_serializer.rb → download_your_data_serializers/download_your_data_user_serializer.rb} +3 -3
  500. data/lib/decidim/download_your_data_serializers.rb +23 -0
  501. data/lib/decidim/engine_router.rb +7 -1
  502. data/lib/decidim/env.rb +56 -0
  503. data/lib/decidim/event_recorder.rb +25 -0
  504. data/lib/decidim/events/base_event.rb +10 -14
  505. data/lib/decidim/events/simple_event.rb +6 -5
  506. data/lib/decidim/exporters/export_data.rb +1 -1
  507. data/lib/decidim/exporters/export_manifest.rb +2 -1
  508. data/lib/decidim/file_validator_humanizer.rb +24 -1
  509. data/lib/decidim/filterable_resource.rb +55 -0
  510. data/lib/decidim/fingerprint_calculator.rb +2 -4
  511. data/lib/decidim/form_builder.rb +160 -139
  512. data/lib/decidim/gamification/badge.rb +2 -3
  513. data/lib/decidim/gamification/badge_scorer.rb +1 -0
  514. data/lib/decidim/has_area.rb +27 -0
  515. data/lib/decidim/has_attachments.rb +2 -2
  516. data/lib/decidim/has_category.rb +36 -0
  517. data/lib/decidim/has_resource_permission.rb +2 -0
  518. data/lib/decidim/importers/import_manifest.rb +3 -21
  519. data/lib/decidim/jsonb_attributes.rb +6 -4
  520. data/lib/decidim/manifest_messages.rb +23 -0
  521. data/lib/decidim/map/autocomplete.rb +6 -4
  522. data/lib/decidim/map/frontend.rb +2 -2
  523. data/lib/decidim/map/provider/dynamic_map/here.rb +1 -46
  524. data/lib/decidim/map/provider/geocoding/here.rb +5 -5
  525. data/lib/decidim/map/provider/here.rb +3 -0
  526. data/lib/decidim/map/provider/osm.rb +3 -0
  527. data/lib/decidim/map.rb +4 -4
  528. data/lib/decidim/menu.rb +1 -1
  529. data/lib/decidim/metric_manifest.rb +2 -2
  530. data/lib/decidim/nicknamizable.rb +1 -1
  531. data/lib/decidim/notification_setting_manifest.rb +19 -0
  532. data/lib/decidim/organization_settings.rb +8 -7
  533. data/lib/decidim/participatory_space_context_manifest.rb +2 -2
  534. data/lib/decidim/participatory_space_manifest.rb +2 -2
  535. data/lib/decidim/participatory_space_resourceable.rb +1 -7
  536. data/lib/decidim/processes_file_locally.rb +31 -0
  537. data/lib/decidim/query.rb +84 -0
  538. data/lib/decidim/record_encryptor.rb +29 -8
  539. data/lib/decidim/reminder_manifest.rb +75 -0
  540. data/lib/decidim/reminder_registry.rb +67 -0
  541. data/lib/decidim/resource_manifest.rb +1 -1
  542. data/lib/decidim/resourceable.rb +15 -5
  543. data/lib/decidim/scopable.rb +15 -0
  544. data/lib/decidim/settings_manifest.rb +4 -4
  545. data/lib/decidim/spring.rb +16 -0
  546. data/lib/decidim/translatable_attributes.rb +13 -16
  547. data/lib/decidim/url_option_resolver.rb +50 -0
  548. data/lib/decidim/view_hooks.rb +1 -1
  549. data/lib/decidim/view_model.rb +5 -1
  550. data/lib/decidim/webpacker/thread_safe_compiler.rb +30 -0
  551. data/lib/decidim/webpacker/webpack/custom.js +15 -3
  552. data/lib/decidim/webpacker/webpacker.yml +1 -1
  553. data/lib/decidim/webpacker.rb +2 -0
  554. data/lib/devise/models/decidim_validatable.rb +1 -1
  555. data/lib/tasks/decidim_active_storage_migration_tasks.rake +1 -3
  556. data/lib/tasks/{decidim_data_portability_tasks.rake → decidim_download_your_data_tasks.rake} +9 -9
  557. data/lib/tasks/decidim_mailers_tasks.rake +23 -0
  558. data/lib/tasks/decidim_metrics_tasks.rake +1 -1
  559. data/lib/tasks/decidim_pwa_tasks.rake +18 -0
  560. data/lib/tasks/decidim_reminders_tasks.rake +17 -0
  561. data/lib/tasks/decidim_tasks.rake +6 -1
  562. data/lib/tasks/decidim_webpacker_tasks.rake +1 -1
  563. data/lib/tasks/upgrade/decidim_fix_nickname_uniqueness.rake +51 -0
  564. metadata +186 -92
  565. data/app/controllers/decidim/cookie_policy_controller.rb +0 -24
  566. data/app/helpers/decidim/cookies_helper.rb +0 -11
  567. data/app/packs/src/decidim/editor/clipboard_override.js +0 -143
  568. data/app/packs/src/decidim/editor/clipboard_utilities.js +0 -119
  569. data/app/packs/src/decidim/vizzs/renders.js +0 -9
  570. data/app/packs/src/decidim/vizzs.js +0 -1
  571. data/app/packs/stylesheets/decidim/modules/_cookie-bar.scss +0 -26
  572. data/app/services/decidim/home_activity_search.rb +0 -80
  573. data/app/services/decidim/participatory_space_search.rb +0 -54
  574. data/app/views/decidim/cookie_policy/accept.js.erb +0 -3
  575. data/app/views/decidim/export_mailer/data_portability_export.html.erb +0 -7
  576. data/config/locales/gn-PY.yml +0 -1
  577. data/config/locales/ka-GE.yml +0 -1
  578. data/config/locales/lo-LA.yml +0 -1
  579. data/config/locales/oc-FR.yml +0 -1
  580. data/lib/decidim/core/test/shared_examples/mcell_examples.rb +0 -17
  581. data/lib/decidim/core/test/shared_examples/resource_endorsed_event_examples.rb +0 -60
  582. data/lib/decidim/core/test/shared_examples/versions_controller_examples.rb +0 -40
  583. data/lib/decidim/data_portability_serializers.rb +0 -23
  584. data/lib/decidim/middleware/rails_cookies.rb +0 -23
  585. /data/app/packs/src/decidim/vizzs/{metrics.js → index.js} +0 -0
  586. /data/app/views/decidim/application/{_document.html.erb → _document.erb} +0 -0
  587. /data/app/views/decidim/application/{_photos.html.erb → _photos.erb} +0 -0
  588. /data/app/views/decidim/{data_portability → download_your_data}/export.html.erb +0 -0
@@ -1,4 +1,4 @@
1
- /* global spyOn, jest */
1
+ /* global spyOn */
2
2
  /* eslint-disable id-length */
3
3
  window.$ = $;
4
4
 
@@ -7,20 +7,6 @@ import DataPicker from "./data_picker"
7
7
 
8
8
  const FormFilterComponent = require("./form_filter.component_for_testing.js");
9
9
 
10
- const expectedPushState = (state, filters) => {
11
- const queryString = Object.keys(filters).map((key) => {
12
- const name = `filter[${key}]`;
13
- const val = filters[key];
14
- if (Array.isArray(val)) {
15
- return val.map((v) => `${encodeURIComponent(`${name}[]`)}=${encodeURIComponent(v)}`).join("&");
16
- }
17
-
18
- return `${encodeURIComponent(name)}=${encodeURIComponent(val)}`;
19
- }).join("&");
20
-
21
- return [state, null, `/filters?${queryString}`];
22
- }
23
-
24
10
  describe("FormFilterComponent", () => {
25
11
  const selector = "form#new_filter";
26
12
  let subject = null;
@@ -29,10 +15,6 @@ describe("FormFilterComponent", () => {
29
15
  beforeEach(() => {
30
16
  let form = `
31
17
  <form id="new_filter" action="/filters" method="get">
32
- <fieldset>
33
- <input id="filter_search_text_cont" placeholder="Search" data-disable-dynamic-change="true" type="search" name="filter[search_text_cont]">
34
- </fieldset>
35
-
36
18
  <fieldset>
37
19
  <div id="filter_somerandomid_scope_id" class="data-picker picker-multiple" data-picker-name="filter[scope_id]">
38
20
  <div class="picker-values">
@@ -85,25 +67,11 @@ describe("FormFilterComponent", () => {
85
67
  `;
86
68
  $("body").append(form);
87
69
 
88
- const $form = $(document).find("form");
89
-
90
70
  window.Decidim = window.Decidim || {};
91
71
 
92
72
  window.theDataPicker = new DataPicker($(".data-picker"));
93
73
  window.theCheckBoxesTree = new CheckBoxesTree();
94
- window.Rails = {
95
- fire: (htmlElement, event) => {
96
- // Hack to call trigger on the correct instance of the form, as fetching
97
- // with the selector does not work.
98
- if (htmlElement === $form[0]) {
99
- $form.trigger(event);
100
- }
101
- }
102
- };
103
-
104
- subject = new FormFilterComponent($form);
105
-
106
- jest.useFakeTimers();
74
+ subject = new FormFilterComponent($(document).find("form"));
107
75
  });
108
76
 
109
77
  it("exists", () => {
@@ -120,19 +88,7 @@ describe("FormFilterComponent", () => {
120
88
 
121
89
  describe("when mounted", () => {
122
90
  beforeEach(() => {
123
- // Jest doesn't implement listening on the form submit event so we need
124
- // to hack it.
125
- const originalOn = subject.$form.on.bind(subject.$form);
126
- jest.spyOn(subject.$form, "on").mockImplementation((...args) => {
127
- if (args[0] === "submit") {
128
- subject.$form.submitHandler = args[1];
129
- } else if (args[0] === "change" && typeof args[1] === "string") {
130
- subject.$form.changeHandler = args[2];
131
- } else {
132
- originalOn(...args);
133
- }
134
- });
135
-
91
+ spyOn(subject.$form, "on");
136
92
  subject.mountComponent();
137
93
  });
138
94
 
@@ -144,98 +100,8 @@ describe("FormFilterComponent", () => {
144
100
  expect(subject.mounted).toBeTruthy();
145
101
  });
146
102
 
147
- it("binds the form change and submit events", () => {
103
+ it("binds the form change event", () => {
148
104
  expect(subject.$form.on).toHaveBeenCalledWith("change", "input:not([data-disable-dynamic-change]), select:not([data-disable-dynamic-change])", subject._onFormChange);
149
- expect(subject.$form.on).toHaveBeenCalledWith("submit", subject._onFormSubmit);
150
- });
151
-
152
- describe("form changes", () => {
153
- beforeEach(() => {
154
- spyOn(window.history, "pushState");
155
-
156
- // This is a hack to be able to trigger the events even somewhat close
157
- // to an actual situation. In real browser environment the change events
158
- // would be triggered by the input/select elements but to simplify the
159
- // test implementation, we trigger them directly on the form.
160
- const originalTrigger = subject.$form.trigger.bind(subject.$form);
161
- jest.spyOn(subject.$form, "trigger").mockImplementation((...args) => {
162
- if (args[0] === "submit") {
163
- subject.$form.submitHandler(
164
- $.event.fix(new CustomEvent("submit", { bubbles: true, cancelable: true }))
165
- );
166
- } else if (args[0] === "change") {
167
- subject.$form.changeHandler();
168
- } else {
169
- originalTrigger(...args);
170
- }
171
-
172
- jest.runAllTimers();
173
- });
174
- });
175
-
176
- it("does not save the state in case there were no changes to previous state", () => {
177
- subject.$form.trigger("change");
178
-
179
- expect(window.history.pushState).not.toHaveBeenCalled();
180
- });
181
-
182
- it("saves the state after dynamic form changes", () => {
183
- $("#filter_somerandomid_category_id").val(2);
184
-
185
- subject.$form.trigger("change");
186
-
187
- const state = {
188
- "filter_somerandomid_scope_id": [
189
- {
190
- "text": "Scope 1",
191
- "url": "picker_url_1",
192
- "value": "3"
193
- },
194
- {
195
- "text": "Scope 2",
196
- "url": "picker_url_2",
197
- "value": "4"
198
- }
199
- ]
200
- };
201
- const filters = {
202
- "search_text_cont": "",
203
- "scope_id": [3, 4],
204
- "category_id": 2,
205
- "state": [""]
206
- };
207
- expect(window.history.pushState).toHaveBeenCalledWith(...expectedPushState(state, filters));
208
- });
209
-
210
- it("saves the state after form submission through input element", () => {
211
- const textInput = document.getElementById("filter_search_text_cont");
212
- textInput.value = "search";
213
-
214
- subject.$form.trigger("submit");
215
-
216
- const state = {
217
- "filter_somerandomid_scope_id": [
218
- {
219
- "text": "Scope 1",
220
- "url": "picker_url_1",
221
- "value": "3"
222
- },
223
- {
224
- "text": "Scope 2",
225
- "url": "picker_url_2",
226
- "value": "4"
227
- }
228
- ]
229
- }
230
- const filters = {
231
- "search_text_cont": "search",
232
- "scope_id": [3, 4],
233
- "category_id": 1,
234
- "state": [""]
235
- }
236
-
237
- expect(window.history.pushState).toHaveBeenCalledWith(...expectedPushState(state, filters));
238
- });
239
105
  });
240
106
 
241
107
  describe("onpopstate event", () => {
@@ -265,14 +131,6 @@ describe("FormFilterComponent", () => {
265
131
  expect(checked.map((input) => input.value)).toEqual(["", "accepted", "evaluating"]);
266
132
  expect(checked.filter((input) => input.indeterminate).map((input) => input.value)).toEqual([""]);
267
133
  });
268
-
269
- it("does not save the state", () => {
270
- spyOn(window.history, "pushState");
271
-
272
- window.onpopstate({ isTrusted: true, state: scopesPickerState});
273
-
274
- expect(window.history.pushState).not.toHaveBeenCalled();
275
- });
276
134
  });
277
135
  });
278
136
 
@@ -287,9 +145,8 @@ describe("FormFilterComponent", () => {
287
145
  expect(subject.mounted).toBeFalsy();
288
146
  });
289
147
 
290
- it("unbinds the form change and submit events", () => {
148
+ it("unbinds the form change event", () => {
291
149
  expect(subject.$form.off).toHaveBeenCalledWith("change", "input, select", subject._onFormChange);
292
- expect(subject.$form.off).toHaveBeenCalledWith("submit", subject._onFormSubmit);
293
150
  });
294
151
  });
295
152
 
@@ -23,7 +23,6 @@ export default class FormFilterComponent {
23
23
 
24
24
  this._updateInitialState();
25
25
  this._onFormChange = delayed(this, this._onFormChange.bind(this));
26
- this._onFormSubmit = delayed(this, this._onFormSubmit.bind(this));
27
26
  this._onPopState = this._onPopState.bind(this);
28
27
 
29
28
  if (window.Decidim.PopStateHandler) {
@@ -43,7 +42,6 @@ export default class FormFilterComponent {
43
42
  if (this.mounted) {
44
43
  this.mounted = false;
45
44
  this.$form.off("change", "input, select", this._onFormChange);
46
- this.$form.off("submit", this._onFormSubmit);
47
45
 
48
46
  unregisterCallback(`filters-${this.id}`)
49
47
  }
@@ -64,7 +62,6 @@ export default class FormFilterComponent {
64
62
  contentContainer = this.$form.data("remoteFill");
65
63
  }
66
64
  this.$form.on("change", "input:not([data-disable-dynamic-change]), select:not([data-disable-dynamic-change])", this._onFormChange);
67
- this.$form.on("submit", this._onFormSubmit);
68
65
 
69
66
  this.currentFormRequest = null;
70
67
  this.$form.on("ajax:beforeSend", (e) => {
@@ -257,16 +254,14 @@ export default class FormFilterComponent {
257
254
 
258
255
  // Only one instance should submit the form on browser history navigation
259
256
  if (this.popStateSubmiter) {
260
- Rails.fire(this.$form[0], "submit", { from: "pop" });
257
+ Rails.fire(this.$form[0], "submit");
261
258
  }
262
259
 
263
260
  this.changeEvents = true;
264
261
  }
265
262
 
266
263
  /**
267
- * Handles the logic to decide whether the form should be submitted or not
268
- * after a form change event. The form is only submitted when changes have
269
- * occurred.
264
+ * Handles the logic to update the current location after a form change event.
270
265
  * @private
271
266
  * @returns {Void} - Returns nothing.
272
267
  */
@@ -275,7 +270,7 @@ export default class FormFilterComponent {
275
270
  return;
276
271
  }
277
272
 
278
- const [newPath] = this._currentStateAndPath();
273
+ const [newPath, newState] = this._currentStateAndPath();
279
274
  const path = this._getLocation(false);
280
275
 
281
276
  if (newPath === path) {
@@ -283,23 +278,6 @@ export default class FormFilterComponent {
283
278
  }
284
279
 
285
280
  Rails.fire(this.$form[0], "submit");
286
- }
287
-
288
- /**
289
- * Saves the current state of the search on form submit to update the search
290
- * parameters to the URL and store the picker states.
291
- * @private
292
- * @param {jQuery.Event} ev The event that caused the form to submit.
293
- * @returns {Void} - Returns nothing.
294
- */
295
- _onFormSubmit(ev) {
296
- const eventDetail = ev.originalEvent.detail;
297
- if (eventDetail && eventDetail.from === "pop") {
298
- return;
299
- }
300
-
301
- const [newPath, newState] = this._currentStateAndPath();
302
-
303
281
  pushState(newPath, newState);
304
282
  this._saveFilters(newPath);
305
283
  }
@@ -336,7 +314,7 @@ export default class FormFilterComponent {
336
314
  * @returns {String} - Returns a unique identifier
337
315
  */
338
316
  _getUID() {
339
- return `filter-form-${new Date().getUTCMilliseconds()}-${Math.floor(Math.random() * 10000000)}`;
317
+ return `filter-form-${new Date().setUTCMilliseconds()}-${Math.floor(Math.random() * 10000000)}`;
340
318
  }
341
319
 
342
320
  /**
@@ -1,43 +1,20 @@
1
- import Tribute from "src/decidim/vendor/tribute"
1
+ import AutoComplete from "src/decidim/autocomplete";
2
2
 
3
3
  $(() => {
4
4
  $("[data-decidim-geocoding]").each((_i, el) => {
5
5
  const $input = $(el);
6
- const $fieldContainer = $input.parent();
7
6
 
8
- $fieldContainer.addClass("has-tribute");
9
-
10
- const tribute = new Tribute(
11
- {
12
- autocompleteMode: true,
13
- // autocompleteSeparator: / \+ /, // See below, requires Tribute update
14
- allowSpaces: true,
15
- positionMenu: false,
16
- replaceTextSuffix: "",
17
- menuContainer: $fieldContainer.get(0),
18
- noMatchTemplate: null,
19
- values: (text, cb) => {
20
- $input.trigger("geocoder-suggest.decidim", [text, cb]);
21
- }
7
+ const autoComplete = new AutoComplete(el, {
8
+ mode: "single",
9
+ dataMatchKeys: ["value"],
10
+ dataSource: (query, callback) => {
11
+ $input.trigger("geocoder-suggest.decidim", [query, callback]);
22
12
  }
23
- );
24
-
25
- // Port https://github.com/zurb/tribute/pull/406
26
- // This changes the autocomplete separator from space to " + " so that
27
- // we can do searches such as "streetname 4" including a space. Otherwise
28
- // this would do two separate searches for "streetname" and "4".
29
- tribute.range.getLastWordInText = (text) => {
30
- const final = text.replace(/\u00A0/g, " ");
31
- const wordsArray = final.split(/ \+ /);
32
- const worldsCount = wordsArray.length - 1;
33
-
34
- return wordsArray[worldsCount].trim();
35
- };
36
-
37
- tribute.attach($input.get(0));
13
+ });
14
+ el.addEventListener("selection", autoComplete);
38
15
 
39
- $input.on("tribute-replaced", function(ev) {
40
- const selectedItem = ev.detail.item.original;
16
+ $input.on("selection", (event) => {
17
+ const selectedItem = event.detail.selection.value;
41
18
  $input.trigger("geocoder-suggest-select.decidim", [selectedItem]);
42
19
 
43
20
  // Not all geocoding autocomplete APIs include the coordinates in the
@@ -48,7 +25,5 @@ $(() => {
48
25
  $input.trigger("geocoder-suggest-coordinates.decidim", [selectedItem.coordinates]);
49
26
  }
50
27
  });
51
-
52
- $input.data("geocoder-tribute", tribute);
53
28
  });
54
29
  });
@@ -1,6 +1,6 @@
1
1
  /**
2
2
  * Makes the #select-identity-button to open a popup for the user to
3
- * select with which identity he wants to perform an action.
3
+ * select with which identity they want to perform an action.
4
4
  *
5
5
  */
6
6
  $(() => {
@@ -1,11 +1,11 @@
1
- import moment from "moment"
1
+ import dayjs from "dayjs"
2
2
 
3
3
  $(() => {
4
4
  const $impersonationWarning = $(".impersonation-warning");
5
5
  if ($impersonationWarning.length) {
6
- const endsAt = moment($impersonationWarning.data("session-ends-at"));
6
+ const endsAt = dayjs($impersonationWarning.data("session-ends-at"));
7
7
  const exitInterval = setInterval(() => {
8
- const diff = (endsAt - moment()) / 60000;
8
+ const diff = (endsAt - dayjs()) / 60000;
9
9
  const diffInMinutes = Math.round(diff);
10
10
  $impersonationWarning.find(".minutes").html(diffInMinutes);
11
11
 
@@ -1,4 +1,14 @@
1
1
  const COUNT_KEY = "%count%";
2
+ // How often SR announces the message in relation to maximum characters. E.g.
3
+ // if max characters is 1000, screen reader announces the remaining characters
4
+ // every 100 (= 0.1 * 1000) characters. This will be "floored" to the closest
5
+ // 100 if the maximum characters > 100. E.g. if max characters is 5500, the
6
+ // threshold is 500 (= Math.floor(550 / 100) * 100). With 100 or less
7
+ // characters, this ratio is omitted and the announce threshold is always set to
8
+ // 10.
9
+ const SR_ANNOUNCE_THRESHOLD_RATIO = 0.1;
10
+ // The number of characters left after which every keystroke will be announced.
11
+ const SR_ANNOUNCE_EVERY_THRESHOLD = 10;
2
12
  const DEFAULT_MESSAGES = {
3
13
  charactersAtLeast: {
4
14
  one: `at least ${COUNT_KEY} character`,
@@ -21,15 +31,35 @@ export default class InputCharacterCounter {
21
31
  this.$target = $(this.$input.data("remaining-characters"));
22
32
  this.minCharacters = parseInt(this.$input.attr("minlength"), 10);
23
33
  this.maxCharacters = parseInt(this.$input.attr("maxlength"), 10);
34
+ this.describeByCounter = typeof this.$input.attr("aria-describedby") === "undefined";
24
35
 
25
- if (this.$target.length < 1) {
26
- let targetId = null;
36
+ // Define the closest length for the input "gaps" defined by the threshold.
37
+ if (this.maxCharacters > 10) {
38
+ if (this.maxCharacters > 100) {
39
+ this.announceThreshold = Math.floor(this.maxCharacters * SR_ANNOUNCE_THRESHOLD_RATIO / 100) * 100;
40
+ } else {
41
+ this.announceThreshold = 10;
42
+ }
43
+
44
+ // The number of characters left after which every keystroke will be announced.
45
+ this.announceEveryThreshold = SR_ANNOUNCE_EVERY_THRESHOLD;
46
+ } else {
47
+ this.announceThreshold = 1;
48
+ this.announceEveryThreshold = 1;
49
+ }
50
+
51
+ let targetId = this.$target.attr("id");
52
+ if (typeof targetId === "undefined") {
27
53
  if (this.$input.attr("id") && this.$input.attr("id").length > 0) {
28
54
  targetId = `${this.$input.attr("id")}_characters`;
29
55
  } else {
30
56
  targetId = `characters_${Math.random().toString(36).substr(2, 9)}`;
31
57
  }
58
+ }
32
59
 
60
+ if (this.$target.length > 0) {
61
+ this.$target.attr("id", targetId)
62
+ } else {
33
63
  this.$target = $(`<span id="${targetId}" class="form-input-extra-before" />`)
34
64
 
35
65
  // If input is a hidden for WYSIWYG editor add it at the end
@@ -48,11 +78,35 @@ export default class InputCharacterCounter {
48
78
  }
49
79
  }
50
80
 
81
+ this.updateInputLength();
82
+ this.previousInputLength = this.inputLength;
83
+
51
84
  if (this.$target.length > 0 && (this.maxCharacters > 0 || this.minCharacters > 0)) {
85
+ // Create the screen reader target element. We don't want to constantly
86
+ // announce every change to screen reader, only occasionally.
87
+ this.$srTarget = $(
88
+ `<span role="status" id="${targetId}_sr" class="show-for-sr remaining-character-count-sr" />`
89
+ );
90
+ this.$target.before(this.$srTarget);
91
+ this.$target.attr("aria-hidden", "true");
92
+ this.setDescribedBy(true);
93
+
52
94
  this.bindEvents();
53
95
  }
54
96
  }
55
97
 
98
+ setDescribedBy(active) {
99
+ if (!this.describeByCounter) {
100
+ return;
101
+ }
102
+
103
+ if (active) {
104
+ this.$input.attr("aria-describedby", this.$srTarget.attr("id"));
105
+ } else {
106
+ this.$input.removeAttr("aria-describedby");
107
+ }
108
+ }
109
+
56
110
  bindEvents() {
57
111
  // In WYSIWYG editors (Quill) we need to find the active editor from the
58
112
  // DOM node. Quill has the experimental "find" method that should work
@@ -69,17 +123,109 @@ export default class InputCharacterCounter {
69
123
  this.$input.on("keyup", () => {
70
124
  this.updateStatus();
71
125
  });
126
+ this.$input.on("input", () => {
127
+ this.updateInputLength();
128
+ this.checkScreenReaderUpdate();
129
+ // If the input is "described by" the character counter, some screen
130
+ // readers (NVDA) announce the status twice when it is updated. By
131
+ // removing the aria-describedby attribute while the user is typing makes
132
+ // the screen reader announce the status only once.
133
+ this.setDescribedBy(false);
134
+ });
135
+ this.$input.on("focus", () => {
136
+ this.updateScreenReaderStatus();
137
+ });
138
+ this.$input.on("blur", () => {
139
+ this.updateScreenReaderStatus();
140
+ this.setDescribedBy(true);
141
+ });
72
142
  if (this.$input.get(0) !== null) {
73
143
  this.$input.get(0).addEventListener("emoji.added", () => {
74
144
  this.updateStatus();
75
145
  });
76
146
  }
77
147
  this.updateStatus();
148
+ this.updateScreenReaderStatus();
78
149
  }
79
150
 
80
- updateStatus() {
81
- const numCharacters = this.$input.val().length;
151
+ getInputLength() {
152
+ return this.inputLength;
153
+ }
154
+
155
+ updateInputLength() {
156
+ this.previousInputLength = this.inputLength;
157
+ this.inputLength = this.$input.val().length;
158
+ }
159
+
160
+ /**
161
+ * This compares the current inputLength to the previous value and decides
162
+ * whether the user is currently adding or deleting characters from the view.
163
+ *
164
+ * @returns {String} The input direction either "ins" for insert or "del" for
165
+ * delete.
166
+ */
167
+ getInputDirection() {
168
+ if (this.inputLength < this.previousInputLength) {
169
+ return "del";
170
+ }
171
+
172
+ return "ins";
173
+ }
174
+
175
+ getScreenReaderLength() {
176
+ const currentLength = this.getInputLength();
177
+ if (this.maxCharacters < 10) {
178
+ return currentLength;
179
+ } else if (this.maxCharacters - currentLength <= this.announceEveryThreshold) {
180
+ return currentLength;
181
+ }
182
+
183
+ const srLength = currentLength - currentLength % this.announceThreshold;
184
+
185
+ // Prevent the screen reader telling too many characters left if the user
186
+ // deletes a characters. This can cause confusing experience e.g. when the
187
+ // user is closing the maximum amount of characters, so if the previous
188
+ // announcement was "10 characters left" and the user removes one character,
189
+ // the screen reader would announce "100 characters left" next time (when
190
+ // they actually have only 11 characters left). Similar when they are
191
+ // deleting a character at 900 characters, the screen reader would announce
192
+ // "1000 characters left" even when they only have 901 characters left.
193
+ if (this.getInputDirection() === "del") {
194
+ // The first branch makes sure that if the SR length matches the actual
195
+ // length, it will be always announced.
196
+ if (srLength === currentLength) {
197
+ return srLength;
198
+ // The second branch checks that if we are at the final threshold, we
199
+ // should not announce "0 characters left" when the user deletes more than
200
+ // the "announce after every stroke" limit (this.announceEveryThreshold).
201
+ } else if (this.maxCharacters - srLength === this.announceThreshold) {
202
+ return this.announcedAt || currentLength;
203
+ // The third branch checks that when deleting characters, we should
204
+ // announce the next threshold to get accurate annoucement. E.g. when we
205
+ // have 750 characters left and the user deletes 100 characters at once,
206
+ // we should announce "700 characters left" after that deletion.
207
+ } else if (srLength < currentLength) {
208
+ return srLength + this.announceThreshold;
209
+ }
210
+ // This fixes an issue in the following situation:
211
+ // 1. 750 characters left
212
+ // 2. Delete 100 characters in a row
213
+ // 3. SR: "800 characters left" (actual 850)
214
+ // 4. Type one additional character
215
+ // 5. Without this, SR would announce "900 characters left" = confusing
216
+ } else if (srLength < this.announcedAt) {
217
+ return this.announcedAt;
218
+ }
219
+
220
+ return srLength;
221
+ }
222
+
223
+ getMessages(currentLength = null) {
82
224
  const showMessages = [];
225
+ let inputLength = currentLength;
226
+ if (inputLength === null) {
227
+ inputLength = this.getInputLength()
228
+ }
83
229
 
84
230
  if (this.minCharacters > 0) {
85
231
  let message = MESSAGES.charactersAtLeast.other;
@@ -90,7 +236,7 @@ export default class InputCharacterCounter {
90
236
  }
91
237
 
92
238
  if (this.maxCharacters > 0) {
93
- const remaining = this.maxCharacters - numCharacters;
239
+ const remaining = this.maxCharacters - inputLength;
94
240
  let message = MESSAGES.charactersLeft.other;
95
241
  if (remaining === 1) {
96
242
  message = MESSAGES.charactersLeft.one;
@@ -101,12 +247,36 @@ export default class InputCharacterCounter {
101
247
  showMessages.push(message.replace(COUNT_KEY, remaining));
102
248
  }
103
249
 
104
- this.$target.text(showMessages.join(", "));
250
+ return showMessages;
251
+ }
252
+
253
+ updateStatus() {
254
+ this.$target.text(this.getMessages().join(", "));
255
+ }
256
+
257
+ checkScreenReaderUpdate() {
258
+ if (this.maxCharacters < 1) {
259
+ return;
260
+ }
261
+
262
+ const currentLength = this.getScreenReaderLength();
263
+ if (currentLength === this.announcedAt) {
264
+ return;
265
+ }
266
+
267
+ this.announcedAt = currentLength;
268
+ this.updateScreenReaderStatus(currentLength);
269
+ }
270
+
271
+ updateScreenReaderStatus(currentLength = null) {
272
+ this.$srTarget.text(this.getMessages(currentLength).join(", "));
105
273
  }
106
274
  }
107
275
 
108
276
  const createCharacterCounter = ($input) => {
109
- $input.data("remaining-characters-counter", new InputCharacterCounter($input));
277
+ if (typeof $input !== "undefined" && $input.length) {
278
+ $input.data("remaining-characters-counter", new InputCharacterCounter($input));
279
+ }
110
280
  }
111
281
 
112
282
  $(() => {