threddedDANIEL 0.14.5
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/MIT-LICENSE +20 -0
- data/README.md +744 -0
- data/app/assets/images/favicons/README.md +3 -0
- data/app/assets/images/favicons/amazon.png +0 -0
- data/app/assets/images/favicons/github.png +0 -0
- data/app/assets/images/favicons/google_branding/logo_calendar_128px.png +0 -0
- data/app/assets/images/favicons/google_branding/logo_docs_48px.png +0 -0
- data/app/assets/images/favicons/google_branding/logo_drive_48px.png +0 -0
- data/app/assets/images/favicons/google_branding/logo_forms_48px.png +0 -0
- data/app/assets/images/favicons/google_branding/logo_sheets_48px.png +0 -0
- data/app/assets/images/favicons/google_branding/logo_slides_48px.png +0 -0
- data/app/assets/images/favicons/stackexchange.png +0 -0
- data/app/assets/images/favicons/twitter.png +0 -0
- data/app/assets/images/favicons/wikipedia.png +0 -0
- data/app/assets/images/thredded/breadcrumb-chevron.svg +1 -0
- data/app/assets/images/thredded/follow.svg +1 -0
- data/app/assets/images/thredded/lock.svg +1 -0
- data/app/assets/images/thredded/moderation.svg +1 -0
- data/app/assets/images/thredded/private-messages.svg +1 -0
- data/app/assets/images/thredded/settings.svg +1 -0
- data/app/assets/images/thredded/three-dot-menu.svg +3 -0
- data/app/assets/images/thredded/unfollow.svg +1 -0
- data/app/assets/javascripts/thredded.es6 +2 -0
- data/app/assets/javascripts/thredded/components/currently_online.es6 +32 -0
- data/app/assets/javascripts/thredded/components/flash_messages.es6 +9 -0
- data/app/assets/javascripts/thredded/components/mention_autocompletion.es6 +40 -0
- data/app/assets/javascripts/thredded/components/post_form.es6 +37 -0
- data/app/assets/javascripts/thredded/components/preview_area.es6 +56 -0
- data/app/assets/javascripts/thredded/components/quote_post.es6 +49 -0
- data/app/assets/javascripts/thredded/components/time_stamps.es6 +25 -0
- data/app/assets/javascripts/thredded/components/topic_form.es6 +94 -0
- data/app/assets/javascripts/thredded/components/topics.es6 +46 -0
- data/app/assets/javascripts/thredded/components/turboforms.es6 +25 -0
- data/app/assets/javascripts/thredded/components/user_preferences_form.es6 +66 -0
- data/app/assets/javascripts/thredded/components/user_textcomplete.es6 +50 -0
- data/app/assets/javascripts/thredded/components/users_select.es6 +122 -0
- data/app/assets/javascripts/thredded/core/csrf_tokens.es6 +9 -0
- data/app/assets/javascripts/thredded/core/debounce.es6 +32 -0
- data/app/assets/javascripts/thredded/core/escape_html.es6 +7 -0
- data/app/assets/javascripts/thredded/core/hide_soft_keyboard.es6 +7 -0
- data/app/assets/javascripts/thredded/core/on_page_load.es6 +54 -0
- data/app/assets/javascripts/thredded/core/serialize_form.es6 +9 -0
- data/app/assets/javascripts/thredded/core/thredded.es6 +1 -0
- data/app/assets/javascripts/thredded/dependencies.js +8 -0
- data/app/assets/javascripts/thredded/dependencies/autosize.js +1 -0
- data/app/assets/javascripts/thredded/dependencies/textcomplete.js +1 -0
- data/app/assets/javascripts/thredded/dependencies/timeago.js +1 -0
- data/app/assets/javascripts/thredded/dependencies/ujs.js +1 -0
- data/app/assets/javascripts/thredded/thredded.es6 +2 -0
- data/app/assets/stylesheets/thredded.scss +3 -0
- data/app/assets/stylesheets/thredded/_base.scss +13 -0
- data/app/assets/stylesheets/thredded/_dependencies.scss +0 -0
- data/app/assets/stylesheets/thredded/_email.scss +51 -0
- data/app/assets/stylesheets/thredded/_thredded.scss +33 -0
- data/app/assets/stylesheets/thredded/base/_alerts.scss +44 -0
- data/app/assets/stylesheets/thredded/base/_buttons.scss +73 -0
- data/app/assets/stylesheets/thredded/base/_dropdown-menu.scss +83 -0
- data/app/assets/stylesheets/thredded/base/_forms.scss +84 -0
- data/app/assets/stylesheets/thredded/base/_grid.scss +58 -0
- data/app/assets/stylesheets/thredded/base/_lists.scss +31 -0
- data/app/assets/stylesheets/thredded/base/_nav.scss +67 -0
- data/app/assets/stylesheets/thredded/base/_tables.scss +17 -0
- data/app/assets/stylesheets/thredded/base/_typography.scss +72 -0
- data/app/assets/stylesheets/thredded/base/_variables.scss +141 -0
- data/app/assets/stylesheets/thredded/components/_alerts.scss +19 -0
- data/app/assets/stylesheets/thredded/components/_base.scss +25 -0
- data/app/assets/stylesheets/thredded/components/_currently-online.scss +54 -0
- data/app/assets/stylesheets/thredded/components/_empty.scss +11 -0
- data/app/assets/stylesheets/thredded/components/_flash-message.scss +19 -0
- data/app/assets/stylesheets/thredded/components/_following.scss +14 -0
- data/app/assets/stylesheets/thredded/components/_form-list.scss +39 -0
- data/app/assets/stylesheets/thredded/components/_icons.scss +3 -0
- data/app/assets/stylesheets/thredded/components/_main-section.scss +9 -0
- data/app/assets/stylesheets/thredded/components/_mention-autocomplete.scss +57 -0
- data/app/assets/stylesheets/thredded/components/_messageboard.scss +155 -0
- data/app/assets/stylesheets/thredded/components/_onebox.scss +275 -0
- data/app/assets/stylesheets/thredded/components/_pagination.scss +35 -0
- data/app/assets/stylesheets/thredded/components/_post-form.scss +27 -0
- data/app/assets/stylesheets/thredded/components/_post.scss +138 -0
- data/app/assets/stylesheets/thredded/components/_preferences.scss +19 -0
- data/app/assets/stylesheets/thredded/components/_preview_area.scss +11 -0
- data/app/assets/stylesheets/thredded/components/_topic-delete.scss +8 -0
- data/app/assets/stylesheets/thredded/components/_topic-header.scss +111 -0
- data/app/assets/stylesheets/thredded/components/_topics.scss +186 -0
- data/app/assets/stylesheets/thredded/layout/_main-container.scss +16 -0
- data/app/assets/stylesheets/thredded/layout/_main-navigation.scss +79 -0
- data/app/assets/stylesheets/thredded/layout/_moderation.scss +97 -0
- data/app/assets/stylesheets/thredded/layout/_navigation.scss +87 -0
- data/app/assets/stylesheets/thredded/layout/_search-navigation.scss +126 -0
- data/app/assets/stylesheets/thredded/layout/_user-navigation.scss +20 -0
- data/app/assets/stylesheets/thredded/layout/_user.scss +10 -0
- data/app/assets/stylesheets/thredded/utilities/_is-compact.scss +26 -0
- data/app/assets/stylesheets/thredded/utilities/_is-expanded.scss +16 -0
- data/app/commands/thredded/at_notification_extractor.rb +23 -0
- data/app/commands/thredded/autofollow_users.rb +62 -0
- data/app/commands/thredded/create_messageboard.rb +42 -0
- data/app/commands/thredded/mark_all_read.rb +22 -0
- data/app/commands/thredded/moderate_post.rb +48 -0
- data/app/commands/thredded/notify_following_users.rb +78 -0
- data/app/commands/thredded/notify_private_topic_users.rb +34 -0
- data/app/controllers/concerns/thredded/new_post_params.rb +21 -0
- data/app/controllers/concerns/thredded/new_private_post_params.rb +21 -0
- data/app/controllers/concerns/thredded/new_private_topic_params.rb +25 -0
- data/app/controllers/concerns/thredded/new_topic_params.rb +19 -0
- data/app/controllers/concerns/thredded/render_preview.rb +16 -0
- data/app/controllers/thredded/application_controller.rb +145 -0
- data/app/controllers/thredded/autocomplete_users_controller.rb +52 -0
- data/app/controllers/thredded/messageboard_groups_controller.rb +31 -0
- data/app/controllers/thredded/messageboards_controller.rb +53 -0
- data/app/controllers/thredded/moderation_controller.rb +103 -0
- data/app/controllers/thredded/post_permalinks_controller.rb +11 -0
- data/app/controllers/thredded/post_previews_controller.rb +29 -0
- data/app/controllers/thredded/posts_controller.rb +97 -0
- data/app/controllers/thredded/preferences_controller.rb +42 -0
- data/app/controllers/thredded/private_post_permalinks_controller.rb +12 -0
- data/app/controllers/thredded/private_post_previews_controller.rb +29 -0
- data/app/controllers/thredded/private_posts_controller.rb +96 -0
- data/app/controllers/thredded/private_topic_previews_controller.rb +15 -0
- data/app/controllers/thredded/private_topics_controller.rb +102 -0
- data/app/controllers/thredded/read_states_controller.rb +13 -0
- data/app/controllers/thredded/theme_previews_controller.rb +34 -0
- data/app/controllers/thredded/topic_previews_controller.rb +15 -0
- data/app/controllers/thredded/topics_controller.rb +205 -0
- data/app/forms/thredded/edit_topic_form.rb +51 -0
- data/app/forms/thredded/post_form.rb +54 -0
- data/app/forms/thredded/private_post_form.rb +50 -0
- data/app/forms/thredded/private_topic_form.rb +160 -0
- data/app/forms/thredded/topic_form.rb +95 -0
- data/app/forms/thredded/user_preferences_form.rb +110 -0
- data/app/helpers/thredded/application_helper.rb +129 -0
- data/app/helpers/thredded/nav_helper.rb +42 -0
- data/app/helpers/thredded/render_helper.rb +15 -0
- data/app/helpers/thredded/urls_helper.rb +134 -0
- data/app/jobs/thredded/activity_updater_job.rb +21 -0
- data/app/jobs/thredded/auto_follow_and_notify_job.rb +14 -0
- data/app/jobs/thredded/notify_private_topic_users_job.rb +12 -0
- data/app/mailer_previews/thredded/base_mailer_preview.rb +118 -0
- data/app/mailer_previews/thredded/post_mailer_preview.rb +13 -0
- data/app/mailer_previews/thredded/private_topic_mailer_preview.rb +13 -0
- data/app/mailers/thredded/base_mailer.rb +18 -0
- data/app/mailers/thredded/post_mailer.rb +21 -0
- data/app/mailers/thredded/private_topic_mailer.rb +21 -0
- data/app/models/concerns/thredded/content_moderation_state.rb +67 -0
- data/app/models/concerns/thredded/friendly_id_reserved_words_and_pagination.rb +17 -0
- data/app/models/concerns/thredded/moderation_state.rb +14 -0
- data/app/models/concerns/thredded/notifier_preference.rb +19 -0
- data/app/models/concerns/thredded/post_common.rb +76 -0
- data/app/models/concerns/thredded/search_parser.rb +41 -0
- data/app/models/concerns/thredded/topic_common.rb +94 -0
- data/app/models/concerns/thredded/topics_search.rb +67 -0
- data/app/models/concerns/thredded/user_topic_read_state_common.rb +42 -0
- data/app/models/thredded/category.rb +18 -0
- data/app/models/thredded/messageboard.rb +120 -0
- data/app/models/thredded/messageboard_group.rb +19 -0
- data/app/models/thredded/messageboard_notifications_for_followed_topics.rb +30 -0
- data/app/models/thredded/messageboard_user.rb +14 -0
- data/app/models/thredded/notifications_for_followed_topics.rb +25 -0
- data/app/models/thredded/notifications_for_private_topics.rb +22 -0
- data/app/models/thredded/null_user.rb +51 -0
- data/app/models/thredded/null_user_topic_read_state.rb +17 -0
- data/app/models/thredded/post.rb +77 -0
- data/app/models/thredded/post_moderation_record.rb +56 -0
- data/app/models/thredded/private_post.rb +63 -0
- data/app/models/thredded/private_topic.rb +89 -0
- data/app/models/thredded/private_user.rb +8 -0
- data/app/models/thredded/stats.rb +25 -0
- data/app/models/thredded/topic.rb +179 -0
- data/app/models/thredded/topic_category.rb +8 -0
- data/app/models/thredded/user_detail.rb +27 -0
- data/app/models/thredded/user_extender.rb +75 -0
- data/app/models/thredded/user_messageboard_preference.rb +45 -0
- data/app/models/thredded/user_permissions/admin/if_admin_column_true.rb +14 -0
- data/app/models/thredded/user_permissions/admin/none.rb +14 -0
- data/app/models/thredded/user_permissions/message/readers_of_writeable_boards.rb +15 -0
- data/app/models/thredded/user_permissions/moderate/if_moderator_column_true.rb +16 -0
- data/app/models/thredded/user_permissions/moderate/none.rb +16 -0
- data/app/models/thredded/user_permissions/read/all.rb +27 -0
- data/app/models/thredded/user_permissions/write/all.rb +16 -0
- data/app/models/thredded/user_permissions/write/none.rb +16 -0
- data/app/models/thredded/user_post_notification.rb +29 -0
- data/app/models/thredded/user_preference.rb +26 -0
- data/app/models/thredded/user_private_topic_read_state.rb +13 -0
- data/app/models/thredded/user_topic_follow.rb +34 -0
- data/app/models/thredded/user_topic_read_state.rb +13 -0
- data/app/notifiers/thredded/email_notifier.rb +25 -0
- data/app/policies/thredded/messageboard_group_policy.rb +16 -0
- data/app/policies/thredded/messageboard_policy.rb +49 -0
- data/app/policies/thredded/post_policy.rb +64 -0
- data/app/policies/thredded/private_post_policy.rb +38 -0
- data/app/policies/thredded/private_topic_policy.rb +24 -0
- data/app/policies/thredded/topic_policy.rb +49 -0
- data/app/view_hooks/thredded/all_view_hooks.rb +125 -0
- data/app/view_models/thredded/base_topic_view.rb +43 -0
- data/app/view_models/thredded/messageboard_group_view.rb +27 -0
- data/app/view_models/thredded/post_view.rb +89 -0
- data/app/view_models/thredded/posts_page_view.rb +27 -0
- data/app/view_models/thredded/private_topic_view.rb +20 -0
- data/app/view_models/thredded/private_topics_page_view.rb +31 -0
- data/app/view_models/thredded/topic_email_view.rb +18 -0
- data/app/view_models/thredded/topic_posts_page_view.rb +17 -0
- data/app/view_models/thredded/topic_view.rb +68 -0
- data/app/view_models/thredded/topics_page_view.rb +38 -0
- data/app/views/layouts/thredded/application.html.erb +18 -0
- data/app/views/thredded/categories/_category.html.erb +1 -0
- data/app/views/thredded/error_pages/forbidden.html.erb +6 -0
- data/app/views/thredded/error_pages/not_found.html.erb +6 -0
- data/app/views/thredded/kaminari/_first_page.html.erb +11 -0
- data/app/views/thredded/kaminari/_gap.html.erb +8 -0
- data/app/views/thredded/kaminari/_last_page.html.erb +11 -0
- data/app/views/thredded/kaminari/_next_page.html.erb +11 -0
- data/app/views/thredded/kaminari/_page.html.erb +12 -0
- data/app/views/thredded/kaminari/_paginator.html.erb +23 -0
- data/app/views/thredded/kaminari/_prev_page.html.erb +11 -0
- data/app/views/thredded/messageboard_groups/new.html.erb +28 -0
- data/app/views/thredded/messageboards/_form.html.erb +30 -0
- data/app/views/thredded/messageboards/_messageboard.html.erb +20 -0
- data/app/views/thredded/messageboards/_messageboard_meta.html.erb +13 -0
- data/app/views/thredded/messageboards/edit.html.erb +15 -0
- data/app/views/thredded/messageboards/index.html.erb +34 -0
- data/app/views/thredded/messageboards/new.html.erb +15 -0
- data/app/views/thredded/moderation/_nav.html.erb +19 -0
- data/app/views/thredded/moderation/_post.html.erb +19 -0
- data/app/views/thredded/moderation/_post_moderation_actions.html.erb +12 -0
- data/app/views/thredded/moderation/_post_moderation_record.html.erb +46 -0
- data/app/views/thredded/moderation/_user_moderation_state.html.erb +3 -0
- data/app/views/thredded/moderation/_user_post.html.erb +12 -0
- data/app/views/thredded/moderation/_users_search_form.html.erb +13 -0
- data/app/views/thredded/moderation/activity.html.erb +20 -0
- data/app/views/thredded/moderation/history.html.erb +13 -0
- data/app/views/thredded/moderation/pending.html.erb +24 -0
- data/app/views/thredded/moderation/user.html.erb +54 -0
- data/app/views/thredded/moderation/users.html.erb +41 -0
- data/app/views/thredded/post_mailer/post_notification.html.erb +26 -0
- data/app/views/thredded/post_mailer/post_notification.text.erb +14 -0
- data/app/views/thredded/post_previews/preview.html.erb +1 -0
- data/app/views/thredded/post_previews/update.html.erb +1 -0
- data/app/views/thredded/posts/_content.html.erb +1 -0
- data/app/views/thredded/posts/_form.html.erb +5 -0
- data/app/views/thredded/posts/_post.html.erb +13 -0
- data/app/views/thredded/posts/_user.html.erb +3 -0
- data/app/views/thredded/posts/edit.html.erb +16 -0
- data/app/views/thredded/posts/new.html.erb +15 -0
- data/app/views/thredded/posts_common/_actions.html.erb +28 -0
- data/app/views/thredded/posts_common/_content.html.erb +3 -0
- data/app/views/thredded/posts_common/_form.html.erb +22 -0
- data/app/views/thredded/posts_common/_header.html.erb +8 -0
- data/app/views/thredded/posts_common/_header_with_topic.html.erb +15 -0
- data/app/views/thredded/posts_common/_header_with_user_and_topic.html.erb +18 -0
- data/app/views/thredded/posts_common/actions/_delete.html.erb +4 -0
- data/app/views/thredded/posts_common/actions/_edit.html.erb +3 -0
- data/app/views/thredded/posts_common/actions/_mark_as_unread.html.erb +2 -0
- data/app/views/thredded/posts_common/actions/_quote.html.erb +4 -0
- data/app/views/thredded/posts_common/form/_after_content.html.erb +8 -0
- data/app/views/thredded/posts_common/form/_before_content.html.erb +8 -0
- data/app/views/thredded/posts_common/form/_content.html.erb +7 -0
- data/app/views/thredded/posts_common/form/_content_field.html.erb +8 -0
- data/app/views/thredded/posts_common/form/_preview_area.html.erb +16 -0
- data/app/views/thredded/preferences/_form.html.erb +95 -0
- data/app/views/thredded/preferences/_messageboards_nav.html.erb +8 -0
- data/app/views/thredded/preferences/_messageboards_nav_item.html.erb +2 -0
- data/app/views/thredded/preferences/edit.html.erb +20 -0
- data/app/views/thredded/private_post_previews/preview.html.erb +1 -0
- data/app/views/thredded/private_post_previews/update.html.erb +1 -0
- data/app/views/thredded/private_posts/_content.html.erb +1 -0
- data/app/views/thredded/private_posts/_form.html.erb +6 -0
- data/app/views/thredded/private_posts/_private_post.html.erb +6 -0
- data/app/views/thredded/private_posts/edit.html.erb +16 -0
- data/app/views/thredded/private_posts/new.html.erb +11 -0
- data/app/views/thredded/private_topic_mailer/message_notification.html.erb +26 -0
- data/app/views/thredded/private_topic_mailer/message_notification.text.erb +15 -0
- data/app/views/thredded/private_topic_previews/preview.html.erb +1 -0
- data/app/views/thredded/private_topics/_breadcrumbs.html.erb +4 -0
- data/app/views/thredded/private_topics/_form.html.erb +39 -0
- data/app/views/thredded/private_topics/_header.html.erb +17 -0
- data/app/views/thredded/private_topics/_no_private_topics.html.erb +6 -0
- data/app/views/thredded/private_topics/_private_topic.html.erb +23 -0
- data/app/views/thredded/private_topics/edit.html.erb +35 -0
- data/app/views/thredded/private_topics/header/_participant.html.erb +1 -0
- data/app/views/thredded/private_topics/index.html.erb +32 -0
- data/app/views/thredded/private_topics/new.html.erb +11 -0
- data/app/views/thredded/private_topics/private_topic/_participant.html.erb +1 -0
- data/app/views/thredded/private_topics/show.html.erb +28 -0
- data/app/views/thredded/search/_form.html.erb +13 -0
- data/app/views/thredded/shared/_breadcrumbs.html.erb +6 -0
- data/app/views/thredded/shared/_content_moderation_blocked_state.html.erb +8 -0
- data/app/views/thredded/shared/_currently_online.html.erb +6 -0
- data/app/views/thredded/shared/_flash_messages.html.erb +7 -0
- data/app/views/thredded/shared/_header.html.erb +3 -0
- data/app/views/thredded/shared/_nav.html.erb +19 -0
- data/app/views/thredded/shared/_page.html.erb +15 -0
- data/app/views/thredded/shared/currently_online/_header.html.erb +5 -0
- data/app/views/thredded/shared/currently_online/_user_list.html.erb +3 -0
- data/app/views/thredded/shared/currently_online/_user_list_item.html.erb +6 -0
- data/app/views/thredded/shared/nav/_moderation.html.erb +14 -0
- data/app/views/thredded/shared/nav/_notification_preferences.html.erb +8 -0
- data/app/views/thredded/shared/nav/_private_topics.html.erb +12 -0
- data/app/views/thredded/shared/nav/_standalone.html.erb +12 -0
- data/app/views/thredded/shared/nav/_standalone_profile.html.erb +3 -0
- data/app/views/thredded/shared/preview.html.erb +10 -0
- data/app/views/thredded/theme_previews/_section_title.html.erb +3 -0
- data/app/views/thredded/theme_previews/show.html.erb +99 -0
- data/app/views/thredded/topic_previews/preview.html.erb +1 -0
- data/app/views/thredded/topics/_followers.html.erb +12 -0
- data/app/views/thredded/topics/_form.html.erb +32 -0
- data/app/views/thredded/topics/_header.html.erb +32 -0
- data/app/views/thredded/topics/_sticky_topics_divider.html.erb +1 -0
- data/app/views/thredded/topics/_topic.html.erb +47 -0
- data/app/views/thredded/topics/_topic_form_admin_options.html.erb +12 -0
- data/app/views/thredded/topics/edit.html.erb +50 -0
- data/app/views/thredded/topics/index.html.erb +35 -0
- data/app/views/thredded/topics/new.html.erb +12 -0
- data/app/views/thredded/topics/search.html.erb +39 -0
- data/app/views/thredded/topics/show.html.erb +46 -0
- data/app/views/thredded/users/_link.html.erb +14 -0
- data/app/views/thredded/users/_post.html.erb +6 -0
- data/app/views/thredded/users/_posts.html.erb +7 -0
- data/bin/rails +6 -0
- data/config/i18n-tasks.yml +16 -0
- data/config/locales/de.yml +257 -0
- data/config/locales/en.yml +253 -0
- data/config/locales/es.yml +257 -0
- data/config/locales/fr.yml +255 -0
- data/config/locales/it.yml +257 -0
- data/config/locales/pl.yml +257 -0
- data/config/locales/pt-BR.yml +258 -0
- data/config/locales/ru.yml +255 -0
- data/config/locales/zh-CN.yml +246 -0
- data/config/routes.rb +87 -0
- data/db/migrate/20160329231848_create_thredded.rb +257 -0
- data/db/seeds.rb +4 -0
- data/db/upgrade_migrations/20160611094616_upgrade_v0_5_to_v0_6.rb +28 -0
- data/db/upgrade_migrations/20160723012349_upgrade_v0_6_to_v0_7.rb +46 -0
- data/db/upgrade_migrations/20161019150201_upgrade_v0_7_to_v0_8.rb +34 -0
- data/db/upgrade_migrations/20161113161801_upgrade_v0_8_to_v0_9.rb +60 -0
- data/db/upgrade_migrations/20170125033319_upgrade_v0_9_to_v0_10.rb +36 -0
- data/db/upgrade_migrations/20170312131417_upgrade_thredded_v0_10_to_v0_11.rb +23 -0
- data/db/upgrade_migrations/20170420163138_upgrade_thredded_v0_11_to_v0_12.rb +28 -0
- data/db/upgrade_migrations/20170811090735_upgrade_thredded_v0_13_to_v_014.rb +21 -0
- data/lib/generators/thredded/install/USAGE +8 -0
- data/lib/generators/thredded/install/install_generator.rb +20 -0
- data/lib/generators/thredded/install/templates/initializer.rb +160 -0
- data/lib/tasks/thredded_tasks.rake +14 -0
- data/lib/thredded.rb +190 -0
- data/lib/thredded/base_migration.rb +14 -0
- data/lib/thredded/base_notifier.rb +28 -0
- data/lib/thredded/collection_to_strings_with_cache_renderer.rb +86 -0
- data/lib/thredded/content_formatter.rb +129 -0
- data/lib/thredded/database_seeder.rb +290 -0
- data/lib/thredded/db_tools.rb +103 -0
- data/lib/thredded/email_transformer.rb +22 -0
- data/lib/thredded/email_transformer/base.rb +47 -0
- data/lib/thredded/email_transformer/onebox.rb +21 -0
- data/lib/thredded/engine.rb +28 -0
- data/lib/thredded/errors.rb +68 -0
- data/lib/thredded/formatting_demo_content.rb +30 -0
- data/lib/thredded/html_pipeline/at_mention_filter.rb +78 -0
- data/lib/thredded/html_pipeline/autolink_filter.rb +15 -0
- data/lib/thredded/html_pipeline/kramdown_filter.rb +39 -0
- data/lib/thredded/html_pipeline/onebox_filter.rb +143 -0
- data/lib/thredded/html_pipeline/wrap_iframes_filter.rb +13 -0
- data/lib/thredded/version.rb +5 -0
- data/lib/thredded/view_hooks/config.rb +37 -0
- data/lib/thredded/view_hooks/renderer.rb +30 -0
- data/vendor/assets/javascripts/autosize.min.js +6 -0
- data/vendor/assets/javascripts/textcomplete.min.js +2 -0
- metadata +1035 -0
@@ -0,0 +1,103 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
module DbTools
|
5
|
+
class << self
|
6
|
+
MIGRATION_SPEC_SOURCE_VERSION = 'v0.8'
|
7
|
+
|
8
|
+
# Runs the migrations in the given paths.
|
9
|
+
def migrate(paths:, quiet:, &filter)
|
10
|
+
verbose_was = ActiveRecord::Migration.verbose
|
11
|
+
ActiveRecord::Migration.verbose = !quiet
|
12
|
+
migrate = lambda do
|
13
|
+
if Rails::VERSION::STRING >= '5.2'
|
14
|
+
ActiveRecord::MigrationContext.new(paths).migrate(nil, &filter)
|
15
|
+
else
|
16
|
+
ActiveRecord::Migrator.migrate(paths, &filter)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
if quiet
|
20
|
+
silence_active_record(&migrate)
|
21
|
+
else
|
22
|
+
migrate.call
|
23
|
+
end
|
24
|
+
ensure
|
25
|
+
ActiveRecord::Migration.verbose = verbose_was
|
26
|
+
end
|
27
|
+
|
28
|
+
def dump_file
|
29
|
+
File.expand_path("../../../spec/migration/#{MIGRATION_SPEC_SOURCE_VERSION}.#{adapter}.dump", __FILE__)
|
30
|
+
end
|
31
|
+
|
32
|
+
def dump(to = dump_file)
|
33
|
+
case adapter
|
34
|
+
when /sqlite/i
|
35
|
+
system ['sqlite3', Rails.root.join(database), '.dump', '>', to].join(' ')
|
36
|
+
when /postgres/i
|
37
|
+
cmd = "pg_dump --dbname=postgresql://#{username}:#{password}@#{host}:5432/#{database}" \
|
38
|
+
"--verbose --clean --no-owner --no-acl --format=c > #{to}"
|
39
|
+
system cmd
|
40
|
+
when /mysql/i
|
41
|
+
system("mysqldump --user #{username} -p#{password} #{database} > #{to}")
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
def restore(from = dump_file)
|
46
|
+
case adapter
|
47
|
+
when /postgres/i
|
48
|
+
cmd = [
|
49
|
+
'pg_restore --verbose --clean --no-owner --no-acl',
|
50
|
+
"--dbname=postgresql://#{username}:#{password}@#{host}:5432/#{database}",
|
51
|
+
from,
|
52
|
+
'>',
|
53
|
+
Rails.root.join('log', 'restore.log')
|
54
|
+
].join(' ')
|
55
|
+
system cmd
|
56
|
+
when /mysql/i, /sqlite/i
|
57
|
+
connection = ActiveRecord::Base.connection
|
58
|
+
statements = File.read(from).split(/;$/)
|
59
|
+
statements.pop
|
60
|
+
silence_active_record do
|
61
|
+
ActiveRecord::Base.transaction do
|
62
|
+
statements.each do |statement|
|
63
|
+
connection.execute(statement) unless statement =~ /(BEGIN TRANSACTION|COMMIT)/
|
64
|
+
end
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
69
|
+
|
70
|
+
def silence_active_record
|
71
|
+
was = ActiveRecord::Base.logger.level
|
72
|
+
ActiveRecord::Base.logger.level = Logger::WARN
|
73
|
+
yield
|
74
|
+
ensure
|
75
|
+
ActiveRecord::Base.logger.level = was
|
76
|
+
end
|
77
|
+
|
78
|
+
def config
|
79
|
+
@config ||= Rails.configuration.database_configuration[Rails.env]
|
80
|
+
end
|
81
|
+
|
82
|
+
def adapter
|
83
|
+
config['adapter']
|
84
|
+
end
|
85
|
+
|
86
|
+
def database
|
87
|
+
config['database']
|
88
|
+
end
|
89
|
+
|
90
|
+
def username
|
91
|
+
config['username']
|
92
|
+
end
|
93
|
+
|
94
|
+
def password
|
95
|
+
config['password']
|
96
|
+
end
|
97
|
+
|
98
|
+
def host
|
99
|
+
config['host']
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
end
|
@@ -0,0 +1,22 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'thredded/email_transformer/onebox'
|
4
|
+
|
5
|
+
module Thredded
|
6
|
+
# This transformer should applied to emails so that they render correctly in the email clients.
|
7
|
+
#
|
8
|
+
# For example, if you use roadie, you can configure it to use the transformer in the initializer:
|
9
|
+
#
|
10
|
+
# # config/initializers/roadie.rb
|
11
|
+
# Rails.application.config.roadie.before_transformation = Thredded::EmailTransformer
|
12
|
+
#
|
13
|
+
module EmailTransformer
|
14
|
+
mattr_accessor :transformers
|
15
|
+
self.transformers = [Onebox]
|
16
|
+
|
17
|
+
# @param doc [Nokogiri::HTML::Document]
|
18
|
+
def self.call(doc)
|
19
|
+
transformers.each { |transformer| transformer.call(doc) }
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
@@ -0,0 +1,47 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
module EmailTransformer
|
5
|
+
# A helper module with common functions for constructing Nokogiri elements.
|
6
|
+
module Helpers
|
7
|
+
# Creates a `<p>` node with the given child
|
8
|
+
def paragraph(child)
|
9
|
+
Nokogiri::XML::Node.new('p', doc).tap do |p|
|
10
|
+
p.add_child child
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
# Creates an `<a>` node with the given attributes and content
|
15
|
+
def anchor(href, target: '_blank', content: href)
|
16
|
+
Nokogiri::XML::Node.new('a', doc).tap do |a|
|
17
|
+
a['href'] = href
|
18
|
+
a['target'] = target
|
19
|
+
a.content = content
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
|
24
|
+
class Base
|
25
|
+
include Helpers
|
26
|
+
|
27
|
+
# @return [Nokogiri::HTML::Document]
|
28
|
+
attr_reader :doc
|
29
|
+
|
30
|
+
# @param doc [Nokogiri::HTML::Document]
|
31
|
+
def initialize(doc)
|
32
|
+
@doc = doc
|
33
|
+
end
|
34
|
+
|
35
|
+
def self.inherited(base)
|
36
|
+
base.extend ClassMethods
|
37
|
+
end
|
38
|
+
|
39
|
+
module ClassMethods
|
40
|
+
# @param doc [Nokogiri::HTML::Document]
|
41
|
+
def call(doc)
|
42
|
+
new(doc).call
|
43
|
+
end
|
44
|
+
end
|
45
|
+
end
|
46
|
+
end
|
47
|
+
end
|
@@ -0,0 +1,21 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'thredded/email_transformer/base'
|
4
|
+
module Thredded
|
5
|
+
module EmailTransformer
|
6
|
+
# Wraps oneboxes with tables, because only tables can have borders in most email clients.
|
7
|
+
class Onebox < Base
|
8
|
+
def call
|
9
|
+
doc.css('aside.onebox').each do |onebox|
|
10
|
+
table = Nokogiri::XML::Node.new('table', doc)
|
11
|
+
table['class'] = 'onebox-wrapper-table'
|
12
|
+
onebox.swap table
|
13
|
+
table
|
14
|
+
.add_child(Nokogiri::XML::Node.new('tr', doc))
|
15
|
+
.add_child(Nokogiri::XML::Node.new('td', doc))
|
16
|
+
.add_child(onebox)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
class Engine < ::Rails::Engine
|
5
|
+
isolate_namespace Thredded
|
6
|
+
|
7
|
+
config.generators do |g|
|
8
|
+
g.test_framework :rspec, fixture: true
|
9
|
+
g.fixture_replacement :factory_bot, dir: 'spec/factories'
|
10
|
+
g.helper false
|
11
|
+
end
|
12
|
+
|
13
|
+
config.to_prepare do
|
14
|
+
Thredded::AllViewHooks.reset_instance!
|
15
|
+
if Thredded.user_class
|
16
|
+
Thredded.user_class.send(:include, Thredded::UserExtender)
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
initializer 'thredded.setup_assets' do
|
21
|
+
Thredded::Engine.config.assets.precompile += %w[
|
22
|
+
thredded.js
|
23
|
+
thredded.css
|
24
|
+
thredded/*.svg
|
25
|
+
]
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
@@ -0,0 +1,68 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
class Error < StandardError
|
5
|
+
end
|
6
|
+
|
7
|
+
module Errors
|
8
|
+
class DatabaseEmpty < Thredded::Error
|
9
|
+
def message
|
10
|
+
'Seed the database with "rake db:seed".'
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
class LoginRequired < Thredded::Error
|
15
|
+
def message
|
16
|
+
I18n.t('thredded.errors.login_required')
|
17
|
+
end
|
18
|
+
end
|
19
|
+
|
20
|
+
class UserNotFound < Thredded::Error
|
21
|
+
def message
|
22
|
+
'This user could not be found. Is their name misspelled?'
|
23
|
+
end
|
24
|
+
end
|
25
|
+
|
26
|
+
class PrivateTopicNotFound < Thredded::Error
|
27
|
+
def message
|
28
|
+
I18n.t('thredded.errors.private_topic_not_found')
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
class TopicNotFound < Thredded::Error
|
33
|
+
def message
|
34
|
+
'This topic does not exist.'
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
class MessageboardNotFound < Thredded::Error
|
39
|
+
def message
|
40
|
+
'This messageboard does not exist.'
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
class MessageboardReadDenied < Thredded::Error
|
45
|
+
def message
|
46
|
+
'You are not authorized access to this messageboard.'
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
class MessageboardCreateDenied < Thredded::Error
|
51
|
+
def message
|
52
|
+
'You are not authorized to create a new messageboard.'
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
class TopicCreateDenied < Thredded::Error
|
57
|
+
def message
|
58
|
+
'You are not authorized to post in this messageboard.'
|
59
|
+
end
|
60
|
+
end
|
61
|
+
|
62
|
+
class PrivateTopicCreateDenied < Thredded::Error
|
63
|
+
def message
|
64
|
+
I18n.t('thredded.errors.private_topic_create_denied')
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|
68
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
# Content for demo-ing formatting functionality
|
5
|
+
module FormattingDemoContent
|
6
|
+
class << self
|
7
|
+
attr_accessor :parts
|
8
|
+
end
|
9
|
+
self.parts = [
|
10
|
+
<<-'MARKDOWN',
|
11
|
+
#### Oneboxes
|
12
|
+
|
13
|
+
URLs of supported resources are replaced with boxes like these:
|
14
|
+
|
15
|
+
**Twitter** `https://twitter.com/thredded/status/838824533477982209`:
|
16
|
+
https://twitter.com/thredded/status/838824533477982209
|
17
|
+
**StackExchange** `http://codegolf.stackexchange.com/questions/45701`:
|
18
|
+
http://codegolf.stackexchange.com/questions/45701
|
19
|
+
**Amazon** `https://www.amazon.co.uk/dp/0521797071`:
|
20
|
+
https://www.amazon.co.uk/dp/0521797071
|
21
|
+
**YouTube** `https://www.youtube.com/watch?v=1QP7elXwpLw`:
|
22
|
+
https://www.youtube.com/watch?v=1QP7elXwpLw
|
23
|
+
**Google Maps** `https://goo.gl/maps/R6nj3Qwf2LR2`:
|
24
|
+
https://goo.gl/maps/R6nj3Qwf2LR2
|
25
|
+
|
26
|
+
Many more resources are [supported](https://github.com/discourse/onebox/tree/master/lib/onebox/engine). Powered by the [onebox](https://github.com/discourse/onebox) library.
|
27
|
+
MARKDOWN
|
28
|
+
]
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,78 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
module HtmlPipeline
|
5
|
+
class AtMentionFilter < ::HTML::Pipeline::Filter
|
6
|
+
DEFAULT_IGNORED_ANCESTOR_TAGS = %w[pre code tt a style].freeze
|
7
|
+
|
8
|
+
# @param context [Hash]
|
9
|
+
# @option context :users_provider [#call(usernames)] given usernames, returns a list of users.
|
10
|
+
def initialize(doc, context = nil, result = nil)
|
11
|
+
super doc, context, result
|
12
|
+
@users_provider = context[:users_provider]
|
13
|
+
@view_context = context[:view_context]
|
14
|
+
end
|
15
|
+
|
16
|
+
def call
|
17
|
+
return doc unless @users_provider
|
18
|
+
process_text_nodes! do |node|
|
19
|
+
content = node.to_html
|
20
|
+
next unless content.include?('@')
|
21
|
+
highlight! content
|
22
|
+
node.replace content
|
23
|
+
end
|
24
|
+
doc
|
25
|
+
end
|
26
|
+
|
27
|
+
# @return [Array<Thredded.user_class>] users that were @-mentioned
|
28
|
+
def mentioned_users
|
29
|
+
return [] unless @users_provider
|
30
|
+
names = []
|
31
|
+
process_text_nodes! { |node| names.concat mentioned_names(node.to_html) }
|
32
|
+
names.uniq!
|
33
|
+
@users_provider.call(names)
|
34
|
+
end
|
35
|
+
|
36
|
+
private
|
37
|
+
|
38
|
+
MATCH_NAME_RE = /(?:^|[\s>])@([\w\-]+|"[\w.,\- ()]+")(?=\W|$)/
|
39
|
+
|
40
|
+
def mentioned_names(text_node_html)
|
41
|
+
text_node_html.scan(MATCH_NAME_RE).map(&:first).map { |m| m.start_with?('"') ? m[1..-2] : m }
|
42
|
+
end
|
43
|
+
|
44
|
+
def highlight!(text_node_html)
|
45
|
+
names = mentioned_names(text_node_html)
|
46
|
+
return if names.blank?
|
47
|
+
@users_provider.call(names).each do |user|
|
48
|
+
name = user.thredded_display_name
|
49
|
+
maybe_quoted_name = name =~ /[., ()]/ ? %("#{name}") : name
|
50
|
+
url = Thredded.user_path(@view_context, user)
|
51
|
+
text_node_html.gsub!(
|
52
|
+
/(^|[\s>])(@#{Regexp.escape maybe_quoted_name})([^a-z\d]|$)/i,
|
53
|
+
%(\\1<a href="#{ERB::Util.html_escape url}">@#{ERB::Util.html_escape maybe_quoted_name}</a>\\3)
|
54
|
+
)
|
55
|
+
end
|
56
|
+
end
|
57
|
+
|
58
|
+
# Yields text nodes that should be processed.
|
59
|
+
def process_text_nodes!
|
60
|
+
doc.search('.//text()').each do |node|
|
61
|
+
next if has_ancestor?(node, ignored_ancestor_tags)
|
62
|
+
yield node
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
# Return ancestor tags to stop the at-mention highlighting.
|
67
|
+
#
|
68
|
+
# @return [Array<String>] Ancestor tags.
|
69
|
+
def ignored_ancestor_tags
|
70
|
+
if @context[:ignored_ancestor_tags]
|
71
|
+
DEFAULT_IGNORED_ANCESTOR_TAGS | @context[:ignored_ancestor_tags]
|
72
|
+
else
|
73
|
+
DEFAULT_IGNORED_ANCESTOR_TAGS
|
74
|
+
end
|
75
|
+
end
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Thredded
|
4
|
+
module HtmlPipeline
|
5
|
+
# HTML Filter for auto_linking urls in HTML.
|
6
|
+
#
|
7
|
+
# AutolinkFilter is required because Kramdown does not autolink by default.
|
8
|
+
# https://github.com/gettalong/kramdown/issues/306
|
9
|
+
class AutolinkFilter < ::HTML::Pipeline::Filter
|
10
|
+
def call
|
11
|
+
Rinku.auto_link(html, :all)
|
12
|
+
end
|
13
|
+
end
|
14
|
+
end
|
15
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'kramdown'
|
4
|
+
module Thredded
|
5
|
+
module HtmlPipeline
|
6
|
+
class KramdownFilter < ::HTML::Pipeline::TextFilter
|
7
|
+
class << self
|
8
|
+
attr_accessor :options
|
9
|
+
end
|
10
|
+
|
11
|
+
# See http://kramdown.gettalong.org/options.html
|
12
|
+
self.options = {
|
13
|
+
input: 'GFM',
|
14
|
+
gfm_quirks: 'paragraph_end',
|
15
|
+
# Smart quotes conflict with @"at mentions". Disable smart quotes.
|
16
|
+
smart_quotes: %w[apos apos quot quot],
|
17
|
+
remove_block_html_tags: false,
|
18
|
+
syntax_highlighter: nil
|
19
|
+
}
|
20
|
+
|
21
|
+
def initialize(text, context = nil, result = nil)
|
22
|
+
super text, context, result
|
23
|
+
if @text.frozen?
|
24
|
+
@text = @text.delete("\r")
|
25
|
+
else
|
26
|
+
@text.delete! "\r"
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
# Convert Markdown to HTML using the best available implementation
|
31
|
+
# and convert into a DocumentFragment.
|
32
|
+
def call
|
33
|
+
result = Kramdown::Document.new(@text, self.class.options.deep_merge(context[:kramdown_options] || {})).to_html
|
34
|
+
result.rstrip!
|
35
|
+
result
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|