decidim-core 0.30.2 → 0.31.0.rc1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/README.md +0 -4
- data/app/cells/decidim/activity_cell.rb +0 -3
- data/app/cells/decidim/address/show.erb +1 -0
- data/app/cells/decidim/address_cell.rb +7 -1
- data/app/cells/decidim/amendable/amend_button_card/show.erb +2 -2
- data/app/cells/decidim/author/likes.erb +4 -0
- data/app/cells/decidim/author/show.erb +1 -3
- data/app/cells/decidim/author_cell.rb +5 -28
- data/app/cells/decidim/badge_cell.rb +1 -7
- data/app/cells/decidim/card_cell.rb +2 -12
- data/app/cells/decidim/card_metadata_cell.rb +4 -4
- data/app/cells/decidim/coauthorships_cell.rb +1 -1
- data/app/cells/decidim/collapsible_authors/show.erb +1 -1
- data/app/cells/decidim/content_blocks/hero_cell.rb +23 -4
- data/app/cells/decidim/content_blocks/hero_settings_form/show.erb +16 -2
- data/app/cells/decidim/content_blocks/highlighted_content_banner/show.erb +6 -6
- data/app/cells/decidim/content_blocks/highlighted_content_banner_cell.rb +28 -1
- data/app/cells/decidim/content_blocks/highlighted_content_banner_settings_form/show.erb +27 -0
- data/app/cells/decidim/content_blocks/highlighted_content_banner_settings_form_cell.rb +13 -0
- data/app/cells/decidim/content_blocks/highlighted_elements_for_component_settings_form/show.erb +9 -2
- data/app/cells/decidim/content_blocks/html_settings_form/show.erb +3 -1
- data/app/cells/decidim/content_blocks/last_activity_cell.rb +1 -1
- data/app/cells/decidim/content_blocks/last_activity_settings_form/show.erb +3 -1
- data/app/cells/decidim/content_blocks/menu_breadcrumb_last_activity_cell.rb +1 -1
- data/app/cells/decidim/content_blocks/participatory_space_extra_data/recent_users.erb +1 -1
- data/app/cells/decidim/content_blocks/participatory_space_hero/show.erb +0 -5
- data/app/cells/decidim/content_blocks/participatory_space_hero_cell.rb +1 -13
- data/app/cells/decidim/content_blocks/participatory_space_hero_settings_form/show.erb +10 -3
- data/app/cells/decidim/content_blocks/participatory_space_last_activity/content.erb +1 -1
- data/app/cells/decidim/content_blocks/participatory_space_last_activity/recent_avatars.erb +1 -1
- data/app/cells/decidim/content_blocks/participatory_space_last_activity_cell.rb +3 -3
- data/app/cells/decidim/content_blocks/participatory_space_main_data/title.erb +17 -5
- data/app/cells/decidim/content_blocks/participatory_space_main_data_cell.rb +4 -0
- data/app/cells/decidim/content_blocks/stats/show.erb +2 -3
- data/app/cells/decidim/data_consent/modal.erb +1 -1
- data/app/cells/decidim/like_block/show.erb +3 -0
- data/app/cells/decidim/{endorsement_block_cell.rb → like_block_cell.rb} +2 -2
- data/app/cells/decidim/like_buttons/button_content.erb +2 -0
- data/app/cells/decidim/like_buttons/show.erb +17 -0
- data/app/cells/decidim/{endorsement_buttons → like_buttons}/verification_modal.erb +3 -3
- data/app/cells/decidim/like_buttons_cell.rb +77 -0
- data/app/cells/decidim/likers_list/empty.erb +3 -0
- data/app/cells/decidim/{endorsers_list → likers_list}/full.erb +3 -3
- data/app/cells/decidim/likers_list/show.erb +20 -0
- data/app/cells/decidim/likers_list_cell.rb +64 -0
- data/app/cells/decidim/nav_links/show.erb +1 -1
- data/app/cells/decidim/notification_actions/buttons_cell.rb +1 -1
- data/app/cells/decidim/onboarding_action_message_cell.rb +1 -1
- data/app/cells/decidim/participatory_space_dropdown_metadata/metadata.erb +0 -8
- data/app/cells/decidim/participatory_space_dropdown_metadata_cell.rb +0 -8
- data/app/cells/decidim/profile/details.erb +1 -1
- data/app/cells/decidim/profile/tabs.erb +1 -1
- data/app/cells/decidim/profile_actions/show.erb +0 -1
- data/app/cells/decidim/profile_actions_cell.rb +5 -91
- data/app/cells/decidim/profile_cell.rb +4 -39
- data/app/cells/decidim/report_button/content.erb +2 -0
- data/app/cells/decidim/report_button/flag_modal.erb +1 -1
- data/app/cells/decidim/report_button_cell.rb +1 -1
- data/app/cells/decidim/report_user_button/flag_modal.erb +1 -1
- data/app/cells/decidim/resource_history_cell.rb +2 -1
- data/app/cells/decidim/resource_types_filter/show.erb +1 -1
- data/app/cells/decidim/share_button_cell.rb +1 -1
- data/app/cells/decidim/share_text_widget/modal.erb +16 -0
- data/app/cells/decidim/share_text_widget/show.erb +2 -0
- data/app/cells/decidim/share_text_widget_cell.rb +27 -0
- data/app/cells/decidim/share_widget/modal.erb +19 -8
- data/app/cells/decidim/share_widget/qr_code_modal.erb +26 -0
- data/app/cells/decidim/share_widget/show.erb +1 -0
- data/app/cells/decidim/share_widget_cell.rb +10 -6
- data/app/cells/decidim/statistic/show.erb +23 -5
- data/app/cells/decidim/statistic_cell.rb +28 -3
- data/app/cells/decidim/tab_panels/show.erb +1 -1
- data/app/cells/decidim/user_profile/show.erb +1 -1
- data/app/cells/decidim/user_profile_cell.rb +1 -9
- data/app/cells/decidim/version_author_cell.rb +0 -7
- data/app/commands/decidim/amendable/accept.rb +1 -3
- data/app/commands/decidim/amendable/create_draft.rb +2 -3
- data/app/commands/decidim/amendable/update_draft.rb +2 -3
- data/app/commands/decidim/create_ephemeral_user.rb +7 -1
- data/app/commands/decidim/create_omniauth_registration.rb +8 -7
- data/app/commands/decidim/create_report.rb +9 -1
- data/app/commands/decidim/destroy_account.rb +0 -5
- data/app/commands/decidim/like_resource.rb +52 -0
- data/app/commands/decidim/messaging/reply_to_conversation.rb +2 -21
- data/app/commands/decidim/messaging/start_conversation.rb +2 -21
- data/app/commands/decidim/unlike_resource.rb +34 -0
- data/app/commands/decidim/update_account.rb +1 -1
- data/app/controllers/concerns/decidim/ephemeral_session_checker.rb +6 -2
- data/app/controllers/concerns/decidim/user_profile.rb +0 -1
- data/app/controllers/decidim/account_controller.rb +1 -1
- data/app/controllers/decidim/amendments_controller.rb +4 -5
- data/app/controllers/decidim/application_controller.rb +2 -1
- data/app/controllers/decidim/devise/confirmations_controller.rb +0 -5
- data/app/controllers/decidim/devise/omniauth_registrations_controller.rb +1 -1
- data/app/controllers/decidim/devise/passwords_controller.rb +1 -1
- data/app/controllers/decidim/devise/registrations_controller.rb +1 -1
- data/app/controllers/decidim/doorkeeper/authorizations_controller.rb +19 -1
- data/app/controllers/decidim/download_your_data_controller.rb +2 -2
- data/app/controllers/decidim/likes_controller.rb +57 -0
- data/app/controllers/decidim/links_controller.rb +1 -0
- data/app/controllers/decidim/messaging/conversations_controller.rb +1 -8
- data/app/controllers/decidim/newsletters_controller.rb +1 -1
- data/app/controllers/decidim/notifications_settings_controller.rb +2 -2
- data/app/controllers/decidim/pages_controller.rb +0 -1
- data/app/controllers/decidim/profiles_controller.rb +2 -57
- data/app/controllers/decidim/qr_controller.rb +73 -0
- data/app/controllers/decidim/resource_autocomplete_controller.rb +56 -0
- data/app/controllers/decidim/user_activities_controller.rb +0 -1
- data/app/events/decidim/resource_liked_event.rb +36 -0
- data/app/forms/decidim/amendable/create_form.rb +0 -1
- data/app/forms/decidim/amendable/edit_form.rb +0 -1
- data/app/forms/decidim/amendable/form.rb +4 -4
- data/app/forms/decidim/ephemeral_user_form.rb +1 -0
- data/app/forms/decidim/notifications_settings_form.rb +1 -1
- data/app/forms/decidim/omniauth_registration_form.rb +1 -1
- data/app/forms/decidim/upload_validation_form.rb +9 -1
- data/app/helpers/decidim/amendments_helper.rb +1 -2
- data/app/helpers/decidim/animations_helper.rb +14 -0
- data/app/helpers/decidim/decidim_form_helper.rb +17 -6
- data/app/helpers/decidim/layout_helper.rb +1 -15
- data/app/helpers/decidim/likeable_helper.rb +46 -0
- data/app/helpers/decidim/messaging/conversation_helper.rb +1 -1
- data/app/helpers/decidim/omniauth_helper.rb +1 -1
- data/app/helpers/decidim/qr_code_helper.rb +37 -0
- data/app/helpers/decidim/rich_text_editor_helper.rb +0 -1
- data/app/helpers/decidim/sanitize_helper.rb +5 -5
- data/app/helpers/decidim/social_share_button_helper.rb +11 -10
- data/app/jobs/decidim/delete_inactive_participants_job.rb +41 -0
- data/app/jobs/decidim/process_inactive_participant_job.rb +48 -0
- data/app/mailers/decidim/participants_account_mailer.rb +38 -0
- data/app/mailers/decidim/user_group_mailer.rb +28 -0
- data/app/models/decidim/action_log.rb +3 -5
- data/app/models/decidim/amendment.rb +1 -1
- data/app/models/decidim/attachment_collection.rb +2 -0
- data/app/models/decidim/authorization_transfer.rb +0 -1
- data/app/models/decidim/coauthorship.rb +1 -1
- data/app/models/decidim/component.rb +4 -1
- data/app/models/decidim/{endorsement.rb → like.rb} +4 -4
- data/app/models/decidim/messaging/conversation.rb +3 -5
- data/app/models/decidim/messaging/message.rb +3 -15
- data/app/models/decidim/omniauth_provider.rb +1 -1
- data/app/models/decidim/organization.rb +3 -8
- data/app/models/decidim/participatory_space_role_config/{valuator.rb → evaluator.rb} +1 -1
- data/app/models/decidim/reminder_record.rb +1 -1
- data/app/models/decidim/searchable_resource.rb +16 -0
- data/app/models/decidim/taxonomy.rb +6 -1
- data/app/models/decidim/user.rb +32 -15
- data/app/models/decidim/user_base_entity.rb +2 -2
- data/app/packs/entrypoints/decidim_conference_diploma.scss +2 -2
- data/app/packs/entrypoints/decidim_core.scss +2 -2
- data/app/packs/entrypoints/decidim_editor.js +0 -4
- data/app/packs/entrypoints/decidim_map.scss +2 -2
- data/app/packs/entrypoints/decidim_overrides.scss +1 -1
- data/app/packs/entrypoints/decidim_qr.js +1 -0
- data/app/packs/images/decidim/remixicon.symbol.svg +3 -3
- data/app/packs/src/decidim/a11y.js +1 -120
- data/app/packs/src/decidim/append_redirect_url_to_modals.js +1 -1
- data/app/packs/src/decidim/attachments/file_or_link_tabs.js +1 -1
- data/app/packs/src/decidim/confirm.js +27 -20
- data/app/packs/src/decidim/controllers/.keep +0 -0
- data/app/packs/src/decidim/controllers/accordion/controller.js +100 -0
- data/app/packs/src/decidim/controllers/account_form/account_form.test.js +423 -0
- data/app/packs/src/decidim/controllers/account_form/controller.js +69 -0
- data/app/packs/src/decidim/{input_character_counter.js → controllers/character_counter/controller.js} +90 -106
- data/app/packs/src/decidim/controllers/clipboard_copy/clipboard_copy.test.js +305 -0
- data/app/packs/src/decidim/controllers/clipboard_copy/controller.js +403 -0
- data/app/packs/src/decidim/controllers/date_picker/controller.js +12 -0
- data/app/packs/src/decidim/controllers/delete_account_form/controller.js +45 -0
- data/app/packs/src/decidim/controllers/delete_account_form/delete_account_form.test.js +216 -0
- data/app/packs/src/decidim/controllers/dropdown/controller.js +125 -0
- data/app/packs/src/decidim/controllers/editor/controller.js +8 -0
- data/app/packs/src/decidim/controllers/emoji/controller.js +26 -0
- data/app/packs/src/decidim/{input_emoji.js → controllers/emoji/emoji.js} +1 -15
- data/app/packs/src/decidim/{form_filter.js → controllers/form_filter/controller.js} +68 -51
- data/app/packs/src/decidim/controllers/form_filter/form_filter.test.js +648 -0
- data/app/packs/src/decidim/controllers/form_validator/controller.js +23 -0
- data/app/packs/src/decidim/controllers/form_validator/form_validator.js +789 -0
- data/app/packs/src/decidim/controllers/form_validator/form_validator.test.js +685 -0
- data/app/packs/src/decidim/controllers/impersonation_warning/controller.js +137 -0
- data/app/packs/src/decidim/controllers/impersonation_warning/impersonation.test.js +382 -0
- data/app/packs/src/decidim/controllers/input_tags/controller.js +28 -0
- data/app/packs/src/decidim/controllers/mention/controller.js +243 -0
- data/app/packs/src/decidim/controllers/mention/input_mentions.test.js +562 -0
- data/app/packs/src/decidim/controllers/multiple_mentions/controller.js +215 -0
- data/app/packs/src/decidim/controllers/multiple_mentions/input_multiple_mentions.test.js +645 -0
- data/app/packs/src/decidim/controllers/notification_action/controller.js +210 -0
- data/app/packs/src/decidim/controllers/notification_action/notification_action.test.js +388 -0
- data/app/packs/src/decidim/controllers/password_toggler/controller.js +232 -0
- data/app/packs/src/decidim/controllers/password_toggler/password_toggler.test.js +229 -0
- data/app/packs/src/decidim/controllers/report_form/controller.js +184 -0
- data/app/packs/src/decidim/controllers/report_form/report_form_manager.test.js +511 -0
- data/app/packs/src/decidim/controllers/scroll_to_last/controller.js +9 -0
- data/app/packs/src/decidim/controllers/scroll_to_last/scroll_to_last_child.test.js +225 -0
- data/app/packs/src/decidim/controllers/sticky_buttons/controller.js +55 -0
- data/app/packs/src/decidim/controllers/sticky_buttons/sticky_buttons.test.js +334 -0
- data/app/packs/src/decidim/controllers/sticky_header/controller.js +116 -0
- data/app/packs/src/decidim/controllers/sticky_header/sticky_header.test.js +320 -0
- data/app/packs/src/decidim/controllers/toggle/controller.js +111 -0
- data/app/packs/src/decidim/controllers/toggle/toggle.test.js +260 -0
- data/app/packs/src/decidim/controllers/tooltip/controller.js +287 -0
- data/app/packs/src/decidim/controllers/tooltip/tooltip.test.js +380 -0
- data/app/packs/src/decidim/controllers/user_registration_form/controller.js +160 -0
- data/app/packs/src/decidim/controllers/user_registration_form/omniauth_registration_form.test.js +441 -0
- data/app/packs/src/decidim/controllers/user_registration_form/user_registration_form.test.js +386 -0
- data/app/packs/src/decidim/data_consent/index.js +1 -1
- data/app/packs/src/decidim/datalist_select.js +1 -1
- data/app/packs/src/decidim/datepicker/form_datepicker.js +1 -1
- data/app/packs/src/decidim/datepicker/generate_datepicker.js +2 -2
- data/app/packs/src/decidim/datepicker/generate_timepicker.js +2 -2
- data/app/packs/src/decidim/direct_uploads/upload_field.js +3 -5
- data/app/packs/src/decidim/direct_uploads/upload_modal.js +1 -1
- data/app/packs/src/decidim/dropdown_menu.js +1 -1
- data/app/packs/src/decidim/editor/common/input_dialog.js +1 -1
- data/app/packs/src/decidim/editor/common/suggestion.js +10 -2
- data/app/packs/src/decidim/editor/common/upload_dialog.js +1 -1
- data/app/packs/src/decidim/editor/extensions/decidim_kit/index.js +7 -7
- data/app/packs/src/decidim/editor/extensions/emoji/index.js +2 -3
- data/app/packs/src/decidim/editor/extensions/image/index.js +1 -1
- data/app/packs/src/decidim/editor/extensions/image/node_view.js +1 -1
- data/app/packs/src/decidim/editor/extensions/link/bubble_menu.js +1 -1
- data/app/packs/src/decidim/editor/extensions/link/index.js +1 -1
- data/app/packs/src/decidim/editor/extensions/mention/index.js +1 -2
- data/app/packs/src/decidim/editor/extensions/mention_resource/index.js +69 -0
- data/app/packs/src/decidim/editor/extensions/video_embed/index.js +1 -1
- data/app/packs/src/decidim/editor/index.js +6 -5
- data/app/packs/src/decidim/editor/test/extensions/decidim_kit.test.js +1 -2
- data/app/packs/src/decidim/editor/test/extensions/emoji.test.js +17 -0
- data/app/packs/src/decidim/editor/test/extensions/heading.test.js +2 -2
- data/app/packs/src/decidim/editor/toolbar.js +1 -1
- data/app/packs/src/decidim/form_attachments.js +1 -1
- data/app/packs/src/decidim/geocoding/attach_input.js +3 -0
- data/app/packs/src/decidim/geocoding/provider/here.js +1 -1
- data/app/packs/src/decidim/geocoding/provider/photon.js +1 -1
- data/app/packs/src/decidim/geocoding/reverse_geocoding.js +2 -2
- data/app/packs/src/decidim/geocoding.js +2 -2
- data/app/packs/src/decidim/index.js +127 -95
- data/app/packs/src/decidim/map/integration/forms.js +1 -1
- data/app/packs/src/decidim/map/legacy.js +1 -1
- data/app/packs/src/decidim/map/provider/default.js +1 -1
- data/app/packs/src/decidim/map/provider/here.js +1 -1
- data/app/packs/src/decidim/map.js +1 -1
- data/app/packs/src/decidim/{external_link.js → refactor/implementation/external_link.js} +1 -1
- data/app/packs/src/decidim/{onboarding_pending_action.js → refactor/integration/onboarding_pending_action.js} +4 -4
- data/app/packs/src/decidim/{check_boxes_tree.js → refactor/moved/check_boxes_tree.js} +24 -9
- data/app/packs/src/decidim/{i18n.test.js → refactor/moved/i18n.test.js} +1 -1
- data/app/packs/src/decidim/{icon.js → refactor/moved/icon.js} +2 -0
- data/app/packs/src/decidim/refactor/support/stimulus.js +39 -0
- data/app/packs/src/decidim/refactor/test/configuration.test.js +360 -0
- data/app/packs/src/decidim/{external_domain_warning.test.js → refactor/test/external_domain_warning.test.js} +1 -1
- data/app/packs/src/decidim/{external_link.test.js → refactor/test/external_link.test.js} +1 -1
- data/app/packs/src/decidim/results_listing.js +2 -2
- data/app/packs/src/decidim/security/selfxss_warning.js +1 -1
- data/app/packs/src/decidim/session_timeouter.js +1 -1
- data/app/packs/src/decidim/sw/push-permissions.js +1 -1
- data/app/packs/src/decidim/utilities/dom.js +1 -1
- data/app/packs/stylesheets/decidim/_accordion.scss +37 -3
- data/app/packs/stylesheets/decidim/_author.scss +0 -21
- data/app/packs/stylesheets/decidim/_buttons.scss +4 -0
- data/app/packs/stylesheets/decidim/_callout.scss +1 -1
- data/app/packs/stylesheets/decidim/_cards.scss +1 -1
- data/app/packs/stylesheets/decidim/_content_blocks.scss +1 -1
- data/app/packs/stylesheets/decidim/_cookies.scss +1 -1
- data/app/packs/stylesheets/decidim/_datepicker.scss +0 -4
- data/app/packs/stylesheets/decidim/_dropdown.scss +56 -19
- data/app/packs/stylesheets/decidim/_forms.scss +6 -6
- data/app/packs/stylesheets/decidim/_header.scss +17 -1
- data/app/packs/stylesheets/decidim/_labels.scss +1 -1
- data/app/packs/stylesheets/decidim/_layout.scss +12 -2
- data/app/packs/stylesheets/decidim/{_endorsers_list.scss → _likes_list.scss} +1 -1
- data/app/packs/stylesheets/decidim/_modal.scss +5 -1
- data/app/packs/stylesheets/decidim/_modal_share.scss +83 -9
- data/app/packs/stylesheets/decidim/_modal_update.scss +1 -5
- data/app/packs/stylesheets/decidim/_participatory_spaces.scss +1 -1
- data/app/packs/stylesheets/decidim/_rich_text.scss +13 -0
- data/app/packs/stylesheets/decidim/_statistics.scss +35 -3
- data/app/packs/stylesheets/decidim/_tooltip.scss +1 -1
- data/app/packs/stylesheets/decidim/_typography.scss +6 -0
- data/app/packs/stylesheets/decidim/_wizard_steps.scss +1 -1
- data/app/packs/stylesheets/decidim/application.scss +76 -76
- data/app/packs/stylesheets/decidim/editor.scss +11 -12
- data/app/packs/stylesheets/decidim/legacy/email.scss +6 -5
- data/app/packs/stylesheets/decidim/map.scss +5 -3
- data/app/packs/stylesheets/decidim/qr.scss +65 -0
- data/app/permissions/decidim/default_permissions.rb +16 -0
- data/app/permissions/decidim/permissions.rb +5 -26
- data/app/presenters/decidim/admin_log/base_user_presenter.rb +2 -2
- data/app/presenters/decidim/admin_log/organization_presenter.rb +0 -15
- data/app/presenters/decidim/home_stats_presenter.rb +4 -54
- data/app/presenters/decidim/log/base_presenter.rb +1 -1
- data/app/presenters/decidim/menu_item_presenter.rb +10 -3
- data/app/presenters/decidim/official_author_presenter.rb +0 -4
- data/app/presenters/decidim/participatory_space_private_user_presenter.rb +0 -4
- data/app/presenters/decidim/resource_locator_presenter.rb +15 -4
- data/app/presenters/decidim/resource_presenter.rb +8 -8
- data/app/presenters/decidim/stats_presenter.rb +39 -60
- data/app/presenters/decidim/taxonomy_presenter.rb +5 -1
- data/app/presenters/decidim/user_presenter.rb +0 -6
- data/app/queries/decidim/authorized_users.rb +26 -0
- data/app/queries/decidim/inactive_users_query.rb +31 -0
- data/app/queries/decidim/public_activities.rb +2 -2
- data/app/queries/decidim/stats_followers_count.rb +43 -0
- data/app/queries/decidim/stats_participants_count.rb +102 -0
- data/app/serializers/decidim/exporters/open_data_user_serializer.rb +1 -5
- data/app/serializers/decidim/exporters/participatory_space_serializer.rb +0 -1
- data/app/services/decidim/action_logger.rb +2 -1
- data/app/services/decidim/base_diff_renderer.rb +0 -17
- data/app/services/decidim/onboarding_manager.rb +4 -0
- data/app/services/decidim/open_data_exporter.rb +19 -11
- data/app/services/decidim/send_push_notification.rb +4 -3
- data/app/services/decidim/traceability.rb +1 -1
- data/app/validators/etiquette_validator.rb +7 -0
- data/app/validators/translated_etiquette_validator.rb +1 -0
- data/app/views/decidim/account/_old_password_field.html.erb +1 -1
- data/app/views/decidim/account/_password_fields.html.erb +1 -1
- data/app/views/decidim/account/delete.html.erb +1 -1
- data/app/views/decidim/account/show.html.erb +3 -3
- data/app/views/decidim/amendments/edit_draft.html.erb +0 -4
- data/app/views/decidim/amendments/new.html.erb +0 -4
- data/app/views/decidim/amendments/preview_draft.html.erb +1 -1
- data/app/views/decidim/application/_accordion_section.html.erb +1 -1
- data/app/views/decidim/application/_collection.html.erb +1 -1
- data/app/views/decidim/application/_radio_accordion.html.erb +30 -0
- data/app/views/decidim/devise/omniauth_registrations/new.html.erb +1 -1
- data/app/views/decidim/devise/omniauth_registrations/new_tos_fields.html.erb +1 -1
- data/app/views/decidim/devise/registrations/new.html.erb +1 -1
- data/app/views/decidim/devise/shared/_links.html.erb +1 -1
- data/app/views/decidim/doorkeeper/authorizations/new.html.erb +39 -8
- data/app/views/decidim/errors/internal_server_error.html.erb +1 -0
- data/app/views/decidim/likes/update_buttons_and_counters.js.erb +21 -0
- data/app/views/decidim/messaging/conversations/_add_conversation_users.html.erb +2 -2
- data/app/views/decidim/messaging/conversations/_reply_form.html.erb +1 -1
- data/app/views/decidim/messaging/conversations/_start.html.erb +1 -1
- data/app/views/decidim/messaging/conversations/create.js.erb +0 -1
- data/app/views/decidim/messaging/conversations/new.html.erb +1 -1
- data/app/views/decidim/messaging/conversations/show.html.erb +1 -1
- data/app/views/decidim/notifications_settings/show.html.erb +3 -3
- data/app/views/decidim/pages/_tabbed.html.erb +1 -1
- data/app/views/decidim/pages/index.html.erb +1 -1
- data/app/views/decidim/participants_account_mailer/inactivity_notification.html.erb +13 -0
- data/app/views/decidim/participants_account_mailer/removal_notification.html.erb +11 -0
- data/app/views/decidim/profiles/show.html.erb +1 -1
- data/app/views/decidim/qr/show.html.erb +28 -0
- data/app/views/decidim/searches/_filters.html.erb +18 -2
- data/app/views/decidim/searches/index.html.erb +5 -2
- data/app/views/decidim/shared/_confirm_modal.html.erb +5 -3
- data/app/views/decidim/shared/_filters.html.erb +2 -2
- data/app/views/decidim/shared/_orders.html.erb +1 -1
- data/app/views/decidim/shared/_resource_actions.html.erb +8 -5
- data/app/views/decidim/user_activities/index.html.erb +1 -1
- data/app/views/decidim/user_group_mailer/notify_deprecation_to_member.html.erb +15 -0
- data/app/views/decidim/user_group_mailer/notify_deprecation_to_owner.html.erb +22 -0
- data/app/views/layouts/decidim/_head.html.erb +15 -12
- data/app/views/layouts/decidim/_impersonation_warning.html.erb +1 -1
- data/app/views/layouts/decidim/_js_configuration.html.erb +17 -19
- data/app/views/layouts/decidim/_meta_tags_context.html.erb +3 -0
- data/app/views/layouts/decidim/_offline_banner.html.erb +1 -1
- data/app/views/layouts/decidim/_wrapper.html.erb +8 -6
- data/app/views/layouts/decidim/footer/_focus_mode_main.html.erb +7 -0
- data/app/views/layouts/decidim/footer/_main_language_chooser.html.erb +1 -1
- data/app/views/layouts/decidim/header/_close_ephemeral_session.html.erb +12 -2
- data/app/views/layouts/decidim/header/_focus_mode_back_button.html.erb +24 -0
- data/app/views/layouts/decidim/header/_main.html.erb +28 -22
- data/app/views/layouts/decidim/header/_main_links_desktop.html.erb +1 -1
- data/app/views/layouts/decidim/header/_main_links_dropdown.html.erb +18 -18
- data/app/views/layouts/decidim/header/_main_links_mobile_item_account.html.erb +1 -1
- data/app/views/layouts/decidim/header/_main_menu_mobile.html.erb +1 -1
- data/app/views/layouts/decidim/header/_menu_breadcrumb_desktop.html.erb +1 -1
- data/app/views/layouts/decidim/header/_menu_breadcrumb_items.html.erb +1 -1
- data/app/views/layouts/decidim/header/_menu_breadcrumb_mobile_tablet.html.erb +1 -1
- data/app/views/layouts/decidim/header/_mobile_language_choose.html.erb +1 -1
- data/app/views/layouts/decidim/mailer.html.erb +2 -2
- data/app/views/layouts/decidim/newsletter_base.html.erb +2 -2
- data/app/views/layouts/decidim/shared/_layout_two_col.html.erb +3 -1
- data/app/views/layouts/decidim/shared/_layout_user_profile.html.erb +7 -3
- data/config/assets.rb +3 -3
- data/config/initializers/devise.rb +8 -0
- data/config/initializers/omniauth.rb +1 -1
- data/config/locales/ar.yml +0 -209
- data/config/locales/bg.yml +0 -238
- data/config/locales/bs-BA.yml +0 -30
- data/config/locales/ca-IT.yml +112 -286
- data/config/locales/ca.yml +112 -286
- data/config/locales/cs.yml +85 -279
- data/config/locales/de.yml +112 -285
- data/config/locales/el.yml +0 -224
- data/config/locales/en.yml +112 -286
- data/config/locales/eo.yml +0 -2
- data/config/locales/es-MX.yml +112 -286
- data/config/locales/es-PY.yml +112 -286
- data/config/locales/es.yml +112 -286
- data/config/locales/eu.yml +125 -299
- data/config/locales/fi-plain.yml +112 -286
- data/config/locales/fi.yml +112 -286
- data/config/locales/fr-CA.yml +85 -275
- data/config/locales/fr.yml +85 -275
- data/config/locales/ga-IE.yml +0 -28
- data/config/locales/gl.yml +0 -140
- data/config/locales/hu.yml +0 -237
- data/config/locales/id-ID.yml +0 -136
- data/config/locales/is-IS.yml +0 -14
- data/config/locales/it.yml +1 -234
- data/config/locales/ja.yml +108 -282
- data/config/locales/lb.yml +0 -189
- data/config/locales/lt.yml +0 -240
- data/config/locales/lv.yml +0 -186
- data/config/locales/nl.yml +0 -189
- data/config/locales/no.yml +0 -189
- data/config/locales/pl.yml +0 -244
- data/config/locales/pt-BR.yml +0 -213
- data/config/locales/pt.yml +0 -189
- data/config/locales/ro-RO.yml +54 -218
- data/config/locales/ru.yml +0 -62
- data/config/locales/sk.yml +0 -164
- data/config/locales/sr-CS.yml +0 -31
- data/config/locales/sv.yml +80 -263
- data/config/locales/tr-TR.yml +0 -189
- data/config/locales/uk.yml +0 -23
- data/config/locales/zh-CN.yml +0 -180
- data/config/locales/zh-TW.yml +0 -229
- data/config/routes.rb +3 -36
- data/db/migrate/20250217192438_convert_user_groups_into_users.rb +38 -0
- data/db/migrate/20250221085317_add_user_type_to_action_logs.rb +12 -0
- data/db/migrate/20250221090905_create_decidim_api_jwt_denylist.rb +11 -0
- data/db/migrate/20250221091348_add_api_key_to_decidim_api_user.rb +7 -0
- data/db/migrate/20250327143657_remove_metrics_content_blocks.rb +9 -0
- data/db/migrate/20250401153444_enable_pkce.rb +8 -0
- data/db/migrate/20250416130426_rename_doorkeeper_scopes.rb +25 -0
- data/db/migrate/20250515105231_rename_endorsements_to_likes.rb +62 -0
- data/db/migrate/20250523104311_move_cta_to_hero_content_block.rb +34 -0
- data/db/migrate/20250527122040_move_highlighted_content_banner_settings_to_content_block.rb +69 -0
- data/db/migrate/20250603103953_add_refresh_tokens_enabled_to_doorkeeper_applications.rb +7 -0
- data/db/migrate/20250609073104_remove_not_null_on_active_storage_blobs_checksum.active_storage.rb +10 -0
- data/db/migrate/20250613122148_remove_hashtag_table_from_core.rb +7 -0
- data/decidim-core.gemspec +4 -4
- data/{app/helpers → lib}/decidim/action_authorization_helper.rb +5 -4
- data/lib/decidim/amendable.rb +4 -4
- data/lib/decidim/api/functions/user_entity_finder.rb +1 -1
- data/lib/decidim/api/input_filters/component_input_filter.rb +3 -3
- data/lib/decidim/api/input_filters/user_entity_input_filter.rb +0 -21
- data/lib/decidim/api/input_sorts/has_likeable_input_sort.rb +15 -0
- data/lib/decidim/api/interfaces/attachable_collection_interface.rb +13 -0
- data/lib/decidim/api/interfaces/author_interface.rb +1 -2
- data/lib/decidim/api/interfaces/authorable_interface.rb +2 -2
- data/lib/decidim/api/interfaces/coauthorable_interface.rb +1 -1
- data/lib/decidim/api/interfaces/component_interface.rb +6 -0
- data/lib/decidim/api/interfaces/followable_interface.rb +13 -0
- data/lib/decidim/api/interfaces/likeable_interface.rb +19 -0
- data/lib/decidim/api/interfaces/localizable_interface.rb +18 -0
- data/lib/decidim/api/interfaces/participatory_space_interface.rb +6 -8
- data/lib/decidim/api/interfaces/referable_interface.rb +13 -0
- data/lib/decidim/api/interfaces/timestamps_interface.rb +0 -1
- data/lib/decidim/api/interfaces/traceable_interface.rb +1 -1
- data/lib/decidim/api/types/attachment_collection_type.rb +14 -0
- data/lib/decidim/api/types/organization_type.rb +3 -1
- data/lib/decidim/api/types/session_type.rb +0 -5
- data/lib/decidim/api/types/statistic_type.rb +29 -3
- data/lib/decidim/api/types/trace_version_type.rb +1 -1
- data/lib/decidim/api/types/user_type.rb +1 -2
- data/lib/decidim/api_response_helper.rb +12 -0
- data/lib/decidim/asset_router/storage.rb +2 -4
- data/lib/decidim/assets/tailwind/instance.rb +1 -1
- data/lib/decidim/assets/tailwind/tailwind.config.js.erb +4 -3
- data/lib/decidim/attribute_encryptor.rb +1 -1
- data/lib/decidim/attributes/time_with_zone.rb +1 -1
- data/lib/decidim/authorable.rb +5 -34
- data/lib/decidim/coauthorable.rb +13 -31
- data/lib/decidim/content_parsers/blob_parser.rb +1 -1
- data/lib/decidim/content_parsers/mention_resource_parser.rb +23 -0
- data/lib/decidim/content_parsers/tag_parser.rb +2 -2
- data/lib/decidim/content_parsers/user_parser.rb +2 -2
- data/lib/decidim/content_parsers.rb +14 -2
- data/lib/decidim/content_processor.rb +1 -1
- data/lib/decidim/content_renderers/mention_resource_renderer.rb +77 -0
- data/lib/decidim/content_renderers.rb +1 -2
- data/lib/decidim/core/api.rb +7 -8
- data/lib/decidim/core/content_blocks/registry_manager.rb +19 -5
- data/lib/decidim/core/engine.rb +72 -104
- data/lib/decidim/core/menu.rb +0 -7
- data/lib/decidim/core/seeds.rb +11 -39
- data/lib/decidim/core/test/factories.rb +22 -121
- data/lib/decidim/core/test/shared_examples/authorable.rb +0 -17
- data/lib/decidim/core/test/shared_examples/coauthorable.rb +5 -48
- data/lib/decidim/core/test/shared_examples/comments_examples.rb +7 -56
- data/lib/decidim/core/test/shared_examples/likeable.rb +22 -0
- data/lib/decidim/core/test/shared_examples/manage_share_tokens_examples.rb +16 -3
- data/lib/decidim/core/test/shared_examples/map_examples.rb +1 -1
- data/lib/decidim/core/test/shared_examples/oauth_application_examples.rb +137 -0
- data/lib/decidim/core/test/shared_examples/participatory_space_dropdown_metadata_cell_examples.rb +0 -18
- data/lib/decidim/core/test/shared_examples/permissions.rb +0 -46
- data/lib/decidim/core/test/shared_examples/reports_examples.rb +2 -2
- data/lib/decidim/core/test/shared_examples/{resource_endorsed_event_examples.rb → resource_liked_event_examples.rb} +7 -7
- data/lib/decidim/core/test/shared_examples/resource_search_examples.rb +0 -14
- data/lib/decidim/core/test/shared_examples/searchable_results_examples.rb +8 -34
- data/lib/decidim/core/test/shared_examples/social_share_examples.rb +79 -0
- data/lib/decidim/core/test/shared_examples/system_like_resource_examples.rb +192 -0
- data/lib/decidim/core/test/shared_examples/uncommentable_component_examples.rb +2 -2
- data/lib/decidim/core/test/shared_examples/{with_endorsable_permissions_examples.rb → with_likeable_permissions_examples.rb} +22 -22
- data/lib/decidim/core/test.rb +6 -9
- data/lib/decidim/core/version.rb +1 -1
- data/lib/decidim/core.rb +237 -88
- data/lib/decidim/deprecations.rb +1 -1
- data/lib/decidim/download_your_data_serializers.rb +6 -3
- data/lib/decidim/env.rb +3 -1
- data/lib/decidim/events.rb +0 -1
- data/lib/decidim/form_builder.rb +26 -43
- data/lib/decidim/gamification/badge_scorer.rb +0 -2
- data/lib/decidim/gamification.rb +1 -1
- data/lib/decidim/has_conversations.rb +0 -44
- data/lib/decidim/has_workflows.rb +42 -0
- data/lib/decidim/icon_registry.rb +4 -4
- data/lib/decidim/legacy_form_builder.rb +17 -2
- data/lib/decidim/likeable.rb +24 -0
- data/lib/decidim/manifest_registry.rb +1 -1
- data/lib/decidim/map/provider/dynamic_map/here.rb +0 -19
- data/lib/decidim/map/provider/static_map/here.rb +2 -10
- data/lib/decidim/menu.rb +1 -1
- data/lib/decidim/moderation_tools.rb +1 -1
- data/lib/decidim/mutation_registry.rb +17 -0
- data/lib/decidim/newsletter_encryptor.rb +1 -1
- data/lib/decidim/oauth/token_generator.rb +35 -0
- data/lib/decidim/oauth.rb +7 -0
- data/lib/decidim/organization_settings.rb +2 -10
- data/lib/decidim/paddable.rb +1 -1
- data/lib/decidim/participatory_space_resourceable.rb +2 -2
- data/lib/decidim/participatory_space_user.rb +1 -1
- data/lib/decidim/pdf_signature_example.rb +108 -0
- data/lib/decidim/query.rb +1 -1
- data/lib/decidim/query_extensions.rb +0 -33
- data/lib/decidim/reportable.rb +1 -1
- data/lib/decidim/searchable.rb +4 -2
- data/lib/decidim/seeds.rb +8 -0
- data/lib/decidim/settings_manifest.rb +3 -2
- data/lib/decidim/{webpacker → shakapacker}/configuration.rb +4 -4
- data/lib/decidim/{webpacker → shakapacker}/runner.rb +3 -3
- data/lib/decidim/shakapacker/shakapacker.rb +8 -0
- data/lib/decidim/{webpacker → shakapacker}/shakapacker.yml +3 -1
- data/lib/decidim/{webpacker → shakapacker}/webpack/custom.js +1 -1
- data/lib/decidim/shakapacker.rb +34 -0
- data/lib/decidim/social_share_service_manifest.rb +7 -12
- data/lib/decidim/stats_registry.rb +15 -3
- data/{app/helpers → lib}/decidim/tooltip_helper.rb +3 -3
- data/lib/decidim/url_option_resolver.rb +1 -1
- data/lib/decidim/view_model.rb +0 -3
- data/lib/decidim/webpacker.rb +11 -20
- data/lib/decidim/workflow_registry.rb +61 -0
- data/lib/tasks/decidim_participants_tasks.rake +22 -0
- data/lib/tasks/{decidim_webpacker_tasks.rake → decidim_shakapacker_tasks.rake} +35 -33
- data/lib/tasks/decidim_tasks.rake +4 -1
- data/lib/tasks/decidim_taxonomies.rake +0 -34
- data/lib/tasks/upgrade/clean.rake +1 -1
- data/lib/tasks/upgrade/decidim_action_log_valuation_assignment.rake +13 -0
- data/lib/tasks/upgrade/decidim_change_valuator_tasks.rake +12 -0
- data/lib/tasks/upgrade/{decidim_deduplicate_endorsements.rake → decidim_deduplicate_likes.rake} +10 -10
- data/lib/tasks/upgrade/decidim_migrate_wysiwyg_content.rake +2 -2
- data/lib/tasks/upgrade/decidim_paper_trail_valuation_assignment.rake +13 -0
- data/lib/tasks/upgrade/user_groups_migration.rake +155 -0
- metadata +204 -256
- data/app/cells/decidim/author/endorsements.erb +0 -4
- data/app/cells/decidim/author/profile_minicard.erb +0 -30
- data/app/cells/decidim/content_blocks/organization_metrics_cell.rb +0 -21
- data/app/cells/decidim/content_blocks/participatory_space_metrics/content.erb +0 -18
- data/app/cells/decidim/content_blocks/participatory_space_metrics_cell.rb +0 -43
- data/app/cells/decidim/endorsement_block/show.erb +0 -3
- data/app/cells/decidim/endorsement_buttons/button_content.erb +0 -2
- data/app/cells/decidim/endorsement_buttons/select_identity_button.erb +0 -7
- data/app/cells/decidim/endorsement_buttons/show.erb +0 -17
- data/app/cells/decidim/endorsement_buttons_cell.rb +0 -87
- data/app/cells/decidim/endorsers_list/empty.erb +0 -3
- data/app/cells/decidim/endorsers_list/show.erb +0 -20
- data/app/cells/decidim/endorsers_list_cell.rb +0 -64
- data/app/cells/decidim/group_admins/show.erb +0 -34
- data/app/cells/decidim/group_admins_cell.rb +0 -8
- data/app/cells/decidim/group_members/show.erb +0 -46
- data/app/cells/decidim/group_members_cell.rb +0 -8
- data/app/cells/decidim/groups/show.erb +0 -14
- data/app/cells/decidim/groups_cell.rb +0 -22
- data/app/cells/decidim/members/show.erb +0 -12
- data/app/cells/decidim/members_cell.rb +0 -34
- data/app/cells/decidim/profile_actions/dropdown_actions.erb +0 -21
- data/app/cells/decidim/represent_user_group/show.erb +0 -11
- data/app/cells/decidim/represent_user_group_cell.rb +0 -46
- data/app/cells/decidim/user_conversation/conversation_header.erb +0 -14
- data/app/cells/decidim/user_conversation/messages.erb +0 -1
- data/app/cells/decidim/user_conversation/new.erb +0 -6
- data/app/cells/decidim/user_conversation/reply.erb +0 -9
- data/app/cells/decidim/user_conversation/show.erb +0 -19
- data/app/cells/decidim/user_conversation/start.erb +0 -12
- data/app/cells/decidim/user_conversation_cell.rb +0 -62
- data/app/cells/decidim/user_conversations/new_conversation_button.erb +0 -20
- data/app/cells/decidim/user_conversations/show.erb +0 -13
- data/app/cells/decidim/user_conversations_cell.rb +0 -28
- data/app/cells/decidim/user_group_pending_invitations_list/show.erb +0 -28
- data/app/cells/decidim/user_group_pending_invitations_list_cell.rb +0 -26
- data/app/cells/decidim/user_group_pending_requests_list/show.erb +0 -27
- data/app/cells/decidim/user_group_pending_requests_list_cell.rb +0 -26
- data/app/commands/decidim/accept_group_invitation.rb +0 -43
- data/app/commands/decidim/accept_user_group_join_request.rb +0 -53
- data/app/commands/decidim/create_user_group.rb +0 -73
- data/app/commands/decidim/demote_membership.rb +0 -57
- data/app/commands/decidim/endorse_resource.rb +0 -66
- data/app/commands/decidim/invite_user_to_group.rb +0 -63
- data/app/commands/decidim/join_user_group.rb +0 -57
- data/app/commands/decidim/leave_user_group.rb +0 -47
- data/app/commands/decidim/promote_membership.rb +0 -55
- data/app/commands/decidim/reject_group_invitation.rb +0 -42
- data/app/commands/decidim/reject_user_group_join_request.rb +0 -53
- data/app/commands/decidim/remove_user_from_group.rb +0 -53
- data/app/commands/decidim/unendorse_resource.rb +0 -39
- data/app/commands/decidim/update_user_group.rb +0 -72
- data/app/controllers/concerns/decidim/user_groups.rb +0 -23
- data/app/controllers/decidim/endorsements_controller.rb +0 -71
- data/app/controllers/decidim/group_email_confirmations_controller.rb +0 -32
- data/app/controllers/decidim/group_invites_controller.rb +0 -104
- data/app/controllers/decidim/group_members_controller.rb +0 -75
- data/app/controllers/decidim/groups_controller.rb +0 -99
- data/app/controllers/decidim/own_user_groups_controller.rb +0 -15
- data/app/controllers/decidim/user_conversations_controller.rb +0 -133
- data/app/controllers/decidim/user_group_join_requests_controller.rb +0 -72
- data/app/events/decidim/demoted_membership_event.rb +0 -28
- data/app/events/decidim/invited_to_group_event.rb +0 -62
- data/app/events/decidim/join_request_accepted_event.rb +0 -28
- data/app/events/decidim/join_request_created_event.rb +0 -28
- data/app/events/decidim/join_request_rejected_event.rb +0 -28
- data/app/events/decidim/promoted_to_admin_event.rb +0 -28
- data/app/events/decidim/removed_from_group_event.rb +0 -28
- data/app/events/decidim/resource_endorsed_event.rb +0 -36
- data/app/events/decidim/user_group_admin_event.rb +0 -37
- data/app/events/decidim/user_group_created_event.rb +0 -6
- data/app/events/decidim/user_group_updated_event.rb +0 -6
- data/app/forms/decidim/invite_user_to_group_form.rb +0 -29
- data/app/forms/decidim/user_group_form.rb +0 -85
- data/app/helpers/decidim/cta_button_helper.rb +0 -31
- data/app/helpers/decidim/endorsable_helper.rb +0 -84
- data/app/helpers/decidim/twitter_search_helper.rb +0 -14
- data/app/helpers/decidim/user_group_helper.rb +0 -22
- data/app/jobs/decidim/metric_job.rb +0 -15
- data/app/models/decidim/metric.rb +0 -25
- data/app/models/decidim/user_group.rb +0 -217
- data/app/models/decidim/user_group_membership.rb +0 -17
- data/app/packs/entrypoints/decidim_widget.js +0 -1
- data/app/packs/src/decidim/abide_form_validator_fixer.js +0 -44
- data/app/packs/src/decidim/account_form.js +0 -96
- data/app/packs/src/decidim/append_elements.js +0 -6
- data/app/packs/src/decidim/back_to_list.js +0 -26
- data/app/packs/src/decidim/change_report_form_behavior.js +0 -45
- data/app/packs/src/decidim/clipboard.js +0 -128
- data/app/packs/src/decidim/editor/extensions/hashtag/index.js +0 -66
- data/app/packs/src/decidim/editor/extensions/heading/index.js +0 -33
- data/app/packs/src/decidim/editor/test/extensions/hashtag.test.js +0 -105
- data/app/packs/src/decidim/form_filter.component_for_testing.js +0 -7
- data/app/packs/src/decidim/form_validator.js +0 -56
- data/app/packs/src/decidim/gallery.js +0 -5
- data/app/packs/src/decidim/impersonation.js +0 -23
- data/app/packs/src/decidim/input_autojump.js +0 -38
- data/app/packs/src/decidim/input_hashtags.js +0 -91
- data/app/packs/src/decidim/input_mentions.js +0 -143
- data/app/packs/src/decidim/input_multiple_mentions.js +0 -115
- data/app/packs/src/decidim/input_tags.js +0 -16
- data/app/packs/src/decidim/notifications_actions.js +0 -40
- data/app/packs/src/decidim/password_toggler.js +0 -104
- data/app/packs/src/decidim/remote_tooltips.js +0 -38
- data/app/packs/src/decidim/responsive_horizontal_tabs.js +0 -12
- data/app/packs/src/decidim/scroll_to_last_child.js +0 -14
- data/app/packs/src/decidim/slug_form.js +0 -9
- data/app/packs/src/decidim/sticky_footer.js +0 -48
- data/app/packs/src/decidim/sticky_header.js +0 -62
- data/app/packs/src/decidim/toggle.js +0 -37
- data/app/packs/src/decidim/tooltips.js +0 -152
- data/app/packs/src/decidim/user_registrations.js +0 -65
- data/app/packs/src/decidim/vizzs/areachart.js +0 -251
- data/app/packs/src/decidim/vizzs/index.js +0 -82
- data/app/packs/src/decidim/widget.js +0 -14
- data/app/packs/stylesheets/decidim/_hashtags.scss +0 -5
- data/app/packs/stylesheets/decidim/_metrics.scss +0 -70
- data/app/packs/stylesheets/decidim/_modal_identities.scss +0 -23
- data/app/presenters/decidim/admin_log/user_group_presenter.rb +0 -31
- data/app/presenters/decidim/hashtag_presenter.rb +0 -37
- data/app/presenters/decidim/metric_charts_presenter.rb +0 -92
- data/app/presenters/decidim/metric_object_presenter.rb +0 -32
- data/app/presenters/decidim/user_group_presenter.rb +0 -38
- data/app/queries/decidim/metric_manage.rb +0 -81
- data/app/queries/decidim/metric_measure.rb +0 -36
- data/app/queries/decidim/metrics/blocked_users_metric_manage.rb +0 -26
- data/app/queries/decidim/metrics/followers_metric_manage.rb +0 -65
- data/app/queries/decidim/metrics/participants_metric_manage.rb +0 -79
- data/app/queries/decidim/metrics/reported_users_metric_manage.rb +0 -26
- data/app/queries/decidim/metrics/user_reports_metric_manage.rb +0 -26
- data/app/queries/decidim/metrics/users_metric_manage.rb +0 -26
- data/app/queries/decidim/user_groups/accepted_memberships.rb +0 -36
- data/app/queries/decidim/user_groups/accepted_user_groups.rb +0 -38
- data/app/queries/decidim/user_groups/accepted_users.rb +0 -36
- data/app/queries/decidim/user_groups/admin_memberships.rb +0 -37
- data/app/queries/decidim/user_groups/invited_memberships.rb +0 -36
- data/app/queries/decidim/user_groups/manageable_user_groups.rb +0 -39
- data/app/queries/decidim/user_groups/member_memberships.rb +0 -37
- data/app/resolvers/decidim/core/metric_resolver.rb +0 -71
- data/app/resolvers/decidim/hashtags_resolver.rb +0 -15
- data/app/serializers/decidim/exporters/open_data_metric_serializer.rb +0 -22
- data/app/serializers/decidim/exporters/open_data_user_group_serializer.rb +0 -38
- data/app/views/decidim/endorsements/_identity.html.erb +0 -4
- data/app/views/decidim/endorsements/identities.html.erb +0 -18
- data/app/views/decidim/endorsements/update_buttons_and_counters.js.erb +0 -25
- data/app/views/decidim/group_invites/index.html.erb +0 -26
- data/app/views/decidim/groups/_form.html.erb +0 -17
- data/app/views/decidim/groups/edit.html.erb +0 -19
- data/app/views/decidim/groups/new.html.erb +0 -23
- data/app/views/decidim/messaging/conversation_mailer/comanagers_new_conversation.html.erb +0 -17
- data/app/views/decidim/messaging/conversation_mailer/comanagers_new_message.html.erb +0 -17
- data/app/views/decidim/messaging/conversation_mailer/new_group_conversation.html.erb +0 -17
- data/app/views/decidim/messaging/conversation_mailer/new_group_message.html.erb +0 -17
- data/app/views/decidim/own_user_groups/index.html.erb +0 -10
- data/app/views/decidim/shared/_extended_navigation_bar.html.erb +0 -50
- data/app/views/decidim/user_conversations/index.html.erb +0 -4
- data/app/views/decidim/user_conversations/show.html.erb +0 -4
- data/app/views/decidim/user_conversations/update.js.erb +0 -8
- data/config/initializers/new_framework_defaults_7_0.rb +0 -145
- data/lib/decidim/api/input_filters/has_hastaggable_input_filter.rb +0 -15
- data/lib/decidim/api/input_sorts/has_endorsable_input_sort.rb +0 -15
- data/lib/decidim/api/interfaces/endorsable_interface.rb +0 -19
- data/lib/decidim/api/types/hashtag_type.rb +0 -13
- data/lib/decidim/api/types/metric_history_type.rb +0 -20
- data/lib/decidim/api/types/metric_type.rb +0 -13
- data/lib/decidim/api/types/user_group_type.rb +0 -55
- data/lib/decidim/content_parsers/hashtag_parser.rb +0 -66
- data/lib/decidim/content_parsers/user_group_parser.rb +0 -33
- data/lib/decidim/content_renderers/hashtag_renderer.rb +0 -78
- data/lib/decidim/content_renderers/user_group_renderer.rb +0 -31
- data/lib/decidim/core/test/shared_examples/amendable_interface_examples.rb +0 -14
- data/lib/decidim/core/test/shared_examples/amendable_proposals_interface_examples.rb +0 -50
- data/lib/decidim/core/test/shared_examples/assembly_announcements_examples.rb +0 -38
- data/lib/decidim/core/test/shared_examples/attachable_interface_examples.rb +0 -16
- data/lib/decidim/core/test/shared_examples/authorable_interface_examples.rb +0 -59
- data/lib/decidim/core/test/shared_examples/categories_container_examples.rb +0 -22
- data/lib/decidim/core/test/shared_examples/categorizable_interface_examples.rb +0 -27
- data/lib/decidim/core/test/shared_examples/coauthorable_interface_examples.rb +0 -91
- data/lib/decidim/core/test/shared_examples/endorsable.rb +0 -69
- data/lib/decidim/core/test/shared_examples/endorsable_interface_examples.rb +0 -22
- data/lib/decidim/core/test/shared_examples/fingerprintable_interface_examples.rb +0 -17
- data/lib/decidim/core/test/shared_examples/input_filter_examples.rb +0 -118
- data/lib/decidim/core/test/shared_examples/input_sort_examples.rb +0 -126
- data/lib/decidim/core/test/shared_examples/metric_manage_shared_context.rb +0 -9
- data/lib/decidim/core/test/shared_examples/participatory_space_resourcable_interface_examples.rb +0 -43
- data/lib/decidim/core/test/shared_examples/process_announcements_examples.rb +0 -37
- data/lib/decidim/core/test/shared_examples/scopable_interface_examples.rb +0 -19
- data/lib/decidim/core/test/shared_examples/system_endorse_resource_examples.rb +0 -298
- data/lib/decidim/core/test/shared_examples/taxonomizable_interface_examples.rb +0 -20
- data/lib/decidim/core/test/shared_examples/timestamps_interface_examples.rb +0 -21
- data/lib/decidim/core/test/shared_examples/traceable_interface_examples.rb +0 -47
- data/lib/decidim/download_your_data_serializers/download_your_data_user_group_serializer.rb +0 -21
- data/lib/decidim/endorsable.rb +0 -29
- data/lib/decidim/events/user_group_event.rb +0 -43
- data/lib/decidim/hashtag.rb +0 -13
- data/lib/decidim/metric_manifest.rb +0 -35
- data/lib/decidim/metric_operation.rb +0 -58
- data/lib/decidim/metric_operation_manifest.rb +0 -21
- data/lib/decidim/metric_registry.rb +0 -109
- data/lib/gem_overrides/shakapacker/runner.rb +0 -9
- data/lib/tasks/decidim_metrics_tasks.rake +0 -89
- /data/app/cells/decidim/{endorsement_buttons/disabled_endorsements.erb → like_buttons/disabled_likes.erb} +0 -0
- /data/app/packs/src/decidim/{configuration.js → refactor/implementation/configuration.js} +0 -0
- /data/app/packs/src/decidim/{external_domain_warning.js → refactor/implementation/external_domain_warning.js} +0 -0
- /data/app/packs/src/decidim/{autocomplete.js → refactor/moved/autocomplete.js} +0 -0
- /data/app/packs/src/decidim/{delayed.js → refactor/moved/delayed.js} +0 -0
- /data/app/packs/src/decidim/{focus_guard.js → refactor/moved/focus_guard.js} +0 -0
- /data/app/packs/src/decidim/{history.js → refactor/moved/history.js} +0 -0
- /data/app/packs/src/decidim/{i18n.js → refactor/moved/i18n.js} +0 -0
- /data/lib/decidim/core/test/shared_examples/{endorsements_controller_shared_context.rb → likes_controller_shared_context.rb} +0 -0
- /data/{app/helpers → lib}/decidim/resource_helper.rb +0 -0
- /data/lib/decidim/{webpacker → shakapacker}/esbuild.config.js +0 -0
- /data/lib/decidim/{webpacker → shakapacker}/postcss.config.js +0 -0
- /data/lib/decidim/{webpacker → shakapacker}/tsconfig.json +0 -0
- /data/lib/decidim/{webpacker → shakapacker}/webpack/.modernizrrc +0 -0
- /data/lib/decidim/{webpacker → shakapacker}/webpack/webpack.config.js +0 -0
@@ -0,0 +1,685 @@
|
|
1
|
+
/* eslint max-lines: ["error", 690] */
|
2
|
+
/* global jest */
|
3
|
+
/**
|
4
|
+
* @jest-environment jsdom
|
5
|
+
*/
|
6
|
+
|
7
|
+
import FormValidator from "src/decidim/controllers/form_validator/form_validator"
|
8
|
+
|
9
|
+
describe("FormValidator", () => {
|
10
|
+
let formElement = null;
|
11
|
+
let validatorInstance = null;
|
12
|
+
|
13
|
+
beforeEach(() => {
|
14
|
+
// Reset DOM
|
15
|
+
document.body.innerHTML = "";
|
16
|
+
|
17
|
+
// Create basic form structure
|
18
|
+
formElement = document.createElement("form");
|
19
|
+
formElement.innerHTML = `
|
20
|
+
<div>
|
21
|
+
<label for="test-input">Test Input</label>
|
22
|
+
<input type="text" id="test-input" name="testInput" required>
|
23
|
+
<span class="form-error">This field is required</span>
|
24
|
+
</div>
|
25
|
+
<div>
|
26
|
+
<label for="email-input">Email</label>
|
27
|
+
<input type="email" id="email-input" name="emailInput" required>
|
28
|
+
<span class="form-error">Please enter a valid email</span>
|
29
|
+
</div>
|
30
|
+
<button type="submit">Submit</button>
|
31
|
+
`;
|
32
|
+
document.body.appendChild(formElement);
|
33
|
+
});
|
34
|
+
|
35
|
+
afterEach(() => {
|
36
|
+
if (validatorInstance) {
|
37
|
+
validatorInstance.destroyValidator();
|
38
|
+
}
|
39
|
+
jest.clearAllMocks();
|
40
|
+
});
|
41
|
+
|
42
|
+
describe("Constructor and Initialization", () => {
|
43
|
+
it("should initialize with form element", () => {
|
44
|
+
validatorInstance = new FormValidator(formElement);
|
45
|
+
|
46
|
+
expect(validatorInstance.element).toBe(formElement);
|
47
|
+
expect(validatorInstance.validationEnabled).toBe(true);
|
48
|
+
expect(validatorInstance.inputElements).toHaveLength(2);
|
49
|
+
});
|
50
|
+
|
51
|
+
it("should initialize with CSS selector", () => {
|
52
|
+
formElement.id = "test-form";
|
53
|
+
validatorInstance = new FormValidator("#test-form");
|
54
|
+
|
55
|
+
expect(validatorInstance.element).toBe(formElement);
|
56
|
+
});
|
57
|
+
|
58
|
+
it("should throw error for nonexistent form", () => {
|
59
|
+
expect(() => {
|
60
|
+
// eslint-disable-next-line no-new
|
61
|
+
new FormValidator("#nonexistent-form");
|
62
|
+
}).toThrow("Form element not found");
|
63
|
+
});
|
64
|
+
|
65
|
+
it("should initialize with custom validation options", () => {
|
66
|
+
const customOptions = {
|
67
|
+
validateOn: "submit",
|
68
|
+
inputErrorClass: "custom-error"
|
69
|
+
};
|
70
|
+
|
71
|
+
validatorInstance = new FormValidator(formElement, customOptions);
|
72
|
+
|
73
|
+
expect(validatorInstance.validationOptions.validateOn).toBe("submit");
|
74
|
+
expect(validatorInstance.validationOptions.inputErrorClass).toBe("custom-error");
|
75
|
+
});
|
76
|
+
});
|
77
|
+
|
78
|
+
describe("Static Methods", () => {
|
79
|
+
it("should configure global messages", () => {
|
80
|
+
Reflect.defineProperty(window, "Decidim", {
|
81
|
+
writable: true,
|
82
|
+
value: {
|
83
|
+
config: {
|
84
|
+
data: {},
|
85
|
+
set: jest.fn(function(config) {
|
86
|
+
// eslint-disable-next-line no-invalid-this
|
87
|
+
this.data = { ...this.data, ...config };
|
88
|
+
}),
|
89
|
+
get: jest.fn(function(key) {
|
90
|
+
return key
|
91
|
+
// eslint-disable-next-line no-invalid-this
|
92
|
+
? this.data[key]
|
93
|
+
// eslint-disable-next-line no-invalid-this
|
94
|
+
: this.data;
|
95
|
+
})
|
96
|
+
}
|
97
|
+
}
|
98
|
+
});
|
99
|
+
|
100
|
+
window.Decidim.config.set({
|
101
|
+
messages: {
|
102
|
+
forms: {
|
103
|
+
"correct_errors": "Custom error message"
|
104
|
+
}
|
105
|
+
}
|
106
|
+
})
|
107
|
+
|
108
|
+
validatorInstance = new FormValidator(formElement);
|
109
|
+
validatorInstance.announceFormErrorForScreenReader();
|
110
|
+
|
111
|
+
setTimeout(() => {
|
112
|
+
const announceElement = formElement.querySelector(".sr-announce");
|
113
|
+
expect(announceElement.textContent).toBe("Custom error message");
|
114
|
+
}, 150);
|
115
|
+
});
|
116
|
+
});
|
117
|
+
|
118
|
+
describe("Form Validation", () => {
|
119
|
+
beforeEach(() => {
|
120
|
+
validatorInstance = new FormValidator(formElement);
|
121
|
+
});
|
122
|
+
|
123
|
+
it("should validate required text input", () => {
|
124
|
+
const textInput = formElement.querySelector("#test-input");
|
125
|
+
|
126
|
+
// Test empty value (should be invalid)
|
127
|
+
textInput.value = "";
|
128
|
+
const isValidEmpty = validatorInstance.validateSingleInput(textInput);
|
129
|
+
expect(isValidEmpty).toBe(false);
|
130
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(true);
|
131
|
+
|
132
|
+
// Test with value (should be valid)
|
133
|
+
textInput.value = "test value";
|
134
|
+
const isValidFilled = validatorInstance.validateSingleInput(textInput);
|
135
|
+
expect(isValidFilled).toBe(true);
|
136
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(false);
|
137
|
+
});
|
138
|
+
|
139
|
+
it("should validate email input pattern", () => {
|
140
|
+
const emailInput = formElement.querySelector("#email-input");
|
141
|
+
|
142
|
+
// Test invalid email
|
143
|
+
emailInput.value = "invalid-email";
|
144
|
+
const isInvalidEmail = validatorInstance.validateSingleInput(emailInput);
|
145
|
+
expect(isInvalidEmail).toBe(false);
|
146
|
+
|
147
|
+
// Test valid email
|
148
|
+
emailInput.value = "test@example.com";
|
149
|
+
const isValidEmail = validatorInstance.validateSingleInput(emailInput);
|
150
|
+
expect(isValidEmail).toBe(true);
|
151
|
+
});
|
152
|
+
|
153
|
+
it("should validate entire form", () => {
|
154
|
+
const textInput = formElement.querySelector("#test-input");
|
155
|
+
const emailInput = formElement.querySelector("#email-input");
|
156
|
+
|
157
|
+
// Empty form should be invalid
|
158
|
+
const isFormValidEmpty = validatorInstance.validateEntireForm();
|
159
|
+
expect(isFormValidEmpty).toBe(false);
|
160
|
+
|
161
|
+
// Fill with valid data
|
162
|
+
textInput.value = "test value";
|
163
|
+
emailInput.value = "test@example.com";
|
164
|
+
|
165
|
+
const isFormValidFilled = validatorInstance.validateEntireForm();
|
166
|
+
expect(isFormValidFilled).toBe(true);
|
167
|
+
});
|
168
|
+
});
|
169
|
+
|
170
|
+
describe("Radio Button Validation", () => {
|
171
|
+
beforeEach(() => {
|
172
|
+
formElement.innerHTML = `
|
173
|
+
<fieldset>
|
174
|
+
<legend>Choose Option</legend>
|
175
|
+
<input type="radio" id="option-1" name="option" value="1" required>
|
176
|
+
<label for="option-1">Option 1</label>
|
177
|
+
<input type="radio" id="option-2" name="option" value="2" required>
|
178
|
+
<label for="option-2">Option 2</label>
|
179
|
+
<span class="form-error">Please select an option</span>
|
180
|
+
</fieldset>
|
181
|
+
<button type="submit">Submit</button>
|
182
|
+
`;
|
183
|
+
validatorInstance = new FormValidator(formElement);
|
184
|
+
});
|
185
|
+
|
186
|
+
it("should validate radio button group", () => {
|
187
|
+
const radio1 = formElement.querySelector("#option-1");
|
188
|
+
|
189
|
+
// No selection should be invalid
|
190
|
+
const isValidEmpty = validatorInstance.validateRadioGroup("option");
|
191
|
+
expect(isValidEmpty).toBe(false);
|
192
|
+
|
193
|
+
// With selection should be valid
|
194
|
+
radio1.checked = true;
|
195
|
+
const isValidChecked = validatorInstance.validateRadioGroup("option");
|
196
|
+
expect(isValidChecked).toBe(true);
|
197
|
+
});
|
198
|
+
});
|
199
|
+
|
200
|
+
describe("Checkbox Validation", () => {
|
201
|
+
beforeEach(() => {
|
202
|
+
formElement.innerHTML = `
|
203
|
+
<fieldset>
|
204
|
+
<legend>Select Options</legend>
|
205
|
+
<input type="checkbox" id="checkbox-1" name="checkboxes" value="1" required>
|
206
|
+
<label for="checkbox-1">Checkbox 1</label>
|
207
|
+
<input type="checkbox" id="checkbox-2" name="checkboxes" value="2" required>
|
208
|
+
<label for="checkbox-2">Checkbox 2</label>
|
209
|
+
<span class="form-error">Please select at least one option</span>
|
210
|
+
</fieldset>
|
211
|
+
<button type="submit">Submit</button>
|
212
|
+
`;
|
213
|
+
validatorInstance = new FormValidator(formElement);
|
214
|
+
});
|
215
|
+
|
216
|
+
it("should validate checkbox group", () => {
|
217
|
+
const checkbox1 = formElement.querySelector("#checkbox-1");
|
218
|
+
|
219
|
+
// No selection should be invalid
|
220
|
+
const isValidEmpty = validatorInstance.validateCheckboxGroup("checkboxes");
|
221
|
+
expect(isValidEmpty).toBe(false);
|
222
|
+
|
223
|
+
// With selection should be valid
|
224
|
+
checkbox1.checked = true;
|
225
|
+
const isValidChecked = validatorInstance.validateCheckboxGroup("checkboxes");
|
226
|
+
expect(isValidChecked).toBe(true);
|
227
|
+
});
|
228
|
+
|
229
|
+
it("should validate minimum required checkboxes", () => {
|
230
|
+
formElement.innerHTML = `
|
231
|
+
<fieldset>
|
232
|
+
<input type="checkbox" id="multi-1" name="multiCheckboxes" value="1" required data-min-required="2">
|
233
|
+
<label for="multi-1">Multi 1</label>
|
234
|
+
<input type="checkbox" id="multi-2" name="multiCheckboxes" value="2" required data-min-required="2">
|
235
|
+
<label for="multi-2">Multi 2</label>
|
236
|
+
<input type="checkbox" id="multi-3" name="multiCheckboxes" value="3" required data-min-required="2">
|
237
|
+
<label for="multi-3">Multi 3</label>
|
238
|
+
</fieldset>
|
239
|
+
`;
|
240
|
+
validatorInstance = new FormValidator(formElement);
|
241
|
+
|
242
|
+
const checkbox1 = formElement.querySelector("#multi-1");
|
243
|
+
const checkbox2 = formElement.querySelector("#multi-2");
|
244
|
+
|
245
|
+
validatorInstance.validateEntireForm();
|
246
|
+
// One checkbox should be invalid (minimum 2 required)
|
247
|
+
checkbox1.checked = true;
|
248
|
+
const isValidOne = validatorInstance.validateCheckboxGroup("multiCheckboxes");
|
249
|
+
expect(isValidOne).toBe(false);
|
250
|
+
|
251
|
+
// Two checkboxes should be valid
|
252
|
+
checkbox2.checked = true;
|
253
|
+
const isValidTwo = validatorInstance.validateCheckboxGroup("multiCheckboxes");
|
254
|
+
expect(isValidTwo).toBe(true);
|
255
|
+
});
|
256
|
+
});
|
257
|
+
|
258
|
+
describe("Custom Validators", () => {
|
259
|
+
beforeEach(() => {
|
260
|
+
formElement.innerHTML = `
|
261
|
+
<div>
|
262
|
+
<label for="password">Password</label>
|
263
|
+
<input type="password" id="password" name="password" required>
|
264
|
+
</div>
|
265
|
+
<div>
|
266
|
+
<label for="confirm-password">Confirm Password</label>
|
267
|
+
<input type="password" id="confirm-password" name="confirmPassword" required data-equalto="password">
|
268
|
+
<span class="form-error">Passwords must match</span>
|
269
|
+
</div>
|
270
|
+
<button type="submit">Submit</button>
|
271
|
+
`;
|
272
|
+
validatorInstance = new FormValidator(formElement);
|
273
|
+
});
|
274
|
+
|
275
|
+
it("should validate equalTo validator", () => {
|
276
|
+
const passwordInput = formElement.querySelector("#password");
|
277
|
+
const confirmPasswordInput = formElement.querySelector("#confirm-password");
|
278
|
+
|
279
|
+
passwordInput.value = "password123";
|
280
|
+
confirmPasswordInput.value = "different";
|
281
|
+
|
282
|
+
// Passwords do not match
|
283
|
+
const isValidDifferent = validatorInstance.validateSingleInput(confirmPasswordInput);
|
284
|
+
expect(isValidDifferent).toBe(false);
|
285
|
+
|
286
|
+
// Passwords match
|
287
|
+
confirmPasswordInput.value = "password123";
|
288
|
+
const isValidMatching = validatorInstance.validateSingleInput(confirmPasswordInput);
|
289
|
+
expect(isValidMatching).toBe(true);
|
290
|
+
});
|
291
|
+
});
|
292
|
+
|
293
|
+
describe("Pattern Validation", () => {
|
294
|
+
beforeEach(() => {
|
295
|
+
formElement.innerHTML = `
|
296
|
+
<div>
|
297
|
+
<label for="alpha-input">Alpha Only</label>
|
298
|
+
<input type="text" id="alpha-input" name="alphaInput" data-pattern="alpha" required>
|
299
|
+
<span class="form-error">Only letters allowed</span>
|
300
|
+
</div>
|
301
|
+
<div>
|
302
|
+
<label for="number-input">Number Only</label>
|
303
|
+
<input type="text" id="number-input" name="numberInput" data-pattern="integer" required>
|
304
|
+
<span class="form-error">Only numbers allowed</span>
|
305
|
+
</div>
|
306
|
+
<button type="submit">Submit</button>
|
307
|
+
`;
|
308
|
+
validatorInstance = new FormValidator(formElement);
|
309
|
+
});
|
310
|
+
|
311
|
+
it("should validate alpha pattern", () => {
|
312
|
+
const alphaInput = formElement.querySelector("#alpha-input");
|
313
|
+
|
314
|
+
// Invalid: contains numbers
|
315
|
+
alphaInput.value = "abc123";
|
316
|
+
const isValidNumbers = validatorInstance.validateTextInput(alphaInput);
|
317
|
+
expect(isValidNumbers).toBe(false);
|
318
|
+
|
319
|
+
// Valid: only letters
|
320
|
+
alphaInput.value = "abcDEF";
|
321
|
+
const isValidLetters = validatorInstance.validateTextInput(alphaInput);
|
322
|
+
expect(isValidLetters).toBe(true);
|
323
|
+
});
|
324
|
+
|
325
|
+
it("should validate integer pattern", () => {
|
326
|
+
const numberInput = formElement.querySelector("#number-input");
|
327
|
+
|
328
|
+
// Invalid: contains letters
|
329
|
+
numberInput.value = "123abc";
|
330
|
+
const isValidLetters = validatorInstance.validateTextInput(numberInput);
|
331
|
+
expect(isValidLetters).toBe(false);
|
332
|
+
|
333
|
+
// Valid: only numbers
|
334
|
+
numberInput.value = "12345";
|
335
|
+
const isValidNumbers = validatorInstance.validateTextInput(numberInput);
|
336
|
+
expect(isValidNumbers).toBe(true);
|
337
|
+
});
|
338
|
+
});
|
339
|
+
|
340
|
+
describe("Error Class Management", () => {
|
341
|
+
beforeEach(() => {
|
342
|
+
validatorInstance = new FormValidator(formElement);
|
343
|
+
});
|
344
|
+
|
345
|
+
it("should add error classes to invalid inputs", () => {
|
346
|
+
const textInput = formElement.querySelector("#test-input");
|
347
|
+
const labelElement = formElement.querySelector('label[for="test-input"]');
|
348
|
+
|
349
|
+
textInput.value = "";
|
350
|
+
validatorInstance.validateSingleInput(textInput);
|
351
|
+
|
352
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(true);
|
353
|
+
expect(textInput.getAttribute("aria-invalid")).toBe("true");
|
354
|
+
expect(labelElement.classList.contains("is-invalid-label")).toBe(true);
|
355
|
+
});
|
356
|
+
|
357
|
+
it("should remove error classes from valid inputs", () => {
|
358
|
+
const textInput = formElement.querySelector("#test-input");
|
359
|
+
const labelElement = formElement.querySelector('label[for="test-input"]');
|
360
|
+
|
361
|
+
// First make it invalid
|
362
|
+
textInput.value = "";
|
363
|
+
validatorInstance.validateSingleInput(textInput);
|
364
|
+
|
365
|
+
// Then make it valid
|
366
|
+
textInput.value = "valid text";
|
367
|
+
validatorInstance.validateSingleInput(textInput);
|
368
|
+
|
369
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(false);
|
370
|
+
expect(textInput.hasAttribute("aria-invalid")).toBe(false);
|
371
|
+
expect(labelElement.classList.contains("is-invalid-label")).toBe(false);
|
372
|
+
});
|
373
|
+
});
|
374
|
+
|
375
|
+
describe("Accessibility Features", () => {
|
376
|
+
beforeEach(() => {
|
377
|
+
validatorInstance = new FormValidator(formElement);
|
378
|
+
});
|
379
|
+
|
380
|
+
it("should add accessibility attributes to inputs", () => {
|
381
|
+
const textInput = formElement.querySelector("#test-input");
|
382
|
+
const errorElement = formElement.querySelector(".form-error");
|
383
|
+
|
384
|
+
validatorInstance.addAccessibilityAttributes(textInput);
|
385
|
+
|
386
|
+
expect(errorElement.getAttribute("role")).toBe("alert");
|
387
|
+
});
|
388
|
+
|
389
|
+
it("should announce form errors for screen readers", () => {
|
390
|
+
validatorInstance.announceFormErrorForScreenReader();
|
391
|
+
|
392
|
+
const announceElement = formElement.querySelector(".sr-announce");
|
393
|
+
expect(announceElement).toBeTruthy();
|
394
|
+
expect(announceElement.getAttribute("aria-live")).toBe("assertive");
|
395
|
+
expect(announceElement.classList.contains("sr-only")).toBe(true);
|
396
|
+
});
|
397
|
+
|
398
|
+
it("should link inputs to error messages with aria-describedby", () => {
|
399
|
+
const textInput = formElement.querySelector("#test-input");
|
400
|
+
const errorElement = formElement.querySelector(".form-error");
|
401
|
+
|
402
|
+
validatorInstance.addAccessibilityErrorDescribe(textInput, errorElement);
|
403
|
+
|
404
|
+
expect(errorElement.id).toBeTruthy();
|
405
|
+
expect(textInput.getAttribute("aria-describedby")).toBe(errorElement.id);
|
406
|
+
});
|
407
|
+
});
|
408
|
+
|
409
|
+
describe("Form Submission Handling", () => {
|
410
|
+
beforeEach(() => {
|
411
|
+
validatorInstance = new FormValidator(formElement);
|
412
|
+
});
|
413
|
+
|
414
|
+
it("should prevent submission of invalid form", () => {
|
415
|
+
const submitEvent = new Event("submit", { cancelable: true });
|
416
|
+
const preventDefault = jest.spyOn(submitEvent, "preventDefault");
|
417
|
+
|
418
|
+
// Empty form should be invalid
|
419
|
+
formElement.dispatchEvent(submitEvent);
|
420
|
+
|
421
|
+
expect(preventDefault).toHaveBeenCalled();
|
422
|
+
});
|
423
|
+
|
424
|
+
it("should allow submission of valid form", () => {
|
425
|
+
const textInput = formElement.querySelector("#test-input");
|
426
|
+
const emailInput = formElement.querySelector("#email-input");
|
427
|
+
const submitEvent = new Event("submit", { cancelable: true });
|
428
|
+
const preventDefault = jest.spyOn(submitEvent, "preventDefault");
|
429
|
+
|
430
|
+
// Fill form with valid data
|
431
|
+
textInput.value = "test value";
|
432
|
+
emailInput.value = "test@example.com";
|
433
|
+
|
434
|
+
formElement.dispatchEvent(submitEvent);
|
435
|
+
|
436
|
+
expect(preventDefault).not.toHaveBeenCalled();
|
437
|
+
});
|
438
|
+
|
439
|
+
it("should respect formnovalidate attribute", () => {
|
440
|
+
const submitButton = formElement.querySelector('button[type="submit"]');
|
441
|
+
submitButton.setAttribute("formnovalidate", "");
|
442
|
+
|
443
|
+
const submitEvent = new Event("submit", { cancelable: true });
|
444
|
+
const preventDefault = jest.spyOn(submitEvent, "preventDefault");
|
445
|
+
|
446
|
+
// Click submit button to set formnovalidate flag
|
447
|
+
submitButton.click();
|
448
|
+
|
449
|
+
// Empty form should still be allowed with formnovalidate
|
450
|
+
formElement.dispatchEvent(submitEvent);
|
451
|
+
|
452
|
+
expect(preventDefault).not.toHaveBeenCalled();
|
453
|
+
});
|
454
|
+
});
|
455
|
+
|
456
|
+
describe("Event Handling", () => {
|
457
|
+
beforeEach(() => {
|
458
|
+
validatorInstance = new FormValidator(formElement);
|
459
|
+
});
|
460
|
+
|
461
|
+
it("should trigger validation events", () => {
|
462
|
+
const textInput = formElement.querySelector("#test-input");
|
463
|
+
let validEventFired = false;
|
464
|
+
let invalidEventFired = false;
|
465
|
+
|
466
|
+
textInput.addEventListener("valid.formvalidator", () => {
|
467
|
+
validEventFired = true;
|
468
|
+
});
|
469
|
+
|
470
|
+
textInput.addEventListener("invalid.formvalidator", () => {
|
471
|
+
invalidEventFired = true;
|
472
|
+
});
|
473
|
+
|
474
|
+
// Test invalid input
|
475
|
+
textInput.value = "";
|
476
|
+
validatorInstance.validateSingleInput(textInput);
|
477
|
+
expect(invalidEventFired).toBe(true);
|
478
|
+
|
479
|
+
// Test valid input
|
480
|
+
textInput.value = "valid text";
|
481
|
+
validatorInstance.validateSingleInput(textInput);
|
482
|
+
expect(validEventFired).toBe(true);
|
483
|
+
});
|
484
|
+
|
485
|
+
it("should trigger form validation events", () => {
|
486
|
+
let formValidEventFired = false;
|
487
|
+
let formInvalidEventFired = false;
|
488
|
+
|
489
|
+
formElement.addEventListener("formvalid.formvalidator", () => {
|
490
|
+
formValidEventFired = true;
|
491
|
+
});
|
492
|
+
|
493
|
+
formElement.addEventListener("forminvalid.formvalidator", () => {
|
494
|
+
formInvalidEventFired = true;
|
495
|
+
});
|
496
|
+
|
497
|
+
// Test invalid form
|
498
|
+
validatorInstance.validateEntireForm();
|
499
|
+
expect(formInvalidEventFired).toBe(true);
|
500
|
+
|
501
|
+
// Test valid form
|
502
|
+
const textInput = formElement.querySelector("#test-input");
|
503
|
+
const emailInput = formElement.querySelector("#email-input");
|
504
|
+
textInput.value = "test value";
|
505
|
+
emailInput.value = "test@example.com";
|
506
|
+
|
507
|
+
validatorInstance.validateEntireForm();
|
508
|
+
expect(formValidEventFired).toBe(true);
|
509
|
+
});
|
510
|
+
});
|
511
|
+
|
512
|
+
describe("Form Reset Functionality", () => {
|
513
|
+
beforeEach(() => {
|
514
|
+
validatorInstance = new FormValidator(formElement);
|
515
|
+
});
|
516
|
+
|
517
|
+
it("should reset form validation state", () => {
|
518
|
+
const textInput = formElement.querySelector("#test-input");
|
519
|
+
|
520
|
+
// Make input invalid first
|
521
|
+
textInput.value = "";
|
522
|
+
validatorInstance.validateSingleInput(textInput);
|
523
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(true);
|
524
|
+
|
525
|
+
// Reset validation
|
526
|
+
validatorInstance.resetFormValidation();
|
527
|
+
|
528
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(false);
|
529
|
+
expect(textInput.value).toBe("");
|
530
|
+
expect(textInput.hasAttribute("aria-invalid")).toBe(false);
|
531
|
+
});
|
532
|
+
|
533
|
+
it("should trigger reset event", () => {
|
534
|
+
let resetEventFired = false;
|
535
|
+
|
536
|
+
formElement.addEventListener("formreset.formvalidator", () => {
|
537
|
+
resetEventFired = true;
|
538
|
+
});
|
539
|
+
|
540
|
+
validatorInstance.resetFormValidation();
|
541
|
+
expect(resetEventFired).toBe(true);
|
542
|
+
});
|
543
|
+
});
|
544
|
+
|
545
|
+
describe("Validation Control", () => {
|
546
|
+
beforeEach(() => {
|
547
|
+
validatorInstance = new FormValidator(formElement);
|
548
|
+
});
|
549
|
+
|
550
|
+
it("should enable and disable validation", () => {
|
551
|
+
validatorInstance.disableValidation();
|
552
|
+
expect(validatorInstance.isValidationCurrentlyDisabled()).toBe(true);
|
553
|
+
|
554
|
+
validatorInstance.enableValidation();
|
555
|
+
expect(validatorInstance.isValidationCurrentlyDisabled()).toBe(false);
|
556
|
+
});
|
557
|
+
|
558
|
+
it("should skip validation for disabled inputs", () => {
|
559
|
+
const textInput = formElement.querySelector("#test-input");
|
560
|
+
textInput.disabled = true;
|
561
|
+
|
562
|
+
const shouldSkip = validatorInstance.shouldSkipInputValidation(textInput);
|
563
|
+
expect(shouldSkip).toBe(true);
|
564
|
+
});
|
565
|
+
|
566
|
+
it("should skip validation for inputs with ignore attribute", () => {
|
567
|
+
const textInput = formElement.querySelector("#test-input");
|
568
|
+
textInput.setAttribute("data-validator-ignore", "");
|
569
|
+
|
570
|
+
const shouldSkip = validatorInstance.shouldSkipInputValidation(textInput);
|
571
|
+
expect(shouldSkip).toBe(true);
|
572
|
+
});
|
573
|
+
});
|
574
|
+
|
575
|
+
describe("Legacy Compatibility", () => {
|
576
|
+
beforeEach(() => {
|
577
|
+
validatorInstance = new FormValidator(formElement);
|
578
|
+
});
|
579
|
+
|
580
|
+
it("should handle legacy form-error.decidim event", () => {
|
581
|
+
const mockAnnounce = jest.spyOn(validatorInstance, "announceFormErrorForScreenReader");
|
582
|
+
|
583
|
+
formElement.dispatchEvent(new CustomEvent("form-error.decidim"));
|
584
|
+
|
585
|
+
expect(mockAnnounce).toHaveBeenCalled();
|
586
|
+
});
|
587
|
+
|
588
|
+
it("should trigger legacy event on form validation failure", () => {
|
589
|
+
let legacyEventFired = false;
|
590
|
+
|
591
|
+
formElement.addEventListener("form-error.decidim", () => {
|
592
|
+
legacyEventFired = true;
|
593
|
+
});
|
594
|
+
|
595
|
+
// Trigger form invalid event which should trigger legacy event
|
596
|
+
formElement.dispatchEvent(new CustomEvent("forminvalid.formvalidator"));
|
597
|
+
|
598
|
+
expect(legacyEventFired).toBe(true);
|
599
|
+
});
|
600
|
+
});
|
601
|
+
|
602
|
+
describe("Utility Methods", () => {
|
603
|
+
beforeEach(() => {
|
604
|
+
validatorInstance = new FormValidator(formElement);
|
605
|
+
});
|
606
|
+
|
607
|
+
it("should generate unique identifiers", () => {
|
608
|
+
const identifier1 = validatorInstance.generateUniqueIdentifier("test");
|
609
|
+
const identifier2 = validatorInstance.generateUniqueIdentifier("test");
|
610
|
+
|
611
|
+
expect(identifier1).toMatch(/^test-/);
|
612
|
+
expect(identifier2).toMatch(/^test-/);
|
613
|
+
expect(identifier1).not.toBe(identifier2);
|
614
|
+
});
|
615
|
+
|
616
|
+
it("should find input labels correctly", () => {
|
617
|
+
const textInput = formElement.querySelector("#test-input");
|
618
|
+
const labelElement = validatorInstance.findInputLabel(textInput);
|
619
|
+
|
620
|
+
expect(labelElement.getAttribute("for")).toBe("test-input");
|
621
|
+
});
|
622
|
+
|
623
|
+
it("should find form error elements", () => {
|
624
|
+
const textInput = formElement.querySelector("#test-input");
|
625
|
+
const errorElements = validatorInstance.findFormErrorElements(textInput);
|
626
|
+
|
627
|
+
expect(errorElements.length).toBeGreaterThan(0);
|
628
|
+
expect(errorElements[0].classList.contains("form-error")).toBe(true);
|
629
|
+
});
|
630
|
+
});
|
631
|
+
|
632
|
+
describe("Destroy Validator", () => {
|
633
|
+
beforeEach(() => {
|
634
|
+
validatorInstance = new FormValidator(formElement);
|
635
|
+
});
|
636
|
+
|
637
|
+
it("should clean up validator instance", () => {
|
638
|
+
const textInput = formElement.querySelector("#test-input");
|
639
|
+
|
640
|
+
// Make input invalid
|
641
|
+
textInput.value = "";
|
642
|
+
validatorInstance.validateSingleInput(textInput);
|
643
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(true);
|
644
|
+
|
645
|
+
// Destroy validator
|
646
|
+
validatorInstance.destroyValidator();
|
647
|
+
|
648
|
+
expect(textInput.classList.contains("is-invalid-input")).toBe(false);
|
649
|
+
});
|
650
|
+
});
|
651
|
+
|
652
|
+
describe("Complex Form Scenarios", () => {
|
653
|
+
it("should handle forms with mixed input types", () => {
|
654
|
+
formElement.innerHTML = `
|
655
|
+
<input type="text" name="text" required>
|
656
|
+
<input type="email" name="email" required>
|
657
|
+
<input type="radio" name="radio" value="1" required>
|
658
|
+
<input type="radio" name="radio" value="2" required>
|
659
|
+
<input type="checkbox" name="checkbox" required>
|
660
|
+
<select name="select" required>
|
661
|
+
<option value="">Choose...</option>
|
662
|
+
<option value="1">Option 1</option>
|
663
|
+
</select>
|
664
|
+
<textarea name="textarea" required></textarea>
|
665
|
+
<button type="submit">Submit</button>
|
666
|
+
`;
|
667
|
+
|
668
|
+
validatorInstance = new FormValidator(formElement);
|
669
|
+
|
670
|
+
// Initially invalid
|
671
|
+
expect(validatorInstance.validateEntireForm()).toBe(false);
|
672
|
+
|
673
|
+
// Fill all fields
|
674
|
+
formElement.querySelector('input[name="text"]').value = "text";
|
675
|
+
formElement.querySelector('input[name="email"]').value = "test@example.com";
|
676
|
+
formElement.querySelector('input[name="radio"][value="1"]').checked = true;
|
677
|
+
formElement.querySelector('input[name="checkbox"]').checked = true;
|
678
|
+
formElement.querySelector('select[name="select"]').value = "1";
|
679
|
+
formElement.querySelector('textarea[name="textarea"]').value = "textarea content";
|
680
|
+
|
681
|
+
// Should now be valid
|
682
|
+
expect(validatorInstance.validateEntireForm()).toBe(true);
|
683
|
+
});
|
684
|
+
});
|
685
|
+
});
|