From 30712c4886a6da7e0628fadbaa855d8c95a1536f Mon Sep 17 00:00:00 2001
From: Ember <me@ember-is.gay>
Date: Thu, 18 Jan 2024 00:41:52 +1100
Subject: [PATCH] add blobfox theme

---
 .../flavours/blobfox/actions/account_notes.ts |   18 +
 .../flavours/blobfox/actions/accounts.js      |  774 +++++++
 .../blobfox/actions/accounts_typed.ts         |   97 +
 .../flavours/blobfox/actions/alerts.js        |   59 +
 .../flavours/blobfox/actions/announcements.js |  181 ++
 .../flavours/blobfox/actions/app.ts           |    9 +
 .../flavours/blobfox/actions/blocks.js        |  100 +
 .../flavours/blobfox/actions/bookmarks.js     |   91 +
 .../flavours/blobfox/actions/boosts.js        |   32 +
 .../flavours/blobfox/actions/bundles.js       |   25 +
 .../flavours/blobfox/actions/columns.js       |   54 +
 .../flavours/blobfox/actions/compose.js       |  840 ++++++++
 .../flavours/blobfox/actions/conversations.js |  113 +
 .../flavours/blobfox/actions/custom_emojis.js |   40 +
 .../flavours/blobfox/actions/directory.js     |   62 +
 .../flavours/blobfox/actions/domain_blocks.js |  152 ++
 .../blobfox/actions/domain_blocks_typed.ts    |   13 +
 .../flavours/blobfox/actions/dropdown_menu.ts |   11 +
 .../flavours/blobfox/actions/emojis.js        |   14 +
 .../flavours/blobfox/actions/favourites.js    |   94 +
 .../flavours/blobfox/actions/featured_tags.js |   34 +
 .../flavours/blobfox/actions/filters.js       |   97 +
 .../flavours/blobfox/actions/height_cache.js  |   17 +
 .../flavours/blobfox/actions/history.js       |   38 +
 .../blobfox/actions/importer/index.js         |   93 +
 .../blobfox/actions/importer/normalizer.js    |  135 ++
 .../flavours/blobfox/actions/interactions.js  |  600 ++++++
 .../flavours/blobfox/actions/languages.js     |   12 +
 .../flavours/blobfox/actions/lists.js         |  373 ++++
 .../blobfox/actions/local_settings.js         |   81 +
 .../flavours/blobfox/actions/markers.js       |  152 ++
 .../flavours/blobfox/actions/modal.ts         |   19 +
 .../flavours/blobfox/actions/mutes.js         |  117 +
 .../flavours/blobfox/actions/notifications.js |  404 ++++
 .../blobfox/actions/notifications_typed.ts    |   23 +
 .../flavours/blobfox/actions/onboarding.js    |    8 +
 .../blobfox/actions/picture_in_picture.js     |   46 +
 .../flavours/blobfox/actions/pin_statuses.js  |   42 +
 .../flavours/blobfox/actions/polls.js         |   61 +
 .../actions/push_notifications/index.js       |   17 +
 .../actions/push_notifications/registerer.js  |  134 ++
 .../actions/push_notifications/setter.js      |   34 +
 .../flavours/blobfox/actions/reports.js       |   42 +
 .../flavours/blobfox/actions/search.js        |  201 ++
 .../flavours/blobfox/actions/server.js        |  131 ++
 .../flavours/blobfox/actions/settings.js      |   36 +
 .../flavours/blobfox/actions/statuses.js      |  351 +++
 .../flavours/blobfox/actions/store.js         |   43 +
 .../flavours/blobfox/actions/streaming.js     |  184 ++
 .../flavours/blobfox/actions/suggestions.js   |   65 +
 .../flavours/blobfox/actions/tags.js          |  172 ++
 .../flavours/blobfox/actions/timelines.js     |  235 ++
 .../flavours/blobfox/actions/trends.js        |  140 ++
 app/javascript/flavours/blobfox/api.ts        |   63 +
 .../flavours/blobfox/api_types/accounts.ts    |   45 +
 .../blobfox/api_types/custom_emoji.ts         |    8 +
 .../blobfox/api_types/relationships.ts        |   18 +
 app/javascript/flavours/blobfox/blurhash.ts   |  111 +
 app/javascript/flavours/blobfox/compare_id.ts |   11 +
 .../components/__tests__/hashtag_bar.tsx      |  214 ++
 .../flavours/blobfox/components/account.jsx   |  182 ++
 .../blobfox/components/admin/Counter.jsx      |  121 ++
 .../blobfox/components/admin/Dimension.jsx    |   95 +
 .../blobfox/components/admin/ImpactReport.jsx |   91 +
 .../components/admin/ReportReasonSelector.jsx |  165 ++
 .../blobfox/components/admin/Retention.jsx    |  155 ++
 .../blobfox/components/admin/Trends.jsx       |   76 +
 .../blobfox/components/animated_number.tsx    |   81 +
 .../blobfox/components/attachment_list.jsx    |   51 +
 .../blobfox/components/autosuggest_emoji.jsx  |   43 +
 .../components/autosuggest_hashtag.tsx        |   42 +
 .../blobfox/components/autosuggest_input.jsx  |  230 ++
 .../components/autosuggest_textarea.jsx       |  240 +++
 .../flavours/blobfox/components/avatar.tsx    |   49 +
 .../blobfox/components/avatar_composite.jsx   |  106 +
 .../blobfox/components/avatar_overlay.tsx     |   57 +
 .../flavours/blobfox/components/blurhash.tsx  |   48 +
 .../flavours/blobfox/components/button.tsx    |   58 +
 .../flavours/blobfox/components/check.tsx     |   13 +
 .../blobfox/components/circular_progress.tsx  |   27 +
 .../flavours/blobfox/components/column.jsx    |   73 +
 .../blobfox/components/column_back_button.jsx |   63 +
 .../components/column_back_button_slim.jsx    |   40 +
 .../blobfox/components/column_header.jsx      |  219 ++
 .../flavours/blobfox/components/counters.tsx  |   45 +
 .../blobfox/components/dismissable_banner.tsx |   66 +
 .../blobfox/components/display_name.tsx       |  127 ++
 .../flavours/blobfox/components/domain.tsx    |   44 +
 .../blobfox/components/dropdown_menu.jsx      |  338 +++
 .../containers/dropdown_menu_container.js     |   32 +
 .../components/edited_timestamp/index.jsx     |   77 +
 .../blobfox/components/empty_account.tsx      |   33 +
 .../blobfox/components/error_boundary.jsx     |  111 +
 .../flavours/blobfox/components/gifv.tsx      |   70 +
 .../flavours/blobfox/components/hashtag.jsx   |  123 ++
 .../blobfox/components/hashtag_bar.tsx        |  234 ++
 .../flavours/blobfox/components/icon.tsx      |   20 +
 .../blobfox/components/icon_button.tsx        |  180 ++
 .../blobfox/components/icon_with_badge.tsx    |   24 +
 .../blobfox/components/inline_account.jsx     |   37 +
 .../intersection_observer_article.jsx         |  131 ++
 .../flavours/blobfox/components/link.jsx      |   97 +
 .../flavours/blobfox/components/load_gap.tsx  |   34 +
 .../flavours/blobfox/components/load_more.tsx |   24 +
 .../blobfox/components/load_pending.tsx       |   18 +
 .../blobfox/components/loading_indicator.tsx  |    7 +
 .../flavours/blobfox/components/logo.jsx      |   14 +
 .../blobfox/components/media_attachments.jsx  |  128 ++
 .../blobfox/components/media_gallery.jsx      |  395 ++++
 .../blobfox/components/modal_root.jsx         |  165 ++
 .../blobfox/components/navigation_portal.tsx  |   25 +
 .../components/not_signed_in_indicator.tsx    |   12 +
 .../components/notification_purge_buttons.jsx |   64 +
 .../flavours/blobfox/components/permalink.jsx |   50 +
 .../picture_in_picture_placeholder.jsx        |   33 +
 .../flavours/blobfox/components/poll.jsx      |  249 +++
 .../blobfox/components/radio_button.tsx       |   33 +
 .../components/regeneration_indicator.jsx     |   18 +
 .../blobfox/components/relative_timestamp.tsx |  282 +++
 .../flavours/blobfox/components/router.tsx    |   80 +
 .../blobfox/components/scrollable_list.jsx    |  405 ++++
 .../blobfox/components/server_banner.jsx      |   97 +
 .../blobfox/components/server_hero_image.tsx  |   35 +
 .../blobfox/components/setting_text.jsx       |   35 +
 .../blobfox/components/short_number.tsx       |   90 +
 .../flavours/blobfox/components/skeleton.tsx  |   10 +
 .../flavours/blobfox/components/status.jsx    |  879 ++++++++
 .../blobfox/components/status_action_bar.jsx  |  371 ++++
 .../blobfox/components/status_content.jsx     |  491 +++++
 .../blobfox/components/status_header.jsx      |   71 +
 .../blobfox/components/status_icons.jsx       |  147 ++
 .../blobfox/components/status_list.jsx        |  137 ++
 .../blobfox/components/status_prepend.jsx     |  158 ++
 .../blobfox/components/status_reactions.jsx   |  175 ++
 .../components/status_visibility_icon.jsx     |   54 +
 .../blobfox/components/timeline_hint.tsx      |   25 +
 .../blobfox/components/verified_badge.tsx     |   27 +
 .../blobfox/containers/account_container.jsx  |   76 +
 .../blobfox/containers/admin_component.jsx    |   22 +
 .../blobfox/containers/compose_container.jsx  |   31 +
 .../blobfox/containers/domain_container.jsx   |   36 +
 .../containers/dropdown_menu_container.js     |   37 +
 ...intersection_observer_article_container.js |   18 +
 .../flavours/blobfox/containers/mastodon.jsx  |   97 +
 .../blobfox/containers/media_container.jsx    |  127 ++
 .../notification_purge_buttons_container.js   |   53 +
 .../blobfox/containers/poll_container.js      |   26 +
 .../blobfox/containers/scroll_container.js    |   18 +
 .../blobfox/containers/status_container.js    |  313 +++
 .../flavours/blobfox/features/about/index.jsx |  225 ++
 .../account/components/account_note.jsx       |  174 ++
 .../account/components/action_bar.jsx         |   85 +
 .../account/components/featured_tags.jsx      |   52 +
 .../components/follow_request_note.jsx        |   38 +
 .../features/account/components/header.jsx    |  404 ++++
 .../components/profile_column_header.jsx      |   36 +
 .../containers/account_note_container.js      |   19 +
 .../containers/featured_tags_container.js     |   17 +
 .../follow_request_note_container.js          |   17 +
 .../blobfox/features/account/navigation.jsx   |   55 +
 .../account_gallery/components/media_item.jsx |  155 ++
 .../features/account_gallery/index.jsx        |  239 +++
 .../account_timeline/components/header.jsx    |  165 ++
 .../components/limited_account_hint.tsx       |   35 +
 .../components/memorial_note.jsx              |   11 +
 .../components/moved_note.jsx                 |   52 +
 .../containers/header_container.jsx           |  186 ++
 .../features/account_timeline/index.jsx       |  210 ++
 .../flavours/blobfox/features/audio/index.jsx |  600 ++++++
 .../blobfox/features/audio/visualizer.js      |  136 ++
 .../blobfox/features/blocks/index.jsx         |   82 +
 .../features/bookmarked_statuses/index.jsx    |  113 +
 .../closed_registrations_modal/index.jsx      |   77 +
 .../components/column_settings.jsx            |   45 +
 .../containers/column_settings_container.js   |   29 +
 .../features/community_timeline/index.jsx     |  166 ++
 .../compose/components/action_bar.jsx         |   73 +
 .../components/autosuggest_account.jsx        |   24 +
 .../compose/components/character_counter.jsx  |   26 +
 .../compose/components/compose_form.jsx       |  356 ++++
 .../features/compose/components/dropdown.jsx  |  243 +++
 .../compose/components/dropdown_menu.jsx      |  200 ++
 .../components/emoji_picker_dropdown.jsx      |  422 ++++
 .../features/compose/components/header.jsx    |  134 ++
 .../compose/components/language_dropdown.jsx  |  334 +++
 .../compose/components/navigation_bar.jsx     |   54 +
 .../features/compose/components/options.jsx   |  311 +++
 .../features/compose/components/poll_form.jsx |  181 ++
 .../compose/components/privacy_dropdown.jsx   |   90 +
 .../features/compose/components/publisher.jsx |   92 +
 .../compose/components/reply_indicator.jsx    |   70 +
 .../features/compose/components/search.jsx    |  400 ++++
 .../compose/components/search_results.jsx     |   89 +
 .../compose/components/text_icon_button.jsx   |   38 +
 .../compose/components/textarea_icons.jsx     |   61 +
 .../features/compose/components/upload.jsx    |   66 +
 .../compose/components/upload_form.jsx        |   32 +
 .../compose/components/upload_progress.jsx    |   56 +
 .../features/compose/components/warning.jsx   |   28 +
 .../autosuggest_account_container.js          |   16 +
 .../containers/compose_form_container.js      |  150 ++
 .../compose/containers/dropdown_container.js  |   14 +
 .../emoji_picker_dropdown_container.js        |   85 +
 .../compose/containers/header_container.js    |   42 +
 .../containers/language_dropdown_container.js |   37 +
 .../containers/navigation_container.js        |   36 +
 .../compose/containers/options_container.js   |   56 +
 .../compose/containers/poll_form_container.js |   53 +
 .../containers/privacy_dropdown_container.js  |   30 +
 .../containers/reply_indicator_container.js   |   33 +
 .../compose/containers/search_container.js    |   53 +
 .../containers/search_results_container.js    |   20 +
 .../containers/sensitive_button_container.jsx |   77 +
 .../compose/containers/upload_container.js    |   26 +
 .../containers/upload_form_container.js       |    9 +
 .../containers/upload_progress_container.js   |   11 +
 .../compose/containers/warning_container.jsx  |   47 +
 .../blobfox/features/compose/index.jsx        |  122 ++
 .../blobfox/features/compose/util/counter.js  |    9 +
 .../features/compose/util/url_regex.js        |   30 +
 .../components/column_settings.jsx            |   47 +
 .../components/conversation.jsx               |  233 ++
 .../components/conversations_list.jsx         |   77 +
 .../containers/column_settings_container.js   |   19 +
 .../containers/conversation_container.js      |   81 +
 .../conversations_list_container.js           |   16 +
 .../features/direct_timeline/index.jsx        |  165 ++
 .../directory/components/account_card.jsx     |  255 +++
 .../blobfox/features/directory/index.jsx      |  179 ++
 .../blobfox/features/domain_blocks/index.jsx  |   87 +
 .../flavours/blobfox/features/emoji/emoji.js  |  166 ++
 .../features/emoji/emoji_compressed.d.ts      |   57 +
 .../features/emoji/emoji_compressed.js        |  135 ++
 .../blobfox/features/emoji/emoji_map.json     |    1 +
 .../features/emoji/emoji_mart_data_light.ts   |   43 +
 .../features/emoji/emoji_mart_search_light.js |  185 ++
 .../blobfox/features/emoji/emoji_picker.js    |    7 +
 .../emoji/emoji_unicode_mapping_light.ts      |   60 +
 .../blobfox/features/emoji/emoji_utils.js     |  258 +++
 .../features/emoji/unicode_to_filename.js     |   29 +
 .../features/emoji/unicode_to_unified_name.js |   24 +
 .../explore/components/search_section.jsx     |   20 +
 .../features/explore/components/story.jsx     |   61 +
 .../blobfox/features/explore/index.jsx        |  114 +
 .../blobfox/features/explore/links.jsx        |   90 +
 .../blobfox/features/explore/results.jsx      |  229 ++
 .../blobfox/features/explore/statuses.jsx     |   78 +
 .../blobfox/features/explore/suggestions.jsx  |   70 +
 .../blobfox/features/explore/tags.jsx         |   76 +
 .../features/favourited_statuses/index.jsx    |  113 +
 .../blobfox/features/favourites/index.jsx     |  114 +
 .../features/filters/added_to_filter.jsx      |  106 +
 .../features/filters/select_filter.jsx        |  196 ++
 .../blobfox/features/firehose/index.jsx       |  229 ++
 .../components/account_authorize.jsx          |   52 +
 .../containers/account_authorize_container.js |   27 +
 .../features/follow_requests/index.jsx        |   96 +
 .../blobfox/features/followed_tags/index.jsx  |   93 +
 .../blobfox/features/followers/index.jsx      |  177 ++
 .../blobfox/features/following/index.jsx      |  177 ++
 .../components/announcements.jsx              |  455 ++++
 .../getting_started/components/trends.jsx     |   54 +
 .../containers/announcements_container.js     |   22 +
 .../containers/trends_container.js            |   15 +
 .../features/getting_started/index.jsx        |  208 ++
 .../features/getting_started_misc/index.jsx   |   68 +
 .../components/column_settings.jsx            |  138 ++
 .../components/hashtag_header.jsx             |   79 +
 .../containers/column_settings_container.js   |   33 +
 .../features/hashtag_timeline/index.jsx       |  226 ++
 .../components/column_settings.tsx            |  109 +
 .../components/critical_update_banner.tsx     |   26 +
 .../components/explore_prompt.tsx             |   46 +
 .../blobfox/features/home_timeline/index.jsx  |  239 +++
 .../features/interaction_modal/index.jsx      |  417 ++++
 .../features/keyboard_shortcuts/index.jsx     |  152 ++
 .../list_adder/components/account.jsx         |   43 +
 .../features/list_adder/components/list.jsx   |   72 +
 .../blobfox/features/list_adder/index.jsx     |   76 +
 .../list_editor/components/account.jsx        |   79 +
 .../list_editor/components/edit_list_form.jsx |   73 +
 .../list_editor/components/search.jsx         |   81 +
 .../blobfox/features/list_editor/index.jsx    |   83 +
 .../blobfox/features/list_timeline/index.jsx  |  244 +++
 .../lists/components/new_list_form.jsx        |   81 +
 .../flavours/blobfox/features/lists/index.jsx |   93 +
 .../blobfox/features/local_settings/index.jsx |   70 +
 .../local_settings/navigation/index.jsx       |   95 +
 .../local_settings/navigation/item/index.jsx  |   73 +
 .../page/deprecated_item/index.jsx            |   83 +
 .../features/local_settings/page/index.jsx    |  505 +++++
 .../local_settings/page/item/index.jsx        |  118 ++
 .../flavours/blobfox/features/mutes/index.jsx |   88 +
 .../notifications/components/admin_report.jsx |  115 +
 .../notifications/components/admin_signup.jsx |  108 +
 .../components/clear_column_button.jsx        |   20 +
 .../components/column_settings.jsx            |  218 ++
 .../notifications/components/filter_bar.jsx   |  121 ++
 .../notifications/components/follow.jsx       |  108 +
 .../components/follow_request.jsx             |  141 ++
 .../components/grant_permission_button.jsx    |   20 +
 .../notifications/components/notification.jsx |  258 +++
 .../notifications_permission_banner.jsx       |   51 +
 .../notifications/components/overlay.jsx      |   61 +
 .../components/pill_bar_button.jsx            |   43 +
 .../notifications/components/report.jsx       |   67 +
 .../components/setting_toggle.jsx             |   38 +
 .../containers/admin_report_container.js      |   15 +
 .../containers/column_settings_container.js   |   78 +
 .../containers/filter_bar_container.js        |   17 +
 .../containers/follow_request_container.js    |   17 +
 .../containers/notification_container.js      |   25 +
 .../containers/overlay_container.js           |   19 +
 .../blobfox/features/notifications/index.jsx  |  368 ++++
 .../components/arrow_small_right.jsx          |    7 +
 .../components/progress_indicator.jsx         |   28 +
 .../features/onboarding/components/step.jsx   |   50 +
 .../blobfox/features/onboarding/follows.jsx   |   80 +
 .../blobfox/features/onboarding/index.jsx     |  149 ++
 .../blobfox/features/onboarding/share.jsx     |  200 ++
 .../picture_in_picture/components/footer.jsx  |  231 ++
 .../picture_in_picture/components/header.jsx  |   50 +
 .../features/picture_in_picture/index.jsx     |   93 +
 .../containers/account_container.js           |   25 +
 .../containers/search_container.js            |   24 +
 .../features/pinned_accounts_editor/index.jsx |   82 +
 .../features/pinned_statuses/index.jsx        |   70 +
 .../blobfox/features/privacy_policy/index.jsx |   65 +
 .../components/column_settings.jsx            |   46 +
 .../containers/column_settings_container.js   |   29 +
 .../features/public_timeline/index.jsx        |  171 ++
 .../blobfox/features/reblogs/index.jsx        |  115 +
 .../blobfox/features/report/category.jsx      |  108 +
 .../blobfox/features/report/comment.jsx       |  121 ++
 .../features/report/components/option.jsx     |   62 +
 .../report/components/status_check_box.jsx    |   67 +
 .../containers/status_check_box_container.js  |   17 +
 .../blobfox/features/report/rules.jsx         |   69 +
 .../blobfox/features/report/statuses.jsx      |   65 +
 .../blobfox/features/report/thanks.jsx        |   88 +
 .../features/standalone/compose/index.jsx     |   21 +
 .../features/status/components/action_bar.jsx |  267 +++
 .../features/status/components/card.jsx       |  256 +++
 .../status/components/detailed_status.jsx     |  362 ++++
 .../containers/detailed_status_container.js   |  176 ++
 .../blobfox/features/status/index.jsx         |  807 +++++++
 .../subscribed_languages_modal/index.jsx      |  127 ++
 .../features/ui/components/actions_modal.jsx  |   94 +
 .../features/ui/components/audio_modal.jsx    |   61 +
 .../features/ui/components/block_modal.jsx    |  100 +
 .../features/ui/components/boost_modal.jsx    |  131 ++
 .../blobfox/features/ui/components/bundle.jsx |  106 +
 .../ui/components/bundle_column_error.jsx     |  166 ++
 .../ui/components/bundle_modal_error.jsx      |   54 +
 .../blobfox/features/ui/components/column.jsx |   78 +
 .../features/ui/components/column_header.jsx  |   40 +
 .../features/ui/components/column_link.jsx    |   57 +
 .../features/ui/components/column_loading.jsx |   32 +
 .../ui/components/column_subheading.jsx       |   15 +
 .../features/ui/components/columns_area.jsx   |  180 ++
 .../ui/components/compare_history_modal.jsx   |  110 +
 .../features/ui/components/compose_panel.jsx  |   62 +
 .../ui/components/confirmation_modal.jsx      |   83 +
 .../components/deprecated_settings_modal.jsx  |   82 +
 .../ui/components/disabled_account_banner.jsx |   99 +
 .../features/ui/components/doodle_modal.jsx   |  619 ++++++
 .../features/ui/components/drawer_loading.jsx |    9 +
 .../features/ui/components/embed_modal.jsx    |  100 +
 .../ui/components/favourite_modal.jsx         |   94 +
 .../features/ui/components/filter_modal.jsx   |  136 ++
 .../ui/components/focal_point_modal.jsx       |  426 ++++
 .../follow_requests_column_link.jsx           |   54 +
 .../blobfox/features/ui/components/header.jsx |  124 ++
 .../features/ui/components/image_loader.jsx   |  174 ++
 .../features/ui/components/image_modal.jsx    |   64 +
 .../features/ui/components/link_footer.jsx    |  111 +
 .../features/ui/components/list_panel.jsx     |   56 +
 .../features/ui/components/media_modal.jsx    |  261 +++
 .../features/ui/components/modal_loading.jsx  |   18 +
 .../features/ui/components/modal_root.jsx     |  142 ++
 .../features/ui/components/mute_modal.jsx     |  139 ++
 .../ui/components/navigation_panel.jsx        |  127 ++
 .../components/notifications_counter_icon.js  |   10 +
 .../features/ui/components/report_modal.jsx   |  227 ++
 .../features/ui/components/sign_in_banner.jsx |   54 +
 .../features/ui/components/upload_area.jsx    |   55 +
 .../features/ui/components/video_modal.jsx    |   74 +
 .../features/ui/components/zoomable_image.jsx |  456 ++++
 .../ui/containers/bundle_container.js         |   18 +
 .../ui/containers/columns_area_container.js   |   22 +
 .../ui/containers/loading_bar_container.js    |    9 +
 .../features/ui/containers/modal_container.js |   36 +
 .../ui/containers/notifications_container.js  |   33 +
 .../ui/containers/status_list_container.js    |   89 +
 .../flavours/blobfox/features/ui/index.jsx    |  675 ++++++
 .../features/ui/util/async-components.js      |  203 ++
 .../blobfox/features/ui/util/fullscreen.js    |   46 +
 .../features/ui/util/get_rect_from_entry.js   |   21 +
 .../ui/util/intersection_observer_wrapper.js  |   57 +
 .../features/ui/util/optional_motion.js       |    7 +
 .../features/ui/util/react_router_helpers.jsx |  105 +
 .../features/ui/util/reduced_motion.jsx       |   45 +
 .../features/ui/util/schedule_idle_task.js    |   29 +
 .../flavours/blobfox/features/video/index.jsx |  665 ++++++
 .../flavours/blobfox/hooks/useHovering.ts     |   17 +
 .../images/elephant_ui_disappointed.svg       |    1 +
 .../blobfox/images/elephant_ui_working.svg    |    1 +
 .../blobfox/images/glitch-preview.jpg         |  Bin 0 -> 197277 bytes
 .../blobfox/images/logo_warn_glitch.svg       |   49 +
 .../flavours/blobfox/images/mbstobon-ui-0.png |  Bin 0 -> 39646 bytes
 .../flavours/blobfox/images/mbstobon-ui-1.png |  Bin 0 -> 43609 bytes
 .../flavours/blobfox/images/mbstobon-ui-2.png |  Bin 0 -> 40376 bytes
 .../flavours/blobfox/images/mbstobon-ui-3.png |  Bin 0 -> 32449 bytes
 .../blobfox/images/wave-drawer-glitched.png   |  Bin 0 -> 3544 bytes
 .../flavours/blobfox/images/wave-drawer.png   |  Bin 0 -> 3269 bytes
 .../flavours/blobfox/initial_state.js         |  144 ++
 app/javascript/flavours/blobfox/is_mobile.ts  |   34 +
 .../blobfox/load_keyboard_extensions.js       |   16 +
 app/javascript/flavours/blobfox/main.jsx      |   47 +
 .../flavours/blobfox/models/account.ts        |  149 ++
 .../flavours/blobfox/models/custom_emoji.ts   |   15 +
 .../flavours/blobfox/models/relationship.ts   |   29 +
 app/javascript/flavours/blobfox/names.yml     |   40 +
 .../flavours/blobfox/packs/admin.jsx          |   25 +
 .../flavours/blobfox/packs/common.js          |    8 +
 .../flavours/blobfox/packs/error.js           |   14 +
 app/javascript/flavours/blobfox/packs/home.js |   11 +
 .../flavours/blobfox/packs/public.jsx         |  242 +++
 .../flavours/blobfox/packs/settings.js        |   42 +
 .../flavours/blobfox/packs/share.jsx          |   27 +
 .../flavours/blobfox/packs/sign_up.js         |   42 +
 .../flavours/blobfox/performance.js           |   30 +
 .../flavours/blobfox/permissions.ts           |    4 +
 .../blobfox/polyfills/extra_polyfills.ts      |    1 +
 .../flavours/blobfox/polyfills/index.ts       |   21 +
 .../flavours/blobfox/polyfills/intl.ts        |  106 +
 app/javascript/flavours/blobfox/ready.js      |   32 +
 .../flavours/blobfox/reducers/accounts.ts     |   84 +
 .../flavours/blobfox/reducers/accounts_map.js |   24 +
 .../flavours/blobfox/reducers/alerts.js       |   30 +
 .../blobfox/reducers/announcements.js         |  103 +
 .../flavours/blobfox/reducers/blocks.js       |   22 +
 .../flavours/blobfox/reducers/boosts.js       |   25 +
 .../flavours/blobfox/reducers/compose.js      |  660 ++++++
 .../flavours/blobfox/reducers/contexts.js     |  107 +
 .../blobfox/reducers/conversations.js         |  118 ++
 .../blobfox/reducers/custom_emojis.js         |   16 +
 .../flavours/blobfox/reducers/domain_lists.js |   26 +
 .../blobfox/reducers/dropdown_menu.ts         |   33 +
 .../flavours/blobfox/reducers/filters.js      |   45 +
 .../blobfox/reducers/followed_tags.js         |   43 +
 .../flavours/blobfox/reducers/height_cache.js |   24 +
 .../flavours/blobfox/reducers/history.js      |   29 +
 .../flavours/blobfox/reducers/index.ts        |  111 +
 .../flavours/blobfox/reducers/list_adder.js   |   48 +
 .../flavours/blobfox/reducers/list_editor.js  |   99 +
 .../flavours/blobfox/reducers/lists.js        |   38 +
 .../blobfox/reducers/local_settings.js        |   79 +
 .../flavours/blobfox/reducers/markers.js      |   26 +
 .../blobfox/reducers/media_attachments.js     |   16 +
 .../flavours/blobfox/reducers/meta.js         |   25 +
 .../flavours/blobfox/reducers/modal.ts        |   83 +
 .../flavours/blobfox/reducers/mutes.js        |   31 +
 .../blobfox/reducers/notifications.js         |  376 ++++
 .../blobfox/reducers/picture_in_picture.js    |   26 +
 .../reducers/pinned_accounts_editor.js        |   58 +
 .../flavours/blobfox/reducers/polls.js        |   45 +
 .../blobfox/reducers/push_notifications.js    |   54 +
 .../blobfox/reducers/relationships.ts         |  123 ++
 .../flavours/blobfox/reducers/search.js       |   82 +
 .../flavours/blobfox/reducers/server.js       |   63 +
 .../flavours/blobfox/reducers/settings.js     |  203 ++
 .../flavours/blobfox/reducers/status_lists.js |  151 ++
 .../flavours/blobfox/reducers/statuses.js     |  191 ++
 .../flavours/blobfox/reducers/suggestions.js  |   40 +
 .../flavours/blobfox/reducers/tags.js         |   26 +
 .../flavours/blobfox/reducers/timelines.js    |  233 ++
 .../flavours/blobfox/reducers/trends.js       |   47 +
 .../flavours/blobfox/reducers/user_lists.js   |  216 ++
 app/javascript/flavours/blobfox/scroll.ts     |   50 +
 .../flavours/blobfox/selectors/accounts.ts    |   47 +
 .../flavours/blobfox/selectors/index.js       |  118 ++
 app/javascript/flavours/blobfox/settings.js   |   50 +
 .../flavours/blobfox/store/index.ts           |    8 +
 .../blobfox/store/middlewares/errors.ts       |   21 +
 .../blobfox/store/middlewares/loading_bar.ts  |   69 +
 .../blobfox/store/middlewares/sounds.ts       |   64 +
 .../flavours/blobfox/store/store.ts           |   39 +
 .../flavours/blobfox/store/typed_functions.ts |   15 +
 app/javascript/flavours/blobfox/stream.js     |  275 +++
 .../flavours/blobfox/styles/_mixins.scss      |   97 +
 .../flavours/blobfox/styles/about.scss        |   56 +
 .../blobfox/styles/accessibility.scss         |   55 +
 .../flavours/blobfox/styles/accounts.scss     |  381 ++++
 .../flavours/blobfox/styles/admin.scss        | 1884 +++++++++++++++++
 .../flavours/blobfox/styles/basics.scss       |  294 +++
 .../flavours/blobfox/styles/branding.scss     |    3 +
 .../blobfox/styles/components/about.scss      |  295 +++
 .../blobfox/styles/components/accounts.scss   |  807 +++++++
 .../styles/components/announcements.scss      |  233 ++
 .../blobfox/styles/components/boost.scss      |   44 +
 .../blobfox/styles/components/columns.scss    | 1364 ++++++++++++
 .../styles/components/compose_form.scss       |  685 ++++++
 .../blobfox/styles/components/directory.scss  |   68 +
 .../blobfox/styles/components/domains.scss    |   23 +
 .../blobfox/styles/components/doodle.scss     |   86 +
 .../blobfox/styles/components/drawer.scss     |  298 +++
 .../blobfox/styles/components/emoji.scss      |  104 +
 .../styles/components/emoji_picker.scss       |  261 +++
 .../blobfox/styles/components/explore.scss    |  147 ++
 .../blobfox/styles/components/index.scss      |   25 +
 .../blobfox/styles/components/lists.scss      |   94 +
 .../styles/components/local_settings.scss     |  173 ++
 .../blobfox/styles/components/media.scss      |  795 +++++++
 .../blobfox/styles/components/misc.scss       | 1740 +++++++++++++++
 .../blobfox/styles/components/modal.scss      | 1502 +++++++++++++
 .../styles/components/privacy_policy.scss     |  209 ++
 .../components/regeneration_indicator.scss    |   43 +
 .../blobfox/styles/components/search.scss     |  337 +++
 .../blobfox/styles/components/sensitive.scss  |   26 +
 .../blobfox/styles/components/signed_out.scss |  110 +
 .../styles/components/single_column.scss      |  335 +++
 .../blobfox/styles/components/status.scss     | 1212 +++++++++++
 .../flavours/blobfox/styles/containers.scss   |  109 +
 .../flavours/blobfox/styles/contrast.scss     |    3 +
 .../blobfox/styles/contrast/diff.scss         |   79 +
 .../blobfox/styles/contrast/variables.scss    |   22 +
 .../flavours/blobfox/styles/dashboard.scss    |  123 ++
 .../flavours/blobfox/styles/forms.scss        | 1224 +++++++++++
 .../flavours/blobfox/styles/index.scss        |   24 +
 .../flavours/blobfox/styles/lists.scss        |   19 +
 .../blobfox/styles/mastodon-light.scss        |    3 +
 .../blobfox/styles/mastodon-light/diff.scss   |  743 +++++++
 .../styles/mastodon-light/variables.scss      |   57 +
 .../flavours/blobfox/styles/modal.scss        |   37 +
 .../flavours/blobfox/styles/polls.scss        |  314 +++
 .../flavours/blobfox/styles/reset.scss        |   95 +
 .../flavours/blobfox/styles/rich_text.scss    |   99 +
 .../flavours/blobfox/styles/rtl.scss          |  123 ++
 .../flavours/blobfox/styles/statuses.scss     |  232 ++
 .../flavours/blobfox/styles/tables.scss       |  376 ++++
 .../flavours/blobfox/styles/variables.scss    |  103 +
 .../flavours/blobfox/styles/widgets.scss      |  402 ++++
 app/javascript/flavours/blobfox/theme.yml     |   48 +
 app/javascript/flavours/blobfox/types/util.ts |    1 +
 .../flavours/blobfox/utils/backend_links.js   |   18 +
 .../flavours/blobfox/utils/base64.ts          |   10 +
 .../flavours/blobfox/utils/config.js          |   10 +
 .../flavours/blobfox/utils/content_warning.js |   31 +
 .../flavours/blobfox/utils/filters.ts         |   16 +
 .../flavours/blobfox/utils/hashtag.js         |    8 +
 .../flavours/blobfox/utils/hashtags.ts        |   29 +
 app/javascript/flavours/blobfox/utils/html.js |    6 +
 .../flavours/blobfox/utils/icons.jsx          |   13 +
 app/javascript/flavours/blobfox/utils/idna.js |   10 +
 .../flavours/blobfox/utils/js_helpers.js      |    5 +
 .../flavours/blobfox/utils/log_out.js         |   35 +
 .../flavours/blobfox/utils/notifications.js   |   30 +
 .../flavours/blobfox/utils/numbers.ts         |   71 +
 .../blobfox/utils/privacy_preference.js       |    5 +
 .../flavours/blobfox/utils/react_helpers.js   |   21 +
 .../flavours/blobfox/utils/react_router.jsx   |   61 +
 .../flavours/blobfox/utils/resize_image.js    |  191 ++
 .../flavours/blobfox/utils/scrollbar.js       |   34 +
 app/javascript/flavours/blobfox/uuid.ts       |    8 +
 .../skins/blobfox/contrast/common.scss        |    1 +
 .../skins/blobfox/contrast/names.yml          |   12 +
 .../skins/blobfox/mastodon-light/common.scss  |    1 +
 .../skins/blobfox/mastodon-light/names.yml    |   12 +
 .../blobfox/queens-pink-contrast/common.scss  |    3 +
 .../blobfox/queens-pink-contrast/diff.scss    |  381 ++++
 .../blobfox/queens-pink-contrast/names.yml    |    4 +
 .../queens-pink-contrast/screenshot.jpg       |  Bin 0 -> 173565 bytes
 .../queens-pink-contrast/variables.scss       |   81 +
 .../skins/blobfox/solarpunk/common.scss       |    3 +
 .../skins/blobfox/solarpunk/diff.scss         |  143 ++
 .../skins/blobfox/solarpunk/names.yml         |    4 +
 .../skins/blobfox/solarpunk/variables.scss    |   75 +
 578 files changed, 71749 insertions(+)
 create mode 100644 app/javascript/flavours/blobfox/actions/account_notes.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/accounts.js
 create mode 100644 app/javascript/flavours/blobfox/actions/accounts_typed.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/alerts.js
 create mode 100644 app/javascript/flavours/blobfox/actions/announcements.js
 create mode 100644 app/javascript/flavours/blobfox/actions/app.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/blocks.js
 create mode 100644 app/javascript/flavours/blobfox/actions/bookmarks.js
 create mode 100644 app/javascript/flavours/blobfox/actions/boosts.js
 create mode 100644 app/javascript/flavours/blobfox/actions/bundles.js
 create mode 100644 app/javascript/flavours/blobfox/actions/columns.js
 create mode 100644 app/javascript/flavours/blobfox/actions/compose.js
 create mode 100644 app/javascript/flavours/blobfox/actions/conversations.js
 create mode 100644 app/javascript/flavours/blobfox/actions/custom_emojis.js
 create mode 100644 app/javascript/flavours/blobfox/actions/directory.js
 create mode 100644 app/javascript/flavours/blobfox/actions/domain_blocks.js
 create mode 100644 app/javascript/flavours/blobfox/actions/domain_blocks_typed.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/dropdown_menu.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/emojis.js
 create mode 100644 app/javascript/flavours/blobfox/actions/favourites.js
 create mode 100644 app/javascript/flavours/blobfox/actions/featured_tags.js
 create mode 100644 app/javascript/flavours/blobfox/actions/filters.js
 create mode 100644 app/javascript/flavours/blobfox/actions/height_cache.js
 create mode 100644 app/javascript/flavours/blobfox/actions/history.js
 create mode 100644 app/javascript/flavours/blobfox/actions/importer/index.js
 create mode 100644 app/javascript/flavours/blobfox/actions/importer/normalizer.js
 create mode 100644 app/javascript/flavours/blobfox/actions/interactions.js
 create mode 100644 app/javascript/flavours/blobfox/actions/languages.js
 create mode 100644 app/javascript/flavours/blobfox/actions/lists.js
 create mode 100644 app/javascript/flavours/blobfox/actions/local_settings.js
 create mode 100644 app/javascript/flavours/blobfox/actions/markers.js
 create mode 100644 app/javascript/flavours/blobfox/actions/modal.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/mutes.js
 create mode 100644 app/javascript/flavours/blobfox/actions/notifications.js
 create mode 100644 app/javascript/flavours/blobfox/actions/notifications_typed.ts
 create mode 100644 app/javascript/flavours/blobfox/actions/onboarding.js
 create mode 100644 app/javascript/flavours/blobfox/actions/picture_in_picture.js
 create mode 100644 app/javascript/flavours/blobfox/actions/pin_statuses.js
 create mode 100644 app/javascript/flavours/blobfox/actions/polls.js
 create mode 100644 app/javascript/flavours/blobfox/actions/push_notifications/index.js
 create mode 100644 app/javascript/flavours/blobfox/actions/push_notifications/registerer.js
 create mode 100644 app/javascript/flavours/blobfox/actions/push_notifications/setter.js
 create mode 100644 app/javascript/flavours/blobfox/actions/reports.js
 create mode 100644 app/javascript/flavours/blobfox/actions/search.js
 create mode 100644 app/javascript/flavours/blobfox/actions/server.js
 create mode 100644 app/javascript/flavours/blobfox/actions/settings.js
 create mode 100644 app/javascript/flavours/blobfox/actions/statuses.js
 create mode 100644 app/javascript/flavours/blobfox/actions/store.js
 create mode 100644 app/javascript/flavours/blobfox/actions/streaming.js
 create mode 100644 app/javascript/flavours/blobfox/actions/suggestions.js
 create mode 100644 app/javascript/flavours/blobfox/actions/tags.js
 create mode 100644 app/javascript/flavours/blobfox/actions/timelines.js
 create mode 100644 app/javascript/flavours/blobfox/actions/trends.js
 create mode 100644 app/javascript/flavours/blobfox/api.ts
 create mode 100644 app/javascript/flavours/blobfox/api_types/accounts.ts
 create mode 100644 app/javascript/flavours/blobfox/api_types/custom_emoji.ts
 create mode 100644 app/javascript/flavours/blobfox/api_types/relationships.ts
 create mode 100644 app/javascript/flavours/blobfox/blurhash.ts
 create mode 100644 app/javascript/flavours/blobfox/compare_id.ts
 create mode 100644 app/javascript/flavours/blobfox/components/__tests__/hashtag_bar.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/account.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/admin/Counter.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/admin/Dimension.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/admin/ImpactReport.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/admin/ReportReasonSelector.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/admin/Retention.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/admin/Trends.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/animated_number.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/attachment_list.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/autosuggest_emoji.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/autosuggest_hashtag.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/autosuggest_input.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/autosuggest_textarea.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/avatar.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/avatar_composite.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/avatar_overlay.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/blurhash.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/button.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/check.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/circular_progress.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/column.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/column_back_button.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/column_back_button_slim.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/column_header.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/counters.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/dismissable_banner.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/display_name.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/domain.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/dropdown_menu.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/edited_timestamp/containers/dropdown_menu_container.js
 create mode 100644 app/javascript/flavours/blobfox/components/edited_timestamp/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/empty_account.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/error_boundary.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/gifv.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/hashtag.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/hashtag_bar.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/icon.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/icon_button.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/icon_with_badge.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/inline_account.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/intersection_observer_article.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/link.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/load_gap.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/load_more.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/load_pending.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/loading_indicator.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/logo.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/media_attachments.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/media_gallery.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/modal_root.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/navigation_portal.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/not_signed_in_indicator.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/notification_purge_buttons.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/permalink.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/picture_in_picture_placeholder.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/poll.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/radio_button.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/regeneration_indicator.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/relative_timestamp.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/router.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/scrollable_list.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/server_banner.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/server_hero_image.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/setting_text.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/short_number.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/skeleton.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/status.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_action_bar.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_content.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_header.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_icons.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_list.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_prepend.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_reactions.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/status_visibility_icon.jsx
 create mode 100644 app/javascript/flavours/blobfox/components/timeline_hint.tsx
 create mode 100644 app/javascript/flavours/blobfox/components/verified_badge.tsx
 create mode 100644 app/javascript/flavours/blobfox/containers/account_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/containers/admin_component.jsx
 create mode 100644 app/javascript/flavours/blobfox/containers/compose_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/containers/domain_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/containers/dropdown_menu_container.js
 create mode 100644 app/javascript/flavours/blobfox/containers/intersection_observer_article_container.js
 create mode 100644 app/javascript/flavours/blobfox/containers/mastodon.jsx
 create mode 100644 app/javascript/flavours/blobfox/containers/media_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/containers/notification_purge_buttons_container.js
 create mode 100644 app/javascript/flavours/blobfox/containers/poll_container.js
 create mode 100644 app/javascript/flavours/blobfox/containers/scroll_container.js
 create mode 100644 app/javascript/flavours/blobfox/containers/status_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/about/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/components/account_note.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/components/action_bar.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/components/featured_tags.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/components/follow_request_note.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/components/header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/components/profile_column_header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account/containers/account_note_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/account/containers/featured_tags_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/account/containers/follow_request_note_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/account/navigation.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_gallery/components/media_item.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_gallery/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_timeline/components/header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_timeline/components/limited_account_hint.tsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_timeline/components/memorial_note.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_timeline/components/moved_note.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_timeline/containers/header_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/account_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/audio/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/audio/visualizer.js
 create mode 100644 app/javascript/flavours/blobfox/features/blocks/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/bookmarked_statuses/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/closed_registrations_modal/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/community_timeline/components/column_settings.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/community_timeline/containers/column_settings_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/community_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/action_bar.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/autosuggest_account.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/character_counter.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/compose_form.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/dropdown.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/dropdown_menu.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/emoji_picker_dropdown.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/language_dropdown.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/navigation_bar.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/options.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/poll_form.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/privacy_dropdown.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/publisher.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/reply_indicator.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/search.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/search_results.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/text_icon_button.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/textarea_icons.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/upload.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/upload_form.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/upload_progress.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/components/warning.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/autosuggest_account_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/compose_form_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/dropdown_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/emoji_picker_dropdown_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/header_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/language_dropdown_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/navigation_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/options_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/poll_form_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/privacy_dropdown_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/reply_indicator_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/search_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/search_results_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/sensitive_button_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/upload_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/upload_form_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/upload_progress_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/containers/warning_container.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/compose/util/counter.js
 create mode 100644 app/javascript/flavours/blobfox/features/compose/util/url_regex.js
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/components/column_settings.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/components/conversation.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/components/conversations_list.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/containers/column_settings_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/containers/conversation_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/containers/conversations_list_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/direct_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/directory/components/account_card.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/directory/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/domain_blocks/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji.js
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_compressed.d.ts
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_compressed.js
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_map.json
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_mart_data_light.ts
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_mart_search_light.js
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_picker.js
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_unicode_mapping_light.ts
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/emoji_utils.js
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/unicode_to_filename.js
 create mode 100644 app/javascript/flavours/blobfox/features/emoji/unicode_to_unified_name.js
 create mode 100644 app/javascript/flavours/blobfox/features/explore/components/search_section.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/components/story.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/links.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/results.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/statuses.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/suggestions.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/explore/tags.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/favourited_statuses/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/favourites/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/filters/added_to_filter.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/filters/select_filter.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/firehose/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/follow_requests/components/account_authorize.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/follow_requests/containers/account_authorize_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/follow_requests/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/followed_tags/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/followers/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/following/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/getting_started/components/announcements.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/getting_started/components/trends.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/getting_started/containers/announcements_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/getting_started/containers/trends_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/getting_started/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/getting_started_misc/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/hashtag_timeline/components/column_settings.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/hashtag_timeline/components/hashtag_header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/hashtag_timeline/containers/column_settings_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/hashtag_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/home_timeline/components/column_settings.tsx
 create mode 100644 app/javascript/flavours/blobfox/features/home_timeline/components/critical_update_banner.tsx
 create mode 100644 app/javascript/flavours/blobfox/features/home_timeline/components/explore_prompt.tsx
 create mode 100644 app/javascript/flavours/blobfox/features/home_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/interaction_modal/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/keyboard_shortcuts/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_adder/components/account.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_adder/components/list.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_adder/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_editor/components/account.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_editor/components/edit_list_form.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_editor/components/search.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_editor/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/list_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/lists/components/new_list_form.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/lists/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/local_settings/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/local_settings/navigation/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/local_settings/navigation/item/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/local_settings/page/deprecated_item/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/local_settings/page/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/local_settings/page/item/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/mutes/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/admin_report.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/admin_signup.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/clear_column_button.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/column_settings.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/filter_bar.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/follow.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/follow_request.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/grant_permission_button.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/notification.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/notifications_permission_banner.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/overlay.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/pill_bar_button.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/report.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/components/setting_toggle.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/containers/admin_report_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/containers/column_settings_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/containers/filter_bar_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/containers/follow_request_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/containers/notification_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/containers/overlay_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/notifications/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/onboarding/components/arrow_small_right.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/onboarding/components/progress_indicator.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/onboarding/components/step.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/onboarding/follows.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/onboarding/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/onboarding/share.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/picture_in_picture/components/footer.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/picture_in_picture/components/header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/picture_in_picture/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/account_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/search_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/pinned_accounts_editor/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/pinned_statuses/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/privacy_policy/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/public_timeline/components/column_settings.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/public_timeline/containers/column_settings_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/public_timeline/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/reblogs/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/category.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/comment.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/components/option.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/components/status_check_box.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/containers/status_check_box_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/report/rules.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/statuses.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/report/thanks.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/standalone/compose/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/status/components/action_bar.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/status/components/card.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/status/components/detailed_status.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/status/containers/detailed_status_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/status/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/subscribed_languages_modal/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/actions_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/audio_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/block_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/boost_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/bundle.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/bundle_column_error.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/bundle_modal_error.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/column.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/column_header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/column_link.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/column_loading.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/column_subheading.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/columns_area.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/compare_history_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/compose_panel.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/confirmation_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/deprecated_settings_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/disabled_account_banner.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/doodle_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/drawer_loading.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/embed_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/favourite_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/filter_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/focal_point_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/follow_requests_column_link.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/header.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/image_loader.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/image_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/link_footer.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/list_panel.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/media_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/modal_loading.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/modal_root.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/mute_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/navigation_panel.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/notifications_counter_icon.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/report_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/sign_in_banner.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/upload_area.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/video_modal.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/components/zoomable_image.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/containers/bundle_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/containers/columns_area_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/containers/loading_bar_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/containers/modal_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/containers/notifications_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/containers/status_list_container.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/async-components.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/fullscreen.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/get_rect_from_entry.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/intersection_observer_wrapper.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/optional_motion.js
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/react_router_helpers.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/reduced_motion.jsx
 create mode 100644 app/javascript/flavours/blobfox/features/ui/util/schedule_idle_task.js
 create mode 100644 app/javascript/flavours/blobfox/features/video/index.jsx
 create mode 100644 app/javascript/flavours/blobfox/hooks/useHovering.ts
 create mode 100644 app/javascript/flavours/blobfox/images/elephant_ui_disappointed.svg
 create mode 100644 app/javascript/flavours/blobfox/images/elephant_ui_working.svg
 create mode 100644 app/javascript/flavours/blobfox/images/glitch-preview.jpg
 create mode 100644 app/javascript/flavours/blobfox/images/logo_warn_glitch.svg
 create mode 100644 app/javascript/flavours/blobfox/images/mbstobon-ui-0.png
 create mode 100644 app/javascript/flavours/blobfox/images/mbstobon-ui-1.png
 create mode 100644 app/javascript/flavours/blobfox/images/mbstobon-ui-2.png
 create mode 100644 app/javascript/flavours/blobfox/images/mbstobon-ui-3.png
 create mode 100644 app/javascript/flavours/blobfox/images/wave-drawer-glitched.png
 create mode 100644 app/javascript/flavours/blobfox/images/wave-drawer.png
 create mode 100644 app/javascript/flavours/blobfox/initial_state.js
 create mode 100644 app/javascript/flavours/blobfox/is_mobile.ts
 create mode 100644 app/javascript/flavours/blobfox/load_keyboard_extensions.js
 create mode 100644 app/javascript/flavours/blobfox/main.jsx
 create mode 100644 app/javascript/flavours/blobfox/models/account.ts
 create mode 100644 app/javascript/flavours/blobfox/models/custom_emoji.ts
 create mode 100644 app/javascript/flavours/blobfox/models/relationship.ts
 create mode 100644 app/javascript/flavours/blobfox/names.yml
 create mode 100644 app/javascript/flavours/blobfox/packs/admin.jsx
 create mode 100644 app/javascript/flavours/blobfox/packs/common.js
 create mode 100644 app/javascript/flavours/blobfox/packs/error.js
 create mode 100644 app/javascript/flavours/blobfox/packs/home.js
 create mode 100644 app/javascript/flavours/blobfox/packs/public.jsx
 create mode 100644 app/javascript/flavours/blobfox/packs/settings.js
 create mode 100644 app/javascript/flavours/blobfox/packs/share.jsx
 create mode 100644 app/javascript/flavours/blobfox/packs/sign_up.js
 create mode 100644 app/javascript/flavours/blobfox/performance.js
 create mode 100644 app/javascript/flavours/blobfox/permissions.ts
 create mode 100644 app/javascript/flavours/blobfox/polyfills/extra_polyfills.ts
 create mode 100644 app/javascript/flavours/blobfox/polyfills/index.ts
 create mode 100644 app/javascript/flavours/blobfox/polyfills/intl.ts
 create mode 100644 app/javascript/flavours/blobfox/ready.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/accounts.ts
 create mode 100644 app/javascript/flavours/blobfox/reducers/accounts_map.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/alerts.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/announcements.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/blocks.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/boosts.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/compose.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/contexts.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/conversations.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/custom_emojis.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/domain_lists.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/dropdown_menu.ts
 create mode 100644 app/javascript/flavours/blobfox/reducers/filters.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/followed_tags.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/height_cache.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/history.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/index.ts
 create mode 100644 app/javascript/flavours/blobfox/reducers/list_adder.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/list_editor.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/lists.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/local_settings.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/markers.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/media_attachments.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/meta.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/modal.ts
 create mode 100644 app/javascript/flavours/blobfox/reducers/mutes.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/notifications.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/picture_in_picture.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/pinned_accounts_editor.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/polls.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/push_notifications.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/relationships.ts
 create mode 100644 app/javascript/flavours/blobfox/reducers/search.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/server.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/settings.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/status_lists.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/statuses.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/suggestions.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/tags.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/timelines.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/trends.js
 create mode 100644 app/javascript/flavours/blobfox/reducers/user_lists.js
 create mode 100644 app/javascript/flavours/blobfox/scroll.ts
 create mode 100644 app/javascript/flavours/blobfox/selectors/accounts.ts
 create mode 100644 app/javascript/flavours/blobfox/selectors/index.js
 create mode 100644 app/javascript/flavours/blobfox/settings.js
 create mode 100644 app/javascript/flavours/blobfox/store/index.ts
 create mode 100644 app/javascript/flavours/blobfox/store/middlewares/errors.ts
 create mode 100644 app/javascript/flavours/blobfox/store/middlewares/loading_bar.ts
 create mode 100644 app/javascript/flavours/blobfox/store/middlewares/sounds.ts
 create mode 100644 app/javascript/flavours/blobfox/store/store.ts
 create mode 100644 app/javascript/flavours/blobfox/store/typed_functions.ts
 create mode 100644 app/javascript/flavours/blobfox/stream.js
 create mode 100644 app/javascript/flavours/blobfox/styles/_mixins.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/about.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/accessibility.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/accounts.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/admin.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/basics.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/branding.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/about.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/accounts.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/announcements.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/boost.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/columns.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/compose_form.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/directory.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/domains.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/doodle.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/drawer.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/emoji.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/emoji_picker.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/explore.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/index.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/lists.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/local_settings.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/media.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/misc.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/modal.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/privacy_policy.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/regeneration_indicator.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/search.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/sensitive.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/signed_out.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/single_column.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/components/status.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/containers.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/contrast.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/contrast/diff.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/contrast/variables.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/dashboard.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/forms.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/index.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/lists.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/mastodon-light.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/mastodon-light/diff.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/mastodon-light/variables.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/modal.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/polls.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/reset.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/rich_text.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/rtl.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/statuses.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/tables.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/variables.scss
 create mode 100644 app/javascript/flavours/blobfox/styles/widgets.scss
 create mode 100644 app/javascript/flavours/blobfox/theme.yml
 create mode 100644 app/javascript/flavours/blobfox/types/util.ts
 create mode 100644 app/javascript/flavours/blobfox/utils/backend_links.js
 create mode 100644 app/javascript/flavours/blobfox/utils/base64.ts
 create mode 100644 app/javascript/flavours/blobfox/utils/config.js
 create mode 100644 app/javascript/flavours/blobfox/utils/content_warning.js
 create mode 100644 app/javascript/flavours/blobfox/utils/filters.ts
 create mode 100644 app/javascript/flavours/blobfox/utils/hashtag.js
 create mode 100644 app/javascript/flavours/blobfox/utils/hashtags.ts
 create mode 100644 app/javascript/flavours/blobfox/utils/html.js
 create mode 100644 app/javascript/flavours/blobfox/utils/icons.jsx
 create mode 100644 app/javascript/flavours/blobfox/utils/idna.js
 create mode 100644 app/javascript/flavours/blobfox/utils/js_helpers.js
 create mode 100644 app/javascript/flavours/blobfox/utils/log_out.js
 create mode 100644 app/javascript/flavours/blobfox/utils/notifications.js
 create mode 100644 app/javascript/flavours/blobfox/utils/numbers.ts
 create mode 100644 app/javascript/flavours/blobfox/utils/privacy_preference.js
 create mode 100644 app/javascript/flavours/blobfox/utils/react_helpers.js
 create mode 100644 app/javascript/flavours/blobfox/utils/react_router.jsx
 create mode 100644 app/javascript/flavours/blobfox/utils/resize_image.js
 create mode 100644 app/javascript/flavours/blobfox/utils/scrollbar.js
 create mode 100644 app/javascript/flavours/blobfox/uuid.ts
 create mode 100644 app/javascript/skins/blobfox/contrast/common.scss
 create mode 100644 app/javascript/skins/blobfox/contrast/names.yml
 create mode 100644 app/javascript/skins/blobfox/mastodon-light/common.scss
 create mode 100644 app/javascript/skins/blobfox/mastodon-light/names.yml
 create mode 100644 app/javascript/skins/blobfox/queens-pink-contrast/common.scss
 create mode 100644 app/javascript/skins/blobfox/queens-pink-contrast/diff.scss
 create mode 100644 app/javascript/skins/blobfox/queens-pink-contrast/names.yml
 create mode 100644 app/javascript/skins/blobfox/queens-pink-contrast/screenshot.jpg
 create mode 100644 app/javascript/skins/blobfox/queens-pink-contrast/variables.scss
 create mode 100644 app/javascript/skins/blobfox/solarpunk/common.scss
 create mode 100644 app/javascript/skins/blobfox/solarpunk/diff.scss
 create mode 100644 app/javascript/skins/blobfox/solarpunk/names.yml
 create mode 100644 app/javascript/skins/blobfox/solarpunk/variables.scss

diff --git a/app/javascript/flavours/blobfox/actions/account_notes.ts b/app/javascript/flavours/blobfox/actions/account_notes.ts
new file mode 100644
index 00000000000000..b49327584764c9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/account_notes.ts
@@ -0,0 +1,18 @@
+import type { ApiRelationshipJSON } from 'flavours/blobfox/api_types/relationships';
+import { createAppAsyncThunk } from 'flavours/blobfox/store/typed_functions';
+
+import api from '../api';
+
+export const submitAccountNote = createAppAsyncThunk(
+  'account_note/submit',
+  async (args: { id: string; value: string }, { getState }) => {
+    const response = await api(getState).post<ApiRelationshipJSON>(
+      `/api/v1/accounts/${args.id}/note`,
+      {
+        comment: args.value,
+      },
+    );
+
+    return { relationship: response.data };
+  },
+);
diff --git a/app/javascript/flavours/blobfox/actions/accounts.js b/app/javascript/flavours/blobfox/actions/accounts.js
new file mode 100644
index 00000000000000..a93c027def6332
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/accounts.js
@@ -0,0 +1,774 @@
+import api, { getLinks } from '../api';
+
+import {
+  followAccountSuccess, unfollowAccountSuccess,
+  authorizeFollowRequestSuccess, rejectFollowRequestSuccess,
+  followAccountRequest, followAccountFail,
+  unfollowAccountRequest, unfollowAccountFail,
+  muteAccountSuccess, unmuteAccountSuccess,
+  blockAccountSuccess, unblockAccountSuccess,
+  pinAccountSuccess, unpinAccountSuccess,
+  fetchRelationshipsSuccess,
+} from './accounts_typed';
+import { importFetchedAccount, importFetchedAccounts } from './importer';
+
+export const ACCOUNT_FETCH_REQUEST = 'ACCOUNT_FETCH_REQUEST';
+export const ACCOUNT_FETCH_SUCCESS = 'ACCOUNT_FETCH_SUCCESS';
+export const ACCOUNT_FETCH_FAIL    = 'ACCOUNT_FETCH_FAIL';
+
+export const ACCOUNT_LOOKUP_REQUEST = 'ACCOUNT_LOOKUP_REQUEST';
+export const ACCOUNT_LOOKUP_SUCCESS = 'ACCOUNT_LOOKUP_SUCCESS';
+export const ACCOUNT_LOOKUP_FAIL    = 'ACCOUNT_LOOKUP_FAIL';
+
+export const ACCOUNT_BLOCK_REQUEST = 'ACCOUNT_BLOCK_REQUEST';
+export const ACCOUNT_BLOCK_FAIL    = 'ACCOUNT_BLOCK_FAIL';
+
+export const ACCOUNT_UNBLOCK_REQUEST = 'ACCOUNT_UNBLOCK_REQUEST';
+export const ACCOUNT_UNBLOCK_FAIL    = 'ACCOUNT_UNBLOCK_FAIL';
+
+export const ACCOUNT_MUTE_REQUEST = 'ACCOUNT_MUTE_REQUEST';
+export const ACCOUNT_MUTE_FAIL    = 'ACCOUNT_MUTE_FAIL';
+
+export const ACCOUNT_UNMUTE_REQUEST = 'ACCOUNT_UNMUTE_REQUEST';
+export const ACCOUNT_UNMUTE_FAIL    = 'ACCOUNT_UNMUTE_FAIL';
+
+export const ACCOUNT_PIN_REQUEST = 'ACCOUNT_PIN_REQUEST';
+export const ACCOUNT_PIN_FAIL    = 'ACCOUNT_PIN_FAIL';
+
+export const ACCOUNT_UNPIN_REQUEST = 'ACCOUNT_UNPIN_REQUEST';
+export const ACCOUNT_UNPIN_FAIL    = 'ACCOUNT_UNPIN_FAIL';
+
+export const FOLLOWERS_FETCH_REQUEST = 'FOLLOWERS_FETCH_REQUEST';
+export const FOLLOWERS_FETCH_SUCCESS = 'FOLLOWERS_FETCH_SUCCESS';
+export const FOLLOWERS_FETCH_FAIL    = 'FOLLOWERS_FETCH_FAIL';
+
+export const FOLLOWERS_EXPAND_REQUEST = 'FOLLOWERS_EXPAND_REQUEST';
+export const FOLLOWERS_EXPAND_SUCCESS = 'FOLLOWERS_EXPAND_SUCCESS';
+export const FOLLOWERS_EXPAND_FAIL    = 'FOLLOWERS_EXPAND_FAIL';
+
+export const FOLLOWING_FETCH_REQUEST = 'FOLLOWING_FETCH_REQUEST';
+export const FOLLOWING_FETCH_SUCCESS = 'FOLLOWING_FETCH_SUCCESS';
+export const FOLLOWING_FETCH_FAIL    = 'FOLLOWING_FETCH_FAIL';
+
+export const FOLLOWING_EXPAND_REQUEST = 'FOLLOWING_EXPAND_REQUEST';
+export const FOLLOWING_EXPAND_SUCCESS = 'FOLLOWING_EXPAND_SUCCESS';
+export const FOLLOWING_EXPAND_FAIL    = 'FOLLOWING_EXPAND_FAIL';
+
+export const RELATIONSHIPS_FETCH_REQUEST = 'RELATIONSHIPS_FETCH_REQUEST';
+export const RELATIONSHIPS_FETCH_FAIL    = 'RELATIONSHIPS_FETCH_FAIL';
+
+export const FOLLOW_REQUESTS_FETCH_REQUEST = 'FOLLOW_REQUESTS_FETCH_REQUEST';
+export const FOLLOW_REQUESTS_FETCH_SUCCESS = 'FOLLOW_REQUESTS_FETCH_SUCCESS';
+export const FOLLOW_REQUESTS_FETCH_FAIL    = 'FOLLOW_REQUESTS_FETCH_FAIL';
+
+export const FOLLOW_REQUESTS_EXPAND_REQUEST = 'FOLLOW_REQUESTS_EXPAND_REQUEST';
+export const FOLLOW_REQUESTS_EXPAND_SUCCESS = 'FOLLOW_REQUESTS_EXPAND_SUCCESS';
+export const FOLLOW_REQUESTS_EXPAND_FAIL    = 'FOLLOW_REQUESTS_EXPAND_FAIL';
+
+export const FOLLOW_REQUEST_AUTHORIZE_REQUEST = 'FOLLOW_REQUEST_AUTHORIZE_REQUEST';
+export const FOLLOW_REQUEST_AUTHORIZE_SUCCESS = 'FOLLOW_REQUEST_AUTHORIZE_SUCCESS';
+export const FOLLOW_REQUEST_AUTHORIZE_FAIL    = 'FOLLOW_REQUEST_AUTHORIZE_FAIL';
+
+export const FOLLOW_REQUEST_REJECT_REQUEST = 'FOLLOW_REQUEST_REJECT_REQUEST';
+export const FOLLOW_REQUEST_REJECT_SUCCESS = 'FOLLOW_REQUEST_REJECT_SUCCESS';
+export const FOLLOW_REQUEST_REJECT_FAIL    = 'FOLLOW_REQUEST_REJECT_FAIL';
+
+export const PINNED_ACCOUNTS_FETCH_REQUEST = 'PINNED_ACCOUNTS_FETCH_REQUEST';
+export const PINNED_ACCOUNTS_FETCH_SUCCESS = 'PINNED_ACCOUNTS_FETCH_SUCCESS';
+export const PINNED_ACCOUNTS_FETCH_FAIL    = 'PINNED_ACCOUNTS_FETCH_FAIL';
+
+export const PINNED_ACCOUNTS_SUGGESTIONS_FETCH_REQUEST  = 'PINNED_ACCOUNTS_SUGGESTIONS_FETCH_REQUEST';
+export const PINNED_ACCOUNTS_SUGGESTIONS_FETCH_SUCCESS  = 'PINNED_ACCOUNTS_SUGGESTIONS_FETCH_SUCCESS';
+export const PINNED_ACCOUNTS_SUGGESTIONS_FETCH_FAIL     = 'PINNED_ACCOUNTS_SUGGESTIONS_FETCH_FAIL';
+
+export const PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CLEAR  = 'PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CLEAR';
+export const PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CHANGE = 'PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CHANGE';
+
+export const PINNED_ACCOUNTS_EDITOR_RESET = 'PINNED_ACCOUNTS_EDITOR_RESET';
+
+export const ACCOUNT_REVEAL = 'ACCOUNT_REVEAL';
+
+export * from './accounts_typed';
+
+export function fetchAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(fetchRelationships([id]));
+
+    if (getState().getIn(['accounts', id], null) !== null) {
+      return;
+    }
+
+    dispatch(fetchAccountRequest(id));
+
+    api(getState).get(`/api/v1/accounts/${id}`).then(response => {
+      dispatch(importFetchedAccount(response.data));
+      dispatch(fetchAccountSuccess());
+    }).catch(error => {
+      dispatch(fetchAccountFail(id, error));
+    });
+  };
+}
+
+export const lookupAccount = acct => (dispatch, getState) => {
+  dispatch(lookupAccountRequest(acct));
+
+  api(getState).get('/api/v1/accounts/lookup', { params: { acct } }).then(response => {
+    dispatch(fetchRelationships([response.data.id]));
+    dispatch(importFetchedAccount(response.data));
+    dispatch(lookupAccountSuccess());
+  }).catch(error => {
+    dispatch(lookupAccountFail(acct, error));
+  });
+};
+
+export const lookupAccountRequest = (acct) => ({
+  type: ACCOUNT_LOOKUP_REQUEST,
+  acct,
+});
+
+export const lookupAccountSuccess = () => ({
+  type: ACCOUNT_LOOKUP_SUCCESS,
+});
+
+export const lookupAccountFail = (acct, error) => ({
+  type: ACCOUNT_LOOKUP_FAIL,
+  acct,
+  error,
+  skipAlert: true,
+});
+
+export function fetchAccountRequest(id) {
+  return {
+    type: ACCOUNT_FETCH_REQUEST,
+    id,
+  };
+}
+
+export function fetchAccountSuccess() {
+  return {
+    type: ACCOUNT_FETCH_SUCCESS,
+  };
+}
+
+export function fetchAccountFail(id, error) {
+  return {
+    type: ACCOUNT_FETCH_FAIL,
+    id,
+    error,
+    skipAlert: true,
+  };
+}
+
+export function followAccount(id, options = { reblogs: true }) {
+  return (dispatch, getState) => {
+    const alreadyFollowing = getState().getIn(['relationships', id, 'following']);
+    const locked = getState().getIn(['accounts', id, 'locked'], false);
+
+    dispatch(followAccountRequest({ id, locked }));
+
+    api(getState).post(`/api/v1/accounts/${id}/follow`, options).then(response => {
+      dispatch(followAccountSuccess({relationship: response.data, alreadyFollowing}));
+    }).catch(error => {
+      dispatch(followAccountFail({ id, error, locked }));
+    });
+  };
+}
+
+export function unfollowAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(unfollowAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/unfollow`).then(response => {
+      dispatch(unfollowAccountSuccess({relationship: response.data, statuses: getState().get('statuses')}));
+    }).catch(error => {
+      dispatch(unfollowAccountFail({ id, error }));
+    });
+  };
+}
+
+export function blockAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(blockAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/block`).then(response => {
+      // Pass in entire statuses map so we can use it to filter stuff in different parts of the reducers
+      dispatch(blockAccountSuccess({ relationship: response.data, statuses: getState().get('statuses') }));
+    }).catch(error => {
+      dispatch(blockAccountFail({ id, error }));
+    });
+  };
+}
+
+export function unblockAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(unblockAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/unblock`).then(response => {
+      dispatch(unblockAccountSuccess({ relationship: response.data }));
+    }).catch(error => {
+      dispatch(unblockAccountFail({ id, error }));
+    });
+  };
+}
+
+export function blockAccountRequest(id) {
+  return {
+    type: ACCOUNT_BLOCK_REQUEST,
+    id,
+  };
+}
+export function blockAccountFail(error) {
+  return {
+    type: ACCOUNT_BLOCK_FAIL,
+    error,
+  };
+}
+
+export function unblockAccountRequest(id) {
+  return {
+    type: ACCOUNT_UNBLOCK_REQUEST,
+    id,
+  };
+}
+
+export function unblockAccountFail(error) {
+  return {
+    type: ACCOUNT_UNBLOCK_FAIL,
+    error,
+  };
+}
+
+
+export function muteAccount(id, notifications, duration=0) {
+  return (dispatch, getState) => {
+    dispatch(muteAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/mute`, { notifications, duration }).then(response => {
+      // Pass in entire statuses map so we can use it to filter stuff in different parts of the reducers
+      dispatch(muteAccountSuccess({ relationship: response.data, statuses: getState().get('statuses') }));
+    }).catch(error => {
+      dispatch(muteAccountFail({ id, error }));
+    });
+  };
+}
+
+export function unmuteAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(unmuteAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/unmute`).then(response => {
+      dispatch(unmuteAccountSuccess({ relationship: response.data }));
+    }).catch(error => {
+      dispatch(unmuteAccountFail({ id, error }));
+    });
+  };
+}
+
+export function muteAccountRequest(id) {
+  return {
+    type: ACCOUNT_MUTE_REQUEST,
+    id,
+  };
+}
+
+export function muteAccountFail(error) {
+  return {
+    type: ACCOUNT_MUTE_FAIL,
+    error,
+  };
+}
+
+export function unmuteAccountRequest(id) {
+  return {
+    type: ACCOUNT_UNMUTE_REQUEST,
+    id,
+  };
+}
+
+export function unmuteAccountFail(error) {
+  return {
+    type: ACCOUNT_UNMUTE_FAIL,
+    error,
+  };
+}
+
+
+export function fetchFollowers(id) {
+  return (dispatch, getState) => {
+    dispatch(fetchFollowersRequest(id));
+
+    api(getState).get(`/api/v1/accounts/${id}/followers`).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchFollowersSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => {
+      dispatch(fetchFollowersFail(id, error));
+    });
+  };
+}
+
+export function fetchFollowersRequest(id) {
+  return {
+    type: FOLLOWERS_FETCH_REQUEST,
+    id,
+  };
+}
+
+export function fetchFollowersSuccess(id, accounts, next) {
+  return {
+    type: FOLLOWERS_FETCH_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function fetchFollowersFail(id, error) {
+  return {
+    type: FOLLOWERS_FETCH_FAIL,
+    id,
+    error,
+    skipNotFound: true,
+  };
+}
+
+export function expandFollowers(id) {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'followers', id, 'next']);
+
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandFollowersRequest(id));
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandFollowersSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => {
+      dispatch(expandFollowersFail(id, error));
+    });
+  };
+}
+
+export function expandFollowersRequest(id) {
+  return {
+    type: FOLLOWERS_EXPAND_REQUEST,
+    id,
+  };
+}
+
+export function expandFollowersSuccess(id, accounts, next) {
+  return {
+    type: FOLLOWERS_EXPAND_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function expandFollowersFail(id, error) {
+  return {
+    type: FOLLOWERS_EXPAND_FAIL,
+    id,
+    error,
+  };
+}
+
+export function fetchFollowing(id) {
+  return (dispatch, getState) => {
+    dispatch(fetchFollowingRequest(id));
+
+    api(getState).get(`/api/v1/accounts/${id}/following`).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchFollowingSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => {
+      dispatch(fetchFollowingFail(id, error));
+    });
+  };
+}
+
+export function fetchFollowingRequest(id) {
+  return {
+    type: FOLLOWING_FETCH_REQUEST,
+    id,
+  };
+}
+
+export function fetchFollowingSuccess(id, accounts, next) {
+  return {
+    type: FOLLOWING_FETCH_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function fetchFollowingFail(id, error) {
+  return {
+    type: FOLLOWING_FETCH_FAIL,
+    id,
+    error,
+    skipNotFound: true,
+  };
+}
+
+export function expandFollowing(id) {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'following', id, 'next']);
+
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandFollowingRequest(id));
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandFollowingSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => {
+      dispatch(expandFollowingFail(id, error));
+    });
+  };
+}
+
+export function expandFollowingRequest(id) {
+  return {
+    type: FOLLOWING_EXPAND_REQUEST,
+    id,
+  };
+}
+
+export function expandFollowingSuccess(id, accounts, next) {
+  return {
+    type: FOLLOWING_EXPAND_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function expandFollowingFail(id, error) {
+  return {
+    type: FOLLOWING_EXPAND_FAIL,
+    id,
+    error,
+  };
+}
+
+export function fetchRelationships(accountIds) {
+  return (dispatch, getState) => {
+    const state = getState();
+    const loadedRelationships = state.get('relationships');
+    const newAccountIds = accountIds.filter(id => loadedRelationships.get(id, null) === null);
+    const signedIn = !!state.getIn(['meta', 'me']);
+
+    if (!signedIn || newAccountIds.length === 0) {
+      return;
+    }
+
+    dispatch(fetchRelationshipsRequest(newAccountIds));
+
+    api(getState).get(`/api/v1/accounts/relationships?with_suspended=true&${newAccountIds.map(id => `id[]=${id}`).join('&')}`).then(response => {
+      dispatch(fetchRelationshipsSuccess({ relationships: response.data }));
+    }).catch(error => {
+      dispatch(fetchRelationshipsFail(error));
+    });
+  };
+}
+
+export function fetchRelationshipsRequest(ids) {
+  return {
+    type: RELATIONSHIPS_FETCH_REQUEST,
+    ids,
+    skipLoading: true,
+  };
+}
+
+export function fetchRelationshipsFail(error) {
+  return {
+    type: RELATIONSHIPS_FETCH_FAIL,
+    error,
+    skipLoading: true,
+    skipNotFound: true,
+  };
+}
+
+export function fetchFollowRequests() {
+  return (dispatch, getState) => {
+    dispatch(fetchFollowRequestsRequest());
+
+    api(getState).get('/api/v1/follow_requests').then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchFollowRequestsSuccess(response.data, next ? next.uri : null));
+    }).catch(error => dispatch(fetchFollowRequestsFail(error)));
+  };
+}
+
+export function fetchFollowRequestsRequest() {
+  return {
+    type: FOLLOW_REQUESTS_FETCH_REQUEST,
+  };
+}
+
+export function fetchFollowRequestsSuccess(accounts, next) {
+  return {
+    type: FOLLOW_REQUESTS_FETCH_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function fetchFollowRequestsFail(error) {
+  return {
+    type: FOLLOW_REQUESTS_FETCH_FAIL,
+    error,
+  };
+}
+
+export function expandFollowRequests() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'follow_requests', 'next']);
+
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandFollowRequestsRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandFollowRequestsSuccess(response.data, next ? next.uri : null));
+    }).catch(error => dispatch(expandFollowRequestsFail(error)));
+  };
+}
+
+export function expandFollowRequestsRequest() {
+  return {
+    type: FOLLOW_REQUESTS_EXPAND_REQUEST,
+  };
+}
+
+export function expandFollowRequestsSuccess(accounts, next) {
+  return {
+    type: FOLLOW_REQUESTS_EXPAND_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function expandFollowRequestsFail(error) {
+  return {
+    type: FOLLOW_REQUESTS_EXPAND_FAIL,
+    error,
+  };
+}
+
+export function authorizeFollowRequest(id) {
+  return (dispatch, getState) => {
+    dispatch(authorizeFollowRequestRequest(id));
+
+    api(getState)
+      .post(`/api/v1/follow_requests/${id}/authorize`)
+      .then(() => dispatch(authorizeFollowRequestSuccess({ id })))
+      .catch(error => dispatch(authorizeFollowRequestFail(id, error)));
+  };
+}
+
+export function authorizeFollowRequestRequest(id) {
+  return {
+    type: FOLLOW_REQUEST_AUTHORIZE_REQUEST,
+    id,
+  };
+}
+
+export function authorizeFollowRequestFail(id, error) {
+  return {
+    type: FOLLOW_REQUEST_AUTHORIZE_FAIL,
+    id,
+    error,
+  };
+}
+
+
+export function rejectFollowRequest(id) {
+  return (dispatch, getState) => {
+    dispatch(rejectFollowRequestRequest(id));
+
+    api(getState)
+      .post(`/api/v1/follow_requests/${id}/reject`)
+      .then(() => dispatch(rejectFollowRequestSuccess({ id })))
+      .catch(error => dispatch(rejectFollowRequestFail(id, error)));
+  };
+}
+
+export function rejectFollowRequestRequest(id) {
+  return {
+    type: FOLLOW_REQUEST_REJECT_REQUEST,
+    id,
+  };
+}
+
+export function rejectFollowRequestFail(id, error) {
+  return {
+    type: FOLLOW_REQUEST_REJECT_FAIL,
+    id,
+    error,
+  };
+}
+
+export function pinAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(pinAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/pin`).then(response => {
+      dispatch(pinAccountSuccess({ relationship: response.data }));
+    }).catch(error => {
+      dispatch(pinAccountFail(error));
+    });
+  };
+}
+
+export function unpinAccount(id) {
+  return (dispatch, getState) => {
+    dispatch(unpinAccountRequest(id));
+
+    api(getState).post(`/api/v1/accounts/${id}/unpin`).then(response => {
+      dispatch(unpinAccountSuccess({ relationship: response.data }));
+    }).catch(error => {
+      dispatch(unpinAccountFail(error));
+    });
+  };
+}
+
+export function pinAccountRequest(id) {
+  return {
+    type: ACCOUNT_PIN_REQUEST,
+    id,
+  };
+}
+
+export function pinAccountFail(error) {
+  return {
+    type: ACCOUNT_PIN_FAIL,
+    error,
+  };
+}
+
+export function unpinAccountRequest(id) {
+  return {
+    type: ACCOUNT_UNPIN_REQUEST,
+    id,
+  };
+}
+
+export function unpinAccountFail(error) {
+  return {
+    type: ACCOUNT_UNPIN_FAIL,
+    error,
+  };
+}
+
+export function fetchPinnedAccounts() {
+  return (dispatch, getState) => {
+    dispatch(fetchPinnedAccountsRequest());
+
+    api(getState).get('/api/v1/endorsements', { params: { limit: 0 } }).then(response => {
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchPinnedAccountsSuccess(response.data));
+    }).catch(err => dispatch(fetchPinnedAccountsFail(err)));
+  };
+}
+
+export function fetchPinnedAccountsRequest() {
+  return {
+    type: PINNED_ACCOUNTS_FETCH_REQUEST,
+  };
+}
+
+export function fetchPinnedAccountsSuccess(accounts, next) {
+  return {
+    type: PINNED_ACCOUNTS_FETCH_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function fetchPinnedAccountsFail(error) {
+  return {
+    type: PINNED_ACCOUNTS_FETCH_FAIL,
+    error,
+  };
+}
+
+export function fetchPinnedAccountsSuggestions(q) {
+  return (dispatch, getState) => {
+    dispatch(fetchPinnedAccountsSuggestionsRequest());
+
+    const params = {
+      q,
+      resolve: false,
+      limit: 4,
+      following: true,
+    };
+
+    api(getState).get('/api/v1/accounts/search', { params }).then(response => {
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchPinnedAccountsSuggestionsSuccess(q, response.data));
+    }).catch(err => dispatch(fetchPinnedAccountsSuggestionsFail(err)));
+  };
+}
+
+export function fetchPinnedAccountsSuggestionsRequest() {
+  return {
+    type: PINNED_ACCOUNTS_SUGGESTIONS_FETCH_REQUEST,
+  };
+}
+
+export function fetchPinnedAccountsSuggestionsSuccess(query, accounts) {
+  return {
+    type: PINNED_ACCOUNTS_SUGGESTIONS_FETCH_SUCCESS,
+    query,
+    accounts,
+  };
+}
+
+export function fetchPinnedAccountsSuggestionsFail(error) {
+  return {
+    type: PINNED_ACCOUNTS_SUGGESTIONS_FETCH_FAIL,
+    error,
+  };
+}
+
+export function clearPinnedAccountsSuggestions() {
+  return {
+    type: PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CLEAR,
+  };
+}
+
+export function changePinnedAccountsSuggestions(value) {
+  return {
+    type: PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CHANGE,
+    value,
+  };
+}
+
+export function resetPinnedAccountsEditor() {
+  return {
+    type: PINNED_ACCOUNTS_EDITOR_RESET,
+  };
+}
+
diff --git a/app/javascript/flavours/blobfox/actions/accounts_typed.ts b/app/javascript/flavours/blobfox/actions/accounts_typed.ts
new file mode 100644
index 00000000000000..f57bf854f8ee58
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/accounts_typed.ts
@@ -0,0 +1,97 @@
+import { createAction } from '@reduxjs/toolkit';
+
+import type { ApiAccountJSON } from 'flavours/blobfox/api_types/accounts';
+import type { ApiRelationshipJSON } from 'flavours/blobfox/api_types/relationships';
+
+export const revealAccount = createAction<{
+  id: string;
+}>('accounts/revealAccount');
+
+export const importAccounts = createAction<{ accounts: ApiAccountJSON[] }>(
+  'accounts/importAccounts',
+);
+
+function actionWithSkipLoadingTrue<Args extends object>(args: Args) {
+  return {
+    payload: {
+      ...args,
+      skipLoading: true,
+    },
+  };
+}
+
+export const followAccountSuccess = createAction(
+  'accounts/followAccount/SUCCESS',
+  actionWithSkipLoadingTrue<{
+    relationship: ApiRelationshipJSON;
+    alreadyFollowing: boolean;
+  }>,
+);
+
+export const unfollowAccountSuccess = createAction(
+  'accounts/unfollowAccount/SUCCESS',
+  actionWithSkipLoadingTrue<{
+    relationship: ApiRelationshipJSON;
+    statuses: unknown;
+    alreadyFollowing?: boolean;
+  }>,
+);
+
+export const authorizeFollowRequestSuccess = createAction<{ id: string }>(
+  'accounts/followRequestAuthorize/SUCCESS',
+);
+
+export const rejectFollowRequestSuccess = createAction<{ id: string }>(
+  'accounts/followRequestReject/SUCCESS',
+);
+
+export const followAccountRequest = createAction(
+  'accounts/follow/REQUEST',
+  actionWithSkipLoadingTrue<{ id: string; locked: boolean }>,
+);
+
+export const followAccountFail = createAction(
+  'accounts/follow/FAIL',
+  actionWithSkipLoadingTrue<{ id: string; error: string; locked: boolean }>,
+);
+
+export const unfollowAccountRequest = createAction(
+  'accounts/unfollow/REQUEST',
+  actionWithSkipLoadingTrue<{ id: string }>,
+);
+
+export const unfollowAccountFail = createAction(
+  'accounts/unfollow/FAIL',
+  actionWithSkipLoadingTrue<{ id: string; error: string }>,
+);
+
+export const blockAccountSuccess = createAction<{
+  relationship: ApiRelationshipJSON;
+  statuses: unknown;
+}>('accounts/block/SUCCESS');
+
+export const unblockAccountSuccess = createAction<{
+  relationship: ApiRelationshipJSON;
+}>('accounts/unblock/SUCCESS');
+
+export const muteAccountSuccess = createAction<{
+  relationship: ApiRelationshipJSON;
+  statuses: unknown;
+}>('accounts/mute/SUCCESS');
+
+export const unmuteAccountSuccess = createAction<{
+  relationship: ApiRelationshipJSON;
+}>('accounts/unmute/SUCCESS');
+
+export const pinAccountSuccess = createAction<{
+  relationship: ApiRelationshipJSON;
+}>('accounts/pin/SUCCESS');
+
+export const unpinAccountSuccess = createAction<{
+  relationship: ApiRelationshipJSON;
+}>('accounts/unpin/SUCCESS');
+
+export const fetchRelationshipsSuccess = createAction(
+  'relationships/fetch/SUCCESS',
+  actionWithSkipLoadingTrue<{ relationships: ApiRelationshipJSON[] }>,
+);
diff --git a/app/javascript/flavours/blobfox/actions/alerts.js b/app/javascript/flavours/blobfox/actions/alerts.js
new file mode 100644
index 00000000000000..42834146bf5ba6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/alerts.js
@@ -0,0 +1,59 @@
+import { defineMessages } from 'react-intl';
+
+const messages = defineMessages({
+  unexpectedTitle: { id: 'alert.unexpected.title', defaultMessage: 'Oops!' },
+  unexpectedMessage: { id: 'alert.unexpected.message', defaultMessage: 'An unexpected error occurred.' },
+  rateLimitedTitle: { id: 'alert.rate_limited.title', defaultMessage: 'Rate limited' },
+  rateLimitedMessage: { id: 'alert.rate_limited.message', defaultMessage: 'Please retry after {retry_time, time, medium}.' },
+});
+
+export const ALERT_SHOW    = 'ALERT_SHOW';
+export const ALERT_DISMISS = 'ALERT_DISMISS';
+export const ALERT_CLEAR   = 'ALERT_CLEAR';
+export const ALERT_NOOP    = 'ALERT_NOOP';
+
+export const dismissAlert = alert => ({
+  type: ALERT_DISMISS,
+  alert,
+});
+
+export const clearAlert = () => ({
+  type: ALERT_CLEAR,
+});
+
+export const showAlert = alert => ({
+  type: ALERT_SHOW,
+  alert,
+});
+
+export const showAlertForError = (error, skipNotFound = false) => {
+  if (error.response) {
+    const { data, status, statusText, headers } = error.response;
+
+    // Skip these errors as they are reflected in the UI
+    if (skipNotFound && (status === 404 || status === 410)) {
+      return { type: ALERT_NOOP };
+    }
+
+    // Rate limit errors
+    if (status === 429 && headers['x-ratelimit-reset']) {
+      return showAlert({
+        title: messages.rateLimitedTitle,
+        message: messages.rateLimitedMessage,
+        values: { 'retry_time': new Date(headers['x-ratelimit-reset']) },
+      });
+    }
+
+    return showAlert({
+      title: `${status}`,
+      message: data.error || statusText,
+    });
+  }
+
+  console.error(error);
+
+  return showAlert({
+    title: messages.unexpectedTitle,
+    message: messages.unexpectedMessage,
+  });
+};
diff --git a/app/javascript/flavours/blobfox/actions/announcements.js b/app/javascript/flavours/blobfox/actions/announcements.js
new file mode 100644
index 00000000000000..339c5f3adc5a8e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/announcements.js
@@ -0,0 +1,181 @@
+import api from '../api';
+
+import { normalizeAnnouncement } from './importer/normalizer';
+
+export const ANNOUNCEMENTS_FETCH_REQUEST = 'ANNOUNCEMENTS_FETCH_REQUEST';
+export const ANNOUNCEMENTS_FETCH_SUCCESS = 'ANNOUNCEMENTS_FETCH_SUCCESS';
+export const ANNOUNCEMENTS_FETCH_FAIL    = 'ANNOUNCEMENTS_FETCH_FAIL';
+export const ANNOUNCEMENTS_UPDATE        = 'ANNOUNCEMENTS_UPDATE';
+export const ANNOUNCEMENTS_DELETE        = 'ANNOUNCEMENTS_DELETE';
+
+export const ANNOUNCEMENTS_DISMISS_REQUEST = 'ANNOUNCEMENTS_DISMISS_REQUEST';
+export const ANNOUNCEMENTS_DISMISS_SUCCESS = 'ANNOUNCEMENTS_DISMISS_SUCCESS';
+export const ANNOUNCEMENTS_DISMISS_FAIL    = 'ANNOUNCEMENTS_DISMISS_FAIL';
+
+export const ANNOUNCEMENTS_REACTION_ADD_REQUEST = 'ANNOUNCEMENTS_REACTION_ADD_REQUEST';
+export const ANNOUNCEMENTS_REACTION_ADD_SUCCESS = 'ANNOUNCEMENTS_REACTION_ADD_SUCCESS';
+export const ANNOUNCEMENTS_REACTION_ADD_FAIL    = 'ANNOUNCEMENTS_REACTION_ADD_FAIL';
+
+export const ANNOUNCEMENTS_REACTION_REMOVE_REQUEST = 'ANNOUNCEMENTS_REACTION_REMOVE_REQUEST';
+export const ANNOUNCEMENTS_REACTION_REMOVE_SUCCESS = 'ANNOUNCEMENTS_REACTION_REMOVE_SUCCESS';
+export const ANNOUNCEMENTS_REACTION_REMOVE_FAIL    = 'ANNOUNCEMENTS_REACTION_REMOVE_FAIL';
+
+export const ANNOUNCEMENTS_REACTION_UPDATE = 'ANNOUNCEMENTS_REACTION_UPDATE';
+
+export const ANNOUNCEMENTS_TOGGLE_SHOW = 'ANNOUNCEMENTS_TOGGLE_SHOW';
+
+const noOp = () => {};
+
+export const fetchAnnouncements = (done = noOp) => (dispatch, getState) => {
+  dispatch(fetchAnnouncementsRequest());
+
+  api(getState).get('/api/v1/announcements').then(response => {
+    dispatch(fetchAnnouncementsSuccess(response.data.map(x => normalizeAnnouncement(x))));
+  }).catch(error => {
+    dispatch(fetchAnnouncementsFail(error));
+  }).finally(() => {
+    done();
+  });
+};
+
+export const fetchAnnouncementsRequest = () => ({
+  type: ANNOUNCEMENTS_FETCH_REQUEST,
+  skipLoading: true,
+});
+
+export const fetchAnnouncementsSuccess = announcements => ({
+  type: ANNOUNCEMENTS_FETCH_SUCCESS,
+  announcements,
+  skipLoading: true,
+});
+
+export const fetchAnnouncementsFail= error => ({
+  type: ANNOUNCEMENTS_FETCH_FAIL,
+  error,
+  skipLoading: true,
+  skipAlert: true,
+});
+
+export const updateAnnouncements = announcement => ({
+  type: ANNOUNCEMENTS_UPDATE,
+  announcement: normalizeAnnouncement(announcement),
+});
+
+export const dismissAnnouncement = announcementId => (dispatch, getState) => {
+  dispatch(dismissAnnouncementRequest(announcementId));
+
+  api(getState).post(`/api/v1/announcements/${announcementId}/dismiss`).then(() => {
+    dispatch(dismissAnnouncementSuccess(announcementId));
+  }).catch(error => {
+    dispatch(dismissAnnouncementFail(announcementId, error));
+  });
+};
+
+export const dismissAnnouncementRequest = announcementId => ({
+  type: ANNOUNCEMENTS_DISMISS_REQUEST,
+  id: announcementId,
+});
+
+export const dismissAnnouncementSuccess = announcementId => ({
+  type: ANNOUNCEMENTS_DISMISS_SUCCESS,
+  id: announcementId,
+});
+
+export const dismissAnnouncementFail = (announcementId, error) => ({
+  type: ANNOUNCEMENTS_DISMISS_FAIL,
+  id: announcementId,
+  error,
+});
+
+export const addReaction = (announcementId, name) => (dispatch, getState) => {
+  const announcement = getState().getIn(['announcements', 'items']).find(x => x.get('id') === announcementId);
+
+  let alreadyAdded = false;
+
+  if (announcement) {
+    const reaction = announcement.get('reactions').find(x => x.get('name') === name);
+    if (reaction && reaction.get('me')) {
+      alreadyAdded = true;
+    }
+  }
+
+  if (!alreadyAdded) {
+    dispatch(addReactionRequest(announcementId, name, alreadyAdded));
+  }
+
+  api(getState).put(`/api/v1/announcements/${announcementId}/reactions/${encodeURIComponent(name)}`).then(() => {
+    dispatch(addReactionSuccess(announcementId, name, alreadyAdded));
+  }).catch(err => {
+    if (!alreadyAdded) {
+      dispatch(addReactionFail(announcementId, name, err));
+    }
+  });
+};
+
+export const addReactionRequest = (announcementId, name) => ({
+  type: ANNOUNCEMENTS_REACTION_ADD_REQUEST,
+  id: announcementId,
+  name,
+  skipLoading: true,
+});
+
+export const addReactionSuccess = (announcementId, name) => ({
+  type: ANNOUNCEMENTS_REACTION_ADD_SUCCESS,
+  id: announcementId,
+  name,
+  skipLoading: true,
+});
+
+export const addReactionFail = (announcementId, name, error) => ({
+  type: ANNOUNCEMENTS_REACTION_ADD_FAIL,
+  id: announcementId,
+  name,
+  error,
+  skipLoading: true,
+});
+
+export const removeReaction = (announcementId, name) => (dispatch, getState) => {
+  dispatch(removeReactionRequest(announcementId, name));
+
+  api(getState).delete(`/api/v1/announcements/${announcementId}/reactions/${encodeURIComponent(name)}`).then(() => {
+    dispatch(removeReactionSuccess(announcementId, name));
+  }).catch(err => {
+    dispatch(removeReactionFail(announcementId, name, err));
+  });
+};
+
+export const removeReactionRequest = (announcementId, name) => ({
+  type: ANNOUNCEMENTS_REACTION_REMOVE_REQUEST,
+  id: announcementId,
+  name,
+  skipLoading: true,
+});
+
+export const removeReactionSuccess = (announcementId, name) => ({
+  type: ANNOUNCEMENTS_REACTION_REMOVE_SUCCESS,
+  id: announcementId,
+  name,
+  skipLoading: true,
+});
+
+export const removeReactionFail = (announcementId, name, error) => ({
+  type: ANNOUNCEMENTS_REACTION_REMOVE_FAIL,
+  id: announcementId,
+  name,
+  error,
+  skipLoading: true,
+});
+
+export const updateReaction = reaction => ({
+  type: ANNOUNCEMENTS_REACTION_UPDATE,
+  reaction,
+});
+
+export const toggleShowAnnouncements = () => ({
+  type: ANNOUNCEMENTS_TOGGLE_SHOW,
+});
+
+export const deleteAnnouncement = id => ({
+  type: ANNOUNCEMENTS_DELETE,
+  id,
+});
diff --git a/app/javascript/flavours/blobfox/actions/app.ts b/app/javascript/flavours/blobfox/actions/app.ts
new file mode 100644
index 00000000000000..6fbfc07f68c931
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/app.ts
@@ -0,0 +1,9 @@
+import { createAction } from '@reduxjs/toolkit';
+
+import type { LayoutType } from '../is_mobile';
+
+interface ChangeLayoutPayload {
+  layout: LayoutType;
+}
+export const changeLayout =
+  createAction<ChangeLayoutPayload>('APP_LAYOUT_CHANGE');
diff --git a/app/javascript/flavours/blobfox/actions/blocks.js b/app/javascript/flavours/blobfox/actions/blocks.js
new file mode 100644
index 00000000000000..e293657ad36ef9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/blocks.js
@@ -0,0 +1,100 @@
+import api, { getLinks } from '../api';
+
+import { fetchRelationships } from './accounts';
+import { importFetchedAccounts } from './importer';
+import { openModal } from './modal';
+
+export const BLOCKS_FETCH_REQUEST = 'BLOCKS_FETCH_REQUEST';
+export const BLOCKS_FETCH_SUCCESS = 'BLOCKS_FETCH_SUCCESS';
+export const BLOCKS_FETCH_FAIL    = 'BLOCKS_FETCH_FAIL';
+
+export const BLOCKS_EXPAND_REQUEST = 'BLOCKS_EXPAND_REQUEST';
+export const BLOCKS_EXPAND_SUCCESS = 'BLOCKS_EXPAND_SUCCESS';
+export const BLOCKS_EXPAND_FAIL    = 'BLOCKS_EXPAND_FAIL';
+
+export const BLOCKS_INIT_MODAL = 'BLOCKS_INIT_MODAL';
+
+export function fetchBlocks() {
+  return (dispatch, getState) => {
+    dispatch(fetchBlocksRequest());
+
+    api(getState).get('/api/v1/blocks').then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchBlocksSuccess(response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => dispatch(fetchBlocksFail(error)));
+  };
+}
+
+export function fetchBlocksRequest() {
+  return {
+    type: BLOCKS_FETCH_REQUEST,
+  };
+}
+
+export function fetchBlocksSuccess(accounts, next) {
+  return {
+    type: BLOCKS_FETCH_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function fetchBlocksFail(error) {
+  return {
+    type: BLOCKS_FETCH_FAIL,
+    error,
+  };
+}
+
+export function expandBlocks() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'blocks', 'next']);
+
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandBlocksRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandBlocksSuccess(response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => dispatch(expandBlocksFail(error)));
+  };
+}
+
+export function expandBlocksRequest() {
+  return {
+    type: BLOCKS_EXPAND_REQUEST,
+  };
+}
+
+export function expandBlocksSuccess(accounts, next) {
+  return {
+    type: BLOCKS_EXPAND_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function expandBlocksFail(error) {
+  return {
+    type: BLOCKS_EXPAND_FAIL,
+    error,
+  };
+}
+
+export function initBlockModal(account) {
+  return dispatch => {
+    dispatch({
+      type: BLOCKS_INIT_MODAL,
+      account,
+    });
+
+    dispatch(openModal({ modalType: 'BLOCK' }));
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/bookmarks.js b/app/javascript/flavours/blobfox/actions/bookmarks.js
new file mode 100644
index 00000000000000..0b16f61e63635a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/bookmarks.js
@@ -0,0 +1,91 @@
+import api, { getLinks } from '../api';
+
+import { importFetchedStatuses } from './importer';
+
+export const BOOKMARKED_STATUSES_FETCH_REQUEST = 'BOOKMARKED_STATUSES_FETCH_REQUEST';
+export const BOOKMARKED_STATUSES_FETCH_SUCCESS = 'BOOKMARKED_STATUSES_FETCH_SUCCESS';
+export const BOOKMARKED_STATUSES_FETCH_FAIL    = 'BOOKMARKED_STATUSES_FETCH_FAIL';
+
+export const BOOKMARKED_STATUSES_EXPAND_REQUEST = 'BOOKMARKED_STATUSES_EXPAND_REQUEST';
+export const BOOKMARKED_STATUSES_EXPAND_SUCCESS = 'BOOKMARKED_STATUSES_EXPAND_SUCCESS';
+export const BOOKMARKED_STATUSES_EXPAND_FAIL    = 'BOOKMARKED_STATUSES_EXPAND_FAIL';
+
+export function fetchBookmarkedStatuses() {
+  return (dispatch, getState) => {
+    if (getState().getIn(['status_lists', 'bookmarks', 'isLoading'])) {
+      return;
+    }
+
+    dispatch(fetchBookmarkedStatusesRequest());
+
+    api(getState).get('/api/v1/bookmarks').then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedStatuses(response.data));
+      dispatch(fetchBookmarkedStatusesSuccess(response.data, next ? next.uri : null));
+    }).catch(error => {
+      dispatch(fetchBookmarkedStatusesFail(error));
+    });
+  };
+}
+
+export function fetchBookmarkedStatusesRequest() {
+  return {
+    type: BOOKMARKED_STATUSES_FETCH_REQUEST,
+  };
+}
+
+export function fetchBookmarkedStatusesSuccess(statuses, next) {
+  return {
+    type: BOOKMARKED_STATUSES_FETCH_SUCCESS,
+    statuses,
+    next,
+  };
+}
+
+export function fetchBookmarkedStatusesFail(error) {
+  return {
+    type: BOOKMARKED_STATUSES_FETCH_FAIL,
+    error,
+  };
+}
+
+export function expandBookmarkedStatuses() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['status_lists', 'bookmarks', 'next'], null);
+
+    if (url === null || getState().getIn(['status_lists', 'bookmarks', 'isLoading'])) {
+      return;
+    }
+
+    dispatch(expandBookmarkedStatusesRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedStatuses(response.data));
+      dispatch(expandBookmarkedStatusesSuccess(response.data, next ? next.uri : null));
+    }).catch(error => {
+      dispatch(expandBookmarkedStatusesFail(error));
+    });
+  };
+}
+
+export function expandBookmarkedStatusesRequest() {
+  return {
+    type: BOOKMARKED_STATUSES_EXPAND_REQUEST,
+  };
+}
+
+export function expandBookmarkedStatusesSuccess(statuses, next) {
+  return {
+    type: BOOKMARKED_STATUSES_EXPAND_SUCCESS,
+    statuses,
+    next,
+  };
+}
+
+export function expandBookmarkedStatusesFail(error) {
+  return {
+    type: BOOKMARKED_STATUSES_EXPAND_FAIL,
+    error,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/boosts.js b/app/javascript/flavours/blobfox/actions/boosts.js
new file mode 100644
index 00000000000000..1fc2e391e26827
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/boosts.js
@@ -0,0 +1,32 @@
+import { openModal } from './modal';
+
+export const BOOSTS_INIT_MODAL = 'BOOSTS_INIT_MODAL';
+export const BOOSTS_CHANGE_PRIVACY = 'BOOSTS_CHANGE_PRIVACY';
+
+export function initBoostModal(props) {
+  return (dispatch, getState) => {
+    const default_privacy = getState().getIn(['compose', 'default_privacy']);
+
+    const privacy = props.status.get('visibility') === 'private' ? 'private' : default_privacy;
+
+    dispatch({
+      type: BOOSTS_INIT_MODAL,
+      privacy,
+    });
+
+    dispatch(openModal({
+      modalType: 'BOOST',
+      modalProps: props,
+    }));
+  };
+}
+
+
+export function changeBoostPrivacy(privacy) {
+  return dispatch => {
+    dispatch({
+      type: BOOSTS_CHANGE_PRIVACY,
+      privacy,
+    });
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/bundles.js b/app/javascript/flavours/blobfox/actions/bundles.js
new file mode 100644
index 00000000000000..ecc9c8f7d3ec22
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/bundles.js
@@ -0,0 +1,25 @@
+export const BUNDLE_FETCH_REQUEST = 'BUNDLE_FETCH_REQUEST';
+export const BUNDLE_FETCH_SUCCESS = 'BUNDLE_FETCH_SUCCESS';
+export const BUNDLE_FETCH_FAIL = 'BUNDLE_FETCH_FAIL';
+
+export function fetchBundleRequest(skipLoading) {
+  return {
+    type: BUNDLE_FETCH_REQUEST,
+    skipLoading,
+  };
+}
+
+export function fetchBundleSuccess(skipLoading) {
+  return {
+    type: BUNDLE_FETCH_SUCCESS,
+    skipLoading,
+  };
+}
+
+export function fetchBundleFail(error, skipLoading) {
+  return {
+    type: BUNDLE_FETCH_FAIL,
+    error,
+    skipLoading,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/columns.js b/app/javascript/flavours/blobfox/actions/columns.js
new file mode 100644
index 00000000000000..302c3f0f9b34cb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/columns.js
@@ -0,0 +1,54 @@
+import { saveSettings } from './settings';
+
+export const COLUMN_ADD           = 'COLUMN_ADD';
+export const COLUMN_REMOVE        = 'COLUMN_REMOVE';
+export const COLUMN_MOVE          = 'COLUMN_MOVE';
+export const COLUMN_PARAMS_CHANGE = 'COLUMN_PARAMS_CHANGE';
+
+export function addColumn(id, params) {
+  return dispatch => {
+    dispatch({
+      type: COLUMN_ADD,
+      id,
+      params,
+    });
+
+    dispatch(saveSettings());
+  };
+}
+
+export function removeColumn(uuid) {
+  return dispatch => {
+    dispatch({
+      type: COLUMN_REMOVE,
+      uuid,
+    });
+
+    dispatch(saveSettings());
+  };
+}
+
+export function moveColumn(uuid, direction) {
+  return dispatch => {
+    dispatch({
+      type: COLUMN_MOVE,
+      uuid,
+      direction,
+    });
+
+    dispatch(saveSettings());
+  };
+}
+
+export function changeColumnParams(uuid, path, value) {
+  return dispatch => {
+    dispatch({
+      type: COLUMN_PARAMS_CHANGE,
+      uuid,
+      path,
+      value,
+    });
+
+    dispatch(saveSettings());
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/compose.js b/app/javascript/flavours/blobfox/actions/compose.js
new file mode 100644
index 00000000000000..0c1cbdd052c800
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/compose.js
@@ -0,0 +1,840 @@
+import { defineMessages } from 'react-intl';
+
+import axios from 'axios';
+import { throttle } from 'lodash';
+
+import api from 'flavours/blobfox/api';
+import { search as emojiSearch } from 'flavours/blobfox/features/emoji/emoji_mart_search_light';
+import { tagHistory } from 'flavours/blobfox/settings';
+import { recoverHashtags } from 'flavours/blobfox/utils/hashtag';
+
+import { showAlert, showAlertForError } from './alerts';
+import { useEmoji } from './emojis';
+import { importFetchedAccounts, importFetchedStatus } from './importer';
+import { openModal } from './modal';
+import { updateTimeline } from './timelines';
+
+/** @type {AbortController | undefined} */
+let fetchComposeSuggestionsAccountsController;
+/** @type {AbortController | undefined} */
+let fetchComposeSuggestionsTagsController;
+
+export const COMPOSE_CHANGE          = 'COMPOSE_CHANGE';
+export const COMPOSE_CYCLE_ELEFRIEND = 'COMPOSE_CYCLE_ELEFRIEND';
+export const COMPOSE_SUBMIT_REQUEST  = 'COMPOSE_SUBMIT_REQUEST';
+export const COMPOSE_SUBMIT_SUCCESS  = 'COMPOSE_SUBMIT_SUCCESS';
+export const COMPOSE_SUBMIT_FAIL     = 'COMPOSE_SUBMIT_FAIL';
+export const COMPOSE_REPLY           = 'COMPOSE_REPLY';
+export const COMPOSE_REPLY_CANCEL    = 'COMPOSE_REPLY_CANCEL';
+export const COMPOSE_DIRECT          = 'COMPOSE_DIRECT';
+export const COMPOSE_MENTION         = 'COMPOSE_MENTION';
+export const COMPOSE_RESET           = 'COMPOSE_RESET';
+
+export const COMPOSE_UPLOAD_REQUEST    = 'COMPOSE_UPLOAD_REQUEST';
+export const COMPOSE_UPLOAD_SUCCESS    = 'COMPOSE_UPLOAD_SUCCESS';
+export const COMPOSE_UPLOAD_FAIL       = 'COMPOSE_UPLOAD_FAIL';
+export const COMPOSE_UPLOAD_PROGRESS   = 'COMPOSE_UPLOAD_PROGRESS';
+export const COMPOSE_UPLOAD_PROCESSING = 'COMPOSE_UPLOAD_PROCESSING';
+export const COMPOSE_UPLOAD_UNDO       = 'COMPOSE_UPLOAD_UNDO';
+
+export const THUMBNAIL_UPLOAD_REQUEST  = 'THUMBNAIL_UPLOAD_REQUEST';
+export const THUMBNAIL_UPLOAD_SUCCESS  = 'THUMBNAIL_UPLOAD_SUCCESS';
+export const THUMBNAIL_UPLOAD_FAIL     = 'THUMBNAIL_UPLOAD_FAIL';
+export const THUMBNAIL_UPLOAD_PROGRESS = 'THUMBNAIL_UPLOAD_PROGRESS';
+
+export const COMPOSE_SUGGESTIONS_CLEAR = 'COMPOSE_SUGGESTIONS_CLEAR';
+export const COMPOSE_SUGGESTIONS_READY = 'COMPOSE_SUGGESTIONS_READY';
+export const COMPOSE_SUGGESTION_SELECT = 'COMPOSE_SUGGESTION_SELECT';
+export const COMPOSE_SUGGESTION_IGNORE = 'COMPOSE_SUGGESTION_IGNORE';
+export const COMPOSE_SUGGESTION_TAGS_UPDATE = 'COMPOSE_SUGGESTION_TAGS_UPDATE';
+
+export const COMPOSE_TAG_HISTORY_UPDATE = 'COMPOSE_TAG_HISTORY_UPDATE';
+
+export const COMPOSE_MOUNT   = 'COMPOSE_MOUNT';
+export const COMPOSE_UNMOUNT = 'COMPOSE_UNMOUNT';
+
+export const COMPOSE_ADVANCED_OPTIONS_CHANGE = 'COMPOSE_ADVANCED_OPTIONS_CHANGE';
+export const COMPOSE_SENSITIVITY_CHANGE  = 'COMPOSE_SENSITIVITY_CHANGE';
+export const COMPOSE_SPOILERNESS_CHANGE  = 'COMPOSE_SPOILERNESS_CHANGE';
+export const COMPOSE_SPOILER_TEXT_CHANGE = 'COMPOSE_SPOILER_TEXT_CHANGE';
+export const COMPOSE_VISIBILITY_CHANGE   = 'COMPOSE_VISIBILITY_CHANGE';
+export const COMPOSE_LISTABILITY_CHANGE  = 'COMPOSE_LISTABILITY_CHANGE';
+export const COMPOSE_CONTENT_TYPE_CHANGE = 'COMPOSE_CONTENT_TYPE_CHANGE';
+export const COMPOSE_LANGUAGE_CHANGE     = 'COMPOSE_LANGUAGE_CHANGE';
+
+export const COMPOSE_EMOJI_INSERT = 'COMPOSE_EMOJI_INSERT';
+
+export const COMPOSE_UPLOAD_CHANGE_REQUEST     = 'COMPOSE_UPLOAD_UPDATE_REQUEST';
+export const COMPOSE_UPLOAD_CHANGE_SUCCESS     = 'COMPOSE_UPLOAD_UPDATE_SUCCESS';
+export const COMPOSE_UPLOAD_CHANGE_FAIL        = 'COMPOSE_UPLOAD_UPDATE_FAIL';
+
+export const COMPOSE_DOODLE_SET        = 'COMPOSE_DOODLE_SET';
+
+export const COMPOSE_POLL_ADD             = 'COMPOSE_POLL_ADD';
+export const COMPOSE_POLL_REMOVE          = 'COMPOSE_POLL_REMOVE';
+export const COMPOSE_POLL_OPTION_ADD      = 'COMPOSE_POLL_OPTION_ADD';
+export const COMPOSE_POLL_OPTION_CHANGE   = 'COMPOSE_POLL_OPTION_CHANGE';
+export const COMPOSE_POLL_OPTION_REMOVE   = 'COMPOSE_POLL_OPTION_REMOVE';
+export const COMPOSE_POLL_SETTINGS_CHANGE = 'COMPOSE_POLL_SETTINGS_CHANGE';
+
+export const INIT_MEDIA_EDIT_MODAL = 'INIT_MEDIA_EDIT_MODAL';
+
+export const COMPOSE_CHANGE_MEDIA_DESCRIPTION = 'COMPOSE_CHANGE_MEDIA_DESCRIPTION';
+export const COMPOSE_CHANGE_MEDIA_FOCUS       = 'COMPOSE_CHANGE_MEDIA_FOCUS';
+
+export const COMPOSE_SET_STATUS = 'COMPOSE_SET_STATUS';
+export const COMPOSE_FOCUS = 'COMPOSE_FOCUS';
+
+const messages = defineMessages({
+  uploadErrorLimit: { id: 'upload_error.limit', defaultMessage: 'File upload limit exceeded.' },
+  uploadErrorPoll:  { id: 'upload_error.poll', defaultMessage: 'File upload not allowed with polls.' },
+  open: { id: 'compose.published.open', defaultMessage: 'Open' },
+  published: { id: 'compose.published.body', defaultMessage: 'Post published.' },
+  saved: { id: 'compose.saved.body', defaultMessage: 'Post saved.' },
+});
+
+export const ensureComposeIsVisible = (getState, routerHistory) => {
+  if (!getState().getIn(['compose', 'mounted'])) {
+    routerHistory.push('/publish');
+  }
+};
+
+export function setComposeToStatus(status, text, spoiler_text, content_type) {
+  return{
+    type: COMPOSE_SET_STATUS,
+    status,
+    text,
+    spoiler_text,
+    content_type,
+  };
+}
+
+export function changeCompose(text) {
+  return {
+    type: COMPOSE_CHANGE,
+    text: text,
+  };
+}
+
+export function cycleElefriendCompose() {
+  return {
+    type: COMPOSE_CYCLE_ELEFRIEND,
+  };
+}
+
+export function replyCompose(status, routerHistory) {
+  return (dispatch, getState) => {
+    const prependCWRe = getState().getIn(['local_settings', 'prepend_cw_re']);
+    dispatch({
+      type: COMPOSE_REPLY,
+      status: status,
+      prependCWRe: prependCWRe,
+    });
+
+    ensureComposeIsVisible(getState, routerHistory);
+  };
+}
+
+export function cancelReplyCompose() {
+  return {
+    type: COMPOSE_REPLY_CANCEL,
+  };
+}
+
+export function resetCompose() {
+  return {
+    type: COMPOSE_RESET,
+  };
+}
+
+export const focusCompose = (routerHistory, defaultText) => dispatch => {
+  dispatch({
+    type: COMPOSE_FOCUS,
+    defaultText,
+  });
+
+  ensureComposeIsVisible(routerHistory);
+};
+
+export function mentionCompose(account, routerHistory) {
+  return (dispatch, getState) => {
+    dispatch({
+      type: COMPOSE_MENTION,
+      account: account,
+    });
+
+    ensureComposeIsVisible(getState, routerHistory);
+  };
+}
+
+export function directCompose(account, routerHistory) {
+  return (dispatch, getState) => {
+    dispatch({
+      type: COMPOSE_DIRECT,
+      account: account,
+    });
+
+    ensureComposeIsVisible(getState, routerHistory);
+  };
+}
+
+export function submitCompose(routerHistory) {
+  return function (dispatch, getState) {
+    let status     = getState().getIn(['compose', 'text'], '');
+    const media    = getState().getIn(['compose', 'media_attachments']);
+    const statusId = getState().getIn(['compose', 'id'], null);
+    const spoilers = getState().getIn(['compose', 'spoiler']) || getState().getIn(['local_settings', 'always_show_spoilers_field']);
+    let spoilerText = spoilers ? getState().getIn(['compose', 'spoiler_text'], '') : '';
+
+    if ((!status || !status.length) && media.size === 0) {
+      return;
+    }
+
+    if (getState().getIn(['compose', 'advanced_options', 'do_not_federate'])) {
+      status = status + ' 👁️';
+    }
+
+    dispatch(submitComposeRequest());
+
+    // If we're editing a post with media attachments, those have not
+    // necessarily been changed on the server. Do it now in the same
+    // API call.
+    let media_attributes;
+    if (statusId !== null) {
+      media_attributes = media.map(item => {
+        let focus;
+
+        if (item.getIn(['meta', 'focus'])) {
+          focus = `${item.getIn(['meta', 'focus', 'x']).toFixed(2)},${item.getIn(['meta', 'focus', 'y']).toFixed(2)}`;
+        }
+
+        return {
+          id: item.get('id'),
+          description: item.get('description'),
+          focus,
+        };
+      });
+    }
+
+    api(getState).request({
+      url: statusId === null ? '/api/v1/statuses' : `/api/v1/statuses/${statusId}`,
+      method: statusId === null ? 'post' : 'put',
+      data: {
+        status,
+        content_type: getState().getIn(['compose', 'content_type']),
+        in_reply_to_id: getState().getIn(['compose', 'in_reply_to'], null),
+        media_ids: media.map(item => item.get('id')),
+        media_attributes,
+        sensitive: getState().getIn(['compose', 'sensitive']) || (spoilerText.length > 0 && media.size !== 0),
+        spoiler_text: spoilerText,
+        visibility: getState().getIn(['compose', 'privacy']),
+        poll: getState().getIn(['compose', 'poll'], null),
+        language: getState().getIn(['compose', 'language']),
+      },
+      headers: {
+        'Idempotency-Key': getState().getIn(['compose', 'idempotencyKey']),
+      },
+    }).then(function (response) {
+      if (routerHistory
+          && (routerHistory.location.pathname === '/publish' || routerHistory.location.pathname === '/statuses/new')
+          && window.history.state
+          && !getState().getIn(['compose', 'advanced_options', 'threaded_mode'])) {
+        routerHistory.goBack();
+      }
+
+      dispatch(insertIntoTagHistory(response.data.tags, status));
+      dispatch(submitComposeSuccess({ ...response.data }));
+
+      //  If the response has no data then we can't do anything else.
+      if (!response.data) {
+        return;
+      }
+
+      // To make the app more responsive, immediately push the status
+      // into the columns
+      const insertIfOnline = timelineId => {
+        const timeline = getState().getIn(['timelines', timelineId]);
+
+        if (timeline && timeline.get('items').size > 0 && timeline.getIn(['items', 0]) !== null && timeline.get('online')) {
+          dispatch(updateTimeline(timelineId, { ...response.data }));
+        }
+      };
+
+      if (statusId) {
+        dispatch(importFetchedStatus({ ...response.data }));
+      }
+
+      if (statusId === null) {
+        insertIfOnline('home');
+      }
+
+      if (statusId === null && response.data.in_reply_to_id === null && response.data.visibility === 'public') {
+        insertIfOnline('community');
+        if (!response.data.local_only) {
+          insertIfOnline('public');
+        }
+      } else if (statusId === null && response.data.visibility === 'direct') {
+        insertIfOnline('direct');
+      }
+
+      dispatch(showAlert({
+        message: statusId === null ? messages.published : messages.saved,
+        action: messages.open,
+        dismissAfter: 10000,
+        onClick: () => routerHistory.push(`/@${response.data.account.username}/${response.data.id}`),
+      }));
+    }).catch(function (error) {
+      dispatch(submitComposeFail(error));
+    });
+  };
+}
+
+export function submitComposeRequest() {
+  return {
+    type: COMPOSE_SUBMIT_REQUEST,
+  };
+}
+
+export function submitComposeSuccess(status) {
+  return {
+    type: COMPOSE_SUBMIT_SUCCESS,
+    status: status,
+  };
+}
+
+export function submitComposeFail(error) {
+  return {
+    type: COMPOSE_SUBMIT_FAIL,
+    error: error,
+  };
+}
+
+export function doodleSet(options) {
+  return {
+    type: COMPOSE_DOODLE_SET,
+    options: options,
+  };
+}
+
+export function uploadCompose(files) {
+  return function (dispatch, getState) {
+    const uploadLimit = 4;
+    const media = getState().getIn(['compose', 'media_attachments']);
+    const pending = getState().getIn(['compose', 'pending_media_attachments']);
+    const progress = new Array(files.length).fill(0);
+
+    let total = Array.from(files).reduce((a, v) => a + v.size, 0);
+
+    if (files.length + media.size + pending > uploadLimit) {
+      dispatch(showAlert({ message: messages.uploadErrorLimit }));
+      return;
+    }
+
+    if (getState().getIn(['compose', 'poll'])) {
+      dispatch(showAlert({ message: messages.uploadErrorPoll }));
+      return;
+    }
+
+    dispatch(uploadComposeRequest());
+
+    for (const [i, file] of Array.from(files).entries()) {
+      if (media.size + i > 3) break;
+
+      const data = new FormData();
+      data.append('file', file);
+
+      api(getState).post('/api/v2/media', data, {
+        onUploadProgress: function({ loaded }){
+          progress[i] = loaded;
+          dispatch(uploadComposeProgress(progress.reduce((a, v) => a + v, 0), total));
+        },
+      }).then(({ status, data }) => {
+        // If server-side processing of the media attachment has not completed yet,
+        // poll the server until it is, before showing the media attachment as uploaded
+
+        if (status === 200) {
+          dispatch(uploadComposeSuccess(data, file));
+        } else if (status === 202) {
+          dispatch(uploadComposeProcessing());
+
+          let tryCount = 1;
+
+          const poll = () => {
+            api(getState).get(`/api/v1/media/${data.id}`).then(response => {
+              if (response.status === 200) {
+                dispatch(uploadComposeSuccess(response.data, file));
+              } else if (response.status === 206) {
+                const retryAfter = (Math.log2(tryCount) || 1) * 1000;
+                tryCount += 1;
+                setTimeout(() => poll(), retryAfter);
+              }
+            }).catch(error => dispatch(uploadComposeFail(error)));
+          };
+
+          poll();
+        }
+        
+      }).catch(error => dispatch(uploadComposeFail(error)));
+    }
+  };
+}
+
+export const uploadComposeProcessing = () => ({
+  type: COMPOSE_UPLOAD_PROCESSING,
+});
+
+export const uploadThumbnail = (id, file) => (dispatch, getState) => {
+  dispatch(uploadThumbnailRequest());
+
+  const total = file.size;
+  const data = new FormData();
+
+  data.append('thumbnail', file);
+
+  api(getState).put(`/api/v1/media/${id}`, data, {
+    onUploadProgress: ({ loaded }) => {
+      dispatch(uploadThumbnailProgress(loaded, total));
+    },
+  }).then(({ data }) => {
+    dispatch(uploadThumbnailSuccess(data));
+  }).catch(error => {
+    dispatch(uploadThumbnailFail(id, error));
+  });
+};
+
+export const uploadThumbnailRequest = () => ({
+  type: THUMBNAIL_UPLOAD_REQUEST,
+  skipLoading: true,
+});
+
+export const uploadThumbnailProgress = (loaded, total) => ({
+  type: THUMBNAIL_UPLOAD_PROGRESS,
+  loaded,
+  total,
+  skipLoading: true,
+});
+
+export const uploadThumbnailSuccess = media => ({
+  type: THUMBNAIL_UPLOAD_SUCCESS,
+  media,
+  skipLoading: true,
+});
+
+export const uploadThumbnailFail = error => ({
+  type: THUMBNAIL_UPLOAD_FAIL,
+  error,
+  skipLoading: true,
+});
+
+export function initMediaEditModal(id) {
+  return dispatch => {
+    dispatch({
+      type: INIT_MEDIA_EDIT_MODAL,
+      id,
+    });
+
+    dispatch(openModal({
+      modalType: 'FOCAL_POINT',
+      modalProps: { id },
+    }));
+  };
+}
+
+export function onChangeMediaDescription(description) {
+  return {
+    type: COMPOSE_CHANGE_MEDIA_DESCRIPTION,
+    description,
+  };
+}
+
+export function onChangeMediaFocus(focusX, focusY) {
+  return {
+    type: COMPOSE_CHANGE_MEDIA_FOCUS,
+    focusX,
+    focusY,
+  };
+}
+
+export function changeUploadCompose(id, params) {
+  return (dispatch, getState) => {
+    dispatch(changeUploadComposeRequest());
+
+    let media = getState().getIn(['compose', 'media_attachments']).find((item) => item.get('id') === id);
+
+    // Editing already-attached media is deferred to editing the post itself.
+    // For simplicity's sake, fake an API reply.
+    if (media && !media.get('unattached')) {
+      const { focus, ...other } = params;
+      const data = { ...media.toJS(), ...other };
+
+      if (focus) {
+        const [x, y] = focus.split(',');
+        data.meta = { focus: { x: parseFloat(x), y: parseFloat(y) } };
+      }
+
+      dispatch(changeUploadComposeSuccess(data, true));
+    } else {
+      api(getState).put(`/api/v1/media/${id}`, params).then(response => {
+        dispatch(changeUploadComposeSuccess(response.data, false));
+      }).catch(error => {
+        dispatch(changeUploadComposeFail(id, error));
+      });
+    }
+  };
+}
+
+export function changeUploadComposeRequest() {
+  return {
+    type: COMPOSE_UPLOAD_CHANGE_REQUEST,
+    skipLoading: true,
+  };
+}
+
+export function changeUploadComposeSuccess(media, attached) {
+  return {
+    type: COMPOSE_UPLOAD_CHANGE_SUCCESS,
+    media: media,
+    attached: attached,
+    skipLoading: true,
+  };
+}
+
+export function changeUploadComposeFail(error) {
+  return {
+    type: COMPOSE_UPLOAD_CHANGE_FAIL,
+    error: error,
+    skipLoading: true,
+  };
+}
+
+export function uploadComposeRequest() {
+  return {
+    type: COMPOSE_UPLOAD_REQUEST,
+    skipLoading: true,
+  };
+}
+
+export function uploadComposeProgress(loaded, total) {
+  return {
+    type: COMPOSE_UPLOAD_PROGRESS,
+    loaded: loaded,
+    total: total,
+  };
+}
+
+export function uploadComposeSuccess(media, file) {
+  return {
+    type: COMPOSE_UPLOAD_SUCCESS,
+    media: media,
+    file: file,
+    skipLoading: true,
+  };
+}
+
+export function uploadComposeFail(error) {
+  return {
+    type: COMPOSE_UPLOAD_FAIL,
+    error: error,
+    skipLoading: true,
+  };
+}
+
+export function undoUploadCompose(media_id) {
+  return {
+    type: COMPOSE_UPLOAD_UNDO,
+    media_id: media_id,
+  };
+}
+
+export function clearComposeSuggestions() {
+  if (fetchComposeSuggestionsAccountsController) {
+    fetchComposeSuggestionsAccountsController.abort();
+  }
+  return {
+    type: COMPOSE_SUGGESTIONS_CLEAR,
+  };
+}
+
+const fetchComposeSuggestionsAccounts = throttle((dispatch, getState, token) => {
+  if (fetchComposeSuggestionsAccountsController) {
+    fetchComposeSuggestionsAccountsController.abort();
+  }
+
+  fetchComposeSuggestionsAccountsController = new AbortController();
+
+  api(getState).get('/api/v1/accounts/search', {
+    signal: fetchComposeSuggestionsAccountsController.signal,
+
+    params: {
+      q: token.slice(1),
+      resolve: false,
+      limit: 4,
+    },
+  }).then(response => {
+    dispatch(importFetchedAccounts(response.data));
+    dispatch(readyComposeSuggestionsAccounts(token, response.data));
+  }).catch(error => {
+    if (!axios.isCancel(error)) {
+      dispatch(showAlertForError(error));
+    }
+  }).finally(() => {
+    fetchComposeSuggestionsAccountsController = undefined;
+  });
+}, 200, { leading: true, trailing: true });
+
+const fetchComposeSuggestionsEmojis = (dispatch, getState, token) => {
+  const results = emojiSearch(token.replace(':', ''), { maxResults: 5 });
+  dispatch(readyComposeSuggestionsEmojis(token, results));
+};
+
+const fetchComposeSuggestionsTags = throttle((dispatch, getState, token) => {
+  if (fetchComposeSuggestionsTagsController) {
+    fetchComposeSuggestionsTagsController.abort();
+  }
+
+  dispatch(updateSuggestionTags(token));
+
+  fetchComposeSuggestionsTagsController = new AbortController();
+
+  api(getState).get('/api/v2/search', {
+    signal: fetchComposeSuggestionsTagsController.signal,
+
+    params: {
+      type: 'hashtags',
+      q: token.slice(1),
+      resolve: false,
+      limit: 4,
+    },
+  }).then(({ data }) => {
+    dispatch(readyComposeSuggestionsTags(token, data.hashtags));
+  }).catch(error => {
+    if (!axios.isCancel(error)) {
+      dispatch(showAlertForError(error));
+    }
+  }).finally(() => {
+    fetchComposeSuggestionsTagsController = undefined;
+  });
+}, 200, { leading: true, trailing: true });
+
+export function fetchComposeSuggestions(token) {
+  return (dispatch, getState) => {
+    switch (token[0]) {
+    case ':':
+      fetchComposeSuggestionsEmojis(dispatch, getState, token);
+      break;
+    case '#':
+      fetchComposeSuggestionsTags(dispatch, getState, token);
+      break;
+    default:
+      fetchComposeSuggestionsAccounts(dispatch, getState, token);
+      break;
+    }
+  };
+}
+
+export function readyComposeSuggestionsEmojis(token, emojis) {
+  return {
+    type: COMPOSE_SUGGESTIONS_READY,
+    token,
+    emojis,
+  };
+}
+
+export function readyComposeSuggestionsAccounts(token, accounts) {
+  return {
+    type: COMPOSE_SUGGESTIONS_READY,
+    token,
+    accounts,
+  };
+}
+
+export const readyComposeSuggestionsTags = (token, tags) => ({
+  type: COMPOSE_SUGGESTIONS_READY,
+  token,
+  tags,
+});
+
+export function selectComposeSuggestion(position, token, suggestion, path) {
+  return (dispatch, getState) => {
+    let completion;
+    if (suggestion.type === 'emoji') {
+      completion = suggestion.native || suggestion.colons;
+
+      dispatch(useEmoji(suggestion));
+    } else if (suggestion.type === 'hashtag') {
+      completion = `#${suggestion.name}`;
+    } else if (suggestion.type === 'account') {
+      completion = '@' + getState().getIn(['accounts', suggestion.id, 'acct']);
+    }
+
+    // We don't want to replace hashtags that vary only in case due to accessibility, but we need to fire off an event so that
+    // the suggestions are dismissed and the cursor moves forward.
+    if (suggestion.type !== 'hashtag' || token.slice(1).localeCompare(suggestion.name, undefined, { sensitivity: 'accent' }) !== 0) {
+      dispatch({
+        type: COMPOSE_SUGGESTION_SELECT,
+        position,
+        token,
+        completion,
+        path,
+      });
+    } else {
+      dispatch({
+        type: COMPOSE_SUGGESTION_IGNORE,
+        position,
+        token,
+        completion,
+        path,
+      });
+    }
+  };
+}
+
+export function updateSuggestionTags(token) {
+  return {
+    type: COMPOSE_SUGGESTION_TAGS_UPDATE,
+    token,
+  };
+}
+
+export function updateTagHistory(tags) {
+  return {
+    type: COMPOSE_TAG_HISTORY_UPDATE,
+    tags,
+  };
+}
+
+export function hydrateCompose() {
+  return (dispatch, getState) => {
+    const me = getState().getIn(['meta', 'me']);
+    const history = tagHistory.get(me);
+
+    if (history !== null) {
+      dispatch(updateTagHistory(history));
+    }
+  };
+}
+
+function insertIntoTagHistory(recognizedTags, text) {
+  return (dispatch, getState) => {
+    const state = getState();
+    const oldHistory = state.getIn(['compose', 'tagHistory']);
+    const me = state.getIn(['meta', 'me']);
+    const names = recoverHashtags(recognizedTags, text);
+    const intersectedOldHistory = oldHistory.filter(name => names.findIndex(newName => newName.toLowerCase() === name.toLowerCase()) === -1);
+
+    names.push(...intersectedOldHistory.toJS());
+
+    const newHistory = names.slice(0, 1000);
+
+    tagHistory.set(me, newHistory);
+    dispatch(updateTagHistory(newHistory));
+  };
+}
+
+export function mountCompose() {
+  return {
+    type: COMPOSE_MOUNT,
+  };
+}
+
+export function unmountCompose() {
+  return {
+    type: COMPOSE_UNMOUNT,
+  };
+}
+
+export function changeComposeAdvancedOption(option, value) {
+  return {
+    option,
+    type: COMPOSE_ADVANCED_OPTIONS_CHANGE,
+    value,
+  };
+}
+
+export function changeComposeSensitivity() {
+  return {
+    type: COMPOSE_SENSITIVITY_CHANGE,
+  };
+}
+
+export const changeComposeLanguage = language => ({
+  type: COMPOSE_LANGUAGE_CHANGE,
+  language,
+});
+
+export function changeComposeSpoilerness() {
+  return {
+    type: COMPOSE_SPOILERNESS_CHANGE,
+  };
+}
+
+export function changeComposeSpoilerText(text) {
+  return {
+    type: COMPOSE_SPOILER_TEXT_CHANGE,
+    text,
+  };
+}
+
+export function changeComposeVisibility(value) {
+  return {
+    type: COMPOSE_VISIBILITY_CHANGE,
+    value,
+  };
+}
+
+export function changeComposeContentType(value) {
+  return {
+    type: COMPOSE_CONTENT_TYPE_CHANGE,
+    value,
+  };
+}
+
+export function insertEmojiCompose(position, emoji) {
+  return {
+    type: COMPOSE_EMOJI_INSERT,
+    position,
+    emoji,
+  };
+}
+
+export function addPoll() {
+  return {
+    type: COMPOSE_POLL_ADD,
+  };
+}
+
+export function removePoll() {
+  return {
+    type: COMPOSE_POLL_REMOVE,
+  };
+}
+
+export function addPollOption(title) {
+  return {
+    type: COMPOSE_POLL_OPTION_ADD,
+    title,
+  };
+}
+
+export function changePollOption(index, title) {
+  return {
+    type: COMPOSE_POLL_OPTION_CHANGE,
+    index,
+    title,
+  };
+}
+
+export function removePollOption(index) {
+  return {
+    type: COMPOSE_POLL_OPTION_REMOVE,
+    index,
+  };
+}
+
+export function changePollSettings(expiresIn, isMultiple) {
+  return {
+    type: COMPOSE_POLL_SETTINGS_CHANGE,
+    expiresIn,
+    isMultiple,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/conversations.js b/app/javascript/flavours/blobfox/actions/conversations.js
new file mode 100644
index 00000000000000..8c4c4529fb7e8b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/conversations.js
@@ -0,0 +1,113 @@
+import api, { getLinks } from '../api';
+
+import {
+  importFetchedAccounts,
+  importFetchedStatuses,
+  importFetchedStatus,
+} from './importer';
+
+export const CONVERSATIONS_MOUNT   = 'CONVERSATIONS_MOUNT';
+export const CONVERSATIONS_UNMOUNT = 'CONVERSATIONS_UNMOUNT';
+
+export const CONVERSATIONS_FETCH_REQUEST = 'CONVERSATIONS_FETCH_REQUEST';
+export const CONVERSATIONS_FETCH_SUCCESS = 'CONVERSATIONS_FETCH_SUCCESS';
+export const CONVERSATIONS_FETCH_FAIL    = 'CONVERSATIONS_FETCH_FAIL';
+export const CONVERSATIONS_UPDATE        = 'CONVERSATIONS_UPDATE';
+
+export const CONVERSATIONS_READ = 'CONVERSATIONS_READ';
+
+export const CONVERSATIONS_DELETE_REQUEST = 'CONVERSATIONS_DELETE_REQUEST';
+export const CONVERSATIONS_DELETE_SUCCESS = 'CONVERSATIONS_DELETE_SUCCESS';
+export const CONVERSATIONS_DELETE_FAIL    = 'CONVERSATIONS_DELETE_FAIL';
+
+export const mountConversations = () => ({
+  type: CONVERSATIONS_MOUNT,
+});
+
+export const unmountConversations = () => ({
+  type: CONVERSATIONS_UNMOUNT,
+});
+
+export const markConversationRead = conversationId => (dispatch, getState) => {
+  dispatch({
+    type: CONVERSATIONS_READ,
+    id: conversationId,
+  });
+
+  api(getState).post(`/api/v1/conversations/${conversationId}/read`);
+};
+
+export const expandConversations = ({ maxId } = {}) => (dispatch, getState) => {
+  dispatch(expandConversationsRequest());
+
+  const params = { max_id: maxId };
+
+  if (!maxId) {
+    params.since_id = getState().getIn(['conversations', 'items', 0, 'last_status']);
+  }
+
+  const isLoadingRecent = !!params.since_id;
+
+  api(getState).get('/api/v1/conversations', { params })
+    .then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data.reduce((aggr, item) => aggr.concat(item.accounts), [])));
+      dispatch(importFetchedStatuses(response.data.map(item => item.last_status).filter(x => !!x)));
+      dispatch(expandConversationsSuccess(response.data, next ? next.uri : null, isLoadingRecent));
+    })
+    .catch(err => dispatch(expandConversationsFail(err)));
+};
+
+export const expandConversationsRequest = () => ({
+  type: CONVERSATIONS_FETCH_REQUEST,
+});
+
+export const expandConversationsSuccess = (conversations, next, isLoadingRecent) => ({
+  type: CONVERSATIONS_FETCH_SUCCESS,
+  conversations,
+  next,
+  isLoadingRecent,
+});
+
+export const expandConversationsFail = error => ({
+  type: CONVERSATIONS_FETCH_FAIL,
+  error,
+});
+
+export const updateConversations = conversation => dispatch => {
+  dispatch(importFetchedAccounts(conversation.accounts));
+
+  if (conversation.last_status) {
+    dispatch(importFetchedStatus(conversation.last_status));
+  }
+
+  dispatch({
+    type: CONVERSATIONS_UPDATE,
+    conversation,
+  });
+};
+
+export const deleteConversation = conversationId => (dispatch, getState) => {
+  dispatch(deleteConversationRequest(conversationId));
+
+  api(getState).delete(`/api/v1/conversations/${conversationId}`)
+    .then(() => dispatch(deleteConversationSuccess(conversationId)))
+    .catch(error => dispatch(deleteConversationFail(conversationId, error)));
+};
+
+export const deleteConversationRequest = id => ({
+  type: CONVERSATIONS_DELETE_REQUEST,
+  id,
+});
+
+export const deleteConversationSuccess = id => ({
+  type: CONVERSATIONS_DELETE_SUCCESS,
+  id,
+});
+
+export const deleteConversationFail = (id, error) => ({
+  type: CONVERSATIONS_DELETE_FAIL,
+  id,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/custom_emojis.js b/app/javascript/flavours/blobfox/actions/custom_emojis.js
new file mode 100644
index 00000000000000..9ec8156b170a21
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/custom_emojis.js
@@ -0,0 +1,40 @@
+import api from '../api';
+
+export const CUSTOM_EMOJIS_FETCH_REQUEST = 'CUSTOM_EMOJIS_FETCH_REQUEST';
+export const CUSTOM_EMOJIS_FETCH_SUCCESS = 'CUSTOM_EMOJIS_FETCH_SUCCESS';
+export const CUSTOM_EMOJIS_FETCH_FAIL = 'CUSTOM_EMOJIS_FETCH_FAIL';
+
+export function fetchCustomEmojis() {
+  return (dispatch, getState) => {
+    dispatch(fetchCustomEmojisRequest());
+
+    api(getState).get('/api/v1/custom_emojis').then(response => {
+      dispatch(fetchCustomEmojisSuccess(response.data));
+    }).catch(error => {
+      dispatch(fetchCustomEmojisFail(error));
+    });
+  };
+}
+
+export function fetchCustomEmojisRequest() {
+  return {
+    type: CUSTOM_EMOJIS_FETCH_REQUEST,
+    skipLoading: true,
+  };
+}
+
+export function fetchCustomEmojisSuccess(custom_emojis) {
+  return {
+    type: CUSTOM_EMOJIS_FETCH_SUCCESS,
+    custom_emojis,
+    skipLoading: true,
+  };
+}
+
+export function fetchCustomEmojisFail(error) {
+  return {
+    type: CUSTOM_EMOJIS_FETCH_FAIL,
+    error,
+    skipLoading: true,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/directory.js b/app/javascript/flavours/blobfox/actions/directory.js
new file mode 100644
index 00000000000000..cda63f2b5a43eb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/directory.js
@@ -0,0 +1,62 @@
+import api from '../api';
+
+import { fetchRelationships } from './accounts';
+import { importFetchedAccounts } from './importer';
+
+export const DIRECTORY_FETCH_REQUEST = 'DIRECTORY_FETCH_REQUEST';
+export const DIRECTORY_FETCH_SUCCESS = 'DIRECTORY_FETCH_SUCCESS';
+export const DIRECTORY_FETCH_FAIL    = 'DIRECTORY_FETCH_FAIL';
+
+export const DIRECTORY_EXPAND_REQUEST = 'DIRECTORY_EXPAND_REQUEST';
+export const DIRECTORY_EXPAND_SUCCESS = 'DIRECTORY_EXPAND_SUCCESS';
+export const DIRECTORY_EXPAND_FAIL    = 'DIRECTORY_EXPAND_FAIL';
+
+export const fetchDirectory = params => (dispatch, getState) => {
+  dispatch(fetchDirectoryRequest());
+
+  api(getState).get('/api/v1/directory', { params: { ...params, limit: 20 } }).then(({ data }) => {
+    dispatch(importFetchedAccounts(data));
+    dispatch(fetchDirectorySuccess(data));
+    dispatch(fetchRelationships(data.map(x => x.id)));
+  }).catch(error => dispatch(fetchDirectoryFail(error)));
+};
+
+export const fetchDirectoryRequest = () => ({
+  type: DIRECTORY_FETCH_REQUEST,
+});
+
+export const fetchDirectorySuccess = accounts => ({
+  type: DIRECTORY_FETCH_SUCCESS,
+  accounts,
+});
+
+export const fetchDirectoryFail = error => ({
+  type: DIRECTORY_FETCH_FAIL,
+  error,
+});
+
+export const expandDirectory = params => (dispatch, getState) => {
+  dispatch(expandDirectoryRequest());
+
+  const loadedItems = getState().getIn(['user_lists', 'directory', 'items']).size;
+
+  api(getState).get('/api/v1/directory', { params: { ...params, offset: loadedItems, limit: 20 } }).then(({ data }) => {
+    dispatch(importFetchedAccounts(data));
+    dispatch(expandDirectorySuccess(data));
+    dispatch(fetchRelationships(data.map(x => x.id)));
+  }).catch(error => dispatch(expandDirectoryFail(error)));
+};
+
+export const expandDirectoryRequest = () => ({
+  type: DIRECTORY_EXPAND_REQUEST,
+});
+
+export const expandDirectorySuccess = accounts => ({
+  type: DIRECTORY_EXPAND_SUCCESS,
+  accounts,
+});
+
+export const expandDirectoryFail = error => ({
+  type: DIRECTORY_EXPAND_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/domain_blocks.js b/app/javascript/flavours/blobfox/actions/domain_blocks.js
new file mode 100644
index 00000000000000..718002613f4053
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/domain_blocks.js
@@ -0,0 +1,152 @@
+import api, { getLinks } from '../api';
+
+import { blockDomainSuccess, unblockDomainSuccess } from "./domain_blocks_typed";
+
+export * from "./domain_blocks_typed";
+
+export const DOMAIN_BLOCK_REQUEST = 'DOMAIN_BLOCK_REQUEST';
+export const DOMAIN_BLOCK_FAIL    = 'DOMAIN_BLOCK_FAIL';
+
+export const DOMAIN_UNBLOCK_REQUEST = 'DOMAIN_UNBLOCK_REQUEST';
+export const DOMAIN_UNBLOCK_FAIL    = 'DOMAIN_UNBLOCK_FAIL';
+
+export const DOMAIN_BLOCKS_FETCH_REQUEST = 'DOMAIN_BLOCKS_FETCH_REQUEST';
+export const DOMAIN_BLOCKS_FETCH_SUCCESS = 'DOMAIN_BLOCKS_FETCH_SUCCESS';
+export const DOMAIN_BLOCKS_FETCH_FAIL    = 'DOMAIN_BLOCKS_FETCH_FAIL';
+
+export const DOMAIN_BLOCKS_EXPAND_REQUEST = 'DOMAIN_BLOCKS_EXPAND_REQUEST';
+export const DOMAIN_BLOCKS_EXPAND_SUCCESS = 'DOMAIN_BLOCKS_EXPAND_SUCCESS';
+export const DOMAIN_BLOCKS_EXPAND_FAIL    = 'DOMAIN_BLOCKS_EXPAND_FAIL';
+
+export function blockDomain(domain) {
+  return (dispatch, getState) => {
+    dispatch(blockDomainRequest(domain));
+
+    api(getState).post('/api/v1/domain_blocks', { domain }).then(() => {
+      const at_domain = '@' + domain;
+      const accounts = getState().get('accounts').filter(item => item.get('acct').endsWith(at_domain)).valueSeq().map(item => item.get('id'));
+
+      dispatch(blockDomainSuccess({ domain, accounts }));
+    }).catch(err => {
+      dispatch(blockDomainFail(domain, err));
+    });
+  };
+}
+
+export function blockDomainRequest(domain) {
+  return {
+    type: DOMAIN_BLOCK_REQUEST,
+    domain,
+  };
+}
+
+export function blockDomainFail(domain, error) {
+  return {
+    type: DOMAIN_BLOCK_FAIL,
+    domain,
+    error,
+  };
+}
+
+export function unblockDomain(domain) {
+  return (dispatch, getState) => {
+    dispatch(unblockDomainRequest(domain));
+
+    api(getState).delete('/api/v1/domain_blocks', { params: { domain } }).then(() => {
+      const at_domain = '@' + domain;
+      const accounts = getState().get('accounts').filter(item => item.get('acct').endsWith(at_domain)).valueSeq().map(item => item.get('id'));
+      dispatch(unblockDomainSuccess({ domain, accounts }));
+    }).catch(err => {
+      dispatch(unblockDomainFail(domain, err));
+    });
+  };
+}
+
+export function unblockDomainRequest(domain) {
+  return {
+    type: DOMAIN_UNBLOCK_REQUEST,
+    domain,
+  };
+}
+
+export function unblockDomainFail(domain, error) {
+  return {
+    type: DOMAIN_UNBLOCK_FAIL,
+    domain,
+    error,
+  };
+}
+
+export function fetchDomainBlocks() {
+  return (dispatch, getState) => {
+    dispatch(fetchDomainBlocksRequest());
+
+    api(getState).get('/api/v1/domain_blocks').then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(fetchDomainBlocksSuccess(response.data, next ? next.uri : null));
+    }).catch(err => {
+      dispatch(fetchDomainBlocksFail(err));
+    });
+  };
+}
+
+export function fetchDomainBlocksRequest() {
+  return {
+    type: DOMAIN_BLOCKS_FETCH_REQUEST,
+  };
+}
+
+export function fetchDomainBlocksSuccess(domains, next) {
+  return {
+    type: DOMAIN_BLOCKS_FETCH_SUCCESS,
+    domains,
+    next,
+  };
+}
+
+export function fetchDomainBlocksFail(error) {
+  return {
+    type: DOMAIN_BLOCKS_FETCH_FAIL,
+    error,
+  };
+}
+
+export function expandDomainBlocks() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['domain_lists', 'blocks', 'next']);
+
+    if (!url) {
+      return;
+    }
+
+    dispatch(expandDomainBlocksRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(expandDomainBlocksSuccess(response.data, next ? next.uri : null));
+    }).catch(err => {
+      dispatch(expandDomainBlocksFail(err));
+    });
+  };
+}
+
+export function expandDomainBlocksRequest() {
+  return {
+    type: DOMAIN_BLOCKS_EXPAND_REQUEST,
+  };
+}
+
+export function expandDomainBlocksSuccess(domains, next) {
+  return {
+    type: DOMAIN_BLOCKS_EXPAND_SUCCESS,
+    domains,
+    next,
+  };
+}
+
+export function expandDomainBlocksFail(error) {
+  return {
+    type: DOMAIN_BLOCKS_EXPAND_FAIL,
+    error,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/domain_blocks_typed.ts b/app/javascript/flavours/blobfox/actions/domain_blocks_typed.ts
new file mode 100644
index 00000000000000..4be6f7ed63495e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/domain_blocks_typed.ts
@@ -0,0 +1,13 @@
+import { createAction } from '@reduxjs/toolkit';
+
+import type { Account } from 'flavours/blobfox/models/account';
+
+export const blockDomainSuccess = createAction<{
+  domain: string;
+  accounts: Account[];
+}>('domain_blocks/block/SUCCESS');
+
+export const unblockDomainSuccess = createAction<{
+  domain: string;
+  accounts: Account[];
+}>('domain_blocks/unblock/SUCCESS');
diff --git a/app/javascript/flavours/blobfox/actions/dropdown_menu.ts b/app/javascript/flavours/blobfox/actions/dropdown_menu.ts
new file mode 100644
index 00000000000000..3694df1ae03e29
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/dropdown_menu.ts
@@ -0,0 +1,11 @@
+import { createAction } from '@reduxjs/toolkit';
+
+export const openDropdownMenu = createAction<{
+  id: string;
+  keyboard: boolean;
+  scrollKey: string;
+}>('dropdownMenu/open');
+
+export const closeDropdownMenu = createAction<{ id: string }>(
+  'dropdownMenu/close',
+);
diff --git a/app/javascript/flavours/blobfox/actions/emojis.js b/app/javascript/flavours/blobfox/actions/emojis.js
new file mode 100644
index 00000000000000..3b5d53996c8204
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/emojis.js
@@ -0,0 +1,14 @@
+import { saveSettings } from './settings';
+
+export const EMOJI_USE = 'EMOJI_USE';
+
+export function useEmoji(emoji) {
+  return dispatch => {
+    dispatch({
+      type: EMOJI_USE,
+      emoji,
+    });
+
+    dispatch(saveSettings());
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/favourites.js b/app/javascript/flavours/blobfox/actions/favourites.js
new file mode 100644
index 00000000000000..2d4d4e6206e845
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/favourites.js
@@ -0,0 +1,94 @@
+import api, { getLinks } from '../api';
+
+import { importFetchedStatuses } from './importer';
+
+export const FAVOURITED_STATUSES_FETCH_REQUEST = 'FAVOURITED_STATUSES_FETCH_REQUEST';
+export const FAVOURITED_STATUSES_FETCH_SUCCESS = 'FAVOURITED_STATUSES_FETCH_SUCCESS';
+export const FAVOURITED_STATUSES_FETCH_FAIL    = 'FAVOURITED_STATUSES_FETCH_FAIL';
+
+export const FAVOURITED_STATUSES_EXPAND_REQUEST = 'FAVOURITED_STATUSES_EXPAND_REQUEST';
+export const FAVOURITED_STATUSES_EXPAND_SUCCESS = 'FAVOURITED_STATUSES_EXPAND_SUCCESS';
+export const FAVOURITED_STATUSES_EXPAND_FAIL    = 'FAVOURITED_STATUSES_EXPAND_FAIL';
+
+export function fetchFavouritedStatuses() {
+  return (dispatch, getState) => {
+    if (getState().getIn(['status_lists', 'favourites', 'isLoading'])) {
+      return;
+    }
+
+    dispatch(fetchFavouritedStatusesRequest());
+
+    api(getState).get('/api/v1/favourites').then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedStatuses(response.data));
+      dispatch(fetchFavouritedStatusesSuccess(response.data, next ? next.uri : null));
+    }).catch(error => {
+      dispatch(fetchFavouritedStatusesFail(error));
+    });
+  };
+}
+
+export function fetchFavouritedStatusesRequest() {
+  return {
+    type: FAVOURITED_STATUSES_FETCH_REQUEST,
+    skipLoading: true,
+  };
+}
+
+export function fetchFavouritedStatusesSuccess(statuses, next) {
+  return {
+    type: FAVOURITED_STATUSES_FETCH_SUCCESS,
+    statuses,
+    next,
+    skipLoading: true,
+  };
+}
+
+export function fetchFavouritedStatusesFail(error) {
+  return {
+    type: FAVOURITED_STATUSES_FETCH_FAIL,
+    error,
+    skipLoading: true,
+  };
+}
+
+export function expandFavouritedStatuses() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['status_lists', 'favourites', 'next'], null);
+
+    if (url === null || getState().getIn(['status_lists', 'favourites', 'isLoading'])) {
+      return;
+    }
+
+    dispatch(expandFavouritedStatusesRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedStatuses(response.data));
+      dispatch(expandFavouritedStatusesSuccess(response.data, next ? next.uri : null));
+    }).catch(error => {
+      dispatch(expandFavouritedStatusesFail(error));
+    });
+  };
+}
+
+export function expandFavouritedStatusesRequest() {
+  return {
+    type: FAVOURITED_STATUSES_EXPAND_REQUEST,
+  };
+}
+
+export function expandFavouritedStatusesSuccess(statuses, next) {
+  return {
+    type: FAVOURITED_STATUSES_EXPAND_SUCCESS,
+    statuses,
+    next,
+  };
+}
+
+export function expandFavouritedStatusesFail(error) {
+  return {
+    type: FAVOURITED_STATUSES_EXPAND_FAIL,
+    error,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/featured_tags.js b/app/javascript/flavours/blobfox/actions/featured_tags.js
new file mode 100644
index 00000000000000..18bb615394571c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/featured_tags.js
@@ -0,0 +1,34 @@
+import api from '../api';
+
+export const FEATURED_TAGS_FETCH_REQUEST = 'FEATURED_TAGS_FETCH_REQUEST';
+export const FEATURED_TAGS_FETCH_SUCCESS = 'FEATURED_TAGS_FETCH_SUCCESS';
+export const FEATURED_TAGS_FETCH_FAIL    = 'FEATURED_TAGS_FETCH_FAIL';
+
+export const fetchFeaturedTags = (id) => (dispatch, getState) => {
+  if (getState().getIn(['user_lists', 'featured_tags', id, 'items'])) {
+    return;
+  }
+
+  dispatch(fetchFeaturedTagsRequest(id));
+
+  api(getState).get(`/api/v1/accounts/${id}/featured_tags`)
+    .then(({ data }) => dispatch(fetchFeaturedTagsSuccess(id, data)))
+    .catch(err => dispatch(fetchFeaturedTagsFail(id, err)));
+};
+
+export const fetchFeaturedTagsRequest = (id) => ({
+  type: FEATURED_TAGS_FETCH_REQUEST,
+  id,
+});
+
+export const fetchFeaturedTagsSuccess = (id, tags) => ({
+  type: FEATURED_TAGS_FETCH_SUCCESS,
+  id,
+  tags,
+});
+
+export const fetchFeaturedTagsFail = (id, error) => ({
+  type: FEATURED_TAGS_FETCH_FAIL,
+  id,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/filters.js b/app/javascript/flavours/blobfox/actions/filters.js
new file mode 100644
index 00000000000000..a11956ac564c6a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/filters.js
@@ -0,0 +1,97 @@
+import api from '../api';
+
+import { openModal } from './modal';
+
+export const FILTERS_FETCH_REQUEST = 'FILTERS_FETCH_REQUEST';
+export const FILTERS_FETCH_SUCCESS = 'FILTERS_FETCH_SUCCESS';
+export const FILTERS_FETCH_FAIL    = 'FILTERS_FETCH_FAIL';
+
+export const FILTERS_STATUS_CREATE_REQUEST = 'FILTERS_STATUS_CREATE_REQUEST';
+export const FILTERS_STATUS_CREATE_SUCCESS = 'FILTERS_STATUS_CREATE_SUCCESS';
+export const FILTERS_STATUS_CREATE_FAIL    = 'FILTERS_STATUS_CREATE_FAIL';
+
+export const FILTERS_CREATE_REQUEST = 'FILTERS_CREATE_REQUEST';
+export const FILTERS_CREATE_SUCCESS = 'FILTERS_CREATE_SUCCESS';
+export const FILTERS_CREATE_FAIL    = 'FILTERS_CREATE_FAIL';
+
+export const initAddFilter = (status, { contextType }) => dispatch =>
+  dispatch(openModal({
+    modalType: 'FILTER',
+    modalProps: {
+      statusId: status?.get('id'),
+      contextType: contextType,
+    },
+  }));
+
+export const fetchFilters = () => (dispatch, getState) => {
+  dispatch({
+    type: FILTERS_FETCH_REQUEST,
+    skipLoading: true,
+  });
+
+  api(getState)
+    .get('/api/v2/filters')
+    .then(({ data }) => dispatch({
+      type: FILTERS_FETCH_SUCCESS,
+      filters: data,
+      skipLoading: true,
+    }))
+    .catch(err => dispatch({
+      type: FILTERS_FETCH_FAIL,
+      err,
+      skipLoading: true,
+      skipAlert: true,
+    }));
+};
+
+export const createFilterStatus = (params, onSuccess, onFail) => (dispatch, getState) => {
+  dispatch(createFilterStatusRequest());
+
+  api(getState).post(`/api/v2/filters/${params.filter_id}/statuses`, params).then(response => {
+    dispatch(createFilterStatusSuccess(response.data));
+    if (onSuccess) onSuccess();
+  }).catch(error => {
+    dispatch(createFilterStatusFail(error));
+    if (onFail) onFail();
+  });
+};
+
+export const createFilterStatusRequest = () => ({
+  type: FILTERS_STATUS_CREATE_REQUEST,
+});
+
+export const createFilterStatusSuccess = filter_status => ({
+  type: FILTERS_STATUS_CREATE_SUCCESS,
+  filter_status,
+});
+
+export const createFilterStatusFail = error => ({
+  type: FILTERS_STATUS_CREATE_FAIL,
+  error,
+});
+
+export const createFilter = (params, onSuccess, onFail) => (dispatch, getState) => {
+  dispatch(createFilterRequest());
+
+  api(getState).post('/api/v2/filters', params).then(response => {
+    dispatch(createFilterSuccess(response.data));
+    if (onSuccess) onSuccess(response.data);
+  }).catch(error => {
+    dispatch(createFilterFail(error));
+    if (onFail) onFail();
+  });
+};
+
+export const createFilterRequest = () => ({
+  type: FILTERS_CREATE_REQUEST,
+});
+
+export const createFilterSuccess = filter => ({
+  type: FILTERS_CREATE_SUCCESS,
+  filter,
+});
+
+export const createFilterFail = error => ({
+  type: FILTERS_CREATE_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/height_cache.js b/app/javascript/flavours/blobfox/actions/height_cache.js
new file mode 100644
index 00000000000000..a8645410c85b1b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/height_cache.js
@@ -0,0 +1,17 @@
+export const HEIGHT_CACHE_SET = 'HEIGHT_CACHE_SET';
+export const HEIGHT_CACHE_CLEAR = 'HEIGHT_CACHE_CLEAR';
+
+export function setHeight (key, id, height) {
+  return {
+    type: HEIGHT_CACHE_SET,
+    key,
+    id,
+    height,
+  };
+}
+
+export function clearHeight () {
+  return {
+    type: HEIGHT_CACHE_CLEAR,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/history.js b/app/javascript/flavours/blobfox/actions/history.js
new file mode 100644
index 00000000000000..52401b7dce3f95
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/history.js
@@ -0,0 +1,38 @@
+import api from '../api';
+
+import { importFetchedAccounts } from './importer';
+
+export const HISTORY_FETCH_REQUEST = 'HISTORY_FETCH_REQUEST';
+export const HISTORY_FETCH_SUCCESS = 'HISTORY_FETCH_SUCCESS';
+export const HISTORY_FETCH_FAIL    = 'HISTORY_FETCH_FAIL';
+
+export const fetchHistory = statusId => (dispatch, getState) => {
+  const loading = getState().getIn(['history', statusId, 'loading']);
+
+  if (loading) {
+    return;
+  }
+
+  dispatch(fetchHistoryRequest(statusId));
+
+  api(getState).get(`/api/v1/statuses/${statusId}/history`).then(({ data }) => {
+    dispatch(importFetchedAccounts(data.map(x => x.account)));
+    dispatch(fetchHistorySuccess(statusId, data));
+  }).catch(error => dispatch(fetchHistoryFail(error)));
+};
+
+export const fetchHistoryRequest = statusId => ({
+  type: HISTORY_FETCH_REQUEST,
+  statusId,
+});
+
+export const fetchHistorySuccess = (statusId, history) => ({
+  type: HISTORY_FETCH_SUCCESS,
+  statusId,
+  history,
+});
+
+export const fetchHistoryFail = error => ({
+  type: HISTORY_FETCH_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/importer/index.js b/app/javascript/flavours/blobfox/actions/importer/index.js
new file mode 100644
index 00000000000000..5fbc9bb5bbb64a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/importer/index.js
@@ -0,0 +1,93 @@
+import { importAccounts } from '../accounts_typed';
+
+import { normalizeStatus, normalizePoll } from './normalizer';
+
+export const STATUS_IMPORT   = 'STATUS_IMPORT';
+export const STATUSES_IMPORT = 'STATUSES_IMPORT';
+export const POLLS_IMPORT    = 'POLLS_IMPORT';
+export const FILTERS_IMPORT  = 'FILTERS_IMPORT';
+
+function pushUnique(array, object) {
+  if (array.every(element => element.id !== object.id)) {
+    array.push(object);
+  }
+}
+
+export function importStatus(status) {
+  return { type: STATUS_IMPORT, status };
+}
+
+export function importStatuses(statuses) {
+  return { type: STATUSES_IMPORT, statuses };
+}
+
+export function importFilters(filters) {
+  return { type: FILTERS_IMPORT, filters };
+}
+
+export function importPolls(polls) {
+  return { type: POLLS_IMPORT, polls };
+}
+
+export function importFetchedAccount(account) {
+  return importFetchedAccounts([account]);
+}
+
+export function importFetchedAccounts(accounts) {
+  const normalAccounts = [];
+
+  function processAccount(account) {
+    pushUnique(normalAccounts, account);
+
+    if (account.moved) {
+      processAccount(account.moved);
+    }
+  }
+
+  accounts.forEach(processAccount);
+
+  return importAccounts({ accounts: normalAccounts });
+}
+
+export function importFetchedStatus(status) {
+  return importFetchedStatuses([status]);
+}
+
+export function importFetchedStatuses(statuses) {
+  return (dispatch, getState) => {
+    const accounts = [];
+    const normalStatuses = [];
+    const polls = [];
+    const filters = [];
+
+    function processStatus(status) {
+      pushUnique(normalStatuses, normalizeStatus(status, getState().getIn(['statuses', status.id]), getState().get('local_settings')));
+      pushUnique(accounts, status.account);
+
+      if (status.filtered) {
+        status.filtered.forEach(result => pushUnique(filters, result.filter));
+      }
+
+      if (status.reblog && status.reblog.id) {
+        processStatus(status.reblog);
+      }
+
+      if (status.poll && status.poll.id) {
+        pushUnique(polls, normalizePoll(status.poll, getState().getIn(['polls', status.poll.id])));
+      }
+    }
+
+    statuses.forEach(processStatus);
+
+    dispatch(importPolls(polls));
+    dispatch(importFetchedAccounts(accounts));
+    dispatch(importStatuses(normalStatuses));
+    dispatch(importFilters(filters));
+  };
+}
+
+export function importFetchedPoll(poll) {
+  return (dispatch, getState) => {
+    dispatch(importPolls([normalizePoll(poll, getState().getIn(['polls', poll.id]))]));
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/importer/normalizer.js b/app/javascript/flavours/blobfox/actions/importer/normalizer.js
new file mode 100644
index 00000000000000..c2ad0f9908a126
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/importer/normalizer.js
@@ -0,0 +1,135 @@
+import escapeTextContentForBrowser from 'escape-html';
+
+import emojify from '../../features/emoji/emoji';
+import { autoHideCW } from '../../utils/content_warning';
+
+const domParser = new DOMParser();
+
+const makeEmojiMap = emojis => emojis.reduce((obj, emoji) => {
+  obj[`:${emoji.shortcode}:`] = emoji;
+  return obj;
+}, {});
+
+export function searchTextFromRawStatus (status) {
+  const spoilerText   = status.spoiler_text || '';
+  const searchContent = ([spoilerText, status.content].concat((status.poll && status.poll.options) ? status.poll.options.map(option => option.title) : [])).concat(status.media_attachments.map(att => att.description)).join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n');
+  return domParser.parseFromString(searchContent, 'text/html').documentElement.textContent;
+}
+
+export function normalizeFilterResult(result) {
+  const normalResult = { ...result };
+
+  normalResult.filter = normalResult.filter.id;
+
+  return normalResult;
+}
+
+export function normalizeStatus(status, normalOldStatus, settings) {
+  const normalStatus   = { ...status };
+  normalStatus.account = status.account.id;
+
+  if (status.reblog && status.reblog.id) {
+    normalStatus.reblog = status.reblog.id;
+  }
+
+  if (status.poll && status.poll.id) {
+    normalStatus.poll = status.poll.id;
+  }
+
+  if (status.filtered) {
+    normalStatus.filtered = status.filtered.map(normalizeFilterResult);
+  }
+
+  // Only calculate these values when status first encountered and
+  // when the underlying values change. Otherwise keep the ones
+  // already in the reducer
+  if (normalOldStatus && normalOldStatus.get('content') === normalStatus.content && normalOldStatus.get('spoiler_text') === normalStatus.spoiler_text) {
+    normalStatus.search_index = normalOldStatus.get('search_index');
+    normalStatus.contentHtml = normalOldStatus.get('contentHtml');
+    normalStatus.spoilerHtml = normalOldStatus.get('spoilerHtml');
+    normalStatus.hidden = normalOldStatus.get('hidden');
+
+    if (normalOldStatus.get('translation')) {
+      normalStatus.translation = normalOldStatus.get('translation');
+    }
+  } else {
+    const spoilerText   = normalStatus.spoiler_text || '';
+    const searchContent = ([spoilerText, status.content].concat((status.poll && status.poll.options) ? status.poll.options.map(option => option.title) : [])).concat(status.media_attachments.map(att => att.description)).join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n');
+    const emojiMap      = makeEmojiMap(normalStatus.emojis);
+
+    normalStatus.search_index = domParser.parseFromString(searchContent, 'text/html').documentElement.textContent;
+    normalStatus.contentHtml  = emojify(normalStatus.content, emojiMap);
+    normalStatus.spoilerHtml  = emojify(escapeTextContentForBrowser(spoilerText), emojiMap);
+    normalStatus.hidden       = (spoilerText.length > 0 || normalStatus.sensitive) && autoHideCW(settings, spoilerText);
+  }
+
+  if (normalOldStatus) {
+    const list = normalOldStatus.get('media_attachments');
+    if (normalStatus.media_attachments && list) {
+      normalStatus.media_attachments.forEach(item => {
+        const oldItem = list.find(i => i.get('id') === item.id);
+        if (oldItem && oldItem.get('description') === item.description) {
+          item.translation = oldItem.get('translation');
+        }
+      });
+    }
+  }
+
+  return normalStatus;
+}
+
+export function normalizeStatusTranslation(translation, status) {
+  const emojiMap = makeEmojiMap(status.get('emojis').toJS());
+
+  const normalTranslation = {
+    detected_source_language: translation.detected_source_language,
+    language: translation.language,
+    provider: translation.provider,
+    contentHtml: emojify(translation.content, emojiMap),
+    spoilerHtml: emojify(escapeTextContentForBrowser(translation.spoiler_text), emojiMap),
+    spoiler_text: translation.spoiler_text,
+  };
+
+  return normalTranslation;
+}
+
+export function normalizePoll(poll, normalOldPoll) {
+  const normalPoll = { ...poll };
+  const emojiMap = makeEmojiMap(poll.emojis);
+
+  normalPoll.options = poll.options.map((option, index) => {
+    const normalOption = {
+      ...option,
+      voted: poll.own_votes && poll.own_votes.includes(index),
+      titleHtml: emojify(escapeTextContentForBrowser(option.title), emojiMap),
+    };
+
+    if (normalOldPoll && normalOldPoll.getIn(['options', index, 'title']) === option.title) {
+      normalOption.translation = normalOldPoll.getIn(['options', index, 'translation']);
+    }
+
+    return normalOption;
+  });
+
+  return normalPoll;
+}
+
+export function normalizePollOptionTranslation(translation, poll) {
+  const emojiMap = makeEmojiMap(poll.get('emojis').toJS());
+
+  const normalTranslation = {
+    ...translation,
+    titleHtml: emojify(escapeTextContentForBrowser(translation.title), emojiMap),
+  };
+
+  return normalTranslation;
+}
+
+export function normalizeAnnouncement(announcement) {
+  const normalAnnouncement = { ...announcement };
+  const emojiMap = makeEmojiMap(normalAnnouncement.emojis);
+
+  normalAnnouncement.contentHtml = emojify(normalAnnouncement.content, emojiMap);
+
+  return normalAnnouncement;
+}
diff --git a/app/javascript/flavours/blobfox/actions/interactions.js b/app/javascript/flavours/blobfox/actions/interactions.js
new file mode 100644
index 00000000000000..7cc7863af6319d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/interactions.js
@@ -0,0 +1,600 @@
+import api, { getLinks } from '../api';
+
+import { fetchRelationships } from './accounts';
+import { importFetchedAccounts, importFetchedStatus } from './importer';
+
+export const REBLOG_REQUEST = 'REBLOG_REQUEST';
+export const REBLOG_SUCCESS = 'REBLOG_SUCCESS';
+export const REBLOG_FAIL    = 'REBLOG_FAIL';
+
+export const REBLOGS_EXPAND_REQUEST = 'REBLOGS_EXPAND_REQUEST';
+export const REBLOGS_EXPAND_SUCCESS = 'REBLOGS_EXPAND_SUCCESS';
+export const REBLOGS_EXPAND_FAIL = 'REBLOGS_EXPAND_FAIL';
+
+export const FAVOURITE_REQUEST = 'FAVOURITE_REQUEST';
+export const FAVOURITE_SUCCESS = 'FAVOURITE_SUCCESS';
+export const FAVOURITE_FAIL    = 'FAVOURITE_FAIL';
+
+export const UNREBLOG_REQUEST = 'UNREBLOG_REQUEST';
+export const UNREBLOG_SUCCESS = 'UNREBLOG_SUCCESS';
+export const UNREBLOG_FAIL    = 'UNREBLOG_FAIL';
+
+export const UNFAVOURITE_REQUEST = 'UNFAVOURITE_REQUEST';
+export const UNFAVOURITE_SUCCESS = 'UNFAVOURITE_SUCCESS';
+export const UNFAVOURITE_FAIL    = 'UNFAVOURITE_FAIL';
+
+export const REBLOGS_FETCH_REQUEST = 'REBLOGS_FETCH_REQUEST';
+export const REBLOGS_FETCH_SUCCESS = 'REBLOGS_FETCH_SUCCESS';
+export const REBLOGS_FETCH_FAIL    = 'REBLOGS_FETCH_FAIL';
+
+export const FAVOURITES_FETCH_REQUEST = 'FAVOURITES_FETCH_REQUEST';
+export const FAVOURITES_FETCH_SUCCESS = 'FAVOURITES_FETCH_SUCCESS';
+export const FAVOURITES_FETCH_FAIL    = 'FAVOURITES_FETCH_FAIL';
+
+export const FAVOURITES_EXPAND_REQUEST = 'FAVOURITES_EXPAND_REQUEST';
+export const FAVOURITES_EXPAND_SUCCESS = 'FAVOURITES_EXPAND_SUCCESS';
+export const FAVOURITES_EXPAND_FAIL = 'FAVOURITES_EXPAND_FAIL';
+
+export const PIN_REQUEST = 'PIN_REQUEST';
+export const PIN_SUCCESS = 'PIN_SUCCESS';
+export const PIN_FAIL    = 'PIN_FAIL';
+
+export const UNPIN_REQUEST = 'UNPIN_REQUEST';
+export const UNPIN_SUCCESS = 'UNPIN_SUCCESS';
+export const UNPIN_FAIL    = 'UNPIN_FAIL';
+
+export const BOOKMARK_REQUEST = 'BOOKMARK_REQUEST';
+export const BOOKMARK_SUCCESS = 'BOOKMARKED_SUCCESS';
+export const BOOKMARK_FAIL    = 'BOOKMARKED_FAIL';
+
+export const UNBOOKMARK_REQUEST = 'UNBOOKMARKED_REQUEST';
+export const UNBOOKMARK_SUCCESS = 'UNBOOKMARKED_SUCCESS';
+export const UNBOOKMARK_FAIL    = 'UNBOOKMARKED_FAIL';
+
+export const REACTION_UPDATE = 'REACTION_UPDATE';
+
+export const REACTION_ADD_REQUEST = 'REACTION_ADD_REQUEST';
+export const REACTION_ADD_SUCCESS = 'REACTION_ADD_SUCCESS';
+export const REACTION_ADD_FAIL    = 'REACTION_ADD_FAIL';
+
+export const REACTION_REMOVE_REQUEST = 'REACTION_REMOVE_REQUEST';
+export const REACTION_REMOVE_SUCCESS = 'REACTION_REMOVE_SUCCESS';
+export const REACTION_REMOVE_FAIL    = 'REACTION_REMOVE_FAIL';
+
+export function reblog(status, visibility) {
+  return function (dispatch, getState) {
+    dispatch(reblogRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/reblog`, { visibility }).then(function (response) {
+      // The reblog API method returns a new status wrapped around the original. In this case we are only
+      // interested in how the original is modified, hence passing it skipping the wrapper
+      dispatch(importFetchedStatus(response.data.reblog));
+      dispatch(reblogSuccess(status));
+    }).catch(function (error) {
+      dispatch(reblogFail(status, error));
+    });
+  };
+}
+
+export function unreblog(status) {
+  return (dispatch, getState) => {
+    dispatch(unreblogRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/unreblog`).then(response => {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(unreblogSuccess(status));
+    }).catch(error => {
+      dispatch(unreblogFail(status, error));
+    });
+  };
+}
+
+export function reblogRequest(status) {
+  return {
+    type: REBLOG_REQUEST,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function reblogSuccess(status) {
+  return {
+    type: REBLOG_SUCCESS,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function reblogFail(status, error) {
+  return {
+    type: REBLOG_FAIL,
+    status: status,
+    error: error,
+    skipLoading: true,
+  };
+}
+
+export function unreblogRequest(status) {
+  return {
+    type: UNREBLOG_REQUEST,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function unreblogSuccess(status) {
+  return {
+    type: UNREBLOG_SUCCESS,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function unreblogFail(status, error) {
+  return {
+    type: UNREBLOG_FAIL,
+    status: status,
+    error: error,
+    skipLoading: true,
+  };
+}
+
+export function favourite(status) {
+  return function (dispatch, getState) {
+    dispatch(favouriteRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/favourite`).then(function (response) {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(favouriteSuccess(status));
+    }).catch(function (error) {
+      dispatch(favouriteFail(status, error));
+    });
+  };
+}
+
+export function unfavourite(status) {
+  return (dispatch, getState) => {
+    dispatch(unfavouriteRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/unfavourite`).then(response => {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(unfavouriteSuccess(status));
+    }).catch(error => {
+      dispatch(unfavouriteFail(status, error));
+    });
+  };
+}
+
+export function favouriteRequest(status) {
+  return {
+    type: FAVOURITE_REQUEST,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function favouriteSuccess(status) {
+  return {
+    type: FAVOURITE_SUCCESS,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function favouriteFail(status, error) {
+  return {
+    type: FAVOURITE_FAIL,
+    status: status,
+    error: error,
+    skipLoading: true,
+  };
+}
+
+export function unfavouriteRequest(status) {
+  return {
+    type: UNFAVOURITE_REQUEST,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function unfavouriteSuccess(status) {
+  return {
+    type: UNFAVOURITE_SUCCESS,
+    status: status,
+    skipLoading: true,
+  };
+}
+
+export function unfavouriteFail(status, error) {
+  return {
+    type: UNFAVOURITE_FAIL,
+    status: status,
+    error: error,
+    skipLoading: true,
+  };
+}
+
+export function bookmark(status) {
+  return function (dispatch, getState) {
+    dispatch(bookmarkRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/bookmark`).then(function (response) {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(bookmarkSuccess(status, response.data));
+    }).catch(function (error) {
+      dispatch(bookmarkFail(status, error));
+    });
+  };
+}
+
+export function unbookmark(status) {
+  return (dispatch, getState) => {
+    dispatch(unbookmarkRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/unbookmark`).then(response => {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(unbookmarkSuccess(status, response.data));
+    }).catch(error => {
+      dispatch(unbookmarkFail(status, error));
+    });
+  };
+}
+
+export function bookmarkRequest(status) {
+  return {
+    type: BOOKMARK_REQUEST,
+    status: status,
+  };
+}
+
+export function bookmarkSuccess(status, response) {
+  return {
+    type: BOOKMARK_SUCCESS,
+    status: status,
+    response: response,
+  };
+}
+
+export function bookmarkFail(status, error) {
+  return {
+    type: BOOKMARK_FAIL,
+    status: status,
+    error: error,
+  };
+}
+
+export function unbookmarkRequest(status) {
+  return {
+    type: UNBOOKMARK_REQUEST,
+    status: status,
+  };
+}
+
+export function unbookmarkSuccess(status, response) {
+  return {
+    type: UNBOOKMARK_SUCCESS,
+    status: status,
+    response: response,
+  };
+}
+
+export function unbookmarkFail(status, error) {
+  return {
+    type: UNBOOKMARK_FAIL,
+    status: status,
+    error: error,
+  };
+}
+
+export function fetchReblogs(id) {
+  return (dispatch, getState) => {
+    dispatch(fetchReblogsRequest(id));
+
+    api(getState).get(`/api/v1/statuses/${id}/reblogged_by`).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchReblogsSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => {
+      dispatch(fetchReblogsFail(id, error));
+    });
+  };
+}
+
+export function fetchReblogsRequest(id) {
+  return {
+    type: REBLOGS_FETCH_REQUEST,
+    id,
+  };
+}
+
+export function fetchReblogsSuccess(id, accounts, next) {
+  return {
+    type: REBLOGS_FETCH_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function fetchReblogsFail(id, error) {
+  return {
+    type: REBLOGS_FETCH_FAIL,
+    id,
+    error,
+  };
+}
+
+export function expandReblogs(id) {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'reblogged_by', id, 'next']);
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandReblogsRequest(id));
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandReblogsSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => dispatch(expandReblogsFail(id, error)));
+  };
+}
+
+export function expandReblogsRequest(id) {
+  return {
+    type: REBLOGS_EXPAND_REQUEST,
+    id,
+  };
+}
+
+export function expandReblogsSuccess(id, accounts, next) {
+  return {
+    type: REBLOGS_EXPAND_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function expandReblogsFail(id, error) {
+  return {
+    type: REBLOGS_EXPAND_FAIL,
+    id,
+    error,
+  };
+}
+
+export function fetchFavourites(id) {
+  return (dispatch, getState) => {
+    dispatch(fetchFavouritesRequest(id));
+
+    api(getState).get(`/api/v1/statuses/${id}/favourited_by`).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchFavouritesSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => {
+      dispatch(fetchFavouritesFail(id, error));
+    });
+  };
+}
+
+export function fetchFavouritesRequest(id) {
+  return {
+    type: FAVOURITES_FETCH_REQUEST,
+    id,
+  };
+}
+
+export function fetchFavouritesSuccess(id, accounts, next) {
+  return {
+    type: FAVOURITES_FETCH_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function fetchFavouritesFail(id, error) {
+  return {
+    type: FAVOURITES_FETCH_FAIL,
+    id,
+    error,
+  };
+}
+
+export function expandFavourites(id) {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'favourited_by', id, 'next']);
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandFavouritesRequest(id));
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandFavouritesSuccess(id, response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => dispatch(expandFavouritesFail(id, error)));
+  };
+}
+
+export function expandFavouritesRequest(id) {
+  return {
+    type: FAVOURITES_EXPAND_REQUEST,
+    id,
+  };
+}
+
+export function expandFavouritesSuccess(id, accounts, next) {
+  return {
+    type: FAVOURITES_EXPAND_SUCCESS,
+    id,
+    accounts,
+    next,
+  };
+}
+
+export function expandFavouritesFail(id, error) {
+  return {
+    type: FAVOURITES_EXPAND_FAIL,
+    id,
+    error,
+  };
+}
+
+export function pin(status) {
+  return (dispatch, getState) => {
+    dispatch(pinRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/pin`).then(response => {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(pinSuccess(status));
+    }).catch(error => {
+      dispatch(pinFail(status, error));
+    });
+  };
+}
+
+export function pinRequest(status) {
+  return {
+    type: PIN_REQUEST,
+    status,
+    skipLoading: true,
+  };
+}
+
+export function pinSuccess(status) {
+  return {
+    type: PIN_SUCCESS,
+    status,
+    skipLoading: true,
+  };
+}
+
+export function pinFail(status, error) {
+  return {
+    type: PIN_FAIL,
+    status,
+    error,
+    skipLoading: true,
+  };
+}
+
+export function unpin (status) {
+  return (dispatch, getState) => {
+    dispatch(unpinRequest(status));
+
+    api(getState).post(`/api/v1/statuses/${status.get('id')}/unpin`).then(response => {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(unpinSuccess(status));
+    }).catch(error => {
+      dispatch(unpinFail(status, error));
+    });
+  };
+}
+
+export function unpinRequest(status) {
+  return {
+    type: UNPIN_REQUEST,
+    status,
+    skipLoading: true,
+  };
+}
+
+export function unpinSuccess(status) {
+  return {
+    type: UNPIN_SUCCESS,
+    status,
+    skipLoading: true,
+  };
+}
+
+export function unpinFail(status, error) {
+  return {
+    type: UNPIN_FAIL,
+    status,
+    error,
+    skipLoading: true,
+  };
+}
+
+export const addReaction = (statusId, name, url) => (dispatch, getState) => {
+  const status = getState().get('statuses').get(statusId);
+  let alreadyAdded = false;
+  if (status) {
+    const reaction = status.get('reactions').find(x => x.get('name') === name);
+    if (reaction && reaction.get('me')) {
+      alreadyAdded = true;
+    }
+  }
+  if (!alreadyAdded) {
+    dispatch(addReactionRequest(statusId, name, url));
+  }
+
+  // encodeURIComponent is required for the Keycap Number Sign emoji, see:
+  // <https://github.com/blobfox-soc/mastodon/pull/1980#issuecomment-1345538932>
+  api(getState).post(`/api/v1/statuses/${statusId}/react/${encodeURIComponent(name)}`).then(() => {
+    dispatch(addReactionSuccess(statusId, name));
+  }).catch(err => {
+    if (!alreadyAdded) {
+      dispatch(addReactionFail(statusId, name, err));
+    }
+  });
+};
+
+export const addReactionRequest = (statusId, name, url) => ({
+  type: REACTION_ADD_REQUEST,
+  id: statusId,
+  name,
+  url,
+});
+
+export const addReactionSuccess = (statusId, name) => ({
+  type: REACTION_ADD_SUCCESS,
+  id: statusId,
+  name,
+});
+
+export const addReactionFail = (statusId, name, error) => ({
+  type: REACTION_ADD_FAIL,
+  id: statusId,
+  name,
+  error,
+});
+
+export const removeReaction = (statusId, name) => (dispatch, getState) => {
+  dispatch(removeReactionRequest(statusId, name));
+
+  api(getState).post(`/api/v1/statuses/${statusId}/unreact/${encodeURIComponent(name)}`).then(() => {
+    dispatch(removeReactionSuccess(statusId, name));
+  }).catch(err => {
+    dispatch(removeReactionFail(statusId, name, err));
+  });
+};
+
+export const removeReactionRequest = (statusId, name) => ({
+  type: REACTION_REMOVE_REQUEST,
+  id: statusId,
+  name,
+});
+
+export const removeReactionSuccess = (statusId, name) => ({
+  type: REACTION_REMOVE_SUCCESS,
+  id: statusId,
+  name,
+});
+
+export const removeReactionFail = (statusId, name) => ({
+  type: REACTION_REMOVE_FAIL,
+  id: statusId,
+  name,
+});
diff --git a/app/javascript/flavours/blobfox/actions/languages.js b/app/javascript/flavours/blobfox/actions/languages.js
new file mode 100644
index 00000000000000..ad186ba0ccd3e5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/languages.js
@@ -0,0 +1,12 @@
+import { saveSettings } from './settings';
+
+export const LANGUAGE_USE = 'LANGUAGE_USE';
+
+export const useLanguage = language => dispatch => {
+  dispatch({
+    type: LANGUAGE_USE,
+    language,
+  });
+
+  dispatch(saveSettings());
+};
diff --git a/app/javascript/flavours/blobfox/actions/lists.js b/app/javascript/flavours/blobfox/actions/lists.js
new file mode 100644
index 00000000000000..b0789cd426450a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/lists.js
@@ -0,0 +1,373 @@
+import api from '../api';
+
+import { showAlertForError } from './alerts';
+import { importFetchedAccounts } from './importer';
+
+export const LIST_FETCH_REQUEST = 'LIST_FETCH_REQUEST';
+export const LIST_FETCH_SUCCESS = 'LIST_FETCH_SUCCESS';
+export const LIST_FETCH_FAIL    = 'LIST_FETCH_FAIL';
+
+export const LISTS_FETCH_REQUEST = 'LISTS_FETCH_REQUEST';
+export const LISTS_FETCH_SUCCESS = 'LISTS_FETCH_SUCCESS';
+export const LISTS_FETCH_FAIL    = 'LISTS_FETCH_FAIL';
+
+export const LIST_EDITOR_TITLE_CHANGE = 'LIST_EDITOR_TITLE_CHANGE';
+export const LIST_EDITOR_RESET        = 'LIST_EDITOR_RESET';
+export const LIST_EDITOR_SETUP        = 'LIST_EDITOR_SETUP';
+
+export const LIST_CREATE_REQUEST = 'LIST_CREATE_REQUEST';
+export const LIST_CREATE_SUCCESS = 'LIST_CREATE_SUCCESS';
+export const LIST_CREATE_FAIL    = 'LIST_CREATE_FAIL';
+
+export const LIST_UPDATE_REQUEST = 'LIST_UPDATE_REQUEST';
+export const LIST_UPDATE_SUCCESS = 'LIST_UPDATE_SUCCESS';
+export const LIST_UPDATE_FAIL    = 'LIST_UPDATE_FAIL';
+
+export const LIST_DELETE_REQUEST = 'LIST_DELETE_REQUEST';
+export const LIST_DELETE_SUCCESS = 'LIST_DELETE_SUCCESS';
+export const LIST_DELETE_FAIL    = 'LIST_DELETE_FAIL';
+
+export const LIST_ACCOUNTS_FETCH_REQUEST = 'LIST_ACCOUNTS_FETCH_REQUEST';
+export const LIST_ACCOUNTS_FETCH_SUCCESS = 'LIST_ACCOUNTS_FETCH_SUCCESS';
+export const LIST_ACCOUNTS_FETCH_FAIL    = 'LIST_ACCOUNTS_FETCH_FAIL';
+
+export const LIST_EDITOR_SUGGESTIONS_CHANGE = 'LIST_EDITOR_SUGGESTIONS_CHANGE';
+export const LIST_EDITOR_SUGGESTIONS_READY  = 'LIST_EDITOR_SUGGESTIONS_READY';
+export const LIST_EDITOR_SUGGESTIONS_CLEAR  = 'LIST_EDITOR_SUGGESTIONS_CLEAR';
+
+export const LIST_EDITOR_ADD_REQUEST = 'LIST_EDITOR_ADD_REQUEST';
+export const LIST_EDITOR_ADD_SUCCESS = 'LIST_EDITOR_ADD_SUCCESS';
+export const LIST_EDITOR_ADD_FAIL    = 'LIST_EDITOR_ADD_FAIL';
+
+export const LIST_EDITOR_REMOVE_REQUEST = 'LIST_EDITOR_REMOVE_REQUEST';
+export const LIST_EDITOR_REMOVE_SUCCESS = 'LIST_EDITOR_REMOVE_SUCCESS';
+export const LIST_EDITOR_REMOVE_FAIL    = 'LIST_EDITOR_REMOVE_FAIL';
+
+export const LIST_ADDER_RESET = 'LIST_ADDER_RESET';
+export const LIST_ADDER_SETUP = 'LIST_ADDER_SETUP';
+
+export const LIST_ADDER_LISTS_FETCH_REQUEST = 'LIST_ADDER_LISTS_FETCH_REQUEST';
+export const LIST_ADDER_LISTS_FETCH_SUCCESS = 'LIST_ADDER_LISTS_FETCH_SUCCESS';
+export const LIST_ADDER_LISTS_FETCH_FAIL    = 'LIST_ADDER_LISTS_FETCH_FAIL';
+
+export const fetchList = id => (dispatch, getState) => {
+  if (getState().getIn(['lists', id])) {
+    return;
+  }
+
+  dispatch(fetchListRequest(id));
+
+  api(getState).get(`/api/v1/lists/${id}`)
+    .then(({ data }) => dispatch(fetchListSuccess(data)))
+    .catch(err => dispatch(fetchListFail(id, err)));
+};
+
+export const fetchListRequest = id => ({
+  type: LIST_FETCH_REQUEST,
+  id,
+});
+
+export const fetchListSuccess = list => ({
+  type: LIST_FETCH_SUCCESS,
+  list,
+});
+
+export const fetchListFail = (id, error) => ({
+  type: LIST_FETCH_FAIL,
+  id,
+  error,
+});
+
+export const fetchLists = () => (dispatch, getState) => {
+  dispatch(fetchListsRequest());
+
+  api(getState).get('/api/v1/lists')
+    .then(({ data }) => dispatch(fetchListsSuccess(data)))
+    .catch(err => dispatch(fetchListsFail(err)));
+};
+
+export const fetchListsRequest = () => ({
+  type: LISTS_FETCH_REQUEST,
+});
+
+export const fetchListsSuccess = lists => ({
+  type: LISTS_FETCH_SUCCESS,
+  lists,
+});
+
+export const fetchListsFail = error => ({
+  type: LISTS_FETCH_FAIL,
+  error,
+});
+
+export const submitListEditor = shouldReset => (dispatch, getState) => {
+  const listId = getState().getIn(['listEditor', 'listId']);
+  const title  = getState().getIn(['listEditor', 'title']);
+
+  if (listId === null) {
+    dispatch(createList(title, shouldReset));
+  } else {
+    dispatch(updateList(listId, title, shouldReset));
+  }
+};
+
+export const setupListEditor = listId => (dispatch, getState) => {
+  dispatch({
+    type: LIST_EDITOR_SETUP,
+    list: getState().getIn(['lists', listId]),
+  });
+
+  dispatch(fetchListAccounts(listId));
+};
+
+export const changeListEditorTitle = value => ({
+  type: LIST_EDITOR_TITLE_CHANGE,
+  value,
+});
+
+export const createList = (title, shouldReset) => (dispatch, getState) => {
+  dispatch(createListRequest());
+
+  api(getState).post('/api/v1/lists', { title }).then(({ data }) => {
+    dispatch(createListSuccess(data));
+
+    if (shouldReset) {
+      dispatch(resetListEditor());
+    }
+  }).catch(err => dispatch(createListFail(err)));
+};
+
+export const createListRequest = () => ({
+  type: LIST_CREATE_REQUEST,
+});
+
+export const createListSuccess = list => ({
+  type: LIST_CREATE_SUCCESS,
+  list,
+});
+
+export const createListFail = error => ({
+  type: LIST_CREATE_FAIL,
+  error,
+});
+
+export const updateList = (id, title, shouldReset, isExclusive, replies_policy) => (dispatch, getState) => {
+  dispatch(updateListRequest(id));
+
+  api(getState).put(`/api/v1/lists/${id}`, { title, replies_policy, exclusive: typeof isExclusive === 'undefined' ? undefined : !!isExclusive }).then(({ data }) => {
+    dispatch(updateListSuccess(data));
+
+    if (shouldReset) {
+      dispatch(resetListEditor());
+    }
+  }).catch(err => dispatch(updateListFail(id, err)));
+};
+
+export const updateListRequest = id => ({
+  type: LIST_UPDATE_REQUEST,
+  id,
+});
+
+export const updateListSuccess = list => ({
+  type: LIST_UPDATE_SUCCESS,
+  list,
+});
+
+export const updateListFail = (id, error) => ({
+  type: LIST_UPDATE_FAIL,
+  id,
+  error,
+});
+
+export const resetListEditor = () => ({
+  type: LIST_EDITOR_RESET,
+});
+
+export const deleteList = id => (dispatch, getState) => {
+  dispatch(deleteListRequest(id));
+
+  api(getState).delete(`/api/v1/lists/${id}`)
+    .then(() => dispatch(deleteListSuccess(id)))
+    .catch(err => dispatch(deleteListFail(id, err)));
+};
+
+export const deleteListRequest = id => ({
+  type: LIST_DELETE_REQUEST,
+  id,
+});
+
+export const deleteListSuccess = id => ({
+  type: LIST_DELETE_SUCCESS,
+  id,
+});
+
+export const deleteListFail = (id, error) => ({
+  type: LIST_DELETE_FAIL,
+  id,
+  error,
+});
+
+export const fetchListAccounts = listId => (dispatch, getState) => {
+  dispatch(fetchListAccountsRequest(listId));
+
+  api(getState).get(`/api/v1/lists/${listId}/accounts`, { params: { limit: 0 } }).then(({ data }) => {
+    dispatch(importFetchedAccounts(data));
+    dispatch(fetchListAccountsSuccess(listId, data));
+  }).catch(err => dispatch(fetchListAccountsFail(listId, err)));
+};
+
+export const fetchListAccountsRequest = id => ({
+  type: LIST_ACCOUNTS_FETCH_REQUEST,
+  id,
+});
+
+export const fetchListAccountsSuccess = (id, accounts, next) => ({
+  type: LIST_ACCOUNTS_FETCH_SUCCESS,
+  id,
+  accounts,
+  next,
+});
+
+export const fetchListAccountsFail = (id, error) => ({
+  type: LIST_ACCOUNTS_FETCH_FAIL,
+  id,
+  error,
+});
+
+export const fetchListSuggestions = q => (dispatch, getState) => {
+  const params = {
+    q,
+    resolve: false,
+    limit: 4,
+    following: true,
+  };
+
+  api(getState).get('/api/v1/accounts/search', { params }).then(({ data }) => {
+    dispatch(importFetchedAccounts(data));
+    dispatch(fetchListSuggestionsReady(q, data));
+  }).catch(error => dispatch(showAlertForError(error)));
+};
+
+export const fetchListSuggestionsReady = (query, accounts) => ({
+  type: LIST_EDITOR_SUGGESTIONS_READY,
+  query,
+  accounts,
+});
+
+export const clearListSuggestions = () => ({
+  type: LIST_EDITOR_SUGGESTIONS_CLEAR,
+});
+
+export const changeListSuggestions = value => ({
+  type: LIST_EDITOR_SUGGESTIONS_CHANGE,
+  value,
+});
+
+export const addToListEditor = accountId => (dispatch, getState) => {
+  dispatch(addToList(getState().getIn(['listEditor', 'listId']), accountId));
+};
+
+export const addToList = (listId, accountId) => (dispatch, getState) => {
+  dispatch(addToListRequest(listId, accountId));
+
+  api(getState).post(`/api/v1/lists/${listId}/accounts`, { account_ids: [accountId] })
+    .then(() => dispatch(addToListSuccess(listId, accountId)))
+    .catch(err => dispatch(addToListFail(listId, accountId, err)));
+};
+
+export const addToListRequest = (listId, accountId) => ({
+  type: LIST_EDITOR_ADD_REQUEST,
+  listId,
+  accountId,
+});
+
+export const addToListSuccess = (listId, accountId) => ({
+  type: LIST_EDITOR_ADD_SUCCESS,
+  listId,
+  accountId,
+});
+
+export const addToListFail = (listId, accountId, error) => ({
+  type: LIST_EDITOR_ADD_FAIL,
+  listId,
+  accountId,
+  error,
+});
+
+export const removeFromListEditor = accountId => (dispatch, getState) => {
+  dispatch(removeFromList(getState().getIn(['listEditor', 'listId']), accountId));
+};
+
+export const removeFromList = (listId, accountId) => (dispatch, getState) => {
+  dispatch(removeFromListRequest(listId, accountId));
+
+  api(getState).delete(`/api/v1/lists/${listId}/accounts`, { params: { account_ids: [accountId] } })
+    .then(() => dispatch(removeFromListSuccess(listId, accountId)))
+    .catch(err => dispatch(removeFromListFail(listId, accountId, err)));
+};
+
+export const removeFromListRequest = (listId, accountId) => ({
+  type: LIST_EDITOR_REMOVE_REQUEST,
+  listId,
+  accountId,
+});
+
+export const removeFromListSuccess = (listId, accountId) => ({
+  type: LIST_EDITOR_REMOVE_SUCCESS,
+  listId,
+  accountId,
+});
+
+export const removeFromListFail = (listId, accountId, error) => ({
+  type: LIST_EDITOR_REMOVE_FAIL,
+  listId,
+  accountId,
+  error,
+});
+
+export const resetListAdder = () => ({
+  type: LIST_ADDER_RESET,
+});
+
+export const setupListAdder = accountId => (dispatch, getState) => {
+  dispatch({
+    type: LIST_ADDER_SETUP,
+    account: getState().getIn(['accounts', accountId]),
+  });
+  dispatch(fetchLists());
+  dispatch(fetchAccountLists(accountId));
+};
+
+export const fetchAccountLists = accountId => (dispatch, getState) => {
+  dispatch(fetchAccountListsRequest(accountId));
+
+  api(getState).get(`/api/v1/accounts/${accountId}/lists`)
+    .then(({ data }) => dispatch(fetchAccountListsSuccess(accountId, data)))
+    .catch(err => dispatch(fetchAccountListsFail(accountId, err)));
+};
+
+export const fetchAccountListsRequest = id => ({
+  type:LIST_ADDER_LISTS_FETCH_REQUEST,
+  id,
+});
+
+export const fetchAccountListsSuccess = (id, lists) => ({
+  type: LIST_ADDER_LISTS_FETCH_SUCCESS,
+  id,
+  lists,
+});
+
+export const fetchAccountListsFail = (id, err) => ({
+  type: LIST_ADDER_LISTS_FETCH_FAIL,
+  id,
+  err,
+});
+
+export const addToListAdder = listId => (dispatch, getState) => {
+  dispatch(addToList(listId, getState().getIn(['listAdder', 'accountId'])));
+};
+
+export const removeFromListAdder = listId => (dispatch, getState) => {
+  dispatch(removeFromList(listId, getState().getIn(['listAdder', 'accountId'])));
+};
+
diff --git a/app/javascript/flavours/blobfox/actions/local_settings.js b/app/javascript/flavours/blobfox/actions/local_settings.js
new file mode 100644
index 00000000000000..a02e9d763eb9cd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/local_settings.js
@@ -0,0 +1,81 @@
+import { expandSpoilers, disableSwiping } from 'flavours/blobfox/initial_state';
+
+import { openModal } from './modal';
+
+export const LOCAL_SETTING_CHANGE = 'LOCAL_SETTING_CHANGE';
+export const LOCAL_SETTING_DELETE = 'LOCAL_SETTING_DELETE';
+
+export function checkDeprecatedLocalSettings() {
+  return (dispatch, getState) => {
+    const local_auto_unfold = getState().getIn(['local_settings', 'content_warnings', 'auto_unfold']);
+    const local_swipe_to_change_columns = getState().getIn(['local_settings', 'swipe_to_change_columns']);
+    let changed_settings = [];
+
+    if (local_auto_unfold !== null && local_auto_unfold !== undefined) {
+      if (local_auto_unfold === expandSpoilers) {
+        dispatch(deleteLocalSetting(['content_warnings', 'auto_unfold']));
+      } else {
+        changed_settings.push('user_setting_expand_spoilers');
+      }
+    }
+
+    if (local_swipe_to_change_columns !== null && local_swipe_to_change_columns !== undefined) {
+      if (local_swipe_to_change_columns === !disableSwiping) {
+        dispatch(deleteLocalSetting(['swipe_to_change_columns']));
+      } else {
+        changed_settings.push('user_setting_disable_swiping');
+      }
+    }
+
+    if (changed_settings.length > 0) {
+      dispatch(openModal({
+        modalType: 'DEPRECATED_SETTINGS',
+        modalProps: {
+          settings: changed_settings,
+          onConfirm: () => dispatch(clearDeprecatedLocalSettings()),
+        },
+      }));
+    }
+  };
+}
+
+export function clearDeprecatedLocalSettings() {
+  return (dispatch) => {
+    dispatch(deleteLocalSetting(['content_warnings', 'auto_unfold']));
+    dispatch(deleteLocalSetting(['swipe_to_change_columns']));
+  };
+}
+
+export function changeLocalSetting(key, value) {
+  return dispatch => {
+    dispatch({
+      type: LOCAL_SETTING_CHANGE,
+      key,
+      value,
+    });
+
+    dispatch(saveLocalSettings());
+  };
+}
+
+export function deleteLocalSetting(key) {
+  return dispatch => {
+    dispatch({
+      type: LOCAL_SETTING_DELETE,
+      key,
+    });
+
+    dispatch(saveLocalSettings());
+  };
+}
+
+//  __TODO :__
+//  Right now `saveLocalSettings()` doesn't keep track of which user
+//  is currently signed in, but it might be better to give each user
+//  their *own* local settings.
+export function saveLocalSettings() {
+  return (_, getState) => {
+    const localSettings = getState().get('local_settings').toJS();
+    localStorage.setItem('mastodon-settings', JSON.stringify(localSettings));
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/markers.js b/app/javascript/flavours/blobfox/actions/markers.js
new file mode 100644
index 00000000000000..ccb1b23d6f9f54
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/markers.js
@@ -0,0 +1,152 @@
+import { List as ImmutableList } from 'immutable';
+
+import { debounce } from 'lodash';
+
+import api from '../api';
+import { compareId } from '../compare_id';
+
+export const MARKERS_FETCH_REQUEST = 'MARKERS_FETCH_REQUEST';
+export const MARKERS_FETCH_SUCCESS = 'MARKERS_FETCH_SUCCESS';
+export const MARKERS_FETCH_FAIL    = 'MARKERS_FETCH_FAIL';
+export const MARKERS_SUBMIT_SUCCESS = 'MARKERS_SUBMIT_SUCCESS';
+
+export const synchronouslySubmitMarkers = () => (dispatch, getState) => {
+  const accessToken = getState().getIn(['meta', 'access_token'], '');
+  const params      = _buildParams(getState());
+
+  if (Object.keys(params).length === 0 || accessToken === '') {
+    return;
+  }
+
+  // The Fetch API allows us to perform requests that will be carried out
+  // after the page closes. But that only works if the `keepalive` attribute
+  // is supported.
+  if (window.fetch && 'keepalive' in new Request('')) {
+    fetch('/api/v1/markers', {
+      keepalive: true,
+      method: 'POST',
+      headers: {
+        'Content-Type': 'application/json',
+        'Authorization': `Bearer ${accessToken}`,
+      },
+      body: JSON.stringify(params),
+    });
+
+    return;
+  } else if (navigator && navigator.sendBeacon) {
+    // Failing that, we can use sendBeacon, but we have to encode the data as
+    // FormData for DoorKeeper to recognize the token.
+    const formData = new FormData();
+
+    formData.append('bearer_token', accessToken);
+
+    for (const [id, value] of Object.entries(params)) {
+      formData.append(`${id}[last_read_id]`, value.last_read_id);
+    }
+
+    if (navigator.sendBeacon('/api/v1/markers', formData)) {
+      return;
+    }
+  }
+
+  // If neither Fetch nor sendBeacon worked, try to perform a synchronous
+  // request.
+  try {
+    const client = new XMLHttpRequest();
+
+    client.open('POST', '/api/v1/markers', false);
+    client.setRequestHeader('Content-Type', 'application/json');
+    client.setRequestHeader('Authorization', `Bearer ${accessToken}`);
+    client.send(JSON.stringify(params));
+  } catch (e) {
+    // Do not make the BeforeUnload handler error out
+  }
+};
+
+const _buildParams = (state) => {
+  const params = {};
+
+  const lastHomeId         = state.getIn(['timelines', 'home', 'items'], ImmutableList()).find(item => item !== null);
+  const lastNotificationId = state.getIn(['notifications', 'lastReadId']);
+
+  if (lastHomeId && compareId(lastHomeId, state.getIn(['markers', 'home'])) > 0) {
+    params.home = {
+      last_read_id: lastHomeId,
+    };
+  }
+
+  if (lastNotificationId && lastNotificationId !== '0' && compareId(lastNotificationId, state.getIn(['markers', 'notifications'])) > 0) {
+    params.notifications = {
+      last_read_id: lastNotificationId,
+    };
+  }
+
+  return params;
+};
+
+const debouncedSubmitMarkers = debounce((dispatch, getState) => {
+  const accessToken = getState().getIn(['meta', 'access_token'], '');
+  const params      = _buildParams(getState());
+
+  if (Object.keys(params).length === 0 || accessToken === '') {
+    return;
+  }
+
+  api(getState).post('/api/v1/markers', params).then(() => {
+    dispatch(submitMarkersSuccess(params));
+  }).catch(() => {});
+}, 300000, { leading: true, trailing: true });
+
+export function submitMarkersSuccess({ home, notifications }) {
+  return {
+    type: MARKERS_SUBMIT_SUCCESS,
+    home: (home || {}).last_read_id,
+    notifications: (notifications || {}).last_read_id,
+  };
+}
+
+export function submitMarkers(params = {}) {
+  const result = (dispatch, getState) => debouncedSubmitMarkers(dispatch, getState);
+
+  if (params.immediate === true) {
+    debouncedSubmitMarkers.flush();
+  }
+
+  return result;
+}
+
+export const fetchMarkers = () => (dispatch, getState) => {
+  const params = { timeline: ['notifications'] };
+
+  dispatch(fetchMarkersRequest());
+
+  api(getState).get('/api/v1/markers', { params }).then(response => {
+    dispatch(fetchMarkersSuccess(response.data));
+  }).catch(error => {
+    dispatch(fetchMarkersFail(error));
+  });
+};
+
+export function fetchMarkersRequest() {
+  return {
+    type: MARKERS_FETCH_REQUEST,
+    skipLoading: true,
+  };
+}
+
+export function fetchMarkersSuccess(markers) {
+  return {
+    type: MARKERS_FETCH_SUCCESS,
+    markers,
+    skipLoading: true,
+  };
+}
+
+export function fetchMarkersFail(error) {
+  return {
+    type: MARKERS_FETCH_FAIL,
+    error,
+    skipLoading: true,
+    skipAlert: true,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/modal.ts b/app/javascript/flavours/blobfox/actions/modal.ts
new file mode 100644
index 00000000000000..441d185a9c808c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/modal.ts
@@ -0,0 +1,19 @@
+import { createAction } from '@reduxjs/toolkit';
+
+import type { ModalProps } from 'flavours/blobfox/reducers/modal';
+
+import type { MODAL_COMPONENTS } from '../features/ui/components/modal_root';
+
+export type ModalType = keyof typeof MODAL_COMPONENTS;
+
+interface OpenModalPayload {
+  modalType: ModalType;
+  modalProps: ModalProps;
+}
+export const openModal = createAction<OpenModalPayload>('MODAL_OPEN');
+
+interface CloseModalPayload {
+  modalType: ModalType | undefined;
+  ignoreFocus: boolean;
+}
+export const closeModal = createAction<CloseModalPayload>('MODAL_CLOSE');
diff --git a/app/javascript/flavours/blobfox/actions/mutes.js b/app/javascript/flavours/blobfox/actions/mutes.js
new file mode 100644
index 00000000000000..fb041078b84488
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/mutes.js
@@ -0,0 +1,117 @@
+import api, { getLinks } from '../api';
+
+import { fetchRelationships } from './accounts';
+import { importFetchedAccounts } from './importer';
+import { openModal } from './modal';
+
+export const MUTES_FETCH_REQUEST = 'MUTES_FETCH_REQUEST';
+export const MUTES_FETCH_SUCCESS = 'MUTES_FETCH_SUCCESS';
+export const MUTES_FETCH_FAIL    = 'MUTES_FETCH_FAIL';
+
+export const MUTES_EXPAND_REQUEST = 'MUTES_EXPAND_REQUEST';
+export const MUTES_EXPAND_SUCCESS = 'MUTES_EXPAND_SUCCESS';
+export const MUTES_EXPAND_FAIL    = 'MUTES_EXPAND_FAIL';
+
+export const MUTES_INIT_MODAL = 'MUTES_INIT_MODAL';
+export const MUTES_TOGGLE_HIDE_NOTIFICATIONS = 'MUTES_TOGGLE_HIDE_NOTIFICATIONS';
+export const MUTES_CHANGE_DURATION = 'MUTES_CHANGE_DURATION';
+
+export function fetchMutes() {
+  return (dispatch, getState) => {
+    dispatch(fetchMutesRequest());
+
+    api(getState).get('/api/v1/mutes').then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(fetchMutesSuccess(response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => dispatch(fetchMutesFail(error)));
+  };
+}
+
+export function fetchMutesRequest() {
+  return {
+    type: MUTES_FETCH_REQUEST,
+  };
+}
+
+export function fetchMutesSuccess(accounts, next) {
+  return {
+    type: MUTES_FETCH_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function fetchMutesFail(error) {
+  return {
+    type: MUTES_FETCH_FAIL,
+    error,
+  };
+}
+
+export function expandMutes() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['user_lists', 'mutes', 'next']);
+
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandMutesRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedAccounts(response.data));
+      dispatch(expandMutesSuccess(response.data, next ? next.uri : null));
+      dispatch(fetchRelationships(response.data.map(item => item.id)));
+    }).catch(error => dispatch(expandMutesFail(error)));
+  };
+}
+
+export function expandMutesRequest() {
+  return {
+    type: MUTES_EXPAND_REQUEST,
+  };
+}
+
+export function expandMutesSuccess(accounts, next) {
+  return {
+    type: MUTES_EXPAND_SUCCESS,
+    accounts,
+    next,
+  };
+}
+
+export function expandMutesFail(error) {
+  return {
+    type: MUTES_EXPAND_FAIL,
+    error,
+  };
+}
+
+export function initMuteModal(account) {
+  return dispatch => {
+    dispatch({
+      type: MUTES_INIT_MODAL,
+      account,
+    });
+
+    dispatch(openModal({ modalType: 'MUTE' }));
+  };
+}
+
+export function toggleHideNotifications() {
+  return dispatch => {
+    dispatch({ type: MUTES_TOGGLE_HIDE_NOTIFICATIONS });
+  };
+}
+
+export function changeMuteDuration(duration) {
+  return dispatch => {
+    dispatch({
+      type: MUTES_CHANGE_DURATION,
+      duration,
+    });
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/notifications.js b/app/javascript/flavours/blobfox/actions/notifications.js
new file mode 100644
index 00000000000000..ced89453d236a5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/notifications.js
@@ -0,0 +1,404 @@
+import { IntlMessageFormat } from 'intl-messageformat';
+import { defineMessages } from 'react-intl';
+
+import { List as ImmutableList } from 'immutable';
+
+import { compareId } from 'flavours/blobfox/compare_id';
+import { usePendingItems as preferPendingItems } from 'flavours/blobfox/initial_state';
+
+import api, { getLinks } from '../api';
+import { unescapeHTML } from '../utils/html';
+import { requestNotificationPermission } from '../utils/notifications';
+
+import { fetchFollowRequests, fetchRelationships } from './accounts';
+import {
+  importFetchedAccount,
+  importFetchedAccounts,
+  importFetchedStatus,
+  importFetchedStatuses,
+} from './importer';
+import { submitMarkers } from './markers';
+import { notificationsUpdate } from "./notifications_typed";
+import { register as registerPushNotifications } from './push_notifications';
+import { saveSettings } from './settings';
+
+export * from "./notifications_typed";
+
+export const NOTIFICATIONS_UPDATE_NOOP = 'NOTIFICATIONS_UPDATE_NOOP';
+
+// tracking the notif cleaning request
+export const NOTIFICATIONS_DELETE_MARKED_REQUEST = 'NOTIFICATIONS_DELETE_MARKED_REQUEST';
+export const NOTIFICATIONS_DELETE_MARKED_SUCCESS = 'NOTIFICATIONS_DELETE_MARKED_SUCCESS';
+export const NOTIFICATIONS_DELETE_MARKED_FAIL = 'NOTIFICATIONS_DELETE_MARKED_FAIL';
+export const NOTIFICATIONS_MARK_ALL_FOR_DELETE = 'NOTIFICATIONS_MARK_ALL_FOR_DELETE';
+export const NOTIFICATIONS_ENTER_CLEARING_MODE = 'NOTIFICATIONS_ENTER_CLEARING_MODE'; // arg: yes
+// Unmark notifications (when the cleaning mode is left)
+export const NOTIFICATIONS_UNMARK_ALL_FOR_DELETE = 'NOTIFICATIONS_UNMARK_ALL_FOR_DELETE';
+// Mark one for delete
+export const NOTIFICATION_MARK_FOR_DELETE = 'NOTIFICATION_MARK_FOR_DELETE';
+
+export const NOTIFICATIONS_EXPAND_REQUEST = 'NOTIFICATIONS_EXPAND_REQUEST';
+export const NOTIFICATIONS_EXPAND_SUCCESS = 'NOTIFICATIONS_EXPAND_SUCCESS';
+export const NOTIFICATIONS_EXPAND_FAIL    = 'NOTIFICATIONS_EXPAND_FAIL';
+
+export const NOTIFICATIONS_FILTER_SET = 'NOTIFICATIONS_FILTER_SET';
+
+export const NOTIFICATIONS_CLEAR        = 'NOTIFICATIONS_CLEAR';
+export const NOTIFICATIONS_SCROLL_TOP   = 'NOTIFICATIONS_SCROLL_TOP';
+export const NOTIFICATIONS_LOAD_PENDING = 'NOTIFICATIONS_LOAD_PENDING';
+
+export const NOTIFICATIONS_MOUNT   = 'NOTIFICATIONS_MOUNT';
+export const NOTIFICATIONS_UNMOUNT = 'NOTIFICATIONS_UNMOUNT';
+
+export const NOTIFICATIONS_SET_VISIBILITY = 'NOTIFICATIONS_SET_VISIBILITY';
+
+export const NOTIFICATIONS_MARK_AS_READ = 'NOTIFICATIONS_MARK_AS_READ';
+
+export const NOTIFICATIONS_SET_BROWSER_SUPPORT    = 'NOTIFICATIONS_SET_BROWSER_SUPPORT';
+export const NOTIFICATIONS_SET_BROWSER_PERMISSION = 'NOTIFICATIONS_SET_BROWSER_PERMISSION';
+
+defineMessages({
+  mention: { id: 'notification.mention', defaultMessage: '{name} mentioned you' },
+});
+
+const fetchRelatedRelationships = (dispatch, notifications) => {
+  const accountIds = notifications.filter(item => ['follow', 'follow_request', 'admin.sign_up'].indexOf(item.type) !== -1).map(item => item.account.id);
+
+  if (accountIds.length > 0) {
+    dispatch(fetchRelationships(accountIds));
+  }
+};
+
+export const loadPending = () => ({
+  type: NOTIFICATIONS_LOAD_PENDING,
+});
+
+export function updateNotifications(notification, intlMessages, intlLocale) {
+  return (dispatch, getState) => {
+    const activeFilter = getState().getIn(['settings', 'notifications', 'quickFilter', 'active']);
+    const showInColumn = activeFilter === 'all' ? getState().getIn(['settings', 'notifications', 'shows', notification.type], true) : activeFilter === notification.type;
+    const showAlert    = getState().getIn(['settings', 'notifications', 'alerts', notification.type], true);
+    const playSound    = getState().getIn(['settings', 'notifications', 'sounds', notification.type], true);
+
+    let filtered = false;
+
+    if (['mention', 'status'].includes(notification.type) && notification.status.filtered) {
+      const filters = notification.status.filtered.filter(result => result.filter.context.includes('notifications'));
+
+      if (filters.some(result => result.filter.filter_action === 'hide')) {
+        return;
+      }
+
+      filtered = filters.length > 0;
+    }
+
+    if (['follow_request'].includes(notification.type)) {
+      dispatch(fetchFollowRequests());
+    }
+
+    dispatch(submitMarkers());
+
+    if (showInColumn) {
+      dispatch(importFetchedAccount(notification.account));
+
+      if (notification.status) {
+        dispatch(importFetchedStatus(notification.status));
+      }
+
+      if (notification.report) {
+        dispatch(importFetchedAccount(notification.report.target_account));
+      }
+
+
+      dispatch(notificationsUpdate({ notification, preferPendingItems, playSound: playSound && !filtered}));
+
+      fetchRelatedRelationships(dispatch, [notification]);
+    } else if (playSound && !filtered) {
+      dispatch({
+        type: NOTIFICATIONS_UPDATE_NOOP,
+        meta: { sound: 'boop' },
+      });
+    }
+
+    // Desktop notifications
+    if (typeof window.Notification !== 'undefined' && showAlert && !filtered) {
+      const title = new IntlMessageFormat(intlMessages[`notification.${notification.type}`], intlLocale).format({ name: notification.account.display_name.length > 0 ? notification.account.display_name : notification.account.username });
+      const body  = (notification.status && notification.status.spoiler_text.length > 0) ? notification.status.spoiler_text : unescapeHTML(notification.status ? notification.status.content : '');
+
+      const notify = new Notification(title, { body, icon: notification.account.avatar, tag: notification.id });
+
+      notify.addEventListener('click', () => {
+        window.focus();
+        notify.close();
+      });
+    }
+  };
+}
+
+const excludeTypesFromSettings = state => state.getIn(['settings', 'notifications', 'shows']).filter(enabled => !enabled).keySeq().toJS();
+
+const excludeTypesFromFilter = filter => {
+  const allTypes = ImmutableList([
+    'follow',
+    'follow_request',
+    'favourite',
+    'reaction',
+    'reblog',
+    'mention',
+    'poll',
+    'status',
+    'update',
+    'admin.sign_up',
+    'admin.report',
+  ]);
+
+  return allTypes.filterNot(item => item === filter).toJS();
+};
+
+const noOp = () => {};
+
+let expandNotificationsController = new AbortController();
+
+export function expandNotifications({ maxId, forceLoad } = {}, done = noOp) {
+  return (dispatch, getState) => {
+    const activeFilter = getState().getIn(['settings', 'notifications', 'quickFilter', 'active']);
+    const notifications = getState().get('notifications');
+    const isLoadingMore = !!maxId;
+
+    if (notifications.get('isLoading')) {
+      if (forceLoad) {
+        expandNotificationsController.abort();
+        expandNotificationsController = new AbortController();
+      } else {
+        done();
+        return;
+      }
+    }
+
+    const params = {
+      max_id: maxId,
+      exclude_types: activeFilter === 'all'
+        ? excludeTypesFromSettings(getState())
+        : excludeTypesFromFilter(activeFilter),
+    };
+
+    if (!params.max_id && (notifications.get('items', ImmutableList()).size + notifications.get('pendingItems', ImmutableList()).size) > 0) {
+      const a = notifications.getIn(['pendingItems', 0, 'id']);
+      const b = notifications.getIn(['items', 0, 'id']);
+
+      if (a && b && compareId(a, b) > 0) {
+        params.since_id = a;
+      } else {
+        params.since_id = b || a;
+      }
+    }
+
+    const isLoadingRecent = !!params.since_id;
+
+    dispatch(expandNotificationsRequest(isLoadingMore));
+
+    api(getState).get('/api/v1/notifications', { params, signal: expandNotificationsController.signal }).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+
+      dispatch(importFetchedAccounts(response.data.map(item => item.account)));
+      dispatch(importFetchedStatuses(response.data.map(item => item.status).filter(status => !!status)));
+      dispatch(importFetchedAccounts(response.data.filter(item => item.report).map(item => item.report.target_account)));
+
+      dispatch(expandNotificationsSuccess(response.data, next ? next.uri : null, isLoadingMore, isLoadingRecent, isLoadingRecent && preferPendingItems));
+      fetchRelatedRelationships(dispatch, response.data);
+      dispatch(submitMarkers());
+    }).catch(error => {
+      dispatch(expandNotificationsFail(error, isLoadingMore));
+    }).finally(() => {
+      done();
+    });
+  };
+}
+
+export function expandNotificationsRequest(isLoadingMore) {
+  return {
+    type: NOTIFICATIONS_EXPAND_REQUEST,
+    skipLoading: !isLoadingMore,
+  };
+}
+
+export function expandNotificationsSuccess(notifications, next, isLoadingMore, isLoadingRecent, usePendingItems) {
+  return {
+    type: NOTIFICATIONS_EXPAND_SUCCESS,
+    notifications,
+    next,
+    isLoadingRecent: isLoadingRecent,
+    usePendingItems,
+    skipLoading: !isLoadingMore,
+  };
+}
+
+export function expandNotificationsFail(error, isLoadingMore) {
+  return {
+    type: NOTIFICATIONS_EXPAND_FAIL,
+    error,
+    skipLoading: !isLoadingMore,
+    skipAlert: !isLoadingMore || error.name === 'AbortError',
+  };
+}
+
+export function clearNotifications() {
+  return (dispatch, getState) => {
+    dispatch({
+      type: NOTIFICATIONS_CLEAR,
+    });
+
+    api(getState).post('/api/v1/notifications/clear');
+  };
+}
+
+export function scrollTopNotifications(top) {
+  return {
+    type: NOTIFICATIONS_SCROLL_TOP,
+    top,
+  };
+}
+
+export function deleteMarkedNotifications() {
+  return (dispatch, getState) => {
+    dispatch(deleteMarkedNotificationsRequest());
+
+    let ids = [];
+    getState().getIn(['notifications', 'items']).forEach((n) => {
+      if (n.get('markedForDelete')) {
+        ids.push(n.get('id'));
+      }
+    });
+
+    if (ids.length === 0) {
+      return;
+    }
+
+    api(getState).delete(`/api/v1/notifications/destroy_multiple?ids[]=${ids.join('&ids[]=')}`).then(() => {
+      dispatch(deleteMarkedNotificationsSuccess());
+    }).catch(error => {
+      console.error(error);
+      dispatch(deleteMarkedNotificationsFail(error));
+    });
+  };
+}
+
+export function enterNotificationClearingMode(yes) {
+  return {
+    type: NOTIFICATIONS_ENTER_CLEARING_MODE,
+    yes: yes,
+  };
+}
+
+export function markAllNotifications(yes) {
+  return {
+    type: NOTIFICATIONS_MARK_ALL_FOR_DELETE,
+    yes: yes, // true, false or null. null = invert
+  };
+}
+
+export function deleteMarkedNotificationsRequest() {
+  return {
+    type: NOTIFICATIONS_DELETE_MARKED_REQUEST,
+  };
+}
+
+export function deleteMarkedNotificationsFail() {
+  return {
+    type: NOTIFICATIONS_DELETE_MARKED_FAIL,
+  };
+}
+
+export function markNotificationForDelete(id, yes) {
+  return {
+    type: NOTIFICATION_MARK_FOR_DELETE,
+    id: id,
+    yes: yes,
+  };
+}
+
+export function deleteMarkedNotificationsSuccess() {
+  return {
+    type: NOTIFICATIONS_DELETE_MARKED_SUCCESS,
+  };
+}
+
+export function mountNotifications() {
+  return {
+    type: NOTIFICATIONS_MOUNT,
+  };
+}
+
+export function unmountNotifications() {
+  return {
+    type: NOTIFICATIONS_UNMOUNT,
+  };
+}
+
+export function notificationsSetVisibility(visibility) {
+  return {
+    type: NOTIFICATIONS_SET_VISIBILITY,
+    visibility: visibility,
+  };
+}
+
+export function setFilter (filterType) {
+  return dispatch => {
+    dispatch({
+      type: NOTIFICATIONS_FILTER_SET,
+      path: ['notifications', 'quickFilter', 'active'],
+      value: filterType,
+    });
+    dispatch(expandNotifications({ forceLoad: true }));
+    dispatch(saveSettings());
+  };
+}
+
+export function markNotificationsAsRead() {
+  return {
+    type: NOTIFICATIONS_MARK_AS_READ,
+  };
+}
+
+// Browser support
+export function setupBrowserNotifications() {
+  return dispatch => {
+    dispatch(setBrowserSupport('Notification' in window));
+    if ('Notification' in window) {
+      dispatch(setBrowserPermission(Notification.permission));
+    }
+
+    if ('Notification' in window && 'permissions' in navigator) {
+      navigator.permissions.query({ name: 'notifications' }).then((status) => {
+        status.onchange = () => dispatch(setBrowserPermission(Notification.permission));
+      }).catch(console.warn);
+    }
+  };
+}
+
+export function requestBrowserPermission(callback = noOp) {
+  return dispatch => {
+    requestNotificationPermission((permission) => {
+      dispatch(setBrowserPermission(permission));
+      callback(permission);
+
+      if (permission === 'granted') {
+        dispatch(registerPushNotifications());
+      }
+    });
+  };
+}
+
+export function setBrowserSupport (value) {
+  return {
+    type: NOTIFICATIONS_SET_BROWSER_SUPPORT,
+    value,
+  };
+}
+
+export function setBrowserPermission (value) {
+  return {
+    type: NOTIFICATIONS_SET_BROWSER_PERMISSION,
+    value,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/notifications_typed.ts b/app/javascript/flavours/blobfox/actions/notifications_typed.ts
new file mode 100644
index 00000000000000..176362f4b1edd4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/notifications_typed.ts
@@ -0,0 +1,23 @@
+import { createAction } from '@reduxjs/toolkit';
+
+import type { ApiAccountJSON } from '../api_types/accounts';
+// To be replaced once ApiNotificationJSON type exists
+interface FakeApiNotificationJSON {
+  type: string;
+  account: ApiAccountJSON;
+}
+
+export const notificationsUpdate = createAction(
+  'notifications/update',
+  ({
+    playSound,
+    ...args
+  }: {
+    notification: FakeApiNotificationJSON;
+    usePendingItems: boolean;
+    playSound: boolean;
+  }) => ({
+    payload: args,
+    meta: { sound: playSound ? 'boop' : undefined },
+  }),
+);
diff --git a/app/javascript/flavours/blobfox/actions/onboarding.js b/app/javascript/flavours/blobfox/actions/onboarding.js
new file mode 100644
index 00000000000000..a1dd3a731eddc1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/onboarding.js
@@ -0,0 +1,8 @@
+import { changeSetting, saveSettings } from './settings';
+
+export const INTRODUCTION_VERSION = 20181216044202;
+
+export const closeOnboarding = () => dispatch => {
+  dispatch(changeSetting(['introductionVersion'], INTRODUCTION_VERSION));
+  dispatch(saveSettings());
+};
diff --git a/app/javascript/flavours/blobfox/actions/picture_in_picture.js b/app/javascript/flavours/blobfox/actions/picture_in_picture.js
new file mode 100644
index 00000000000000..898375abeb9337
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/picture_in_picture.js
@@ -0,0 +1,46 @@
+// @ts-check
+
+export const PICTURE_IN_PICTURE_DEPLOY = 'PICTURE_IN_PICTURE_DEPLOY';
+export const PICTURE_IN_PICTURE_REMOVE = 'PICTURE_IN_PICTURE_REMOVE';
+
+/**
+ * @typedef MediaProps
+ * @property {string} src
+ * @property {boolean} muted
+ * @property {number} volume
+ * @property {number} currentTime
+ * @property {string} poster
+ * @property {string} backgroundColor
+ * @property {string} foregroundColor
+ * @property {string} accentColor
+ */
+
+/**
+ * @param {string} statusId
+ * @param {string} accountId
+ * @param {string} playerType
+ * @param {MediaProps} props
+ * @returns {object}
+ */
+export const deployPictureInPicture = (statusId, accountId, playerType, props) => {
+  // @ts-expect-error
+  return (dispatch, getState) => {
+    // Do not open a player for a toot that does not exist
+    if (getState().hasIn(['statuses', statusId])) {
+      dispatch({
+        type: PICTURE_IN_PICTURE_DEPLOY,
+        statusId,
+        accountId,
+        playerType,
+        props,
+      });
+    }
+  };
+};
+
+/*
+ * @return {object}
+ */
+export const removePictureInPicture = () => ({
+  type: PICTURE_IN_PICTURE_REMOVE,
+});
diff --git a/app/javascript/flavours/blobfox/actions/pin_statuses.js b/app/javascript/flavours/blobfox/actions/pin_statuses.js
new file mode 100644
index 00000000000000..baa10d15627d66
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/pin_statuses.js
@@ -0,0 +1,42 @@
+import api from '../api';
+import { me } from '../initial_state';
+
+import { importFetchedStatuses } from './importer';
+
+export const PINNED_STATUSES_FETCH_REQUEST = 'PINNED_STATUSES_FETCH_REQUEST';
+export const PINNED_STATUSES_FETCH_SUCCESS = 'PINNED_STATUSES_FETCH_SUCCESS';
+export const PINNED_STATUSES_FETCH_FAIL = 'PINNED_STATUSES_FETCH_FAIL';
+
+export function fetchPinnedStatuses() {
+  return (dispatch, getState) => {
+    dispatch(fetchPinnedStatusesRequest());
+
+    api(getState).get(`/api/v1/accounts/${me}/statuses`, { params: { pinned: true } }).then(response => {
+      dispatch(importFetchedStatuses(response.data));
+      dispatch(fetchPinnedStatusesSuccess(response.data, null));
+    }).catch(error => {
+      dispatch(fetchPinnedStatusesFail(error));
+    });
+  };
+}
+
+export function fetchPinnedStatusesRequest() {
+  return {
+    type: PINNED_STATUSES_FETCH_REQUEST,
+  };
+}
+
+export function fetchPinnedStatusesSuccess(statuses, next) {
+  return {
+    type: PINNED_STATUSES_FETCH_SUCCESS,
+    statuses,
+    next,
+  };
+}
+
+export function fetchPinnedStatusesFail(error) {
+  return {
+    type: PINNED_STATUSES_FETCH_FAIL,
+    error,
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/polls.js b/app/javascript/flavours/blobfox/actions/polls.js
new file mode 100644
index 00000000000000..a37410dc90fa4d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/polls.js
@@ -0,0 +1,61 @@
+import api from '../api';
+
+import { importFetchedPoll } from './importer';
+
+export const POLL_VOTE_REQUEST = 'POLL_VOTE_REQUEST';
+export const POLL_VOTE_SUCCESS = 'POLL_VOTE_SUCCESS';
+export const POLL_VOTE_FAIL    = 'POLL_VOTE_FAIL';
+
+export const POLL_FETCH_REQUEST = 'POLL_FETCH_REQUEST';
+export const POLL_FETCH_SUCCESS = 'POLL_FETCH_SUCCESS';
+export const POLL_FETCH_FAIL    = 'POLL_FETCH_FAIL';
+
+export const vote = (pollId, choices) => (dispatch, getState) => {
+  dispatch(voteRequest());
+
+  api(getState).post(`/api/v1/polls/${pollId}/votes`, { choices })
+    .then(({ data }) => {
+      dispatch(importFetchedPoll(data));
+      dispatch(voteSuccess(data));
+    })
+    .catch(err => dispatch(voteFail(err)));
+};
+
+export const fetchPoll = pollId => (dispatch, getState) => {
+  dispatch(fetchPollRequest());
+
+  api(getState).get(`/api/v1/polls/${pollId}`)
+    .then(({ data }) => {
+      dispatch(importFetchedPoll(data));
+      dispatch(fetchPollSuccess(data));
+    })
+    .catch(err => dispatch(fetchPollFail(err)));
+};
+
+export const voteRequest = () => ({
+  type: POLL_VOTE_REQUEST,
+});
+
+export const voteSuccess = poll => ({
+  type: POLL_VOTE_SUCCESS,
+  poll,
+});
+
+export const voteFail = error => ({
+  type: POLL_VOTE_FAIL,
+  error,
+});
+
+export const fetchPollRequest = () => ({
+  type: POLL_FETCH_REQUEST,
+});
+
+export const fetchPollSuccess = poll => ({
+  type: POLL_FETCH_SUCCESS,
+  poll,
+});
+
+export const fetchPollFail = error => ({
+  type: POLL_FETCH_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/push_notifications/index.js b/app/javascript/flavours/blobfox/actions/push_notifications/index.js
new file mode 100644
index 00000000000000..46b63867f1c224
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/push_notifications/index.js
@@ -0,0 +1,17 @@
+import { saveSettings } from './registerer';
+import { setAlerts } from './setter';
+
+export function changeAlerts(path, value) {
+  return dispatch => {
+    dispatch(setAlerts(path, value));
+    dispatch(saveSettings());
+  };
+}
+
+export {
+  CLEAR_SUBSCRIPTION,
+  SET_BROWSER_SUPPORT,
+  SET_SUBSCRIPTION,
+  SET_ALERTS,
+} from './setter';
+export { register } from './registerer';
diff --git a/app/javascript/flavours/blobfox/actions/push_notifications/registerer.js b/app/javascript/flavours/blobfox/actions/push_notifications/registerer.js
new file mode 100644
index 00000000000000..b3d3850e31d115
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/push_notifications/registerer.js
@@ -0,0 +1,134 @@
+import api from '../../api';
+import { me } from '../../initial_state';
+import { pushNotificationsSetting } from '../../settings';
+import { decode as decodeBase64 } from '../../utils/base64';
+
+import { setBrowserSupport, setSubscription, clearSubscription } from './setter';
+
+// Taken from https://www.npmjs.com/package/web-push
+const urlBase64ToUint8Array = (base64String) => {
+  const padding = '='.repeat((4 - base64String.length % 4) % 4);
+  const base64 = (base64String + padding)
+    .replace(/-/g, '+')
+    .replace(/_/g, '/');
+
+  return decodeBase64(base64);
+};
+
+const getApplicationServerKey = () => document.querySelector('[name="applicationServerKey"]').getAttribute('content');
+
+const getRegistration = () => navigator.serviceWorker.ready;
+
+const getPushSubscription = (registration) =>
+  registration.pushManager.getSubscription()
+    .then(subscription => ({ registration, subscription }));
+
+const subscribe = (registration) =>
+  registration.pushManager.subscribe({
+    userVisibleOnly: true,
+    applicationServerKey: urlBase64ToUint8Array(getApplicationServerKey()),
+  });
+
+const unsubscribe = ({ registration, subscription }) =>
+  subscription ? subscription.unsubscribe().then(() => registration) : registration;
+
+const sendSubscriptionToBackend = (subscription) => {
+  const params = { subscription };
+
+  if (me) {
+    const data = pushNotificationsSetting.get(me);
+    if (data) {
+      params.data = data;
+    }
+  }
+
+  return api().post('/api/web/push_subscriptions', params).then(response => response.data);
+};
+
+// Last one checks for payload support: https://web-push-book.gauntface.com/chapter-06/01-non-standards-browsers/#no-payload
+const supportsPushNotifications = ('serviceWorker' in navigator && 'PushManager' in window && 'getKey' in PushSubscription.prototype);
+
+export function register () {
+  return (dispatch, getState) => {
+    dispatch(setBrowserSupport(supportsPushNotifications));
+
+    if (supportsPushNotifications) {
+      if (!getApplicationServerKey()) {
+        console.error('The VAPID public key is not set. You will not be able to receive Web Push Notifications.');
+        return;
+      }
+
+      getRegistration()
+        .then(getPushSubscription)
+        .then(({ registration, subscription }) => {
+          if (subscription !== null) {
+            // We have a subscription, check if it is still valid
+            const currentServerKey = (new Uint8Array(subscription.options.applicationServerKey)).toString();
+            const subscriptionServerKey = urlBase64ToUint8Array(getApplicationServerKey()).toString();
+            const serverEndpoint = getState().getIn(['push_notifications', 'subscription', 'endpoint']);
+
+            // If the VAPID public key did not change and the endpoint corresponds
+            // to the endpoint saved in the backend, the subscription is valid
+            if (subscriptionServerKey === currentServerKey && subscription.endpoint === serverEndpoint) {
+              return subscription;
+            } else {
+              // Something went wrong, try to subscribe again
+              return unsubscribe({ registration, subscription }).then(subscribe).then(
+                subscription => sendSubscriptionToBackend(subscription));
+            }
+          }
+
+          // No subscription, try to subscribe
+          return subscribe(registration).then(
+            subscription => sendSubscriptionToBackend(subscription));
+        })
+        .then(subscription => {
+          // If we got a PushSubscription (and not a subscription object from the backend)
+          // it means that the backend subscription is valid (and was set during hydration)
+          if (!(subscription instanceof PushSubscription)) {
+            dispatch(setSubscription(subscription));
+            if (me) {
+              pushNotificationsSetting.set(me, { alerts: subscription.alerts });
+            }
+          }
+        })
+        .catch(error => {
+          if (error.code === 20 && error.name === 'AbortError') {
+            console.warn('Your browser supports Web Push Notifications, but does not seem to implement the VAPID protocol.');
+          } else if (error.code === 5 && error.name === 'InvalidCharacterError') {
+            console.error('The VAPID public key seems to be invalid:', getApplicationServerKey());
+          }
+
+          // Clear alerts and hide UI settings
+          dispatch(clearSubscription());
+          if (me) {
+            pushNotificationsSetting.remove(me);
+          }
+
+          return getRegistration()
+            .then(getPushSubscription)
+            .then(unsubscribe);
+        })
+        .catch(console.warn);
+    } else {
+      console.warn('Your browser does not support Web Push Notifications.');
+    }
+  };
+}
+
+export function saveSettings() {
+  return (_, getState) => {
+    const state = getState().get('push_notifications');
+    const subscription = state.get('subscription');
+    const alerts = state.get('alerts');
+    const data = { alerts };
+
+    api().put(`/api/web/push_subscriptions/${subscription.get('id')}`, {
+      data,
+    }).then(() => {
+      if (me) {
+        pushNotificationsSetting.set(me, data);
+      }
+    }).catch(console.warn);
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/push_notifications/setter.js b/app/javascript/flavours/blobfox/actions/push_notifications/setter.js
new file mode 100644
index 00000000000000..5561766bff42b3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/push_notifications/setter.js
@@ -0,0 +1,34 @@
+export const SET_BROWSER_SUPPORT = 'PUSH_NOTIFICATIONS_SET_BROWSER_SUPPORT';
+export const SET_SUBSCRIPTION = 'PUSH_NOTIFICATIONS_SET_SUBSCRIPTION';
+export const CLEAR_SUBSCRIPTION = 'PUSH_NOTIFICATIONS_CLEAR_SUBSCRIPTION';
+export const SET_ALERTS = 'PUSH_NOTIFICATIONS_SET_ALERTS';
+
+export function setBrowserSupport (value) {
+  return {
+    type: SET_BROWSER_SUPPORT,
+    value,
+  };
+}
+
+export function setSubscription (subscription) {
+  return {
+    type: SET_SUBSCRIPTION,
+    subscription,
+  };
+}
+
+export function clearSubscription () {
+  return {
+    type: CLEAR_SUBSCRIPTION,
+  };
+}
+
+export function setAlerts (path, value) {
+  return dispatch => {
+    dispatch({
+      type: SET_ALERTS,
+      path,
+      value,
+    });
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/reports.js b/app/javascript/flavours/blobfox/actions/reports.js
new file mode 100644
index 00000000000000..756b8cd05e1648
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/reports.js
@@ -0,0 +1,42 @@
+import api from '../api';
+
+import { openModal } from './modal';
+
+export const REPORT_SUBMIT_REQUEST = 'REPORT_SUBMIT_REQUEST';
+export const REPORT_SUBMIT_SUCCESS = 'REPORT_SUBMIT_SUCCESS';
+export const REPORT_SUBMIT_FAIL    = 'REPORT_SUBMIT_FAIL';
+
+export const initReport = (account, status) => dispatch =>
+  dispatch(openModal({
+    modalType: 'REPORT',
+    modalProps: {
+      accountId: account.get('id'),
+      statusId: status?.get('id'),
+    },
+  }));
+
+export const submitReport = (params, onSuccess, onFail) => (dispatch, getState) => {
+  dispatch(submitReportRequest());
+
+  api(getState).post('/api/v1/reports', params).then(response => {
+    dispatch(submitReportSuccess(response.data));
+    if (onSuccess) onSuccess();
+  }).catch(error => {
+    dispatch(submitReportFail(error));
+    if (onFail) onFail();
+  });
+};
+
+export const submitReportRequest = () => ({
+  type: REPORT_SUBMIT_REQUEST,
+});
+
+export const submitReportSuccess = report => ({
+  type: REPORT_SUBMIT_SUCCESS,
+  report,
+});
+
+export const submitReportFail = error => ({
+  type: REPORT_SUBMIT_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/search.js b/app/javascript/flavours/blobfox/actions/search.js
new file mode 100644
index 00000000000000..7fcc1edcd8db07
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/search.js
@@ -0,0 +1,201 @@
+import { fromJS } from 'immutable';
+
+import { searchHistory } from 'flavours/blobfox/settings';
+
+import api from '../api';
+
+import { fetchRelationships } from './accounts';
+import { importFetchedAccounts, importFetchedStatuses } from './importer';
+
+export const SEARCH_CHANGE = 'SEARCH_CHANGE';
+export const SEARCH_CLEAR  = 'SEARCH_CLEAR';
+export const SEARCH_SHOW   = 'SEARCH_SHOW';
+
+export const SEARCH_FETCH_REQUEST = 'SEARCH_FETCH_REQUEST';
+export const SEARCH_FETCH_SUCCESS = 'SEARCH_FETCH_SUCCESS';
+export const SEARCH_FETCH_FAIL    = 'SEARCH_FETCH_FAIL';
+
+export const SEARCH_EXPAND_REQUEST = 'SEARCH_EXPAND_REQUEST';
+export const SEARCH_EXPAND_SUCCESS = 'SEARCH_EXPAND_SUCCESS';
+export const SEARCH_EXPAND_FAIL    = 'SEARCH_EXPAND_FAIL';
+
+export const SEARCH_HISTORY_UPDATE  = 'SEARCH_HISTORY_UPDATE';
+
+export function changeSearch(value) {
+  return {
+    type: SEARCH_CHANGE,
+    value,
+  };
+}
+
+export function clearSearch() {
+  return {
+    type: SEARCH_CLEAR,
+  };
+}
+
+export function submitSearch(type) {
+  return (dispatch, getState) => {
+    const value    = getState().getIn(['search', 'value']);
+    const signedIn = !!getState().getIn(['meta', 'me']);
+
+    if (value.length === 0) {
+      dispatch(fetchSearchSuccess({ accounts: [], statuses: [], hashtags: [] }, '', type));
+      return;
+    }
+
+    dispatch(fetchSearchRequest(type));
+
+    api(getState).get('/api/v2/search', {
+      params: {
+        q: value,
+        resolve: signedIn,
+        limit: 11,
+        type,
+      },
+    }).then(response => {
+      if (response.data.accounts) {
+        dispatch(importFetchedAccounts(response.data.accounts));
+      }
+
+      if (response.data.statuses) {
+        dispatch(importFetchedStatuses(response.data.statuses));
+      }
+
+      dispatch(fetchSearchSuccess(response.data, value, type));
+      dispatch(fetchRelationships(response.data.accounts.map(item => item.id)));
+    }).catch(error => {
+      dispatch(fetchSearchFail(error));
+    });
+  };
+}
+
+export function fetchSearchRequest(searchType) {
+  return {
+    type: SEARCH_FETCH_REQUEST,
+    searchType,
+  };
+}
+
+export function fetchSearchSuccess(results, searchTerm, searchType) {
+  return {
+    type: SEARCH_FETCH_SUCCESS,
+    results,
+    searchType,
+    searchTerm,
+  };
+}
+
+export function fetchSearchFail(error) {
+  return {
+    type: SEARCH_FETCH_FAIL,
+    error,
+  };
+}
+
+export const expandSearch = type => (dispatch, getState) => {
+  const value  = getState().getIn(['search', 'value']);
+  const offset = getState().getIn(['search', 'results', type]).size - 1;
+
+  dispatch(expandSearchRequest(type));
+
+  api(getState).get('/api/v2/search', {
+    params: {
+      q: value,
+      type,
+      offset,
+      limit: 11,
+    },
+  }).then(({ data }) => {
+    if (data.accounts) {
+      dispatch(importFetchedAccounts(data.accounts));
+    }
+
+    if (data.statuses) {
+      dispatch(importFetchedStatuses(data.statuses));
+    }
+
+    dispatch(expandSearchSuccess(data, value, type));
+    dispatch(fetchRelationships(data.accounts.map(item => item.id)));
+  }).catch(error => {
+    dispatch(expandSearchFail(error));
+  });
+};
+
+export const expandSearchRequest = (searchType) => ({
+  type: SEARCH_EXPAND_REQUEST,
+  searchType,
+});
+
+export const expandSearchSuccess = (results, searchTerm, searchType) => ({
+  type: SEARCH_EXPAND_SUCCESS,
+  results,
+  searchTerm,
+  searchType,
+});
+
+export const expandSearchFail = error => ({
+  type: SEARCH_EXPAND_FAIL,
+  error,
+});
+
+export const showSearch = () => ({
+  type: SEARCH_SHOW,
+});
+
+export const openURL = routerHistory => (dispatch, getState) => {
+  const value = getState().getIn(['search', 'value']);
+  const signedIn = !!getState().getIn(['meta', 'me']);
+
+  if (!signedIn) {
+    return;
+  }
+
+  dispatch(fetchSearchRequest());
+
+  api(getState).get('/api/v2/search', { params: { q: value, resolve: true } }).then(response => {
+    if (response.data.accounts?.length > 0) {
+      dispatch(importFetchedAccounts(response.data.accounts));
+      routerHistory.push(`/@${response.data.accounts[0].acct}`);
+    } else if (response.data.statuses?.length > 0) {
+      dispatch(importFetchedStatuses(response.data.statuses));
+      routerHistory.push(`/@${response.data.statuses[0].account.acct}/${response.data.statuses[0].id}`);
+    }
+
+    dispatch(fetchSearchSuccess(response.data, value));
+  }).catch(err => {
+    dispatch(fetchSearchFail(err));
+  });
+};
+
+export const clickSearchResult = (q, type) => (dispatch, getState) => {
+  const previous = getState().getIn(['search', 'recent']);
+  const me = getState().getIn(['meta', 'me']);
+  const current = previous.add(fromJS({ type, q })).takeLast(4);
+
+  searchHistory.set(me, current.toJS());
+  dispatch(updateSearchHistory(current));
+};
+
+export const forgetSearchResult = q => (dispatch, getState) => {
+  const previous = getState().getIn(['search', 'recent']);
+  const me = getState().getIn(['meta', 'me']);
+  const current = previous.filterNot(result => result.get('q') === q);
+
+  searchHistory.set(me, current.toJS());
+  dispatch(updateSearchHistory(current));
+};
+
+export const updateSearchHistory = recent => ({
+  type: SEARCH_HISTORY_UPDATE,
+  recent,
+});
+
+export const hydrateSearch = () => (dispatch, getState) => {
+  const me = getState().getIn(['meta', 'me']);
+  const history = searchHistory.get(me);
+
+  if (history !== null) {
+    dispatch(updateSearchHistory(history));
+  }
+};
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/actions/server.js b/app/javascript/flavours/blobfox/actions/server.js
new file mode 100644
index 00000000000000..65f3efc3a72a3d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/server.js
@@ -0,0 +1,131 @@
+import api from '../api';
+
+import { importFetchedAccount } from './importer';
+
+export const SERVER_FETCH_REQUEST = 'Server_FETCH_REQUEST';
+export const SERVER_FETCH_SUCCESS = 'Server_FETCH_SUCCESS';
+export const SERVER_FETCH_FAIL    = 'Server_FETCH_FAIL';
+
+export const SERVER_TRANSLATION_LANGUAGES_FETCH_REQUEST = 'SERVER_TRANSLATION_LANGUAGES_FETCH_REQUEST';
+export const SERVER_TRANSLATION_LANGUAGES_FETCH_SUCCESS = 'SERVER_TRANSLATION_LANGUAGES_FETCH_SUCCESS';
+export const SERVER_TRANSLATION_LANGUAGES_FETCH_FAIL    = 'SERVER_TRANSLATION_LANGUAGES_FETCH_FAIL';
+
+export const EXTENDED_DESCRIPTION_REQUEST = 'EXTENDED_DESCRIPTION_REQUEST';
+export const EXTENDED_DESCRIPTION_SUCCESS = 'EXTENDED_DESCRIPTION_SUCCESS';
+export const EXTENDED_DESCRIPTION_FAIL    = 'EXTENDED_DESCRIPTION_FAIL';
+
+export const SERVER_DOMAIN_BLOCKS_FETCH_REQUEST = 'SERVER_DOMAIN_BLOCKS_FETCH_REQUEST';
+export const SERVER_DOMAIN_BLOCKS_FETCH_SUCCESS = 'SERVER_DOMAIN_BLOCKS_FETCH_SUCCESS';
+export const SERVER_DOMAIN_BLOCKS_FETCH_FAIL    = 'SERVER_DOMAIN_BLOCKS_FETCH_FAIL';
+
+export const fetchServer = () => (dispatch, getState) => {
+  if (getState().getIn(['server', 'server', 'isLoading'])) {
+    return;
+  }
+
+  dispatch(fetchServerRequest());
+
+  api(getState)
+    .get('/api/v2/instance').then(({ data }) => {
+      if (data.contact.account) dispatch(importFetchedAccount(data.contact.account));
+      dispatch(fetchServerSuccess(data));
+    }).catch(err => dispatch(fetchServerFail(err)));
+};
+
+const fetchServerRequest = () => ({
+  type: SERVER_FETCH_REQUEST,
+});
+
+const fetchServerSuccess = server => ({
+  type: SERVER_FETCH_SUCCESS,
+  server,
+});
+
+const fetchServerFail = error => ({
+  type: SERVER_FETCH_FAIL,
+  error,
+});
+
+export const fetchServerTranslationLanguages = () => (dispatch, getState) => {
+  dispatch(fetchServerTranslationLanguagesRequest());
+
+  api(getState)
+    .get('/api/v1/instance/translation_languages').then(({ data }) => {
+      dispatch(fetchServerTranslationLanguagesSuccess(data));
+    }).catch(err => dispatch(fetchServerTranslationLanguagesFail(err)));
+};
+
+const fetchServerTranslationLanguagesRequest = () => ({
+  type: SERVER_TRANSLATION_LANGUAGES_FETCH_REQUEST,
+});
+
+const fetchServerTranslationLanguagesSuccess = translationLanguages => ({
+  type: SERVER_TRANSLATION_LANGUAGES_FETCH_SUCCESS,
+  translationLanguages,
+});
+
+const fetchServerTranslationLanguagesFail = error => ({
+  type: SERVER_TRANSLATION_LANGUAGES_FETCH_FAIL,
+  error,
+});
+
+export const fetchExtendedDescription = () => (dispatch, getState) => {
+  if (getState().getIn(['server', 'extendedDescription', 'isLoading'])) {
+    return;
+  }
+
+  dispatch(fetchExtendedDescriptionRequest());
+
+  api(getState)
+    .get('/api/v1/instance/extended_description')
+    .then(({ data }) => dispatch(fetchExtendedDescriptionSuccess(data)))
+    .catch(err => dispatch(fetchExtendedDescriptionFail(err)));
+};
+
+const fetchExtendedDescriptionRequest = () => ({
+  type: EXTENDED_DESCRIPTION_REQUEST,
+});
+
+const fetchExtendedDescriptionSuccess = description => ({
+  type: EXTENDED_DESCRIPTION_SUCCESS,
+  description,
+});
+
+const fetchExtendedDescriptionFail = error => ({
+  type: EXTENDED_DESCRIPTION_FAIL,
+  error,
+});
+
+export const fetchDomainBlocks = () => (dispatch, getState) => {
+  if (getState().getIn(['server', 'domainBlocks', 'isLoading'])) {
+    return;
+  }
+
+  dispatch(fetchDomainBlocksRequest());
+
+  api(getState)
+    .get('/api/v1/instance/domain_blocks')
+    .then(({ data }) => dispatch(fetchDomainBlocksSuccess(true, data)))
+    .catch(err => {
+      if (err.response.status === 404) {
+        dispatch(fetchDomainBlocksSuccess(false, []));
+      } else {
+        dispatch(fetchDomainBlocksFail(err));
+      }
+    });
+};
+
+const fetchDomainBlocksRequest = () => ({
+  type: SERVER_DOMAIN_BLOCKS_FETCH_REQUEST,
+});
+
+const fetchDomainBlocksSuccess = (isAvailable, blocks) => ({
+  type: SERVER_DOMAIN_BLOCKS_FETCH_SUCCESS,
+  isAvailable,
+  blocks,
+});
+
+const fetchDomainBlocksFail = error => ({
+  type: SERVER_DOMAIN_BLOCKS_FETCH_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/settings.js b/app/javascript/flavours/blobfox/actions/settings.js
new file mode 100644
index 00000000000000..3685b0684e0b83
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/settings.js
@@ -0,0 +1,36 @@
+import { debounce } from 'lodash';
+
+import api from '../api';
+
+import { showAlertForError } from './alerts';
+
+export const SETTING_CHANGE = 'SETTING_CHANGE';
+export const SETTING_SAVE   = 'SETTING_SAVE';
+
+export function changeSetting(path, value) {
+  return dispatch => {
+    dispatch({
+      type: SETTING_CHANGE,
+      path,
+      value,
+    });
+
+    dispatch(saveSettings());
+  };
+}
+
+const debouncedSave = debounce((dispatch, getState) => {
+  if (getState().getIn(['settings', 'saved'])) {
+    return;
+  }
+
+  const data = getState().get('settings').filter((_, path) => path !== 'saved').toJS();
+
+  api().put('/api/web/settings', { data })
+    .then(() => dispatch({ type: SETTING_SAVE }))
+    .catch(error => dispatch(showAlertForError(error)));
+}, 5000, { trailing: true });
+
+export function saveSettings() {
+  return (dispatch, getState) => debouncedSave(dispatch, getState);
+}
diff --git a/app/javascript/flavours/blobfox/actions/statuses.js b/app/javascript/flavours/blobfox/actions/statuses.js
new file mode 100644
index 00000000000000..5bdd31c3438cbe
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/statuses.js
@@ -0,0 +1,351 @@
+import api from '../api';
+
+import { ensureComposeIsVisible, setComposeToStatus } from './compose';
+import { importFetchedStatus, importFetchedStatuses } from './importer';
+import { deleteFromTimelines } from './timelines';
+
+export const STATUS_FETCH_REQUEST = 'STATUS_FETCH_REQUEST';
+export const STATUS_FETCH_SUCCESS = 'STATUS_FETCH_SUCCESS';
+export const STATUS_FETCH_FAIL    = 'STATUS_FETCH_FAIL';
+
+export const STATUS_DELETE_REQUEST = 'STATUS_DELETE_REQUEST';
+export const STATUS_DELETE_SUCCESS = 'STATUS_DELETE_SUCCESS';
+export const STATUS_DELETE_FAIL    = 'STATUS_DELETE_FAIL';
+
+export const CONTEXT_FETCH_REQUEST = 'CONTEXT_FETCH_REQUEST';
+export const CONTEXT_FETCH_SUCCESS = 'CONTEXT_FETCH_SUCCESS';
+export const CONTEXT_FETCH_FAIL    = 'CONTEXT_FETCH_FAIL';
+
+export const STATUS_MUTE_REQUEST = 'STATUS_MUTE_REQUEST';
+export const STATUS_MUTE_SUCCESS = 'STATUS_MUTE_SUCCESS';
+export const STATUS_MUTE_FAIL    = 'STATUS_MUTE_FAIL';
+
+export const STATUS_UNMUTE_REQUEST = 'STATUS_UNMUTE_REQUEST';
+export const STATUS_UNMUTE_SUCCESS = 'STATUS_UNMUTE_SUCCESS';
+export const STATUS_UNMUTE_FAIL    = 'STATUS_UNMUTE_FAIL';
+
+export const STATUS_REVEAL   = 'STATUS_REVEAL';
+export const STATUS_HIDE     = 'STATUS_HIDE';
+export const STATUS_COLLAPSE = 'STATUS_COLLAPSE';
+
+export const REDRAFT = 'REDRAFT';
+
+export const STATUS_FETCH_SOURCE_REQUEST = 'STATUS_FETCH_SOURCE_REQUEST';
+export const STATUS_FETCH_SOURCE_SUCCESS = 'STATUS_FETCH_SOURCE_SUCCESS';
+export const STATUS_FETCH_SOURCE_FAIL    = 'STATUS_FETCH_SOURCE_FAIL';
+
+export const STATUS_TRANSLATE_REQUEST = 'STATUS_TRANSLATE_REQUEST';
+export const STATUS_TRANSLATE_SUCCESS = 'STATUS_TRANSLATE_SUCCESS';
+export const STATUS_TRANSLATE_FAIL    = 'STATUS_TRANSLATE_FAIL';
+export const STATUS_TRANSLATE_UNDO    = 'STATUS_TRANSLATE_UNDO';
+
+export function fetchStatusRequest(id, skipLoading) {
+  return {
+    type: STATUS_FETCH_REQUEST,
+    id,
+    skipLoading,
+  };
+}
+
+export function fetchStatus(id, forceFetch = false) {
+  return (dispatch, getState) => {
+    const skipLoading = !forceFetch && getState().getIn(['statuses', id], null) !== null;
+
+    dispatch(fetchContext(id));
+
+    if (skipLoading) {
+      return;
+    }
+
+    dispatch(fetchStatusRequest(id, skipLoading));
+
+    api(getState).get(`/api/v1/statuses/${id}`).then(response => {
+      dispatch(importFetchedStatus(response.data));
+      dispatch(fetchStatusSuccess(skipLoading));
+    }).catch(error => {
+      dispatch(fetchStatusFail(id, error, skipLoading));
+    });
+  };
+}
+
+export function fetchStatusSuccess(skipLoading) {
+  return {
+    type: STATUS_FETCH_SUCCESS,
+    skipLoading,
+  };
+}
+
+export function fetchStatusFail(id, error, skipLoading) {
+  return {
+    type: STATUS_FETCH_FAIL,
+    id,
+    error,
+    skipLoading,
+    skipAlert: true,
+  };
+}
+
+export function redraft(status, raw_text, content_type) {
+  return {
+    type: REDRAFT,
+    status,
+    raw_text,
+    content_type,
+  };
+}
+
+export const editStatus = (id, routerHistory) => (dispatch, getState) => {
+  let status = getState().getIn(['statuses', id]);
+
+  if (status.get('poll')) {
+    status = status.set('poll', getState().getIn(['polls', status.get('poll')]));
+  }
+
+  dispatch(fetchStatusSourceRequest());
+
+  api(getState).get(`/api/v1/statuses/${id}/source`).then(response => {
+    dispatch(fetchStatusSourceSuccess());
+    ensureComposeIsVisible(getState, routerHistory);
+    dispatch(setComposeToStatus(status, response.data.text, response.data.spoiler_text, response.data.content_type));
+  }).catch(error => {
+    dispatch(fetchStatusSourceFail(error));
+  });
+};
+
+export const fetchStatusSourceRequest = () => ({
+  type: STATUS_FETCH_SOURCE_REQUEST,
+});
+
+export const fetchStatusSourceSuccess = () => ({
+  type: STATUS_FETCH_SOURCE_SUCCESS,
+});
+
+export const fetchStatusSourceFail = error => ({
+  type: STATUS_FETCH_SOURCE_FAIL,
+  error,
+});
+
+export function deleteStatus(id, routerHistory, withRedraft = false) {
+  return (dispatch, getState) => {
+    let status = getState().getIn(['statuses', id]);
+
+    if (status.get('poll')) {
+      status = status.set('poll', getState().getIn(['polls', status.get('poll')]));
+    }
+
+    dispatch(deleteStatusRequest(id));
+
+    api(getState).delete(`/api/v1/statuses/${id}`).then(response => {
+      dispatch(deleteStatusSuccess(id));
+      dispatch(deleteFromTimelines(id));
+
+      if (withRedraft) {
+        dispatch(redraft(status, response.data.text, response.data.content_type));
+
+        ensureComposeIsVisible(getState, routerHistory);
+      }
+    }).catch(error => {
+      dispatch(deleteStatusFail(id, error));
+    });
+  };
+}
+
+export function deleteStatusRequest(id) {
+  return {
+    type: STATUS_DELETE_REQUEST,
+    id: id,
+  };
+}
+
+export function deleteStatusSuccess(id) {
+  return {
+    type: STATUS_DELETE_SUCCESS,
+    id: id,
+  };
+}
+
+export function deleteStatusFail(id, error) {
+  return {
+    type: STATUS_DELETE_FAIL,
+    id: id,
+    error: error,
+  };
+}
+
+export const updateStatus = status => dispatch =>
+  dispatch(importFetchedStatus(status));
+
+export function fetchContext(id) {
+  return (dispatch, getState) => {
+    dispatch(fetchContextRequest(id));
+
+    api(getState).get(`/api/v1/statuses/${id}/context`).then(response => {
+      dispatch(importFetchedStatuses(response.data.ancestors.concat(response.data.descendants)));
+      dispatch(fetchContextSuccess(id, response.data.ancestors, response.data.descendants));
+
+    }).catch(error => {
+      if (error.response && error.response.status === 404) {
+        dispatch(deleteFromTimelines(id));
+      }
+
+      dispatch(fetchContextFail(id, error));
+    });
+  };
+}
+
+export function fetchContextRequest(id) {
+  return {
+    type: CONTEXT_FETCH_REQUEST,
+    id,
+  };
+}
+
+export function fetchContextSuccess(id, ancestors, descendants) {
+  return {
+    type: CONTEXT_FETCH_SUCCESS,
+    id,
+    ancestors,
+    descendants,
+    statuses: ancestors.concat(descendants),
+  };
+}
+
+export function fetchContextFail(id, error) {
+  return {
+    type: CONTEXT_FETCH_FAIL,
+    id,
+    error,
+    skipAlert: true,
+  };
+}
+
+export function muteStatus(id) {
+  return (dispatch, getState) => {
+    dispatch(muteStatusRequest(id));
+
+    api(getState).post(`/api/v1/statuses/${id}/mute`).then(() => {
+      dispatch(muteStatusSuccess(id));
+    }).catch(error => {
+      dispatch(muteStatusFail(id, error));
+    });
+  };
+}
+
+export function muteStatusRequest(id) {
+  return {
+    type: STATUS_MUTE_REQUEST,
+    id,
+  };
+}
+
+export function muteStatusSuccess(id) {
+  return {
+    type: STATUS_MUTE_SUCCESS,
+    id,
+  };
+}
+
+export function muteStatusFail(id, error) {
+  return {
+    type: STATUS_MUTE_FAIL,
+    id,
+    error,
+  };
+}
+
+export function unmuteStatus(id) {
+  return (dispatch, getState) => {
+    dispatch(unmuteStatusRequest(id));
+
+    api(getState).post(`/api/v1/statuses/${id}/unmute`).then(() => {
+      dispatch(unmuteStatusSuccess(id));
+    }).catch(error => {
+      dispatch(unmuteStatusFail(id, error));
+    });
+  };
+}
+
+export function unmuteStatusRequest(id) {
+  return {
+    type: STATUS_UNMUTE_REQUEST,
+    id,
+  };
+}
+
+export function unmuteStatusSuccess(id) {
+  return {
+    type: STATUS_UNMUTE_SUCCESS,
+    id,
+  };
+}
+
+export function unmuteStatusFail(id, error) {
+  return {
+    type: STATUS_UNMUTE_FAIL,
+    id,
+    error,
+  };
+}
+
+export function hideStatus(ids) {
+  if (!Array.isArray(ids)) {
+    ids = [ids];
+  }
+
+  return {
+    type: STATUS_HIDE,
+    ids,
+  };
+}
+
+export function revealStatus(ids) {
+  if (!Array.isArray(ids)) {
+    ids = [ids];
+  }
+
+  return {
+    type: STATUS_REVEAL,
+    ids,
+  };
+}
+
+export function toggleStatusCollapse(id, isCollapsed) {
+  return {
+    type: STATUS_COLLAPSE,
+    id,
+    isCollapsed,
+  };
+}
+
+export const translateStatus = id => (dispatch, getState) => {
+  dispatch(translateStatusRequest(id));
+
+  api(getState).post(`/api/v1/statuses/${id}/translate`).then(response => {
+    dispatch(translateStatusSuccess(id, response.data));
+  }).catch(error => {
+    dispatch(translateStatusFail(id, error));
+  });
+};
+
+export const translateStatusRequest = id => ({
+  type: STATUS_TRANSLATE_REQUEST,
+  id,
+});
+
+export const translateStatusSuccess = (id, translation) => ({
+  type: STATUS_TRANSLATE_SUCCESS,
+  id,
+  translation,
+});
+
+export const translateStatusFail = (id, error) => ({
+  type: STATUS_TRANSLATE_FAIL,
+  id,
+  error,
+});
+
+export const undoStatusTranslation = (id, pollId) => ({
+  type: STATUS_TRANSLATE_UNDO,
+  id,
+  pollId,
+});
diff --git a/app/javascript/flavours/blobfox/actions/store.js b/app/javascript/flavours/blobfox/actions/store.js
new file mode 100644
index 00000000000000..86a42433d25092
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/store.js
@@ -0,0 +1,43 @@
+import { Iterable, fromJS } from 'immutable';
+
+import { hydrateCompose } from './compose';
+import { importFetchedAccounts } from './importer';
+import { hydrateSearch } from './search';
+import { saveSettings } from './settings';
+
+export const STORE_HYDRATE = 'STORE_HYDRATE';
+export const STORE_HYDRATE_LAZY = 'STORE_HYDRATE_LAZY';
+
+const convertState = rawState =>
+  fromJS(rawState, (k, v) =>
+    Iterable.isIndexed(v) ? v.toList() : v.toMap());
+
+const applyMigrations = (state) => {
+  return state.withMutations(state => {
+    // Migrate blobfox-soc local-only “Show unread marker” setting to Mastodon's setting
+    if (state.getIn(['local_settings', 'notifications', 'show_unread']) !== undefined) {
+      // Only change if the Mastodon setting does not deviate from default
+      if (state.getIn(['settings', 'notifications', 'showUnread']) !== false) {
+        state.setIn(['settings', 'notifications', 'showUnread'], state.getIn(['local_settings', 'notifications', 'show_unread']));
+      }
+      state.removeIn(['local_settings', 'notifications', 'show_unread']);
+    }
+  });
+};
+
+
+export function hydrateStore(rawState) {
+  return dispatch => {
+    const state = applyMigrations(convertState(rawState));
+
+    dispatch({
+      type: STORE_HYDRATE,
+      state,
+    });
+
+    dispatch(hydrateCompose());
+    dispatch(hydrateSearch());
+    dispatch(importFetchedAccounts(Object.values(rawState.accounts)));
+    dispatch(saveSettings());
+  };
+}
diff --git a/app/javascript/flavours/blobfox/actions/streaming.js b/app/javascript/flavours/blobfox/actions/streaming.js
new file mode 100644
index 00000000000000..d8341a5c1682ce
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/streaming.js
@@ -0,0 +1,184 @@
+// @ts-check
+
+import { getLocale } from '../locales';
+import { connectStream } from '../stream';
+
+import {
+  fetchAnnouncements,
+  updateAnnouncements,
+  updateReaction as updateAnnouncementsReaction,
+  deleteAnnouncement,
+} from './announcements';
+import { updateConversations } from './conversations';
+import { updateNotifications, expandNotifications } from './notifications';
+import { updateStatus } from './statuses';
+import {
+  updateTimeline,
+  deleteFromTimelines,
+  expandHomeTimeline,
+  connectTimeline,
+  disconnectTimeline,
+  fillHomeTimelineGaps,
+  fillPublicTimelineGaps,
+  fillCommunityTimelineGaps,
+  fillListTimelineGaps,
+} from './timelines';
+
+/**
+ * @param {number} max
+ * @returns {number}
+ */
+const randomUpTo = max =>
+  Math.floor(Math.random() * Math.floor(max));
+
+/**
+ * @param {string} timelineId
+ * @param {string} channelName
+ * @param {Object.<string, string>} params
+ * @param {Object} options
+ * @param {function(Function, Function): void} [options.fallback]
+ * @param {function(): void} [options.fillGaps]
+ * @param {function(object): boolean} [options.accept]
+ * @returns {function(): void}
+ */
+export const connectTimelineStream = (timelineId, channelName, params = {}, options = {}) => {
+  const { messages } = getLocale();
+
+  return connectStream(channelName, params, (dispatch, getState) => {
+    const locale = getState().getIn(['meta', 'locale']);
+
+    // @ts-expect-error
+    let pollingId;
+
+    /**
+     * @param {function(Function, Function): void} fallback
+     */
+
+    const useFallback = fallback => {
+      fallback(dispatch, () => {
+        // eslint-disable-next-line react-hooks/rules-of-hooks -- this is not a react hook
+        pollingId = setTimeout(() => useFallback(fallback), 20000 + randomUpTo(20000));
+      });
+    };
+
+    return {
+      onConnect() {
+        dispatch(connectTimeline(timelineId));
+
+        // @ts-expect-error
+        if (pollingId) {
+          // @ts-ignore
+          clearTimeout(pollingId); pollingId = null;
+        }
+
+        if (options.fillGaps) {
+          dispatch(options.fillGaps());
+        }
+      },
+
+      onDisconnect() {
+        dispatch(disconnectTimeline(timelineId));
+
+        if (options.fallback) {
+          // @ts-expect-error
+          pollingId = setTimeout(() => useFallback(options.fallback), randomUpTo(40000));
+        }
+      },
+
+      onReceive(data) {
+        switch (data.event) {
+        case 'update':
+          // @ts-expect-error
+          dispatch(updateTimeline(timelineId, JSON.parse(data.payload), options.accept));
+          break;
+        case 'status.update':
+          // @ts-expect-error
+          dispatch(updateStatus(JSON.parse(data.payload)));
+          break;
+        case 'delete':
+          dispatch(deleteFromTimelines(data.payload));
+          break;
+        case 'notification':
+          // @ts-expect-error
+          dispatch(updateNotifications(JSON.parse(data.payload), messages, locale));
+          break;
+        case 'conversation':
+          // @ts-expect-error
+          dispatch(updateConversations(JSON.parse(data.payload)));
+          break;
+        case 'announcement':
+          // @ts-expect-error
+          dispatch(updateAnnouncements(JSON.parse(data.payload)));
+          break;
+        case 'announcement.reaction':
+          // @ts-expect-error
+          dispatch(updateAnnouncementsReaction(JSON.parse(data.payload)));
+          break;
+        case 'announcement.delete':
+          dispatch(deleteAnnouncement(data.payload));
+          break;
+        }
+      },
+    };
+  });
+};
+
+/**
+ * @param {Function} dispatch
+ * @param {function(): void} done
+ */
+const refreshHomeTimelineAndNotification = (dispatch, done) => {
+  // @ts-expect-error
+  dispatch(expandHomeTimeline({}, () =>
+    // @ts-expect-error
+    dispatch(expandNotifications({}, () =>
+      dispatch(fetchAnnouncements(done))))));
+};
+
+/**
+ * @returns {function(): void}
+ */
+export const connectUserStream = () =>
+  // @ts-expect-error
+  connectTimelineStream('home', 'user', {}, { fallback: refreshHomeTimelineAndNotification, fillGaps: fillHomeTimelineGaps });
+
+/**
+ * @param {Object} options
+ * @param {boolean} [options.onlyMedia]
+ * @returns {function(): void}
+ */
+export const connectCommunityStream = ({ onlyMedia } = {}) =>
+  connectTimelineStream(`community${onlyMedia ? ':media' : ''}`, `public:local${onlyMedia ? ':media' : ''}`, {}, { fillGaps: () => (fillCommunityTimelineGaps({ onlyMedia })) });
+
+/**
+ * @param {Object} options
+ * @param {boolean} [options.onlyMedia]
+ * @param {boolean} [options.onlyRemote]
+ * @param {boolean} [options.allowLocalOnly]
+ * @returns {function(): void}
+ */
+export const connectPublicStream = ({ onlyMedia, onlyRemote, allowLocalOnly } = {}) =>
+  connectTimelineStream(`public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`, `public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`, {}, { fillGaps: () => fillPublicTimelineGaps({ onlyMedia, onlyRemote, allowLocalOnly }) });
+
+/**
+ * @param {string} columnId
+ * @param {string} tagName
+ * @param {boolean} onlyLocal
+ * @param {function(object): boolean} accept
+ * @returns {function(): void}
+ */
+export const connectHashtagStream = (columnId, tagName, onlyLocal, accept) =>
+  connectTimelineStream(`hashtag:${columnId}${onlyLocal ? ':local' : ''}`, `hashtag${onlyLocal ? ':local' : ''}`, { tag: tagName }, { accept });
+
+/**
+ * @returns {function(): void}
+ */
+export const connectDirectStream = () =>
+  connectTimelineStream('direct', 'direct');
+
+/**
+ * @param {string} listId
+ * @returns {function(): void}
+ */
+export const connectListStream = listId =>
+  connectTimelineStream(`list:${listId}`, 'list', { list: listId }, { fillGaps: () => fillListTimelineGaps(listId) });
diff --git a/app/javascript/flavours/blobfox/actions/suggestions.js b/app/javascript/flavours/blobfox/actions/suggestions.js
new file mode 100644
index 00000000000000..870a311024d12f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/suggestions.js
@@ -0,0 +1,65 @@
+import api from '../api';
+
+import { fetchRelationships } from './accounts';
+import { importFetchedAccounts } from './importer';
+
+export const SUGGESTIONS_FETCH_REQUEST = 'SUGGESTIONS_FETCH_REQUEST';
+export const SUGGESTIONS_FETCH_SUCCESS = 'SUGGESTIONS_FETCH_SUCCESS';
+export const SUGGESTIONS_FETCH_FAIL    = 'SUGGESTIONS_FETCH_FAIL';
+
+export const SUGGESTIONS_DISMISS = 'SUGGESTIONS_DISMISS';
+
+export function fetchSuggestions(withRelationships = false) {
+  return (dispatch, getState) => {
+    dispatch(fetchSuggestionsRequest());
+
+    api(getState).get('/api/v2/suggestions', { params: { limit: 20 } }).then(response => {
+      dispatch(importFetchedAccounts(response.data.map(x => x.account)));
+      dispatch(fetchSuggestionsSuccess(response.data));
+
+      if (withRelationships) {
+        dispatch(fetchRelationships(response.data.map(item => item.account.id)));
+      }
+    }).catch(error => dispatch(fetchSuggestionsFail(error)));
+  };
+}
+
+export function fetchSuggestionsRequest() {
+  return {
+    type: SUGGESTIONS_FETCH_REQUEST,
+    skipLoading: true,
+  };
+}
+
+export function fetchSuggestionsSuccess(suggestions) {
+  return {
+    type: SUGGESTIONS_FETCH_SUCCESS,
+    suggestions,
+    skipLoading: true,
+  };
+}
+
+export function fetchSuggestionsFail(error) {
+  return {
+    type: SUGGESTIONS_FETCH_FAIL,
+    error,
+    skipLoading: true,
+    skipAlert: true,
+  };
+}
+
+export const dismissSuggestion = accountId => (dispatch, getState) => {
+  dispatch({
+    type: SUGGESTIONS_DISMISS,
+    id: accountId,
+  });
+
+  api(getState).delete(`/api/v1/suggestions/${accountId}`).then(() => {
+    dispatch(fetchSuggestionsRequest());
+
+    api(getState).get('/api/v2/suggestions').then(response => {
+      dispatch(importFetchedAccounts(response.data.map(x => x.account)));
+      dispatch(fetchSuggestionsSuccess(response.data));
+    }).catch(error => dispatch(fetchSuggestionsFail(error)));
+  }).catch(() => {});
+};
diff --git a/app/javascript/flavours/blobfox/actions/tags.js b/app/javascript/flavours/blobfox/actions/tags.js
new file mode 100644
index 00000000000000..dda8c924bb59a4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/tags.js
@@ -0,0 +1,172 @@
+import api, { getLinks } from '../api';
+
+export const HASHTAG_FETCH_REQUEST = 'HASHTAG_FETCH_REQUEST';
+export const HASHTAG_FETCH_SUCCESS = 'HASHTAG_FETCH_SUCCESS';
+export const HASHTAG_FETCH_FAIL    = 'HASHTAG_FETCH_FAIL';
+
+export const FOLLOWED_HASHTAGS_FETCH_REQUEST = 'FOLLOWED_HASHTAGS_FETCH_REQUEST';
+export const FOLLOWED_HASHTAGS_FETCH_SUCCESS = 'FOLLOWED_HASHTAGS_FETCH_SUCCESS';
+export const FOLLOWED_HASHTAGS_FETCH_FAIL    = 'FOLLOWED_HASHTAGS_FETCH_FAIL';
+
+export const FOLLOWED_HASHTAGS_EXPAND_REQUEST = 'FOLLOWED_HASHTAGS_EXPAND_REQUEST';
+export const FOLLOWED_HASHTAGS_EXPAND_SUCCESS = 'FOLLOWED_HASHTAGS_EXPAND_SUCCESS';
+export const FOLLOWED_HASHTAGS_EXPAND_FAIL    = 'FOLLOWED_HASHTAGS_EXPAND_FAIL';
+
+export const HASHTAG_FOLLOW_REQUEST = 'HASHTAG_FOLLOW_REQUEST';
+export const HASHTAG_FOLLOW_SUCCESS = 'HASHTAG_FOLLOW_SUCCESS';
+export const HASHTAG_FOLLOW_FAIL    = 'HASHTAG_FOLLOW_FAIL';
+
+export const HASHTAG_UNFOLLOW_REQUEST = 'HASHTAG_UNFOLLOW_REQUEST';
+export const HASHTAG_UNFOLLOW_SUCCESS = 'HASHTAG_UNFOLLOW_SUCCESS';
+export const HASHTAG_UNFOLLOW_FAIL    = 'HASHTAG_UNFOLLOW_FAIL';
+
+export const fetchHashtag = name => (dispatch, getState) => {
+  dispatch(fetchHashtagRequest());
+
+  api(getState).get(`/api/v1/tags/${name}`).then(({ data }) => {
+    dispatch(fetchHashtagSuccess(name, data));
+  }).catch(err => {
+    dispatch(fetchHashtagFail(err));
+  });
+};
+
+export const fetchHashtagRequest = () => ({
+  type: HASHTAG_FETCH_REQUEST,
+});
+
+export const fetchHashtagSuccess = (name, tag) => ({
+  type: HASHTAG_FETCH_SUCCESS,
+  name,
+  tag,
+});
+
+export const fetchHashtagFail = error => ({
+  type: HASHTAG_FETCH_FAIL,
+  error,
+});
+
+export const fetchFollowedHashtags = () => (dispatch, getState) => {
+  dispatch(fetchFollowedHashtagsRequest());
+
+  api(getState).get('/api/v1/followed_tags').then(response => {
+    const next = getLinks(response).refs.find(link => link.rel === 'next');
+    dispatch(fetchFollowedHashtagsSuccess(response.data, next ? next.uri : null));
+  }).catch(err => {
+    dispatch(fetchFollowedHashtagsFail(err));
+  });
+};
+
+export function fetchFollowedHashtagsRequest() {
+  return {
+    type: FOLLOWED_HASHTAGS_FETCH_REQUEST,
+  };
+}
+
+export function fetchFollowedHashtagsSuccess(followed_tags, next) {
+  return {
+    type: FOLLOWED_HASHTAGS_FETCH_SUCCESS,
+    followed_tags,
+    next,
+  };
+}
+
+export function fetchFollowedHashtagsFail(error) {
+  return {
+    type: FOLLOWED_HASHTAGS_FETCH_FAIL,
+    error,
+  };
+}
+
+export function expandFollowedHashtags() {
+  return (dispatch, getState) => {
+    const url = getState().getIn(['followed_tags', 'next']);
+
+    if (url === null) {
+      return;
+    }
+
+    dispatch(expandFollowedHashtagsRequest());
+
+    api(getState).get(url).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(expandFollowedHashtagsSuccess(response.data, next ? next.uri : null));
+    }).catch(error => {
+      dispatch(expandFollowedHashtagsFail(error));
+    });
+  };
+}
+
+export function expandFollowedHashtagsRequest() {
+  return {
+    type: FOLLOWED_HASHTAGS_EXPAND_REQUEST,
+  };
+}
+
+export function expandFollowedHashtagsSuccess(followed_tags, next) {
+  return {
+    type: FOLLOWED_HASHTAGS_EXPAND_SUCCESS,
+    followed_tags,
+    next,
+  };
+}
+
+export function expandFollowedHashtagsFail(error) {
+  return {
+    type: FOLLOWED_HASHTAGS_EXPAND_FAIL,
+    error,
+  };
+}
+
+export const followHashtag = name => (dispatch, getState) => {
+  dispatch(followHashtagRequest(name));
+
+  api(getState).post(`/api/v1/tags/${name}/follow`).then(({ data }) => {
+    dispatch(followHashtagSuccess(name, data));
+  }).catch(err => {
+    dispatch(followHashtagFail(name, err));
+  });
+};
+
+export const followHashtagRequest = name => ({
+  type: HASHTAG_FOLLOW_REQUEST,
+  name,
+});
+
+export const followHashtagSuccess = (name, tag) => ({
+  type: HASHTAG_FOLLOW_SUCCESS,
+  name,
+  tag,
+});
+
+export const followHashtagFail = (name, error) => ({
+  type: HASHTAG_FOLLOW_FAIL,
+  name,
+  error,
+});
+
+export const unfollowHashtag = name => (dispatch, getState) => {
+  dispatch(unfollowHashtagRequest(name));
+
+  api(getState).post(`/api/v1/tags/${name}/unfollow`).then(({ data }) => {
+    dispatch(unfollowHashtagSuccess(name, data));
+  }).catch(err => {
+    dispatch(unfollowHashtagFail(name, err));
+  });
+};
+
+export const unfollowHashtagRequest = name => ({
+  type: HASHTAG_UNFOLLOW_REQUEST,
+  name,
+});
+
+export const unfollowHashtagSuccess = (name, tag) => ({
+  type: HASHTAG_UNFOLLOW_SUCCESS,
+  name,
+  tag,
+});
+
+export const unfollowHashtagFail = (name, error) => ({
+  type: HASHTAG_UNFOLLOW_FAIL,
+  name,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/actions/timelines.js b/app/javascript/flavours/blobfox/actions/timelines.js
new file mode 100644
index 00000000000000..429efc5a0ef10b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/timelines.js
@@ -0,0 +1,235 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import api, { getLinks } from 'flavours/blobfox/api';
+import { compareId } from 'flavours/blobfox/compare_id';
+import { usePendingItems as preferPendingItems } from 'flavours/blobfox/initial_state';
+import { toServerSideType } from 'flavours/blobfox/utils/filters';
+
+import { importFetchedStatus, importFetchedStatuses } from './importer';
+import { submitMarkers } from './markers';
+
+export const TIMELINE_UPDATE  = 'TIMELINE_UPDATE';
+export const TIMELINE_DELETE  = 'TIMELINE_DELETE';
+export const TIMELINE_CLEAR   = 'TIMELINE_CLEAR';
+
+export const TIMELINE_EXPAND_REQUEST = 'TIMELINE_EXPAND_REQUEST';
+export const TIMELINE_EXPAND_SUCCESS = 'TIMELINE_EXPAND_SUCCESS';
+export const TIMELINE_EXPAND_FAIL    = 'TIMELINE_EXPAND_FAIL';
+
+export const TIMELINE_SCROLL_TOP   = 'TIMELINE_SCROLL_TOP';
+export const TIMELINE_LOAD_PENDING = 'TIMELINE_LOAD_PENDING';
+export const TIMELINE_DISCONNECT   = 'TIMELINE_DISCONNECT';
+export const TIMELINE_CONNECT      = 'TIMELINE_CONNECT';
+
+export const TIMELINE_MARK_AS_PARTIAL = 'TIMELINE_MARK_AS_PARTIAL';
+
+export const loadPending = timeline => ({
+  type: TIMELINE_LOAD_PENDING,
+  timeline,
+});
+
+export function updateTimeline(timeline, status, accept) {
+  return (dispatch, getState) => {
+    if (typeof accept === 'function' && !accept(status)) {
+      return;
+    }
+
+    if (getState().getIn(['timelines', timeline, 'isPartial'])) {
+      // Prevent new items from being added to a partial timeline,
+      // since it will be reloaded anyway
+
+      return;
+    }
+
+    let filtered = false;
+
+    if (status.filtered) {
+      const contextType = toServerSideType(timeline);
+      const filters = status.filtered.filter(result => result.filter.context.includes(contextType));
+
+      filtered = filters.length > 0;
+    }
+
+    dispatch(importFetchedStatus(status));
+
+    dispatch({
+      type: TIMELINE_UPDATE,
+      timeline,
+      status,
+      usePendingItems: preferPendingItems,
+      filtered,
+    });
+
+    if (timeline === 'home') {
+      dispatch(submitMarkers());
+    }
+  };
+}
+
+export function deleteFromTimelines(id) {
+  return (dispatch, getState) => {
+    const accountId  = getState().getIn(['statuses', id, 'account']);
+    const references = getState().get('statuses').filter(status => status.get('reblog') === id).map(status => status.get('id'));
+    const reblogOf   = getState().getIn(['statuses', id, 'reblog'], null);
+
+    dispatch({
+      type: TIMELINE_DELETE,
+      id,
+      accountId,
+      references,
+      reblogOf,
+    });
+  };
+}
+
+export function clearTimeline(timeline) {
+  return (dispatch) => {
+    dispatch({ type: TIMELINE_CLEAR, timeline });
+  };
+}
+
+const noOp = () => {};
+
+const parseTags = (tags = {}, mode) => {
+  return (tags[mode] || []).map((tag) => {
+    return tag.value;
+  });
+};
+
+export function expandTimeline(timelineId, path, params = {}, done = noOp) {
+  return (dispatch, getState) => {
+    const timeline = getState().getIn(['timelines', timelineId], ImmutableMap());
+    const isLoadingMore = !!params.max_id;
+
+    if (timeline.get('isLoading')) {
+      done();
+      return;
+    }
+
+    if (!params.max_id && !params.pinned && (timeline.get('items', ImmutableList()).size + timeline.get('pendingItems', ImmutableList()).size) > 0) {
+      const a = timeline.getIn(['pendingItems', 0]);
+      const b = timeline.getIn(['items', 0]);
+
+      if (a && b && compareId(a, b) > 0) {
+        params.since_id = a;
+      } else {
+        params.since_id = b || a;
+      }
+    }
+
+    const isLoadingRecent = !!params.since_id;
+
+    dispatch(expandTimelineRequest(timelineId, isLoadingMore));
+
+    api(getState).get(path, { params }).then(response => {
+      const next = getLinks(response).refs.find(link => link.rel === 'next');
+      dispatch(importFetchedStatuses(response.data));
+      dispatch(expandTimelineSuccess(timelineId, response.data, next ? next.uri : null, response.status === 206, isLoadingRecent, isLoadingMore, isLoadingRecent && preferPendingItems));
+
+      if (timelineId === 'home') {
+        dispatch(submitMarkers());
+      }
+    }).catch(error => {
+      dispatch(expandTimelineFail(timelineId, error, isLoadingMore));
+    }).finally(() => {
+      done();
+    });
+  };
+}
+
+export function fillTimelineGaps(timelineId, path, params = {}, done = noOp) {
+  return (dispatch, getState) => {
+    const timeline = getState().getIn(['timelines', timelineId], ImmutableMap());
+    const items = timeline.get('items');
+    const nullIndexes = items.map((statusId, index) => statusId === null ? index : null);
+    const gaps = nullIndexes.map(index => index > 0 ? items.get(index - 1) : null);
+
+    // Only expand at most two gaps to avoid doing too many requests
+    done = gaps.take(2).reduce((done, maxId) => {
+      return (() => dispatch(expandTimeline(timelineId, path, { ...params, maxId }, done)));
+    }, done);
+
+    done();
+  };
+}
+
+export const expandHomeTimeline            = ({ maxId } = {}, done = noOp) => expandTimeline('home', '/api/v1/timelines/home', { max_id: maxId }, done);
+export const expandPublicTimeline          = ({ maxId, onlyMedia, onlyRemote, allowLocalOnly } = {}, done = noOp) => expandTimeline(`public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { remote: !!onlyRemote, allow_local_only: !!allowLocalOnly, max_id: maxId, only_media: !!onlyMedia }, done);
+export const expandCommunityTimeline       = ({ maxId, onlyMedia } = {}, done = noOp) => expandTimeline(`community${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { local: true, max_id: maxId, only_media: !!onlyMedia }, done);
+export const expandDirectTimeline          = ({ maxId } = {}, done = noOp) => expandTimeline('direct', '/api/v1/timelines/direct', { max_id: maxId }, done);
+export const expandAccountTimeline         = (accountId, { maxId, withReplies, tagged } = {}) => expandTimeline(`account:${accountId}${withReplies ? ':with_replies' : ''}${tagged ? `:${tagged}` : ''}`, `/api/v1/accounts/${accountId}/statuses`, { exclude_replies: !withReplies, exclude_reblogs: withReplies, tagged, max_id: maxId });
+export const expandAccountFeaturedTimeline = (accountId, { tagged } = {}) => expandTimeline(`account:${accountId}:pinned`, `/api/v1/accounts/${accountId}/statuses`, { pinned: true, tagged });
+export const expandAccountMediaTimeline    = (accountId, { maxId } = {}) => expandTimeline(`account:${accountId}:media`, `/api/v1/accounts/${accountId}/statuses`, { max_id: maxId, only_media: true, limit: 40 });
+export const expandListTimeline            = (id, { maxId } = {}, done = noOp) => expandTimeline(`list:${id}`, `/api/v1/timelines/list/${id}`, { max_id: maxId }, done);
+export const expandHashtagTimeline         = (hashtag, { maxId, tags, local } = {}, done = noOp) => {
+  return expandTimeline(`hashtag:${hashtag}${local ? ':local' : ''}`, `/api/v1/timelines/tag/${hashtag}`, {
+    max_id: maxId,
+    any:    parseTags(tags, 'any'),
+    all:    parseTags(tags, 'all'),
+    none:   parseTags(tags, 'none'),
+    local:  local,
+  }, done);
+};
+
+export const fillHomeTimelineGaps      = (done = noOp) => fillTimelineGaps('home', '/api/v1/timelines/home', {}, done);
+export const fillPublicTimelineGaps    = ({ onlyMedia, onlyRemote, allowLocalOnly } = {}, done = noOp) => fillTimelineGaps(`public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { remote: !!onlyRemote, only_media: !!onlyMedia, allow_local_only: !!allowLocalOnly }, done);
+export const fillCommunityTimelineGaps = ({ onlyMedia } = {}, done = noOp) => fillTimelineGaps(`community${onlyMedia ? ':media' : ''}`, '/api/v1/timelines/public', { local: true, only_media: !!onlyMedia }, done);
+export const fillListTimelineGaps      = (id, done = noOp) => fillTimelineGaps(`list:${id}`, `/api/v1/timelines/list/${id}`, {}, done);
+
+export function expandTimelineRequest(timeline, isLoadingMore) {
+  return {
+    type: TIMELINE_EXPAND_REQUEST,
+    timeline,
+    skipLoading: !isLoadingMore,
+  };
+}
+
+export function expandTimelineSuccess(timeline, statuses, next, partial, isLoadingRecent, isLoadingMore, usePendingItems) {
+  return {
+    type: TIMELINE_EXPAND_SUCCESS,
+    timeline,
+    statuses,
+    next,
+    partial,
+    isLoadingRecent,
+    usePendingItems,
+    skipLoading: !isLoadingMore,
+  };
+}
+
+export function expandTimelineFail(timeline, error, isLoadingMore) {
+  return {
+    type: TIMELINE_EXPAND_FAIL,
+    timeline,
+    error,
+    skipLoading: !isLoadingMore,
+    skipNotFound: timeline.startsWith('account:'),
+  };
+}
+
+export function scrollTopTimeline(timeline, top) {
+  return {
+    type: TIMELINE_SCROLL_TOP,
+    timeline,
+    top,
+  };
+}
+
+export function connectTimeline(timeline) {
+  return {
+    type: TIMELINE_CONNECT,
+    timeline,
+    usePendingItems: preferPendingItems,
+  };
+}
+
+export const disconnectTimeline = timeline => ({
+  type: TIMELINE_DISCONNECT,
+  timeline,
+  usePendingItems: preferPendingItems,
+});
+
+export const markAsPartial = timeline => ({
+  type: TIMELINE_MARK_AS_PARTIAL,
+  timeline,
+});
diff --git a/app/javascript/flavours/blobfox/actions/trends.js b/app/javascript/flavours/blobfox/actions/trends.js
new file mode 100644
index 00000000000000..d314423884efe1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/actions/trends.js
@@ -0,0 +1,140 @@
+import api, { getLinks } from '../api';
+
+import { importFetchedStatuses } from './importer';
+
+export const TRENDS_TAGS_FETCH_REQUEST = 'TRENDS_TAGS_FETCH_REQUEST';
+export const TRENDS_TAGS_FETCH_SUCCESS = 'TRENDS_TAGS_FETCH_SUCCESS';
+export const TRENDS_TAGS_FETCH_FAIL    = 'TRENDS_TAGS_FETCH_FAIL';
+
+export const TRENDS_LINKS_FETCH_REQUEST = 'TRENDS_LINKS_FETCH_REQUEST';
+export const TRENDS_LINKS_FETCH_SUCCESS = 'TRENDS_LINKS_FETCH_SUCCESS';
+export const TRENDS_LINKS_FETCH_FAIL    = 'TRENDS_LINKS_FETCH_FAIL';
+
+export const TRENDS_STATUSES_FETCH_REQUEST = 'TRENDS_STATUSES_FETCH_REQUEST';
+export const TRENDS_STATUSES_FETCH_SUCCESS = 'TRENDS_STATUSES_FETCH_SUCCESS';
+export const TRENDS_STATUSES_FETCH_FAIL    = 'TRENDS_STATUSES_FETCH_FAIL';
+
+export const TRENDS_STATUSES_EXPAND_REQUEST = 'TRENDS_STATUSES_EXPAND_REQUEST';
+export const TRENDS_STATUSES_EXPAND_SUCCESS = 'TRENDS_STATUSES_EXPAND_SUCCESS';
+export const TRENDS_STATUSES_EXPAND_FAIL    = 'TRENDS_STATUSES_EXPAND_FAIL';
+
+export const fetchTrendingHashtags = () => (dispatch, getState) => {
+  dispatch(fetchTrendingHashtagsRequest());
+
+  api(getState)
+    .get('/api/v1/trends/tags')
+    .then(({ data }) => dispatch(fetchTrendingHashtagsSuccess(data)))
+    .catch(err => dispatch(fetchTrendingHashtagsFail(err)));
+};
+
+export const fetchTrendingHashtagsRequest = () => ({
+  type: TRENDS_TAGS_FETCH_REQUEST,
+  skipLoading: true,
+});
+
+export const fetchTrendingHashtagsSuccess = trends => ({
+  type: TRENDS_TAGS_FETCH_SUCCESS,
+  trends,
+  skipLoading: true,
+});
+
+export const fetchTrendingHashtagsFail = error => ({
+  type: TRENDS_TAGS_FETCH_FAIL,
+  error,
+  skipLoading: true,
+  skipAlert: true,
+});
+
+export const fetchTrendingLinks = () => (dispatch, getState) => {
+  dispatch(fetchTrendingLinksRequest());
+
+  api(getState)
+    .get('/api/v1/trends/links')
+    .then(({ data }) => dispatch(fetchTrendingLinksSuccess(data)))
+    .catch(err => dispatch(fetchTrendingLinksFail(err)));
+};
+
+export const fetchTrendingLinksRequest = () => ({
+  type: TRENDS_LINKS_FETCH_REQUEST,
+  skipLoading: true,
+});
+
+export const fetchTrendingLinksSuccess = trends => ({
+  type: TRENDS_LINKS_FETCH_SUCCESS,
+  trends,
+  skipLoading: true,
+});
+
+export const fetchTrendingLinksFail = error => ({
+  type: TRENDS_LINKS_FETCH_FAIL,
+  error,
+  skipLoading: true,
+  skipAlert: true,
+});
+
+export const fetchTrendingStatuses = () => (dispatch, getState) => {
+  if (getState().getIn(['status_lists', 'trending', 'isLoading'])) {
+    return;
+  }
+
+  dispatch(fetchTrendingStatusesRequest());
+
+  api(getState).get('/api/v1/trends/statuses').then(response => {
+    const next = getLinks(response).refs.find(link => link.rel === 'next');
+    dispatch(importFetchedStatuses(response.data));
+    dispatch(fetchTrendingStatusesSuccess(response.data, next ? next.uri : null));
+  }).catch(err => dispatch(fetchTrendingStatusesFail(err)));
+};
+
+export const fetchTrendingStatusesRequest = () => ({
+  type: TRENDS_STATUSES_FETCH_REQUEST,
+  skipLoading: true,
+});
+
+export const fetchTrendingStatusesSuccess = (statuses, next) => ({
+  type: TRENDS_STATUSES_FETCH_SUCCESS,
+  statuses,
+  next,
+  skipLoading: true,
+});
+
+export const fetchTrendingStatusesFail = error => ({
+  type: TRENDS_STATUSES_FETCH_FAIL,
+  error,
+  skipLoading: true,
+  skipAlert: true,
+});
+
+
+export const expandTrendingStatuses = () => (dispatch, getState) => {
+  const url = getState().getIn(['status_lists', 'trending', 'next'], null);
+
+  if (url === null || getState().getIn(['status_lists', 'trending', 'isLoading'])) {
+    return;
+  }
+
+  dispatch(expandTrendingStatusesRequest());
+
+  api(getState).get(url).then(response => {
+    const next = getLinks(response).refs.find(link => link.rel === 'next');
+    dispatch(importFetchedStatuses(response.data));
+    dispatch(expandTrendingStatusesSuccess(response.data, next ? next.uri : null));
+  }).catch(error => {
+    dispatch(expandTrendingStatusesFail(error));
+  });
+};
+
+export const expandTrendingStatusesRequest = () => ({
+  type: TRENDS_STATUSES_EXPAND_REQUEST,
+});
+
+export const expandTrendingStatusesSuccess = (statuses, next) => ({
+  type: TRENDS_STATUSES_EXPAND_SUCCESS,
+  statuses,
+  next,
+});
+
+export const expandTrendingStatusesFail = error => ({
+  type: TRENDS_STATUSES_EXPAND_FAIL,
+  error,
+});
diff --git a/app/javascript/flavours/blobfox/api.ts b/app/javascript/flavours/blobfox/api.ts
new file mode 100644
index 00000000000000..f262fd85707941
--- /dev/null
+++ b/app/javascript/flavours/blobfox/api.ts
@@ -0,0 +1,63 @@
+import type { AxiosResponse, RawAxiosRequestHeaders } from 'axios';
+import axios from 'axios';
+import LinkHeader from 'http-link-header';
+
+import ready from './ready';
+import type { GetState } from './store';
+
+export const getLinks = (response: AxiosResponse) => {
+  const value = response.headers.link as string | undefined;
+
+  if (!value) {
+    return new LinkHeader();
+  }
+
+  return LinkHeader.parse(value);
+};
+
+const csrfHeader: RawAxiosRequestHeaders = {};
+
+const setCSRFHeader = () => {
+  const csrfToken = document.querySelector<HTMLMetaElement>(
+    'meta[name=csrf-token]',
+  );
+
+  if (csrfToken) {
+    csrfHeader['X-CSRF-Token'] = csrfToken.content;
+  }
+};
+
+void ready(setCSRFHeader);
+
+const authorizationHeaderFromState = (getState?: GetState) => {
+  const accessToken =
+    getState && (getState().meta.get('access_token', '') as string);
+
+  if (!accessToken) {
+    return {};
+  }
+
+  return {
+    Authorization: `Bearer ${accessToken}`,
+  } as RawAxiosRequestHeaders;
+};
+
+// eslint-disable-next-line import/no-default-export
+export default function api(getState: GetState) {
+  return axios.create({
+    headers: {
+      ...csrfHeader,
+      ...authorizationHeaderFromState(getState),
+    },
+
+    transformResponse: [
+      function (data: unknown) {
+        try {
+          return JSON.parse(data as string) as unknown;
+        } catch {
+          return data;
+        }
+      },
+    ],
+  });
+}
diff --git a/app/javascript/flavours/blobfox/api_types/accounts.ts b/app/javascript/flavours/blobfox/api_types/accounts.ts
new file mode 100644
index 00000000000000..ce55dc604ad001
--- /dev/null
+++ b/app/javascript/flavours/blobfox/api_types/accounts.ts
@@ -0,0 +1,45 @@
+import type { ApiCustomEmojiJSON } from './custom_emoji';
+
+export interface ApiAccountFieldJSON {
+  name: string;
+  value: string;
+  verified_at: string | null;
+}
+
+export interface ApiAccountRoleJSON {
+  color: string;
+  id: string;
+  name: string;
+}
+
+// See app/serializers/rest/account_serializer.rb
+export interface ApiAccountJSON {
+  acct: string;
+  avatar: string;
+  avatar_static: string;
+  bot: boolean;
+  created_at: string;
+  discoverable: boolean;
+  display_name: string;
+  emojis: ApiCustomEmojiJSON[];
+  fields: ApiAccountFieldJSON[];
+  followers_count: number;
+  following_count: number;
+  group: boolean;
+  header: string;
+  header_static: string;
+  id: string;
+  last_status_at: string;
+  locked: boolean;
+  noindex?: boolean;
+  note: string;
+  roles?: ApiAccountJSON[];
+  statuses_count: number;
+  uri: string;
+  url: string;
+  username: string;
+  moved?: ApiAccountJSON;
+  suspended?: boolean;
+  limited?: boolean;
+  memorial?: boolean;
+}
diff --git a/app/javascript/flavours/blobfox/api_types/custom_emoji.ts b/app/javascript/flavours/blobfox/api_types/custom_emoji.ts
new file mode 100644
index 00000000000000..05144d6f68d0e8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/api_types/custom_emoji.ts
@@ -0,0 +1,8 @@
+// See app/serializers/rest/account_serializer.rb
+export interface ApiCustomEmojiJSON {
+  shortcode: string;
+  static_url: string;
+  url: string;
+  category?: string;
+  visible_in_picker: boolean;
+}
diff --git a/app/javascript/flavours/blobfox/api_types/relationships.ts b/app/javascript/flavours/blobfox/api_types/relationships.ts
new file mode 100644
index 00000000000000..9f26a0ce9b333d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/api_types/relationships.ts
@@ -0,0 +1,18 @@
+// See app/serializers/rest/relationship_serializer.rb
+export interface ApiRelationshipJSON {
+  blocked_by: boolean;
+  blocking: boolean;
+  domain_blocking: boolean;
+  endorsed: boolean;
+  followed_by: boolean;
+  following: boolean;
+  id: string;
+  languages: string[] | null;
+  muting_notifications: boolean;
+  muting: boolean;
+  note: string;
+  notifying: boolean;
+  requested_by: boolean;
+  requested: boolean;
+  showing_reblogs: boolean;
+}
diff --git a/app/javascript/flavours/blobfox/blurhash.ts b/app/javascript/flavours/blobfox/blurhash.ts
new file mode 100644
index 00000000000000..cafe7b12dcff8b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/blurhash.ts
@@ -0,0 +1,111 @@
+const DIGIT_CHARACTERS = [
+  '0',
+  '1',
+  '2',
+  '3',
+  '4',
+  '5',
+  '6',
+  '7',
+  '8',
+  '9',
+  'A',
+  'B',
+  'C',
+  'D',
+  'E',
+  'F',
+  'G',
+  'H',
+  'I',
+  'J',
+  'K',
+  'L',
+  'M',
+  'N',
+  'O',
+  'P',
+  'Q',
+  'R',
+  'S',
+  'T',
+  'U',
+  'V',
+  'W',
+  'X',
+  'Y',
+  'Z',
+  'a',
+  'b',
+  'c',
+  'd',
+  'e',
+  'f',
+  'g',
+  'h',
+  'i',
+  'j',
+  'k',
+  'l',
+  'm',
+  'n',
+  'o',
+  'p',
+  'q',
+  'r',
+  's',
+  't',
+  'u',
+  'v',
+  'w',
+  'x',
+  'y',
+  'z',
+  '#',
+  '$',
+  '%',
+  '*',
+  '+',
+  ',',
+  '-',
+  '.',
+  ':',
+  ';',
+  '=',
+  '?',
+  '@',
+  '[',
+  ']',
+  '^',
+  '_',
+  '{',
+  '|',
+  '}',
+  '~',
+];
+
+export const decode83 = (str: string) => {
+  let value = 0;
+  let digit;
+
+  for (const c of str) {
+    digit = DIGIT_CHARACTERS.indexOf(c);
+    value = value * 83 + digit;
+  }
+
+  return value;
+};
+
+export const intToRGB = (int: number) => ({
+  r: Math.max(0, int >> 16),
+  g: Math.max(0, (int >> 8) & 255),
+  b: Math.max(0, int & 255),
+});
+
+export const getAverageFromBlurhash = (blurhash: string) => {
+  if (!blurhash) {
+    return null;
+  }
+
+  return intToRGB(decode83(blurhash.slice(2, 6)));
+};
diff --git a/app/javascript/flavours/blobfox/compare_id.ts b/app/javascript/flavours/blobfox/compare_id.ts
new file mode 100644
index 00000000000000..30b05724817892
--- /dev/null
+++ b/app/javascript/flavours/blobfox/compare_id.ts
@@ -0,0 +1,11 @@
+export function compareId(id1: string, id2: string) {
+  if (id1 === id2) {
+    return 0;
+  }
+
+  if (id1.length === id2.length) {
+    return id1 > id2 ? 1 : -1;
+  } else {
+    return id1.length > id2.length ? 1 : -1;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/components/__tests__/hashtag_bar.tsx b/app/javascript/flavours/blobfox/components/__tests__/hashtag_bar.tsx
new file mode 100644
index 00000000000000..b7225fc92e01e4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/__tests__/hashtag_bar.tsx
@@ -0,0 +1,214 @@
+import { fromJS } from 'immutable';
+
+import type { StatusLike } from '../hashtag_bar';
+import { computeHashtagBarForStatus } from '../hashtag_bar';
+
+function createStatus(
+  content: string,
+  hashtags: string[],
+  hasMedia = false,
+  spoilerText?: string,
+) {
+  return fromJS({
+    tags: hashtags.map((name) => ({ name })),
+    contentHtml: content,
+    media_attachments: hasMedia ? ['fakeMedia'] : [],
+    spoiler_text: spoilerText,
+  }) as unknown as StatusLike; // need to force the type here, as it is not properly defined
+}
+
+describe('computeHashtagBarForStatus', () => {
+  it('does nothing when there are no tags', () => {
+    const status = createStatus('<p>Simple text</p>', []);
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Simple text</p>"`,
+    );
+  });
+
+  it('displays out of band hashtags in the bar', () => {
+    const status = createStatus(
+      '<p>Simple text <a href="test">#hashtag</a></p>',
+      ['hashtag', 'test'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual(['test']);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Simple text <a href="test">#hashtag</a></p>"`,
+    );
+  });
+
+  it('does not truncate the contents when the last child is a text node', () => {
+    const status = createStatus(
+      'this is a #<a class="zrl" href="https://example.com/search?tag=test">test</a>. Some more text',
+      ['test'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"this is a #<a class="zrl" href="https://example.com/search?tag=test">test</a>. Some more text"`,
+    );
+  });
+
+  it('extract tags from the last line', () => {
+    const status = createStatus(
+      '<p>Simple text</p><p><a href="test">#hashtag</a></p>',
+      ['hashtag'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual(['hashtag']);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Simple text</p>"`,
+    );
+  });
+
+  it('does not include tags from content', () => {
+    const status = createStatus(
+      '<p>Simple text with a <a href="test">#hashtag</a></p><p><a href="test">#hashtag</a></p>',
+      ['hashtag'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Simple text with a <a href="test">#hashtag</a></p>"`,
+    );
+  });
+
+  it('works with one line status and hashtags', () => {
+    const status = createStatus(
+      '<p><a href="test">#test</a>. And another <a href="test">#hashtag</a></p>',
+      ['hashtag', 'test'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p><a href="test">#test</a>. And another <a href="test">#hashtag</a></p>"`,
+    );
+  });
+
+  it('de-duplicate accentuated characters with case differences', () => {
+    const status = createStatus(
+      '<p>Text</p><p><a href="test">#éaa</a> <a href="test">#Éaa</a></p>',
+      ['éaa'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual(['Éaa']);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Text</p>"`,
+    );
+  });
+
+  it('handles server-side normalized tags with accentuated characters', () => {
+    const status = createStatus(
+      '<p>Text</p><p><a href="test">#éaa</a> <a href="test">#Éaa</a></p>',
+      ['eaa'], // The server may normalize the hashtags in the `tags` attribute
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual(['Éaa']);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Text</p>"`,
+    );
+  });
+
+  it('does not display in bar a hashtag in content with a case difference', () => {
+    const status = createStatus(
+      '<p>Text <a href="test">#Éaa</a></p><p><a href="test">#éaa</a></p>',
+      ['éaa'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>Text <a href="test">#Éaa</a></p>"`,
+    );
+  });
+
+  it('does not modify a status with a line of hashtags only', () => {
+    const status = createStatus(
+      '<p><a href="test">#test</a>  <a href="test">#hashtag</a></p>',
+      ['test', 'hashtag'],
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p><a href="test">#test</a>  <a href="test">#hashtag</a></p>"`,
+    );
+  });
+
+  it('puts the hashtags in the bar if a status content has hashtags in the only line and has a media', () => {
+    const status = createStatus(
+      '<p>This is my content! <a href="test">#hashtag</a></p>',
+      ['hashtag'],
+      true,
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p>This is my content! <a href="test">#hashtag</a></p>"`,
+    );
+  });
+
+  it('puts the hashtags in the bar if a status content is only hashtags and has a media', () => {
+    const status = createStatus(
+      '<p><a href="test">#test</a>  <a href="test">#hashtag</a></p>',
+      ['test', 'hashtag'],
+      true,
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual(['test', 'hashtag']);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(`""`);
+  });
+
+  it('does not use the hashtag bar if the status content is only hashtags, has a CW and a media', () => {
+    const status = createStatus(
+      '<p><a href="test">#test</a>  <a href="test">#hashtag</a></p>',
+      ['test', 'hashtag'],
+      true,
+      'My CW text',
+    );
+
+    const { hashtagsInBar, statusContentProps } =
+      computeHashtagBarForStatus(status);
+
+    expect(hashtagsInBar).toEqual([]);
+    expect(statusContentProps.statusContent).toMatchInlineSnapshot(
+      `"<p><a href="test">#test</a>  <a href="test">#hashtag</a></p>"`,
+    );
+  });
+});
diff --git a/app/javascript/flavours/blobfox/components/account.jsx b/app/javascript/flavours/blobfox/components/account.jsx
new file mode 100644
index 00000000000000..a83c9c066d87ad
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/account.jsx
@@ -0,0 +1,182 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { EmptyAccount } from 'flavours/blobfox/components/empty_account';
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+import { VerifiedBadge } from 'flavours/blobfox/components/verified_badge';
+
+import { me } from '../initial_state';
+
+import { Avatar } from './avatar';
+import { Button } from './button';
+import { FollowersCounter } from './counters';
+import { DisplayName } from './display_name';
+import Permalink from './permalink';
+import { RelativeTimestamp } from './relative_timestamp';
+
+const messages = defineMessages({
+  follow: { id: 'account.follow', defaultMessage: 'Follow' },
+  unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' },
+  cancel_follow_request: { id: 'account.cancel_follow_request', defaultMessage: 'Withdraw follow request' },
+  unblock: { id: 'account.unblock_short', defaultMessage: 'Unblock' },
+  unmute: { id: 'account.unmute_short', defaultMessage: 'Unmute' },
+  mute_notifications: { id: 'account.mute_notifications_short', defaultMessage: 'Mute notifications' },
+  unmute_notifications: { id: 'account.unmute_notifications_short', defaultMessage: 'Unmute notifications' },
+  mute: { id: 'account.mute_short', defaultMessage: 'Mute' },
+  block: { id: 'account.block_short', defaultMessage: 'Block' },
+});
+
+class Account extends ImmutablePureComponent {
+
+  static propTypes = {
+    size: PropTypes.number,
+    account: ImmutablePropTypes.record,
+    onFollow: PropTypes.func.isRequired,
+    onBlock: PropTypes.func.isRequired,
+    onMute: PropTypes.func.isRequired,
+    onMuteNotifications: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    hidden: PropTypes.bool,
+    minimal: PropTypes.bool,
+    defaultAction: PropTypes.string,
+    withBio: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    size: 46,
+  };
+
+  handleFollow = () => {
+    this.props.onFollow(this.props.account);
+  };
+
+  handleBlock = () => {
+    this.props.onBlock(this.props.account);
+  };
+
+  handleMute = () => {
+    this.props.onMute(this.props.account);
+  };
+
+  handleMuteNotifications = () => {
+    this.props.onMuteNotifications(this.props.account, true);
+  };
+
+  handleUnmuteNotifications = () => {
+    this.props.onMuteNotifications(this.props.account, false);
+  };
+
+  render () {
+    const { account, intl, hidden, withBio, defaultAction, size, minimal } = this.props;
+
+    if (!account) {
+      return <EmptyAccount size={size} minimal={minimal} />;
+    }
+
+    if (hidden) {
+      return (
+        <>
+          {account.get('display_name')}
+          {account.get('username')}
+        </>
+      );
+    }
+
+    let buttons;
+
+    if (account.get('id') !== me && account.get('relationship', null) !== null) {
+      const following = account.getIn(['relationship', 'following']);
+      const requested = account.getIn(['relationship', 'requested']);
+      const blocking  = account.getIn(['relationship', 'blocking']);
+      const muting  = account.getIn(['relationship', 'muting']);
+
+      if (requested) {
+        buttons = <Button text={intl.formatMessage(messages.cancel_follow_request)} onClick={this.handleFollow} />;
+      } else if (blocking) {
+        buttons = <Button text={intl.formatMessage(messages.unblock)} onClick={this.handleBlock} />;
+      } else if (muting) {
+        let hidingNotificationsButton;
+
+        if (account.getIn(['relationship', 'muting_notifications'])) {
+          hidingNotificationsButton = <Button text={intl.formatMessage(messages.unmute_notifications)} onClick={this.handleUnmuteNotifications} />;
+        } else {
+          hidingNotificationsButton = <Button text={intl.formatMessage(messages.mute_notifications)} onClick={this.handleMuteNotifications} />;
+        }
+
+        buttons = (
+          <>
+            <Button text={intl.formatMessage(messages.unmute)} onClick={this.handleMute} />
+            {hidingNotificationsButton}
+          </>
+        );
+      } else if (defaultAction === 'mute') {
+        buttons = <Button title={intl.formatMessage(messages.mute)} onClick={this.handleMute} />;
+      } else if (defaultAction === 'block') {
+        buttons = <Button text={intl.formatMessage(messages.block)} onClick={this.handleBlock} />;
+      } else if (!account.get('suspended') && !account.get('moved') || following) {
+        buttons = <Button text={intl.formatMessage(following ? messages.unfollow : messages.follow)} onClick={this.handleFollow} />;
+      }
+    }
+
+    let muteTimeRemaining;
+
+    if (account.get('mute_expires_at')) {
+      muteTimeRemaining = <>· <RelativeTimestamp timestamp={account.get('mute_expires_at')} futureDate /></>;
+    }
+
+    let verification;
+
+    const firstVerifiedField = account.get('fields').find(item => !!item.get('verified_at'));
+
+    if (firstVerifiedField) {
+      verification = <VerifiedBadge link={firstVerifiedField.get('value')} />;
+    }
+
+    return (
+      <div className={classNames('account', { 'account--minimal': minimal })}>
+        <div className='account__wrapper'>
+          <Permalink key={account.get('id')} className='account__display-name' title={account.get('acct')} href={account.get('url')} to={`/@${account.get('acct')}`}>
+            <div className='account__avatar-wrapper'>
+              <Avatar account={account} size={size} />
+            </div>
+
+            <div className='account__contents'>
+              <DisplayName account={account} inline />
+              {!minimal && (
+                <div className='account__details'>
+                  {account.get('followers_count') !== -1 && (
+                    <ShortNumber value={account.get('followers_count')} renderer={FollowersCounter} />
+                  )} {verification} {muteTimeRemaining}
+                </div>
+              )}
+            </div>
+          </Permalink>
+
+          {!minimal && (
+            <div className='account__relationship'>
+              {buttons}
+            </div>
+          )}
+        </div>
+
+        {withBio && (account.get('note').length > 0 ? (
+          <div
+            className='account__note translate'
+            dangerouslySetInnerHTML={{ __html: account.get('note_emojified') }}
+          />
+        ) : (
+          <div className='account__note account__note--missing'><FormattedMessage id='account.no_bio' defaultMessage='No description provided.' /></div>
+        ))}
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Account);
diff --git a/app/javascript/flavours/blobfox/components/admin/Counter.jsx b/app/javascript/flavours/blobfox/components/admin/Counter.jsx
new file mode 100644
index 00000000000000..f9a5fc53962a6c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/admin/Counter.jsx
@@ -0,0 +1,121 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedNumber } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { Sparklines, SparklinesCurve } from 'react-sparklines';
+
+import api from 'flavours/blobfox/api';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+
+const percIncrease = (a, b) => {
+  let percent;
+
+  if (b !== 0) {
+    if (a !== 0) {
+      percent = (b - a) / a;
+    } else {
+      percent = 1;
+    }
+  } else if (b === 0 && a === 0) {
+    percent = 0;
+  } else {
+    percent = - 1;
+  }
+
+  return percent;
+};
+
+export default class Counter extends PureComponent {
+
+  static propTypes = {
+    measure: PropTypes.string.isRequired,
+    start_at: PropTypes.string.isRequired,
+    end_at: PropTypes.string.isRequired,
+    label: PropTypes.string.isRequired,
+    href: PropTypes.string,
+    params: PropTypes.object,
+    target: PropTypes.string,
+  };
+
+  state = {
+    loading: true,
+    data: null,
+  };
+
+  componentDidMount () {
+    const { measure, start_at, end_at, params } = this.props;
+
+    api().post('/api/v1/admin/measures', { keys: [measure], start_at, end_at, [measure]: params }).then(res => {
+      this.setState({
+        loading: false,
+        data: res.data,
+      });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  render () {
+    const { label, href, target } = this.props;
+    const { loading, data } = this.state;
+
+    let content;
+
+    if (loading) {
+      content = (
+        <>
+          <span className='sparkline__value__total'><Skeleton width={43} /></span>
+          <span className='sparkline__value__change'><Skeleton width={43} /></span>
+        </>
+      );
+    } else {
+      const measure = data[0];
+      const percentChange = measure.previous_total && percIncrease(measure.previous_total * 1, measure.total * 1);
+
+      content = (
+        <>
+          <span className='sparkline__value__total'>{measure.human_value || <FormattedNumber value={measure.total} />}</span>
+          {measure.previous_total && (<span className={classNames('sparkline__value__change', { positive: percentChange > 0, negative: percentChange < 0 })}>{percentChange > 0 && '+'}<FormattedNumber value={percentChange} style='percent' /></span>)}
+        </>
+      );
+    }
+
+    const inner = (
+      <>
+        <div className='sparkline__value'>
+          {content}
+        </div>
+
+        <div className='sparkline__label'>
+          {label}
+        </div>
+
+        <div className='sparkline__graph'>
+          {!loading && (
+            <Sparklines width={259} height={55} data={data[0].data.map(x => x.value * 1)}>
+              <SparklinesCurve />
+            </Sparklines>
+          )}
+        </div>
+      </>
+    );
+
+    if (href) {
+      return (
+        <a href={href} className='sparkline' target={target}>
+          {inner}
+        </a>
+      );
+    } else {
+      return (
+        <div className='sparkline'>
+          {inner}
+        </div>
+      );
+    }
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/admin/Dimension.jsx b/app/javascript/flavours/blobfox/components/admin/Dimension.jsx
new file mode 100644
index 00000000000000..bcd4afa3ac16eb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/admin/Dimension.jsx
@@ -0,0 +1,95 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedNumber } from 'react-intl';
+
+import api from 'flavours/blobfox/api';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+import { roundTo10 } from 'flavours/blobfox/utils/numbers';
+
+export default class Dimension extends PureComponent {
+
+  static propTypes = {
+    dimension: PropTypes.string.isRequired,
+    start_at: PropTypes.string.isRequired,
+    end_at: PropTypes.string.isRequired,
+    limit: PropTypes.number.isRequired,
+    label: PropTypes.string.isRequired,
+    params: PropTypes.object,
+  };
+
+  state = {
+    loading: true,
+    data: null,
+  };
+
+  componentDidMount () {
+    const { start_at, end_at, dimension, limit, params } = this.props;
+
+    api().post('/api/v1/admin/dimensions', { keys: [dimension], start_at, end_at, limit, [dimension]: params }).then(res => {
+      this.setState({
+        loading: false,
+        data: res.data,
+      });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  render () {
+    const { label, limit } = this.props;
+    const { loading, data } = this.state;
+
+    let content;
+
+    if (loading) {
+      content = (
+        <table>
+          <tbody>
+            {Array.from(Array(limit)).map((_, i) => (
+              <tr className='dimension__item' key={i}>
+                <td className='dimension__item__key'>
+                  <Skeleton width={100} />
+                </td>
+
+                <td className='dimension__item__value'>
+                  <Skeleton width={60} />
+                </td>
+              </tr>
+            ))}
+          </tbody>
+        </table>
+      );
+    } else {
+      const sum = data[0].data.reduce((sum, cur) => sum + (cur.value * 1), 0);
+
+      content = (
+        <table>
+          <tbody>
+            {data[0].data.map(item => (
+              <tr className='dimension__item' key={item.key}>
+                <td className='dimension__item__key'>
+                  <span className={`dimension__item__indicator dimension__item__indicator--${roundTo10(((item.value * 1) / sum) * 100)}`} />
+                  <span title={item.key}>{item.human_key}</span>
+                </td>
+
+                <td className='dimension__item__value'>
+                  {typeof item.human_value !== 'undefined' ? item.human_value : <FormattedNumber value={item.value} />}
+                </td>
+              </tr>
+            ))}
+          </tbody>
+        </table>
+      );
+    }
+
+    return (
+      <div className='dimension'>
+        <h4>{label}</h4>
+
+        {content}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/admin/ImpactReport.jsx b/app/javascript/flavours/blobfox/components/admin/ImpactReport.jsx
new file mode 100644
index 00000000000000..fe3a2ea29ff5f5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/admin/ImpactReport.jsx
@@ -0,0 +1,91 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedNumber, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import api from 'flavours/blobfox/api';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+
+export default class ImpactReport extends PureComponent {
+
+  static propTypes = {
+    domain: PropTypes.string.isRequired,
+  };
+
+  state = {
+    loading: true,
+    data: null,
+  };
+
+  componentDidMount () {
+    const { domain } = this.props;
+
+    const params = {
+      domain: domain,
+      include_subdomains: true,
+    };
+
+    api().post('/api/v1/admin/measures', {
+      keys: ['instance_accounts', 'instance_follows', 'instance_followers'],
+      start_at: null,
+      end_at: null,
+      instance_accounts: params,
+      instance_follows: params,
+      instance_followers: params,
+    }).then(res => {
+      this.setState({
+        loading: false,
+        data: res.data,
+      });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  render () {
+    const { loading, data } = this.state;
+
+    return (
+      <div className='dimension'>
+        <h4><FormattedMessage id='admin.impact_report.title' defaultMessage='Impact summary' /></h4>
+
+        <table>
+          <tbody>
+            <tr className='dimension__item'>
+              <td className='dimension__item__key'>
+                <FormattedMessage id='admin.impact_report.instance_accounts' defaultMessage='Accounts profiles this would delete' />
+              </td>
+
+              <td className='dimension__item__value'>
+                {loading ? <Skeleton width={60} /> : <FormattedNumber value={data[0].total} />}
+              </td>
+            </tr>
+
+            <tr className={classNames('dimension__item', { negative: !loading && data[1].total > 0 })}>
+              <td className='dimension__item__key'>
+                <FormattedMessage id='admin.impact_report.instance_follows' defaultMessage='Followers their users would lose' />
+              </td>
+
+              <td className='dimension__item__value'>
+                {loading ? <Skeleton width={60} /> : <FormattedNumber value={data[1].total} />}
+              </td>
+            </tr>
+
+            <tr className={classNames('dimension__item', { negative: !loading && data[2].total > 0 })}>
+              <td className='dimension__item__key'>
+                <FormattedMessage id='admin.impact_report.instance_followers' defaultMessage='Followers our users would lose' />
+              </td>
+
+              <td className='dimension__item__value'>
+                {loading ? <Skeleton width={60} /> : <FormattedNumber value={data[2].total} />}
+              </td>
+            </tr>
+          </tbody>
+        </table>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/admin/ReportReasonSelector.jsx b/app/javascript/flavours/blobfox/components/admin/ReportReasonSelector.jsx
new file mode 100644
index 00000000000000..6ef9c912330469
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/admin/ReportReasonSelector.jsx
@@ -0,0 +1,165 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import classNames from 'classnames';
+
+import api from 'flavours/blobfox/api';
+
+const messages = defineMessages({
+  legal: { id: 'report.categories.legal', defaultMessage: 'Legal' },
+  other: { id: 'report.categories.other', defaultMessage: 'Other' },
+  spam: { id: 'report.categories.spam', defaultMessage: 'Spam' },
+  violation: { id: 'report.categories.violation', defaultMessage: 'Content violates one or more server rules' },
+});
+
+class Category extends PureComponent {
+
+  static propTypes = {
+    id: PropTypes.string.isRequired,
+    text: PropTypes.string.isRequired,
+    selected: PropTypes.bool,
+    disabled: PropTypes.bool,
+    onSelect: PropTypes.func,
+    children: PropTypes.node,
+  };
+
+  handleClick = () => {
+    const { id, disabled, onSelect } = this.props;
+
+    if (!disabled) {
+      onSelect(id);
+    }
+  };
+
+  render () {
+    const { id, text, disabled, selected, children } = this.props;
+
+    return (
+      <div tabIndex={0} role='button' className={classNames('report-reason-selector__category', { selected, disabled })} onClick={this.handleClick}>
+        {selected && <input type='hidden' name='report[category]' value={id} />}
+
+        <div className='report-reason-selector__category__label'>
+          <span className={classNames('poll__input', { active: selected, disabled })} />
+          {text}
+        </div>
+
+        {(selected && children) && (
+          <div className='report-reason-selector__category__rules'>
+            {children}
+          </div>
+        )}
+      </div>
+    );
+  }
+
+}
+
+class Rule extends PureComponent {
+
+  static propTypes = {
+    id: PropTypes.string.isRequired,
+    text: PropTypes.string.isRequired,
+    selected: PropTypes.bool,
+    disabled: PropTypes.bool,
+    onToggle: PropTypes.func,
+  };
+
+  handleClick = () => {
+    const { id, disabled, onToggle } = this.props;
+
+    if (!disabled) {
+      onToggle(id);
+    }
+  };
+
+  render () {
+    const { id, text, disabled, selected } = this.props;
+
+    return (
+      <div tabIndex={0} role='button' className={classNames('report-reason-selector__rule', { selected, disabled })} onClick={this.handleClick}>
+        <span className={classNames('poll__input', { checkbox: true, active: selected, disabled })} />
+        {selected && <input type='hidden' name='report[rule_ids][]' value={id} />}
+        {text}
+      </div>
+    );
+  }
+
+}
+
+class ReportReasonSelector extends PureComponent {
+
+  static propTypes = {
+    id: PropTypes.string.isRequired,
+    category: PropTypes.string.isRequired,
+    rule_ids: PropTypes.arrayOf(PropTypes.string),
+    disabled: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    category: this.props.category,
+    rule_ids: this.props.rule_ids || [],
+    rules: [],
+  };
+
+  componentDidMount() {
+    api().get('/api/v1/instance').then(res => {
+      this.setState({
+        rules: res.data.rules,
+      });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  _save = () => {
+    const { id, disabled } = this.props;
+    const { category, rule_ids } = this.state;
+
+    if (disabled) {
+      return;
+    }
+
+    api().put(`/api/v1/admin/reports/${id}`, {
+      category,
+      rule_ids,
+    }).catch(err => {
+      console.error(err);
+    });
+  };
+
+  handleSelect = id => {
+    this.setState({ category: id }, () => this._save());
+  };
+
+  handleToggle = id => {
+    const { rule_ids } = this.state;
+
+    if (rule_ids.includes(id)) {
+      this.setState({ rule_ids: rule_ids.filter(x => x !== id ) }, () => this._save());
+    } else {
+      this.setState({ rule_ids: [...rule_ids, id] }, () => this._save());
+    }
+  };
+
+  render () {
+    const { disabled, intl } = this.props;
+    const { rules, category, rule_ids } = this.state;
+
+    return (
+      <div className='report-reason-selector'>
+        <Category id='other' text={intl.formatMessage(messages.other)} selected={category === 'other'} onSelect={this.handleSelect} disabled={disabled} />
+        <Category id='legal' text={intl.formatMessage(messages.legal)} selected={category === 'legal'} onSelect={this.handleSelect} disabled={disabled} />
+        <Category id='spam' text={intl.formatMessage(messages.spam)} selected={category === 'spam'} onSelect={this.handleSelect} disabled={disabled} />
+        <Category id='violation' text={intl.formatMessage(messages.violation)} selected={category === 'violation'} onSelect={this.handleSelect} disabled={disabled}>
+          {rules.map(rule => <Rule key={rule.id} id={rule.id} text={rule.text} selected={rule_ids.includes(rule.id)} onToggle={this.handleToggle} disabled={disabled} />)}
+        </Category>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ReportReasonSelector);
diff --git a/app/javascript/flavours/blobfox/components/admin/Retention.jsx b/app/javascript/flavours/blobfox/components/admin/Retention.jsx
new file mode 100644
index 00000000000000..3c1bec0fdd2c29
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/admin/Retention.jsx
@@ -0,0 +1,155 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, FormattedNumber, FormattedDate } from 'react-intl';
+
+import classNames from 'classnames';
+
+import api from 'flavours/blobfox/api';
+import { roundTo10 } from 'flavours/blobfox/utils/numbers';
+
+const dateForCohort = cohort => {
+  const timeZone = 'UTC';
+  switch(cohort.frequency) {
+  case 'day':
+    return <FormattedDate value={cohort.period} month='long' day='2-digit' timeZone={timeZone} />;
+  default:
+    return <FormattedDate value={cohort.period} month='long' year='numeric' timeZone={timeZone} />;
+  }
+};
+
+export default class Retention extends PureComponent {
+
+  static propTypes = {
+    start_at: PropTypes.string,
+    end_at: PropTypes.string,
+    frequency: PropTypes.string,
+  };
+
+  state = {
+    loading: true,
+    data: null,
+  };
+
+  componentDidMount () {
+    const { start_at, end_at, frequency } = this.props;
+
+    api().post('/api/v1/admin/retention', { start_at, end_at, frequency }).then(res => {
+      this.setState({
+        loading: false,
+        data: res.data,
+      });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  render () {
+    const { loading, data } = this.state;
+    const { frequency } = this.props;
+
+    let content;
+
+    if (loading) {
+      content = <FormattedMessage id='loading_indicator.label' defaultMessage='Loading...' />;
+    } else {
+      content = (
+        <table className='retention__table'>
+          <thead>
+            <tr>
+              <th>
+                <div className='retention__table__date retention__table__label'>
+                  <FormattedMessage id='admin.dashboard.retention.cohort' defaultMessage='Sign-up month' />
+                </div>
+              </th>
+
+              <th>
+                <div className='retention__table__number retention__table__label'>
+                  <FormattedMessage id='admin.dashboard.retention.cohort_size' defaultMessage='New users' />
+                </div>
+              </th>
+
+              {data[0].data.slice(1).map((retention, i) => (
+                <th key={retention.date}>
+                  <div className='retention__table__number retention__table__label'>
+                    {i + 1}
+                  </div>
+                </th>
+              ))}
+            </tr>
+
+            <tr>
+              <td>
+                <div className='retention__table__date retention__table__average'>
+                  <FormattedMessage id='admin.dashboard.retention.average' defaultMessage='Average' />
+                </div>
+              </td>
+
+              <td>
+                <div className='retention__table__size'>
+                  <FormattedNumber value={data.reduce((sum, cohort, i) => sum + ((cohort.data[0].value * 1) - sum) / (i + 1), 0)} maximumFractionDigits={0} />
+                </div>
+              </td>
+
+              {data[0].data.slice(1).map((retention, i) => {
+                const average = data.reduce((sum, cohort, k) => cohort.data[i + 1] ? sum + (cohort.data[i + 1].rate - sum)/(k + 1) : sum, 0);
+
+                return (
+                  <td key={retention.date}>
+                    <div className={classNames('retention__table__box', 'retention__table__average', `retention__table__box--${roundTo10(average * 100)}`)}>
+                      <FormattedNumber value={average} style='percent' />
+                    </div>
+                  </td>
+                );
+              })}
+            </tr>
+          </thead>
+
+          <tbody>
+            {data.slice(0, -1).map(cohort => (
+              <tr key={cohort.period}>
+                <td>
+                  <div className='retention__table__date'>
+                    {dateForCohort(cohort)}
+                  </div>
+                </td>
+
+                <td>
+                  <div className='retention__table__size'>
+                    <FormattedNumber value={cohort.data[0].value} />
+                  </div>
+                </td>
+
+                {cohort.data.slice(1).map(retention => (
+                  <td key={retention.date}>
+                    <div className={classNames('retention__table__box', `retention__table__box--${roundTo10(retention.rate * 100)}`)}>
+                      <FormattedNumber value={retention.rate} style='percent' />
+                    </div>
+                  </td>
+                ))}
+              </tr>
+            ))}
+          </tbody>
+        </table>
+      );
+    }
+
+    let title = null;
+    switch(frequency) {
+    case 'day':
+      title = <FormattedMessage id='admin.dashboard.daily_retention' defaultMessage='User retention rate by day after sign-up' />;
+      break;
+    default:
+      title = <FormattedMessage id='admin.dashboard.monthly_retention' defaultMessage='User retention rate by month after sign-up' />;
+    }
+
+    return (
+      <div className='retention'>
+        <h4>{title}</h4>
+
+        {content}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/admin/Trends.jsx b/app/javascript/flavours/blobfox/components/admin/Trends.jsx
new file mode 100644
index 00000000000000..6dad32390d8017
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/admin/Trends.jsx
@@ -0,0 +1,76 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import api from 'flavours/blobfox/api';
+import Hashtag from 'flavours/blobfox/components/hashtag';
+
+export default class Trends extends PureComponent {
+
+  static propTypes = {
+    limit: PropTypes.number.isRequired,
+  };
+
+  state = {
+    loading: true,
+    data: null,
+  };
+
+  componentDidMount () {
+    const { limit } = this.props;
+
+    api().get('/api/v1/admin/trends/tags', { params: { limit } }).then(res => {
+      this.setState({
+        loading: false,
+        data: res.data,
+      });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  render () {
+    const { limit } = this.props;
+    const { loading, data } = this.state;
+
+    let content;
+
+    if (loading) {
+      content = (
+        <div>
+          {Array.from(Array(limit)).map((_, i) => (
+            <Hashtag key={i} />
+          ))}
+        </div>
+      );
+    } else {
+      content = (
+        <div>
+          {data.map(hashtag => (
+            <Hashtag
+              key={hashtag.name}
+              name={hashtag.name}
+              href={hashtag.id === undefined ? undefined : `/admin/tags/${hashtag.id}`}
+              people={hashtag.history[0].accounts * 1 + hashtag.history[1].accounts * 1}
+              uses={hashtag.history[0].uses * 1 + hashtag.history[1].uses * 1}
+              history={hashtag.history.reverse().map(day => day.uses)}
+              className={classNames(hashtag.requires_review && 'trends__item--requires-review', !hashtag.trendable && !hashtag.requires_review && 'trends__item--disabled')}
+            />
+          ))}
+        </div>
+      );
+    }
+
+    return (
+      <div className='trends trends--compact'>
+        <h4><FormattedMessage id='trends.trending_now' defaultMessage='Trending now' /></h4>
+
+        {content}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/animated_number.tsx b/app/javascript/flavours/blobfox/components/animated_number.tsx
new file mode 100644
index 00000000000000..05a7e01898411e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/animated_number.tsx
@@ -0,0 +1,81 @@
+import { useCallback, useState } from 'react';
+
+import { TransitionMotion, spring } from 'react-motion';
+
+import { reduceMotion } from '../initial_state';
+
+import { ShortNumber } from './short_number';
+
+const obfuscatedCount = (count: number) => {
+  if (count < 0) {
+    return 0;
+  } else if (count <= 1) {
+    return count;
+  } else {
+    return '1+';
+  }
+};
+
+interface Props {
+  value: number;
+  obfuscate?: boolean;
+}
+export const AnimatedNumber: React.FC<Props> = ({ value, obfuscate }) => {
+  const [previousValue, setPreviousValue] = useState(value);
+  const [direction, setDirection] = useState<1 | -1>(1);
+
+  if (previousValue !== value) {
+    setPreviousValue(value);
+    setDirection(value > previousValue ? 1 : -1);
+  }
+
+  const willEnter = useCallback(() => ({ y: -1 * direction }), [direction]);
+  const willLeave = useCallback(
+    () => ({ y: spring(1 * direction, { damping: 35, stiffness: 400 }) }),
+    [direction],
+  );
+
+  if (reduceMotion) {
+    return obfuscate ? (
+      <>{obfuscatedCount(value)}</>
+    ) : (
+      <ShortNumber value={value} />
+    );
+  }
+
+  const styles = [
+    {
+      key: `${value}`,
+      data: value,
+      style: { y: spring(0, { damping: 35, stiffness: 400 }) },
+    },
+  ];
+
+  return (
+    <TransitionMotion
+      styles={styles}
+      willEnter={willEnter}
+      willLeave={willLeave}
+    >
+      {(items) => (
+        <span className='animated-number'>
+          {items.map(({ key, data, style }) => (
+            <span
+              key={key}
+              style={{
+                position: direction * style.y > 0 ? 'absolute' : 'static',
+                transform: `translateY(${style.y * 100}%)`,
+              }}
+            >
+              {obfuscate ? (
+                obfuscatedCount(data as number)
+              ) : (
+                <ShortNumber value={data as number} />
+              )}
+            </span>
+          ))}
+        </span>
+      )}
+    </TransitionMotion>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/attachment_list.jsx b/app/javascript/flavours/blobfox/components/attachment_list.jsx
new file mode 100644
index 00000000000000..e54584f766e190
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/attachment_list.jsx
@@ -0,0 +1,51 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const filename = url => url.split('/').pop().split('#')[0].split('?')[0];
+
+export default class AttachmentList extends ImmutablePureComponent {
+
+  static propTypes = {
+    media: ImmutablePropTypes.list.isRequired,
+    compact: PropTypes.bool,
+  };
+
+  render () {
+    const { media, compact } = this.props;
+
+    return (
+      <div className={classNames('attachment-list', { compact })}>
+        {!compact && (
+          <div className='attachment-list__icon'>
+            <Icon id='link' />
+          </div>
+        )}
+
+        <ul className='attachment-list__list'>
+          {media.map(attachment => {
+            const displayUrl = attachment.get('remote_url') || attachment.get('url');
+
+            return (
+              <li key={attachment.get('id')}>
+                <a href={displayUrl} target='_blank' rel='noopener noreferrer'>
+                  {compact && <Icon id='link' />}
+                  {compact && ' ' }
+                  {displayUrl ? filename(displayUrl) : <FormattedMessage id='attachments_list.unprocessed' defaultMessage='(unprocessed)' />}
+                </a>
+              </li>
+            );
+          })}
+        </ul>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/autosuggest_emoji.jsx b/app/javascript/flavours/blobfox/components/autosuggest_emoji.jsx
new file mode 100644
index 00000000000000..1d56f9139c1387
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/autosuggest_emoji.jsx
@@ -0,0 +1,43 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { assetHost } from 'flavours/blobfox/utils/config';
+
+import { unicodeMapping } from '../features/emoji/emoji_unicode_mapping_light';
+
+export default class AutosuggestEmoji extends PureComponent {
+
+  static propTypes = {
+    emoji: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { emoji } = this.props;
+    let url;
+
+    if (emoji.custom) {
+      url = emoji.imageUrl;
+    } else {
+      const mapping = unicodeMapping[emoji.native] || unicodeMapping[emoji.native.replace(/\uFE0F$/, '')];
+
+      if (!mapping) {
+        return null;
+      }
+
+      url = `${assetHost}/emoji/${mapping.filename}.svg`;
+    }
+
+    return (
+      <div className='autosuggest-emoji'>
+        <img
+          className='emojione'
+          src={url}
+          alt={emoji.native || emoji.colons}
+        />
+
+        {emoji.colons}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/autosuggest_hashtag.tsx b/app/javascript/flavours/blobfox/components/autosuggest_hashtag.tsx
new file mode 100644
index 00000000000000..f45aaf1714cd45
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/autosuggest_hashtag.tsx
@@ -0,0 +1,42 @@
+import { FormattedMessage } from 'react-intl';
+
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+
+interface Props {
+  tag: {
+    name: string;
+    url?: string;
+    history?: {
+      uses: number;
+      accounts: string;
+      day: string;
+    }[];
+    following?: boolean;
+    type: 'hashtag';
+  };
+}
+
+export const AutosuggestHashtag: React.FC<Props> = ({ tag }) => {
+  const weeklyUses = tag.history && (
+    <ShortNumber
+      value={tag.history.reduce((total, day) => total + day.uses * 1, 0)}
+    />
+  );
+
+  return (
+    <div className='autosuggest-hashtag'>
+      <div className='autosuggest-hashtag__name'>
+        #<strong>{tag.name}</strong>
+      </div>
+      {tag.history !== undefined && (
+        <div className='autosuggest-hashtag__uses'>
+          <FormattedMessage
+            id='autosuggest_hashtag.per_week'
+            defaultMessage='{count} per week'
+            values={{ count: weeklyUses }}
+          />
+        </div>
+      )}
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/autosuggest_input.jsx b/app/javascript/flavours/blobfox/components/autosuggest_input.jsx
new file mode 100644
index 00000000000000..6d2474b4426bb0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/autosuggest_input.jsx
@@ -0,0 +1,230 @@
+import PropTypes from 'prop-types';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import AutosuggestAccountContainer from '../features/compose/containers/autosuggest_account_container';
+
+import AutosuggestEmoji from './autosuggest_emoji';
+import { AutosuggestHashtag } from './autosuggest_hashtag';
+
+const textAtCursorMatchesToken = (str, caretPosition, searchTokens) => {
+  let word;
+
+  let left  = str.slice(0, caretPosition).search(/[^\s\u200B]+$/);
+  let right = str.slice(caretPosition).search(/[\s\u200B]/);
+
+  if (right < 0) {
+    word = str.slice(left);
+  } else {
+    word = str.slice(left, right + caretPosition);
+  }
+
+  if (!word || word.trim().length < 3 || searchTokens.indexOf(word[0]) === -1) {
+    return [null, null];
+  }
+
+  word = word.trim().toLowerCase();
+
+  if (word.length > 0) {
+    return [left, word];
+  } else {
+    return [null, null];
+  }
+};
+
+export default class AutosuggestInput extends ImmutablePureComponent {
+
+  static propTypes = {
+    value: PropTypes.string,
+    suggestions: ImmutablePropTypes.list,
+    disabled: PropTypes.bool,
+    placeholder: PropTypes.string,
+    onSuggestionSelected: PropTypes.func.isRequired,
+    onSuggestionsClearRequested: PropTypes.func.isRequired,
+    onSuggestionsFetchRequested: PropTypes.func.isRequired,
+    onChange: PropTypes.func.isRequired,
+    onKeyUp: PropTypes.func,
+    onKeyDown: PropTypes.func,
+    autoFocus: PropTypes.bool,
+    className: PropTypes.string,
+    id: PropTypes.string,
+    searchTokens: PropTypes.arrayOf(PropTypes.string),
+    maxLength: PropTypes.number,
+    lang: PropTypes.string,
+    spellCheck: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    autoFocus: true,
+    searchTokens: ['@', ':', '#'],
+  };
+
+  state = {
+    suggestionsHidden: true,
+    focused: false,
+    selectedSuggestion: 0,
+    lastToken: null,
+    tokenStart: 0,
+  };
+
+  onChange = (e) => {
+    const [ tokenStart, token ] = textAtCursorMatchesToken(e.target.value, e.target.selectionStart, this.props.searchTokens);
+
+    if (token !== null && this.state.lastToken !== token) {
+      this.setState({ lastToken: token, selectedSuggestion: 0, tokenStart });
+      this.props.onSuggestionsFetchRequested(token);
+    } else if (token === null) {
+      this.setState({ lastToken: null });
+      this.props.onSuggestionsClearRequested();
+    }
+
+    this.props.onChange(e);
+  };
+
+  onKeyDown = (e) => {
+    const { suggestions, disabled } = this.props;
+    const { selectedSuggestion, suggestionsHidden } = this.state;
+
+    if (disabled) {
+      e.preventDefault();
+      return;
+    }
+
+    if (e.which === 229 || e.isComposing) {
+      // Ignore key events during text composition
+      // e.key may be a name of the physical key even in this case (e.x. Safari / Chrome on Mac)
+      return;
+    }
+
+    switch(e.key) {
+    case 'Escape':
+      if (suggestions.size === 0 || suggestionsHidden) {
+        document.querySelector('.ui').parentElement.focus();
+      } else {
+        e.preventDefault();
+        this.setState({ suggestionsHidden: true });
+      }
+
+      break;
+    case 'ArrowDown':
+      if (suggestions.size > 0 && !suggestionsHidden) {
+        e.preventDefault();
+        this.setState({ selectedSuggestion: Math.min(selectedSuggestion + 1, suggestions.size - 1) });
+      }
+
+      break;
+    case 'ArrowUp':
+      if (suggestions.size > 0 && !suggestionsHidden) {
+        e.preventDefault();
+        this.setState({ selectedSuggestion: Math.max(selectedSuggestion - 1, 0) });
+      }
+
+      break;
+    case 'Enter':
+    case 'Tab':
+      // Select suggestion
+      if (this.state.lastToken !== null && suggestions.size > 0 && !suggestionsHidden) {
+        e.preventDefault();
+        e.stopPropagation();
+        this.props.onSuggestionSelected(this.state.tokenStart, this.state.lastToken, suggestions.get(selectedSuggestion));
+      }
+
+      break;
+    }
+
+    if (e.defaultPrevented || !this.props.onKeyDown) {
+      return;
+    }
+
+    this.props.onKeyDown(e);
+  };
+
+  onBlur = () => {
+    this.setState({ suggestionsHidden: true, focused: false });
+  };
+
+  onFocus = () => {
+    this.setState({ focused: true });
+  };
+
+  onSuggestionClick = (e) => {
+    const suggestion = this.props.suggestions.get(e.currentTarget.getAttribute('data-index'));
+    e.preventDefault();
+    this.props.onSuggestionSelected(this.state.tokenStart, this.state.lastToken, suggestion);
+    this.input.focus();
+  };
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (nextProps.suggestions !== this.props.suggestions && nextProps.suggestions.size > 0 && this.state.suggestionsHidden && this.state.focused) {
+      this.setState({ suggestionsHidden: false });
+    }
+  }
+
+  setInput = (c) => {
+    this.input = c;
+  };
+
+  renderSuggestion = (suggestion, i) => {
+    const { selectedSuggestion } = this.state;
+    let inner, key;
+
+    if (suggestion.type === 'emoji') {
+      inner = <AutosuggestEmoji emoji={suggestion} />;
+      key   = suggestion.id;
+    } else if (suggestion.type ==='hashtag') {
+      inner = <AutosuggestHashtag tag={suggestion} />;
+      key   = suggestion.name;
+    } else if (suggestion.type === 'account') {
+      inner = <AutosuggestAccountContainer id={suggestion.id} />;
+      key   = suggestion.id;
+    }
+
+    return (
+      <div role='button' tabIndex={0} key={key} data-index={i} className={classNames('autosuggest-textarea__suggestions__item', { selected: i === selectedSuggestion })} onMouseDown={this.onSuggestionClick}>
+        {inner}
+      </div>
+    );
+  };
+
+  render () {
+    const { value, suggestions, disabled, placeholder, onKeyUp, autoFocus, className, id, maxLength, lang, spellCheck } = this.props;
+    const { suggestionsHidden } = this.state;
+
+    return (
+      <div className='autosuggest-input'>
+        <label>
+          <span style={{ display: 'none' }}>{placeholder}</span>
+
+          <input
+            type='text'
+            ref={this.setInput}
+            disabled={disabled}
+            placeholder={placeholder}
+            autoFocus={autoFocus}
+            value={value}
+            onChange={this.onChange}
+            onKeyDown={this.onKeyDown}
+            onKeyUp={onKeyUp}
+            onFocus={this.onFocus}
+            onBlur={this.onBlur}
+            dir='auto'
+            aria-autocomplete='list'
+            id={id}
+            className={className}
+            maxLength={maxLength}
+            lang={lang}
+            spellCheck={spellCheck}
+          />
+        </label>
+
+        <div className={`autosuggest-textarea__suggestions ${suggestionsHidden || suggestions.isEmpty() ? '' : 'autosuggest-textarea__suggestions--visible'}`}>
+          {suggestions.map(this.renderSuggestion)}
+        </div>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/autosuggest_textarea.jsx b/app/javascript/flavours/blobfox/components/autosuggest_textarea.jsx
new file mode 100644
index 00000000000000..28384075c3c16e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/autosuggest_textarea.jsx
@@ -0,0 +1,240 @@
+import PropTypes from 'prop-types';
+import { useCallback, useRef, useState, useEffect, forwardRef } from 'react';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import Textarea from 'react-textarea-autosize';
+
+import AutosuggestAccountContainer from '../features/compose/containers/autosuggest_account_container';
+
+import AutosuggestEmoji from './autosuggest_emoji';
+import { AutosuggestHashtag } from './autosuggest_hashtag';
+
+const textAtCursorMatchesToken = (str, caretPosition) => {
+  let word;
+
+  let left  = str.slice(0, caretPosition).search(/[^\s\u200B]+$/);
+  let right = str.slice(caretPosition).search(/[\s\u200B]/);
+
+  if (right < 0) {
+    word = str.slice(left);
+  } else {
+    word = str.slice(left, right + caretPosition);
+  }
+
+  if (!word || word.trim().length < 3 || ['@', ':', '#'].indexOf(word[0]) === -1) {
+    return [null, null];
+  }
+
+  word = word.trim().toLowerCase();
+
+  if (word.length > 0) {
+    return [left, word];
+  } else {
+    return [null, null];
+  }
+};
+
+const AutosuggestTextarea = forwardRef(({
+  value,
+  suggestions,
+  disabled,
+  placeholder,
+  onSuggestionSelected,
+  onSuggestionsClearRequested,
+  onSuggestionsFetchRequested,
+  onChange,
+  onKeyUp,
+  onKeyDown,
+  onPaste,
+  onFocus,
+  autoFocus = true,
+  lang,
+  children,
+}, textareaRef) => {
+
+  const [suggestionsHidden, setSuggestionsHidden] = useState(true);
+  const [selectedSuggestion, setSelectedSuggestion] = useState(0);
+  const lastTokenRef = useRef(null);
+  const tokenStartRef = useRef(0);
+
+  const handleChange = useCallback((e) => {
+    const [ tokenStart, token ] = textAtCursorMatchesToken(e.target.value, e.target.selectionStart);
+
+    if (token !== null && lastTokenRef.current !== token) {
+      tokenStartRef.current = tokenStart;
+      lastTokenRef.current = token;
+      setSelectedSuggestion(0);
+      onSuggestionsFetchRequested(token);
+    } else if (token === null) {
+      lastTokenRef.current = null;
+      onSuggestionsClearRequested();
+    }
+
+    onChange(e);
+  }, [onSuggestionsFetchRequested, onSuggestionsClearRequested, onChange, setSelectedSuggestion]);
+
+  const handleKeyDown = useCallback((e) => {
+    if (disabled) {
+      e.preventDefault();
+      return;
+    }
+
+    if (e.which === 229 || e.isComposing) {
+      // Ignore key events during text composition
+      // e.key may be a name of the physical key even in this case (e.x. Safari / Chrome on Mac)
+      return;
+    }
+
+    switch(e.key) {
+    case 'Escape':
+      if (suggestions.size === 0 || suggestionsHidden) {
+        document.querySelector('.ui').parentElement.focus();
+      } else {
+        e.preventDefault();
+        setSuggestionsHidden(true);
+      }
+
+      break;
+    case 'ArrowDown':
+      if (suggestions.size > 0 && !suggestionsHidden) {
+        e.preventDefault();
+        setSelectedSuggestion(Math.min(selectedSuggestion + 1, suggestions.size - 1));
+      }
+
+      break;
+    case 'ArrowUp':
+      if (suggestions.size > 0 && !suggestionsHidden) {
+        e.preventDefault();
+        setSelectedSuggestion(Math.max(selectedSuggestion - 1, 0));
+      }
+
+      break;
+    case 'Enter':
+    case 'Tab':
+      // Select suggestion
+      if (lastTokenRef.current !== null && suggestions.size > 0 && !suggestionsHidden) {
+        e.preventDefault();
+        e.stopPropagation();
+        onSuggestionSelected(tokenStartRef.current, lastTokenRef.current, suggestions.get(selectedSuggestion));
+      }
+
+      break;
+    }
+
+    if (e.defaultPrevented || !onKeyDown) {
+      return;
+    }
+
+    onKeyDown(e);
+  }, [disabled, suggestions, suggestionsHidden, selectedSuggestion, setSelectedSuggestion, setSuggestionsHidden, onSuggestionSelected, onKeyDown]);
+
+  const handleBlur = useCallback(() => {
+    setSuggestionsHidden(true);
+  }, [setSuggestionsHidden]);
+
+  const handleFocus = useCallback((e) => {
+    if (onFocus) {
+      onFocus(e);
+    }
+  }, [onFocus]);
+
+  const handleSuggestionClick = useCallback((e) => {
+    const suggestion = suggestions.get(e.currentTarget.getAttribute('data-index'));
+    e.preventDefault();
+    onSuggestionSelected(tokenStartRef.current, lastTokenRef.current, suggestion);
+    textareaRef.current?.focus();
+  }, [suggestions, onSuggestionSelected, textareaRef]);
+
+  const handlePaste = useCallback((e) => {
+    if (e.clipboardData && e.clipboardData.files.length === 1) {
+      onPaste(e.clipboardData.files);
+      e.preventDefault();
+    }
+  }, [onPaste]);
+
+  // Show the suggestions again whenever they change and the textarea is focused
+  useEffect(() => {
+    if (suggestions.size > 0 && textareaRef.current === document.activeElement) {
+      setSuggestionsHidden(false);
+    }
+  }, [suggestions, textareaRef, setSuggestionsHidden]);
+
+  const renderSuggestion = (suggestion, i) => {
+    let inner, key;
+
+    if (suggestion.type === 'emoji') {
+      inner = <AutosuggestEmoji emoji={suggestion} />;
+      key   = suggestion.id;
+    } else if (suggestion.type === 'hashtag') {
+      inner = <AutosuggestHashtag tag={suggestion} />;
+      key   = suggestion.name;
+    } else if (suggestion.type === 'account') {
+      inner = <AutosuggestAccountContainer id={suggestion.id} />;
+      key   = suggestion.id;
+    }
+
+    return (
+      <div role='button' tabIndex={0} key={key} data-index={i} className={classNames('autosuggest-textarea__suggestions__item', { selected: i === selectedSuggestion })} onMouseDown={handleSuggestionClick}>
+        {inner}
+      </div>
+    );
+  };
+
+  return [
+    <div className='compose-form__autosuggest-wrapper' key='autosuggest-wrapper'>
+      <div className='autosuggest-textarea'>
+        <label>
+          <span style={{ display: 'none' }}>{placeholder}</span>
+
+          <Textarea
+            ref={textareaRef}
+            className='autosuggest-textarea__textarea'
+            disabled={disabled}
+            placeholder={placeholder}
+            autoFocus={autoFocus}
+            value={value}
+            onChange={handleChange}
+            onKeyDown={handleKeyDown}
+            onKeyUp={onKeyUp}
+            onFocus={handleFocus}
+            onBlur={handleBlur}
+            onPaste={handlePaste}
+            dir='auto'
+            aria-autocomplete='list'
+            lang={lang}
+          />
+        </label>
+      </div>
+      {children}
+    </div>,
+
+    <div className='autosuggest-textarea__suggestions-wrapper' key='suggestions-wrapper'>
+      <div className={`autosuggest-textarea__suggestions ${suggestionsHidden || suggestions.isEmpty() ? '' : 'autosuggest-textarea__suggestions--visible'}`}>
+        {suggestions.map(renderSuggestion)}
+      </div>
+    </div>,
+  ];
+});
+
+AutosuggestTextarea.propTypes = {
+  value: PropTypes.string,
+  suggestions: ImmutablePropTypes.list,
+  disabled: PropTypes.bool,
+  placeholder: PropTypes.string,
+  onSuggestionSelected: PropTypes.func.isRequired,
+  onSuggestionsClearRequested: PropTypes.func.isRequired,
+  onSuggestionsFetchRequested: PropTypes.func.isRequired,
+  onChange: PropTypes.func.isRequired,
+  onKeyUp: PropTypes.func,
+  onKeyDown: PropTypes.func,
+  onPaste: PropTypes.func.isRequired,
+  onFocus:PropTypes.func,
+  children: PropTypes.node,
+  autoFocus: PropTypes.bool,
+  lang: PropTypes.string,
+};
+
+export default AutosuggestTextarea;
diff --git a/app/javascript/flavours/blobfox/components/avatar.tsx b/app/javascript/flavours/blobfox/components/avatar.tsx
new file mode 100644
index 00000000000000..5a4b595fcfa6d2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/avatar.tsx
@@ -0,0 +1,49 @@
+import classNames from 'classnames';
+
+import type { Account } from 'flavours/blobfox/models/account';
+
+import { useHovering } from '../hooks/useHovering';
+import { autoPlayGif } from '../initial_state';
+
+interface Props {
+  account: Account | undefined; // FIXME: remove `undefined` once we know for sure its always there
+  size: number;
+  style?: React.CSSProperties;
+  inline?: boolean;
+  animate?: boolean;
+}
+
+export const Avatar: React.FC<Props> = ({
+  account,
+  animate = autoPlayGif,
+  size = 20,
+  inline = false,
+  style: styleFromParent,
+}) => {
+  const { hovering, handleMouseEnter, handleMouseLeave } = useHovering(animate);
+
+  const style = {
+    ...styleFromParent,
+    width: `${size}px`,
+    height: `${size}px`,
+  };
+
+  const src =
+    hovering || animate
+      ? account?.get('avatar')
+      : account?.get('avatar_static');
+
+  return (
+    <div
+      className={classNames('account__avatar', {
+        'account__avatar-inline': inline,
+      })}
+      onMouseEnter={handleMouseEnter}
+      onMouseLeave={handleMouseLeave}
+      style={style}
+      data-avatar-of={account && `@${account.get('acct')}`}
+    >
+      {src && <img src={src} alt={account?.get('acct')} />}
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/avatar_composite.jsx b/app/javascript/flavours/blobfox/components/avatar_composite.jsx
new file mode 100644
index 00000000000000..c736f1dd53200d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/avatar_composite.jsx
@@ -0,0 +1,106 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { autoPlayGif } from '../initial_state';
+
+import { Avatar } from './avatar';
+
+export default class AvatarComposite extends PureComponent {
+
+  static propTypes = {
+    accounts: ImmutablePropTypes.list.isRequired,
+    animate: PropTypes.bool,
+    size: PropTypes.number.isRequired,
+  };
+
+  static defaultProps = {
+    animate: autoPlayGif,
+  };
+
+  renderItem (account, size, index) {
+    const { animate } = this.props;
+
+    let width  = 50;
+    let height = 100;
+    let top    = 'auto';
+    let left   = 'auto';
+    let bottom = 'auto';
+    let right  = 'auto';
+
+    if (size === 1) {
+      width = 100;
+    }
+
+    if (size === 4 || (size === 3 && index > 0)) {
+      height = 50;
+    }
+
+    if (size === 2) {
+      if (index === 0) {
+        right = '1px';
+      } else {
+        left = '1px';
+      }
+    } else if (size === 3) {
+      if (index === 0) {
+        right = '1px';
+      } else if (index > 0) {
+        left = '1px';
+      }
+
+      if (index === 1) {
+        bottom = '1px';
+      } else if (index > 1) {
+        top = '1px';
+      }
+    } else if (size === 4) {
+      if (index === 0 || index === 2) {
+        right = '1px';
+      }
+
+      if (index === 1 || index === 3) {
+        left = '1px';
+      }
+
+      if (index < 2) {
+        bottom = '1px';
+      } else {
+        top = '1px';
+      }
+    }
+
+    const style = {
+      left: left,
+      top: top,
+      right: right,
+      bottom: bottom,
+      width: `${width}%`,
+      height: `${height}%`,
+    };
+
+    return (
+      <div key={account.get('id')} style={style}>
+        <Avatar account={account} animate={animate} />
+      </div>
+    );
+  }
+
+  render() {
+    const { accounts, size } = this.props;
+
+    return (
+      <div className='account__avatar-composite' style={{ width: `${size}px`, height: `${size}px` }}>
+        {accounts.take(4).map((account, i) => this.renderItem(account, Math.min(accounts.size, 4), i))}
+
+        {accounts.size > 4 && (
+          <span className='account__avatar-composite__label'>
+            +{accounts.size - 4}
+          </span>
+        )}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/avatar_overlay.tsx b/app/javascript/flavours/blobfox/components/avatar_overlay.tsx
new file mode 100644
index 00000000000000..d2d26aaab40e22
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/avatar_overlay.tsx
@@ -0,0 +1,57 @@
+import type { Account } from 'flavours/blobfox/models/account';
+
+import { useHovering } from '../hooks/useHovering';
+import { autoPlayGif } from '../initial_state';
+
+interface Props {
+  account: Account | undefined; // FIXME: remove `undefined` once we know for sure its always there
+  friend: Account | undefined; // FIXME: remove `undefined` once we know for sure its always there
+  size?: number;
+  baseSize?: number;
+  overlaySize?: number;
+}
+
+export const AvatarOverlay: React.FC<Props> = ({
+  account,
+  friend,
+  size = 46,
+  baseSize = 36,
+  overlaySize = 24,
+}) => {
+  const { hovering, handleMouseEnter, handleMouseLeave } =
+    useHovering(autoPlayGif);
+  const accountSrc = hovering
+    ? account?.get('avatar')
+    : account?.get('avatar_static');
+  const friendSrc = hovering
+    ? friend?.get('avatar')
+    : friend?.get('avatar_static');
+
+  return (
+    <div
+      className='account__avatar-overlay'
+      style={{ width: size, height: size }}
+      onMouseEnter={handleMouseEnter}
+      onMouseLeave={handleMouseLeave}
+    >
+      <div className='account__avatar-overlay-base'>
+        <div
+          className='account__avatar'
+          style={{ width: `${baseSize}px`, height: `${baseSize}px` }}
+          data-avatar-of={`@${account?.get('acct')}`}
+        >
+          {accountSrc && <img src={accountSrc} alt={account?.get('acct')} />}
+        </div>
+      </div>
+      <div className='account__avatar-overlay-overlay'>
+        <div
+          className='account__avatar'
+          style={{ width: `${overlaySize}px`, height: `${overlaySize}px` }}
+          data-avatar-of={`@${friend?.get('acct')}`}
+        >
+          {friendSrc && <img src={friendSrc} alt={friend?.get('acct')} />}
+        </div>
+      </div>
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/blurhash.tsx b/app/javascript/flavours/blobfox/components/blurhash.tsx
new file mode 100644
index 00000000000000..8e2a8af23e5937
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/blurhash.tsx
@@ -0,0 +1,48 @@
+import { memo, useRef, useEffect } from 'react';
+
+import { decode } from 'blurhash';
+
+interface Props extends React.HTMLAttributes<HTMLCanvasElement> {
+  hash: string;
+  width?: number;
+  height?: number;
+  dummy?: boolean; // Whether dummy mode is enabled. If enabled, nothing is rendered and canvas left untouched
+  children?: never;
+}
+const Blurhash: React.FC<Props> = ({
+  hash,
+  width = 32,
+  height = width,
+  dummy = false,
+  ...canvasProps
+}) => {
+  const canvasRef = useRef<HTMLCanvasElement>(null);
+
+  useEffect(() => {
+    // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
+    const canvas = canvasRef.current!;
+
+    // eslint-disable-next-line no-self-assign
+    canvas.width = canvas.width; // resets canvas
+
+    if (dummy || !hash) return;
+
+    try {
+      const pixels = decode(hash, width, height);
+      const ctx = canvas.getContext('2d');
+      const imageData = new ImageData(pixels, width, height);
+
+      ctx?.putImageData(imageData, 0, 0);
+    } catch (err) {
+      console.error('Blurhash decoding failure', { err, hash });
+    }
+  }, [dummy, hash, width, height]);
+
+  return (
+    <canvas {...canvasProps} ref={canvasRef} width={width} height={height} />
+  );
+};
+
+const MemoizedBlurhash = memo(Blurhash);
+
+export { MemoizedBlurhash as Blurhash };
diff --git a/app/javascript/flavours/blobfox/components/button.tsx b/app/javascript/flavours/blobfox/components/button.tsx
new file mode 100644
index 00000000000000..0b6a0f267ebd6e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/button.tsx
@@ -0,0 +1,58 @@
+import { useCallback } from 'react';
+
+import classNames from 'classnames';
+
+interface BaseProps extends React.ButtonHTMLAttributes<HTMLButtonElement> {
+  block?: boolean;
+  secondary?: boolean;
+  text?: JSX.Element;
+}
+
+interface PropsWithChildren extends BaseProps {
+  text?: never;
+}
+
+interface PropsWithText extends BaseProps {
+  text: JSX.Element;
+  children: never;
+}
+
+type Props = PropsWithText | PropsWithChildren;
+
+export const Button: React.FC<Props> = ({
+  text,
+  type = 'button',
+  onClick,
+  disabled,
+  block,
+  secondary,
+  className,
+  title,
+  children,
+  ...props
+}) => {
+  const handleClick = useCallback<React.MouseEventHandler<HTMLButtonElement>>(
+    (e) => {
+      if (!disabled && onClick) {
+        onClick(e);
+      }
+    },
+    [disabled, onClick],
+  );
+
+  return (
+    <button
+      className={classNames('button', className, {
+        'button-secondary': secondary,
+        'button--block': block,
+      })}
+      disabled={disabled}
+      onClick={handleClick}
+      title={title}
+      type={type}
+      {...props}
+    >
+      {text ?? children}
+    </button>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/check.tsx b/app/javascript/flavours/blobfox/components/check.tsx
new file mode 100644
index 00000000000000..901f89fc5b2bfa
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/check.tsx
@@ -0,0 +1,13 @@
+export const Check: React.FC = () => (
+  <svg
+    xmlns='http://www.w3.org/2000/svg'
+    viewBox='0 0 20 20'
+    fill='currentColor'
+  >
+    <path
+      fillRule='evenodd'
+      d='M16.704 4.153a.75.75 0 01.143 1.052l-8 10.5a.75.75 0 01-1.127.075l-4.5-4.5a.75.75 0 011.06-1.06l3.894 3.893 7.48-9.817a.75.75 0 011.05-.143z'
+      clipRule='evenodd'
+    />
+  </svg>
+);
diff --git a/app/javascript/flavours/blobfox/components/circular_progress.tsx b/app/javascript/flavours/blobfox/components/circular_progress.tsx
new file mode 100644
index 00000000000000..850eb93e482cf1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/circular_progress.tsx
@@ -0,0 +1,27 @@
+interface Props {
+  size: number;
+  strokeWidth: number;
+}
+
+export const CircularProgress: React.FC<Props> = ({ size, strokeWidth }) => {
+  const viewBox = `0 0 ${size} ${size}`;
+  const radius = (size - strokeWidth) / 2;
+
+  return (
+    <svg
+      width={size}
+      height={size}
+      viewBox={viewBox}
+      className='circular-progress'
+      role='progressbar'
+    >
+      <circle
+        fill='none'
+        cx={size / 2}
+        cy={size / 2}
+        r={radius}
+        strokeWidth={`${strokeWidth}px`}
+      />
+    </svg>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/column.jsx b/app/javascript/flavours/blobfox/components/column.jsx
new file mode 100644
index 00000000000000..22d6eabed7bd3c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/column.jsx
@@ -0,0 +1,73 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+
+import { scrollTop } from '../scroll';
+
+const listenerOptions = supportsPassiveEvents ? { passive: true } : false;
+
+export default class Column extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node,
+    extraClasses: PropTypes.string,
+    label: PropTypes.string,
+    bindToDocument: PropTypes.bool,
+  };
+
+  scrollTop () {
+    let scrollable = null;
+
+    if (this.props.bindToDocument) {
+      scrollable = document.scrollingElement;
+    } else {
+      scrollable = this.node.querySelector('.scrollable');
+    }
+
+    if (!scrollable) {
+      return;
+    }
+
+    this._interruptScrollAnimation = scrollTop(scrollable);
+  }
+
+  handleWheel = () => {
+    if (typeof this._interruptScrollAnimation !== 'function') {
+      return;
+    }
+
+    this._interruptScrollAnimation();
+  };
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  componentDidMount () {
+    if (this.props.bindToDocument) {
+      document.addEventListener('wheel', this.handleWheel, listenerOptions);
+    } else {
+      this.node.addEventListener('wheel', this.handleWheel, listenerOptions);
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.props.bindToDocument) {
+      document.removeEventListener('wheel', this.handleWheel, listenerOptions);
+    } else {
+      this.node.removeEventListener('wheel', this.handleWheel, listenerOptions);
+    }
+  }
+
+  render () {
+    const { label, children, extraClasses } = this.props;
+
+    return (
+      <div role='region' aria-label={label} className={`column ${extraClasses || ''}`} ref={this.setRef}>
+        {children}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/column_back_button.jsx b/app/javascript/flavours/blobfox/components/column_back_button.jsx
new file mode 100644
index 00000000000000..d51d6be09dba94
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/column_back_button.jsx
@@ -0,0 +1,63 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+import { createPortal } from 'react-dom';
+
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+export class ColumnBackButton extends PureComponent {
+
+  static propTypes = {
+    multiColumn: PropTypes.bool,
+    onClick: PropTypes.func,
+    ...WithRouterPropTypes,
+  };
+
+  handleClick = () => {
+    const { onClick, history } = this.props;
+
+    if (onClick) {
+      onClick();
+    } else if (history.location?.state?.fromMastodon) {
+      history.goBack();
+    } else {
+      history.push('/');
+    }
+  };
+
+  render () {
+    const { multiColumn } = this.props;
+
+    const component = (
+      <button onClick={this.handleClick} className='column-back-button'>
+        <Icon id='chevron-left' className='column-back-button__icon' fixedWidth />
+        <FormattedMessage id='column_back_button.label' defaultMessage='Back' />
+      </button>
+    );
+
+    if (multiColumn) {
+      return component;
+    } else {
+      // The portal container and the component may be rendered to the DOM in
+      // the same React render pass, so the container might not be available at
+      // the time `render()` is called.
+      const container = document.getElementById('tabs-bar__portal');
+      if (container === null) {
+        // The container wasn't available, force a re-render so that the
+        // component can eventually be inserted in the container and not scroll
+        // with the rest of the area.
+        this.forceUpdate();
+        return component;
+      } else {
+        return createPortal(component, container);
+      }
+    }
+  }
+
+}
+
+export default withRouter(ColumnBackButton);
diff --git a/app/javascript/flavours/blobfox/components/column_back_button_slim.jsx b/app/javascript/flavours/blobfox/components/column_back_button_slim.jsx
new file mode 100644
index 00000000000000..82961df5561568
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/column_back_button_slim.jsx
@@ -0,0 +1,40 @@
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+class ColumnBackButtonSlim extends PureComponent {
+
+  static propTypes = {
+    ...WithRouterPropTypes,
+  };
+
+  handleClick = () => {
+    const { location, history } = this.props;
+
+    // Check if there is a previous page in the app to go back to per https://stackoverflow.com/a/70532858/9703201
+    // When upgrading to V6, check `location.key !== 'default'` instead per https://github.com/remix-run/history/blob/main/docs/api-reference.md#location
+    if (location.key) {
+      history.goBack();
+    } else {
+      history.push('/');
+    }
+  };
+
+  render () {
+    return (
+      <div className='column-back-button--slim'>
+        <div role='button' tabIndex={0} onClick={this.handleClick} className='column-back-button column-back-button--slim-button'>
+          <Icon id='chevron-left' className='column-back-button__icon' fixedWidth />
+          <FormattedMessage id='column_back_button.label' defaultMessage='Back' />
+        </div>
+      </div>
+    );
+  }
+}
+
+export default withRouter(ColumnBackButtonSlim);
diff --git a/app/javascript/flavours/blobfox/components/column_header.jsx b/app/javascript/flavours/blobfox/components/column_header.jsx
new file mode 100644
index 00000000000000..dce67ab9950d82
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/column_header.jsx
@@ -0,0 +1,219 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+import { createPortal } from 'react-dom';
+
+import { FormattedMessage, injectIntl, defineMessages } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const messages = defineMessages({
+  show: { id: 'column_header.show_settings', defaultMessage: 'Show settings' },
+  hide: { id: 'column_header.hide_settings', defaultMessage: 'Hide settings' },
+  moveLeft: { id: 'column_header.moveLeft_settings', defaultMessage: 'Move column to the left' },
+  moveRight: { id: 'column_header.moveRight_settings', defaultMessage: 'Move column to the right' },
+});
+
+class ColumnHeader extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    title: PropTypes.node,
+    icon: PropTypes.string,
+    active: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    extraButton: PropTypes.node,
+    showBackButton: PropTypes.bool,
+    children: PropTypes.node,
+    pinned: PropTypes.bool,
+    placeholder: PropTypes.bool,
+    onPin: PropTypes.func,
+    onMove: PropTypes.func,
+    onClick: PropTypes.func,
+    appendContent: PropTypes.node,
+    collapseIssues: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  state = {
+    collapsed: true,
+    animating: false,
+  };
+
+  handleToggleClick = (e) => {
+    e.stopPropagation();
+    this.setState({ collapsed: !this.state.collapsed, animating: true });
+  };
+
+  handleTitleClick = () => {
+    this.props.onClick?.();
+  };
+
+  handleMoveLeft = () => {
+    this.props.onMove(-1);
+  };
+
+  handleMoveRight = () => {
+    this.props.onMove(1);
+  };
+
+  handleBackClick = () => {
+    const { history } = this.props;
+
+    if (history.location?.state?.fromMastodon) {
+      history.goBack();
+    } else {
+      history.push('/');
+    }
+  };
+
+  handleTransitionEnd = () => {
+    this.setState({ animating: false });
+  };
+
+  handlePin = () => {
+    if (!this.props.pinned) {
+      this.props.history.replace('/');
+    }
+
+    this.props.onPin();
+  };
+
+  render () {
+    const { title, icon, active, children, pinned, multiColumn, extraButton, showBackButton, intl: { formatMessage }, placeholder, appendContent, collapseIssues, history } = this.props;
+    const { collapsed, animating } = this.state;
+
+    const wrapperClassName = classNames('column-header__wrapper', {
+      'active': active,
+    });
+
+    const buttonClassName = classNames('column-header', {
+      'active': active,
+    });
+
+    const collapsibleClassName = classNames('column-header__collapsible', {
+      'collapsed': collapsed,
+      'animating': animating,
+    });
+
+    const collapsibleButtonClassName = classNames('column-header__button', {
+      'active': !collapsed,
+    });
+
+    let extraContent, pinButton, moveButtons, backButton, collapseButton;
+
+    if (children) {
+      extraContent = (
+        <div key='extra-content' className='column-header__collapsible__extra'>
+          {children}
+        </div>
+      );
+    }
+
+    if (multiColumn && pinned) {
+      pinButton = <button key='pin-button' className='text-btn column-header__setting-btn' onClick={this.handlePin}><Icon id='times' /> <FormattedMessage id='column_header.unpin' defaultMessage='Unpin' /></button>;
+
+      moveButtons = (
+        <div key='move-buttons' className='column-header__setting-arrows'>
+          <button title={formatMessage(messages.moveLeft)} aria-label={formatMessage(messages.moveLeft)} className='icon-button column-header__setting-btn' onClick={this.handleMoveLeft}><Icon id='chevron-left' /></button>
+          <button title={formatMessage(messages.moveRight)} aria-label={formatMessage(messages.moveRight)} className='icon-button column-header__setting-btn' onClick={this.handleMoveRight}><Icon id='chevron-right' /></button>
+        </div>
+      );
+    } else if (multiColumn && this.props.onPin) {
+      pinButton = <button key='pin-button' className='text-btn column-header__setting-btn' onClick={this.handlePin}><Icon id='plus' /> <FormattedMessage id='column_header.pin' defaultMessage='Pin' /></button>;
+    }
+
+    if (!pinned && ((multiColumn && history.location?.state?.fromMastodon) || showBackButton)) {
+      backButton = (
+        <button onClick={this.handleBackClick} className='column-header__back-button'>
+          <Icon id='chevron-left' className='column-back-button__icon' fixedWidth />
+          <FormattedMessage id='column_back_button.label' defaultMessage='Back' />
+        </button>
+      );
+    }
+
+    const collapsedContent = [
+      extraContent,
+    ];
+
+    if (multiColumn) {
+      collapsedContent.push(pinButton);
+      collapsedContent.push(moveButtons);
+    }
+
+    if (this.context.identity.signedIn && (children || (multiColumn && this.props.onPin))) {
+      collapseButton = (
+        <button
+          className={collapsibleButtonClassName}
+          title={formatMessage(collapsed ? messages.show : messages.hide)}
+          aria-label={formatMessage(collapsed ? messages.show : messages.hide)}
+          onClick={this.handleToggleClick}
+        >
+          <i className='icon-with-badge'>
+            <Icon id='sliders' />
+            {collapseIssues && <i className='icon-with-badge__issue-badge' />}
+          </i>
+        </button>
+      );
+    }
+
+    const hasTitle = icon && title;
+
+    const component = (
+      <div className={wrapperClassName}>
+        <h1 className={buttonClassName}>
+          {hasTitle && (
+            <button onClick={this.handleTitleClick}>
+              <Icon id={icon} fixedWidth className='column-header__icon' />
+              {title}
+            </button>
+          )}
+
+          {!hasTitle && backButton}
+
+          <div className='column-header__buttons'>
+            {hasTitle && backButton}
+            {extraButton}
+            {collapseButton}
+          </div>
+        </h1>
+
+        <div className={collapsibleClassName} tabIndex={collapsed ? -1 : null} onTransitionEnd={this.handleTransitionEnd}>
+          <div className='column-header__collapsible-inner'>
+            {(!collapsed || animating) && collapsedContent}
+          </div>
+        </div>
+
+        {appendContent}
+      </div>
+    );
+
+    if (multiColumn || placeholder) {
+      return component;
+    } else {
+      // The portal container and the component may be rendered to the DOM in
+      // the same React render pass, so the container might not be available at
+      // the time `render()` is called.
+      const container = document.getElementById('tabs-bar__portal');
+      if (container === null) {
+        // The container wasn't available, force a re-render so that the
+        // component can eventually be inserted in the container and not scroll
+        // with the rest of the area.
+        this.forceUpdate();
+        return component;
+      } else {
+        return createPortal(component, container);
+      }
+    }
+  }
+
+}
+
+export default injectIntl(withRouter(ColumnHeader));
diff --git a/app/javascript/flavours/blobfox/components/counters.tsx b/app/javascript/flavours/blobfox/components/counters.tsx
new file mode 100644
index 00000000000000..35b0ad8d607421
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/counters.tsx
@@ -0,0 +1,45 @@
+import React from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+export const StatusesCounter = (
+  displayNumber: React.ReactNode,
+  pluralReady: number,
+) => (
+  <FormattedMessage
+    id='account.statuses_counter'
+    defaultMessage='{count, plural, one {{counter} Post} other {{counter} Posts}}'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+    }}
+  />
+);
+
+export const FollowingCounter = (
+  displayNumber: React.ReactNode,
+  pluralReady: number,
+) => (
+  <FormattedMessage
+    id='account.following_counter'
+    defaultMessage='{count, plural, one {{counter} Following} other {{counter} Following}}'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+    }}
+  />
+);
+
+export const FollowersCounter = (
+  displayNumber: React.ReactNode,
+  pluralReady: number,
+) => (
+  <FormattedMessage
+    id='account.followers_counter'
+    defaultMessage='{count, plural, one {{counter} Follower} other {{counter} Followers}}'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+    }}
+  />
+);
diff --git a/app/javascript/flavours/blobfox/components/dismissable_banner.tsx b/app/javascript/flavours/blobfox/components/dismissable_banner.tsx
new file mode 100644
index 00000000000000..40ee47fb120834
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/dismissable_banner.tsx
@@ -0,0 +1,66 @@
+/* eslint-disable @typescript-eslint/no-unsafe-call,
+                  @typescript-eslint/no-unsafe-return,
+                  @typescript-eslint/no-unsafe-assignment,
+                  @typescript-eslint/no-unsafe-member-access
+                  -- the settings store is not yet typed */
+import type { PropsWithChildren } from 'react';
+import { useCallback, useState, useEffect } from 'react';
+
+import { defineMessages, useIntl } from 'react-intl';
+
+import { changeSetting } from 'flavours/blobfox/actions/settings';
+import { bannerSettings } from 'flavours/blobfox/settings';
+import { useAppSelector, useAppDispatch } from 'flavours/blobfox/store';
+
+import { IconButton } from './icon_button';
+
+const messages = defineMessages({
+  dismiss: { id: 'dismissable_banner.dismiss', defaultMessage: 'Dismiss' },
+});
+
+interface Props {
+  id: string;
+}
+
+export const DismissableBanner: React.FC<PropsWithChildren<Props>> = ({
+  id,
+  children,
+}) => {
+  const dismissed = useAppSelector((state) =>
+    state.settings.getIn(['dismissed_banners', id], false),
+  );
+  const dispatch = useAppDispatch();
+
+  const [visible, setVisible] = useState(!bannerSettings.get(id) && !dismissed);
+  const intl = useIntl();
+
+  const handleDismiss = useCallback(() => {
+    setVisible(false);
+    bannerSettings.set(id, true);
+    dispatch(changeSetting(['dismissed_banners', id], true));
+  }, [id, dispatch]);
+
+  useEffect(() => {
+    if (!visible && !dismissed) {
+      dispatch(changeSetting(['dismissed_banners', id], true));
+    }
+  }, [id, dispatch, visible, dismissed]);
+
+  if (!visible) {
+    return null;
+  }
+
+  return (
+    <div className='dismissable-banner'>
+      <div className='dismissable-banner__action'>
+        <IconButton
+          icon='times'
+          title={intl.formatMessage(messages.dismiss)}
+          onClick={handleDismiss}
+        />
+      </div>
+
+      <div className='dismissable-banner__message'>{children}</div>
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/display_name.tsx b/app/javascript/flavours/blobfox/components/display_name.tsx
new file mode 100644
index 00000000000000..3eae2e14504640
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/display_name.tsx
@@ -0,0 +1,127 @@
+import React from 'react';
+
+import classNames from 'classnames';
+
+import type { List } from 'immutable';
+
+import type { Account } from 'flavours/blobfox/models/account';
+
+import { autoPlayGif } from '../initial_state';
+
+import { Skeleton } from './skeleton';
+
+interface Props {
+  account?: Account;
+  others?: List<Account>;
+  localDomain?: string;
+  inline?: boolean;
+}
+
+export class DisplayName extends React.PureComponent<Props> {
+  handleMouseEnter: React.ReactEventHandler<HTMLSpanElement> = ({
+    currentTarget,
+  }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis =
+      currentTarget.querySelectorAll<HTMLImageElement>('img.custom-emoji');
+
+    emojis.forEach((emoji) => {
+      const originalSrc = emoji.getAttribute('data-original');
+      if (originalSrc != null) emoji.src = originalSrc;
+    });
+  };
+
+  handleMouseLeave: React.ReactEventHandler<HTMLSpanElement> = ({
+    currentTarget,
+  }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis =
+      currentTarget.querySelectorAll<HTMLImageElement>('img.custom-emoji');
+
+    emojis.forEach((emoji) => {
+      const staticSrc = emoji.getAttribute('data-static');
+      if (staticSrc != null) emoji.src = staticSrc;
+    });
+  };
+
+  render() {
+    const { others, localDomain, inline } = this.props;
+
+    let displayName: React.ReactNode,
+      suffix: React.ReactNode,
+      account: Account | undefined;
+
+    if (others && others.size > 0) {
+      account = others.first();
+    } else if (this.props.account) {
+      account = this.props.account;
+    }
+
+    if (others && others.size > 1) {
+      displayName = others
+        .take(2)
+        .map((a) => (
+          <bdi key={a.get('id')}>
+            <strong
+              className='display-name__html'
+              dangerouslySetInnerHTML={{ __html: a.get('display_name_html') }}
+            />
+          </bdi>
+        ))
+        .reduce((prev, cur) => [prev, ', ', cur]);
+
+      if (others.size - 2 > 0) {
+        suffix = `+${others.size - 2}`;
+      }
+    } else if (account) {
+      let acct = account.get('acct');
+
+      if (!acct.includes('@') && localDomain) {
+        acct = `${acct}@${localDomain}`;
+      }
+
+      displayName = (
+        <bdi>
+          <strong
+            className='display-name__html'
+            dangerouslySetInnerHTML={{
+              __html: account.get('display_name_html'),
+            }}
+          />
+        </bdi>
+      );
+      suffix = <span className='display-name__account'>@{acct}</span>;
+    } else {
+      displayName = (
+        <bdi>
+          <strong className='display-name__html'>
+            <Skeleton width='10ch' />
+          </strong>
+        </bdi>
+      );
+      suffix = (
+        <span className='display-name__account'>
+          <Skeleton width='7ch' />
+        </span>
+      );
+    }
+
+    return (
+      <span
+        className={classNames('display-name', { inline })}
+        onMouseEnter={this.handleMouseEnter}
+        onMouseLeave={this.handleMouseLeave}
+      >
+        {displayName}
+        {inline ? ' ' : null}
+        {suffix}
+      </span>
+    );
+  }
+}
diff --git a/app/javascript/flavours/blobfox/components/domain.tsx b/app/javascript/flavours/blobfox/components/domain.tsx
new file mode 100644
index 00000000000000..f4a3b9d4b69d2a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/domain.tsx
@@ -0,0 +1,44 @@
+import { useCallback } from 'react';
+
+import { defineMessages, useIntl } from 'react-intl';
+
+import { IconButton } from './icon_button';
+
+const messages = defineMessages({
+  unblockDomain: {
+    id: 'account.unblock_domain',
+    defaultMessage: 'Unblock domain {domain}',
+  },
+});
+
+interface Props {
+  domain: string;
+  onUnblockDomain: (domain: string) => void;
+}
+
+export const Domain: React.FC<Props> = ({ domain, onUnblockDomain }) => {
+  const intl = useIntl();
+
+  const handleDomainUnblock = useCallback(() => {
+    onUnblockDomain(domain);
+  }, [domain, onUnblockDomain]);
+
+  return (
+    <div className='domain'>
+      <div className='domain__wrapper'>
+        <span className='domain__domain-name'>
+          <strong>{domain}</strong>
+        </span>
+
+        <div className='domain__buttons'>
+          <IconButton
+            active
+            icon='unlock'
+            title={intl.formatMessage(messages.unblockDomain, { domain })}
+            onClick={handleDomainUnblock}
+          />
+        </div>
+      </div>
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/dropdown_menu.jsx b/app/javascript/flavours/blobfox/components/dropdown_menu.jsx
new file mode 100644
index 00000000000000..9640b4fdfeca45
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/dropdown_menu.jsx
@@ -0,0 +1,338 @@
+import PropTypes from 'prop-types';
+import { PureComponent, cloneElement, Children } from 'react';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+import Overlay from 'react-overlays/Overlay';
+
+import { CircularProgress } from 'flavours/blobfox/components/circular_progress';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { IconButton } from './icon_button';
+
+const listenerOptions = supportsPassiveEvents ? { passive: true, capture: true } : true;
+let id = 0;
+
+class DropdownMenu extends PureComponent {
+
+  static propTypes = {
+    items: PropTypes.oneOfType([PropTypes.array, ImmutablePropTypes.list]).isRequired,
+    loading: PropTypes.bool,
+    scrollable: PropTypes.bool,
+    onClose: PropTypes.func.isRequired,
+    style: PropTypes.object,
+    openedViaKeyboard: PropTypes.bool,
+    renderItem: PropTypes.func,
+    renderHeader: PropTypes.func,
+    onItemClick: PropTypes.func.isRequired,
+  };
+
+  static defaultProps = {
+    style: {},
+  };
+
+  handleDocumentClick = e => {
+    if (this.node && !this.node.contains(e.target)) {
+      this.props.onClose();
+      e.stopPropagation();
+    }
+  };
+
+  componentDidMount () {
+    document.addEventListener('click', this.handleDocumentClick, { capture: true });
+    document.addEventListener('keydown', this.handleKeyDown, { capture: true });
+    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
+
+    if (this.focusedItem && this.props.openedViaKeyboard) {
+      this.focusedItem.focus({ preventScroll: true });
+    }
+  }
+
+  componentWillUnmount () {
+    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
+    document.removeEventListener('keydown', this.handleKeyDown, { capture: true });
+    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  setFocusRef = c => {
+    this.focusedItem = c;
+  };
+
+  handleKeyDown = e => {
+    const items = Array.from(this.node.querySelectorAll('a, button'));
+    const index = items.indexOf(document.activeElement);
+    let element = null;
+
+    switch(e.key) {
+    case 'ArrowDown':
+      element = items[index+1] || items[0];
+      break;
+    case 'ArrowUp':
+      element = items[index-1] || items[items.length-1];
+      break;
+    case 'Tab':
+      if (e.shiftKey) {
+        element = items[index-1] || items[items.length-1];
+      } else {
+        element = items[index+1] || items[0];
+      }
+      break;
+    case 'Home':
+      element = items[0];
+      break;
+    case 'End':
+      element = items[items.length-1];
+      break;
+    case 'Escape':
+      this.props.onClose();
+      break;
+    }
+
+    if (element) {
+      element.focus();
+      e.preventDefault();
+      e.stopPropagation();
+    }
+  };
+
+  handleItemKeyPress = e => {
+    if (e.key === 'Enter' || e.key === ' ') {
+      this.handleClick(e);
+    }
+  };
+
+  handleClick = e => {
+    const { onItemClick } = this.props;
+    onItemClick(e);
+  };
+
+  renderItem = (option, i) => {
+    if (option === null) {
+      return <li key={`sep-${i}`} className='dropdown-menu__separator' />;
+    }
+
+    const { text, href = '#', target = '_blank', method, dangerous } = option;
+
+    return (
+      <li className={classNames('dropdown-menu__item', { 'dropdown-menu__item--dangerous': dangerous })} key={`${text}-${i}`}>
+        <a href={href} target={target} data-method={method} rel='noopener noreferrer' role='button' tabIndex={0} ref={i === 0 ? this.setFocusRef : null} onClick={this.handleClick} onKeyPress={this.handleItemKeyPress} data-index={i}>
+          {text}
+        </a>
+      </li>
+    );
+  };
+
+  render () {
+    const { items, scrollable, renderHeader, loading } = this.props;
+
+    let renderItem = this.props.renderItem || this.renderItem;
+
+    return (
+      <div className={classNames('dropdown-menu__container', { 'dropdown-menu__container--loading': loading })} ref={this.setRef}>
+        {loading && (
+          <CircularProgress size={30} strokeWidth={3.5} />
+        )}
+
+        {!loading && renderHeader && (
+          <div className='dropdown-menu__container__header'>
+            {renderHeader(items)}
+          </div>
+        )}
+
+        {!loading && (
+          <ul className={classNames('dropdown-menu__container__list', { 'dropdown-menu__container__list--scrollable': scrollable })}>
+            {items.map((option, i) => renderItem(option, i, { onClick: this.handleClick, onKeyPress: this.handleItemKeyPress }))}
+          </ul>
+        )}
+      </div>
+    );
+  }
+
+}
+
+class Dropdown extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node,
+    icon: PropTypes.string,
+    items: PropTypes.oneOfType([PropTypes.array, ImmutablePropTypes.list]).isRequired,
+    loading: PropTypes.bool,
+    size: PropTypes.number,
+    title: PropTypes.string,
+    disabled: PropTypes.bool,
+    scrollable: PropTypes.bool,
+    status: ImmutablePropTypes.map,
+    isUserTouching: PropTypes.func,
+    onOpen: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    openDropdownId: PropTypes.number,
+    openedViaKeyboard: PropTypes.bool,
+    renderItem: PropTypes.func,
+    renderHeader: PropTypes.func,
+    onItemClick: PropTypes.func,
+    ...WithRouterPropTypes
+  };
+
+  static defaultProps = {
+    title: 'Menu',
+  };
+
+  state = {
+    id: id++,
+  };
+
+  handleClick = ({ type }) => {
+    if (this.state.id === this.props.openDropdownId) {
+      this.handleClose();
+    } else {
+      this.props.onOpen(this.state.id, this.handleItemClick, type !== 'click');
+    }
+  };
+
+  handleClose = () => {
+    if (this.activeElement) {
+      this.activeElement.focus({ preventScroll: true });
+      this.activeElement = null;
+    }
+    this.props.onClose(this.state.id);
+  };
+
+  handleMouseDown = () => {
+    if (!this.state.open) {
+      this.activeElement = document.activeElement;
+    }
+  };
+
+  handleButtonKeyDown = (e) => {
+    switch(e.key) {
+    case ' ':
+    case 'Enter':
+      this.handleMouseDown();
+      break;
+    }
+  };
+
+  handleKeyPress = (e) => {
+    switch(e.key) {
+    case ' ':
+    case 'Enter':
+      this.handleClick(e);
+      e.stopPropagation();
+      e.preventDefault();
+      break;
+    }
+  };
+
+  handleItemClick = e => {
+    const { onItemClick } = this.props;
+    const i = Number(e.currentTarget.getAttribute('data-index'));
+    const item = this.props.items[i];
+
+    this.handleClose();
+
+    if (typeof onItemClick === 'function') {
+      e.preventDefault();
+      onItemClick(item, i);
+    } else if (item && typeof item.action === 'function') {
+      e.preventDefault();
+      item.action();
+    } else if (item && item.to) {
+      e.preventDefault();
+      this.props.history.push(item.to);
+    }
+  };
+
+  setTargetRef = c => {
+    this.target = c;
+  };
+
+  findTarget = () => {
+    return this.target;
+  };
+
+  componentWillUnmount = () => {
+    if (this.state.id === this.props.openDropdownId) {
+      this.handleClose();
+    }
+  };
+
+  close = () => {
+    this.handleClose();
+  };
+
+  render () {
+    const {
+      icon,
+      items,
+      size,
+      title,
+      disabled,
+      loading,
+      scrollable,
+      openDropdownId,
+      openedViaKeyboard,
+      children,
+      renderItem,
+      renderHeader,
+    } = this.props;
+
+    const open = this.state.id === openDropdownId;
+
+    const button = children ? cloneElement(Children.only(children), {
+      onClick: this.handleClick,
+      onMouseDown: this.handleMouseDown,
+      onKeyDown: this.handleButtonKeyDown,
+      onKeyPress: this.handleKeyPress,
+    }) : (
+      <IconButton
+        icon={icon}
+        title={title}
+        active={open}
+        disabled={disabled}
+        size={size}
+        onClick={this.handleClick}
+        onMouseDown={this.handleMouseDown}
+        onKeyDown={this.handleButtonKeyDown}
+        onKeyPress={this.handleKeyPress}
+      />
+    );
+
+    return (
+      <>
+        <span ref={this.setTargetRef}>
+          {button}
+        </span>
+        <Overlay show={open} offset={[5, 5]} placement={'bottom'} flip target={this.findTarget} popperConfig={{ strategy: 'fixed' }}>
+          {({ props, arrowProps, placement }) => (
+            <div {...props}>
+              <div className={`dropdown-animation dropdown-menu ${placement}`}>
+                <div className={`dropdown-menu__arrow ${placement}`} {...arrowProps} />
+                <DropdownMenu
+                  items={items}
+                  loading={loading}
+                  scrollable={scrollable}
+                  onClose={this.handleClose}
+                  openedViaKeyboard={openedViaKeyboard}
+                  renderItem={renderItem}
+                  renderHeader={renderHeader}
+                  onItemClick={this.handleItemClick}
+                />
+              </div>
+            </div>
+          )}
+        </Overlay>
+      </>
+    );
+  }
+
+}
+
+export default withRouter(Dropdown);
diff --git a/app/javascript/flavours/blobfox/components/edited_timestamp/containers/dropdown_menu_container.js b/app/javascript/flavours/blobfox/components/edited_timestamp/containers/dropdown_menu_container.js
new file mode 100644
index 00000000000000..b52f19e46705c4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/edited_timestamp/containers/dropdown_menu_container.js
@@ -0,0 +1,32 @@
+import { connect } from 'react-redux';
+
+import { openDropdownMenu, closeDropdownMenu } from 'flavours/blobfox/actions/dropdown_menu';
+import { fetchHistory } from 'flavours/blobfox/actions/history';
+import DropdownMenu from 'flavours/blobfox/components/dropdown_menu';
+
+/**
+ *
+ * @param {import('flavours/blobfox/store').RootState} state
+ * @param {*} props
+ */
+const mapStateToProps = (state, { statusId }) => ({
+  openDropdownId: state.dropdownMenu.openId,
+  openedViaKeyboard: state.dropdownMenu.keyboard,
+  items: state.getIn(['history', statusId, 'items']),
+  loading: state.getIn(['history', statusId, 'loading']),
+});
+
+const mapDispatchToProps = (dispatch, { statusId }) => ({
+
+  onOpen (id, onItemClick, keyboard) {
+    dispatch(fetchHistory(statusId));
+    dispatch(openDropdownMenu({ id, keyboard }));
+  },
+
+  onClose (id) {
+    dispatch(closeDropdownMenu({ id }));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(DropdownMenu);
diff --git a/app/javascript/flavours/blobfox/components/edited_timestamp/index.jsx b/app/javascript/flavours/blobfox/components/edited_timestamp/index.jsx
new file mode 100644
index 00000000000000..d38b2865b4a2fa
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/edited_timestamp/index.jsx
@@ -0,0 +1,77 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import InlineAccount from 'flavours/blobfox/components/inline_account';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+
+import DropdownMenu from './containers/dropdown_menu_container';
+
+const mapDispatchToProps = (dispatch, { statusId }) => ({
+
+  onItemClick (index) {
+    dispatch(openModal({
+      modalType: 'COMPARE_HISTORY',
+      modalProps: { index, statusId },
+    }));
+  },
+
+});
+
+class EditedTimestamp extends PureComponent {
+
+  static propTypes = {
+    statusId: PropTypes.string.isRequired,
+    timestamp: PropTypes.string.isRequired,
+    intl: PropTypes.object.isRequired,
+    onItemClick: PropTypes.func.isRequired,
+  };
+
+  handleItemClick = (item, i) => {
+    const { onItemClick } = this.props;
+    onItemClick(i);
+  };
+
+  renderHeader = items => {
+    return (
+      <FormattedMessage id='status.edited_x_times' defaultMessage='Edited {count, plural, one {# time} other {# times}}' values={{ count: items.size - 1 }} />
+    );
+  };
+
+  renderItem = (item, index, { onClick, onKeyPress }) => {
+    const formattedDate = <RelativeTimestamp timestamp={item.get('created_at')} short={false} />;
+    const formattedName = <InlineAccount accountId={item.get('account')} />;
+
+    const label = item.get('original') ? (
+      <FormattedMessage id='status.history.created' defaultMessage='{name} created {date}' values={{ name: formattedName, date: formattedDate }} />
+    ) : (
+      <FormattedMessage id='status.history.edited' defaultMessage='{name} edited {date}' values={{ name: formattedName, date: formattedDate }} />
+    );
+
+    return (
+      <li className='dropdown-menu__item edited-timestamp__history__item' key={item.get('created_at')}>
+        <button data-index={index} onClick={onClick} onKeyPress={onKeyPress}>{label}</button>
+      </li>
+    );
+  };
+
+  render () {
+    const { timestamp, intl, statusId } = this.props;
+
+    return (
+      <DropdownMenu statusId={statusId} renderItem={this.renderItem} scrollable renderHeader={this.renderHeader} onItemClick={this.handleItemClick}>
+        <button className='dropdown-menu__text-button'>
+          <FormattedMessage id='status.edited' defaultMessage='Edited {date}' values={{ date: intl.formatDate(timestamp, { hour12: false, month: 'short', day: '2-digit', hour: '2-digit', minute: '2-digit' }) }} /> <Icon id='caret-down' />
+        </button>
+      </DropdownMenu>
+    );
+  }
+
+}
+
+export default connect(null, mapDispatchToProps)(injectIntl(EditedTimestamp));
diff --git a/app/javascript/flavours/blobfox/components/empty_account.tsx b/app/javascript/flavours/blobfox/components/empty_account.tsx
new file mode 100644
index 00000000000000..1c2ddc1aed2035
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/empty_account.tsx
@@ -0,0 +1,33 @@
+import React from 'react';
+
+import classNames from 'classnames';
+
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+
+interface Props {
+  size?: number;
+  minimal?: boolean;
+}
+
+export const EmptyAccount: React.FC<Props> = ({
+  size = 46,
+  minimal = false,
+}) => {
+  return (
+    <div className={classNames('account', { 'account--minimal': minimal })}>
+      <div className='account__wrapper'>
+        <div className='account__display-name'>
+          <div className='account__avatar-wrapper'>
+            <Skeleton width={size} height={size} />
+          </div>
+
+          <div>
+            <DisplayName />
+            <Skeleton width='7ch' />
+          </div>
+        </div>
+      </div>
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/error_boundary.jsx b/app/javascript/flavours/blobfox/components/error_boundary.jsx
new file mode 100644
index 00000000000000..3cea43c06ecf39
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/error_boundary.jsx
@@ -0,0 +1,111 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import StackTrace from 'stacktrace-js';
+
+import { version, source_url } from 'flavours/blobfox/initial_state';
+
+export default class ErrorBoundary extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node,
+  };
+
+  state = {
+    hasError: false,
+    errorMessage: undefined,
+    stackTrace: undefined,
+    mappedStackTrace: undefined,
+    componentStack: undefined,
+  };
+
+  componentDidCatch (error, info) {
+    this.setState({
+      hasError: true,
+      errorMessage: error.toString(),
+      stackTrace: error.stack,
+      componentStack: info && info.componentStack,
+      mappedStackTrace: undefined,
+    });
+
+    StackTrace.fromError(error).then((stackframes) => {
+      this.setState({
+        mappedStackTrace: stackframes.map((sf) => sf.toString()).join('\n'),
+      });
+    }).catch(() => {
+      this.setState({
+        mappedStackTrace: undefined,
+      });
+    });
+  }
+
+  handleCopyStackTrace = () => {
+    const { errorMessage, stackTrace, mappedStackTrace } = this.state;
+    const textarea = document.createElement('textarea');
+
+    let contents = [errorMessage, stackTrace];
+    if (mappedStackTrace) {
+      contents.push(mappedStackTrace);
+    }
+
+    textarea.textContent    = contents.join('\n\n\n');
+    textarea.style.position = 'fixed';
+
+    document.body.appendChild(textarea);
+
+    try {
+      textarea.select();
+      document.execCommand('copy');
+    } catch (e) {
+
+    } finally {
+      document.body.removeChild(textarea);
+    }
+
+    this.setState({ copied: true });
+    setTimeout(() => this.setState({ copied: false }), 700);
+  };
+
+  render() {
+    const { hasError, copied, errorMessage } = this.state;
+
+    if (!hasError) {
+      return this.props.children;
+    }
+
+    const likelyBrowserAddonIssue = errorMessage && errorMessage.includes('NotFoundError');
+
+    return (
+      <div className='error-boundary'>
+        <div>
+          <p className='error-boundary__error'>
+            { likelyBrowserAddonIssue ? (
+              <FormattedMessage id='error.unexpected_crash.explanation_addons' defaultMessage='This page could not be displayed correctly. This error is likely caused by a browser add-on or automatic translation tools.' />
+            ) : (
+              <FormattedMessage id='error.unexpected_crash.explanation' defaultMessage='Due to a bug in our code or a browser compatibility issue, this page could not be displayed correctly.' />
+            )}
+          </p>
+
+          <p>
+            { likelyBrowserAddonIssue ? (
+              <FormattedMessage id='error.unexpected_crash.next_steps_addons' defaultMessage='Try disabling them and refreshing the page. If that does not help, you may still be able to use Mastodon through a different browser or native app.' />
+            ) : (
+              <FormattedMessage id='error.unexpected_crash.next_steps' defaultMessage='Try refreshing the page. If that does not help, you may still be able to use Mastodon through a different browser or native app.' />
+            )}
+          </p>
+
+          <p className='error-boundary__footer'>Mastodon v{version} · <a href={source_url} rel='noopener noreferrer' target='_blank'><FormattedMessage id='errors.unexpected_crash.report_issue' defaultMessage='Report issue' /></a> · <button onClick={this.handleCopyStackTrace} className={copied ? 'copied' : ''}><FormattedMessage id='errors.unexpected_crash.copy_stacktrace' defaultMessage='Copy stacktrace to clipboard' /></button></p>
+        </div>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/gifv.tsx b/app/javascript/flavours/blobfox/components/gifv.tsx
new file mode 100644
index 00000000000000..c2be591128f458
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/gifv.tsx
@@ -0,0 +1,70 @@
+import { useCallback, useState } from 'react';
+
+interface Props {
+  src: string;
+  key: string;
+  alt?: string;
+  lang?: string;
+  width: number;
+  height: number;
+  onClick?: () => void;
+}
+
+export const GIFV: React.FC<Props> = ({
+  src,
+  alt,
+  lang,
+  width,
+  height,
+  onClick,
+}) => {
+  const [loading, setLoading] = useState(true);
+
+  const handleLoadedData: React.ReactEventHandler<HTMLVideoElement> =
+    useCallback(() => {
+      setLoading(false);
+    }, [setLoading]);
+
+  const handleClick: React.MouseEventHandler = useCallback(
+    (e) => {
+      if (onClick) {
+        e.stopPropagation();
+        onClick();
+      }
+    },
+    [onClick],
+  );
+
+  return (
+    <div className='gifv' style={{ position: 'relative' }}>
+      {loading && (
+        <canvas
+          width={width}
+          height={height}
+          role='button'
+          tabIndex={0}
+          aria-label={alt}
+          title={alt}
+          lang={lang}
+          onClick={handleClick}
+        />
+      )}
+
+      <video
+        src={src}
+        role='button'
+        tabIndex={0}
+        aria-label={alt}
+        title={alt}
+        lang={lang}
+        muted
+        loop
+        autoPlay
+        playsInline
+        onClick={handleClick}
+        onLoadedData={handleLoadedData}
+        style={{ position: loading ? 'absolute' : 'static', top: 0, left: 0 }}
+      />
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/hashtag.jsx b/app/javascript/flavours/blobfox/components/hashtag.jsx
new file mode 100644
index 00000000000000..101c8c518a4efd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/hashtag.jsx
@@ -0,0 +1,123 @@
+// @ts-check
+import PropTypes from 'prop-types';
+import { Component } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Sparklines, SparklinesCurve } from 'react-sparklines';
+
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+
+import Permalink from './permalink';
+
+class SilentErrorBoundary extends Component {
+
+  static propTypes = {
+    children: PropTypes.node,
+  };
+
+  state = {
+    error: false,
+  };
+
+  componentDidCatch() {
+    this.setState({ error: true });
+  }
+
+  render() {
+    if (this.state.error) {
+      return null;
+    }
+
+    return this.props.children;
+  }
+
+}
+
+/**
+ * Used to render counter of how much people are talking about hashtag
+ * @type {(displayNumber: JSX.Element, pluralReady: number) => JSX.Element}
+ */
+export const accountsCountRenderer = (displayNumber, pluralReady) => (
+  <FormattedMessage
+    id='trends.counter_by_accounts'
+    defaultMessage='{count, plural, one {{counter} person} other {{counter} people}} in the past {days, plural, one {day} other {# days}}'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+      days: 2,
+    }}
+  />
+);
+
+// @ts-expect-error
+export const ImmutableHashtag = ({ hashtag }) => (
+  <Hashtag
+    name={hashtag.get('name')}
+    href={hashtag.get('url')}
+    to={`/tags/${hashtag.get('name')}`}
+    people={hashtag.getIn(['history', 0, 'accounts']) * 1 + hashtag.getIn(['history', 1, 'accounts']) * 1}
+    // @ts-expect-error
+    history={hashtag.get('history').reverse().map((day) => day.get('uses')).toArray()}
+  />
+);
+
+ImmutableHashtag.propTypes = {
+  hashtag: ImmutablePropTypes.map.isRequired,
+};
+
+// @ts-expect-error
+const Hashtag = ({ name, href, to, people, uses, history, className, description, withGraph }) => (
+  <div className={classNames('trends__item', className)}>
+    <div className='trends__item__name'>
+      <Permalink href={href} to={to}>
+        {name ? <>#<span>{name}</span></> : <Skeleton width={50} />}
+      </Permalink>
+
+      {description ? (
+        <span>{description}</span>
+      ) : (
+        typeof people !== 'undefined' ? <ShortNumber value={people} renderer={accountsCountRenderer} /> : <Skeleton width={100} />
+      )}
+    </div>
+
+    {typeof uses !== 'undefined' && (
+      <div className='trends__item__current'>
+        <ShortNumber value={uses} />
+      </div>
+    )}
+
+    {withGraph && (
+      <div className='trends__item__sparkline'>
+        <SilentErrorBoundary>
+          <Sparklines width={50} height={28} data={history ? history : Array.from(Array(7)).map(() => 0)}>
+            <SparklinesCurve style={{ fill: 'none' }} />
+          </Sparklines>
+        </SilentErrorBoundary>
+      </div>
+    )}
+  </div>
+);
+
+Hashtag.propTypes = {
+  name: PropTypes.string,
+  href: PropTypes.string,
+  to: PropTypes.string,
+  people: PropTypes.number,
+  description: PropTypes.node,
+  uses: PropTypes.number,
+  history: PropTypes.arrayOf(PropTypes.number),
+  className: PropTypes.string,
+  withGraph: PropTypes.bool,
+};
+
+Hashtag.defaultProps = {
+  withGraph: true,
+};
+
+export default Hashtag;
diff --git a/app/javascript/flavours/blobfox/components/hashtag_bar.tsx b/app/javascript/flavours/blobfox/components/hashtag_bar.tsx
new file mode 100644
index 00000000000000..91fa9221983a70
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/hashtag_bar.tsx
@@ -0,0 +1,234 @@
+import { useState, useCallback } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import type { List, Record } from 'immutable';
+
+import { groupBy, minBy } from 'lodash';
+
+import { getStatusContent } from './status_content';
+
+// Fit on a single line on desktop
+const VISIBLE_HASHTAGS = 3;
+
+// Those types are not correct, they need to be replaced once this part of the state is typed
+export type TagLike = Record<{ name: string }>;
+export type StatusLike = Record<{
+  tags: List<TagLike>;
+  contentHTML: string;
+  media_attachments: List<unknown>;
+  spoiler_text?: string;
+}>;
+
+function normalizeHashtag(hashtag: string) {
+  return (
+    hashtag && hashtag.startsWith('#') ? hashtag.slice(1) : hashtag
+  ).normalize('NFKC');
+}
+
+function isNodeLinkHashtag(element: Node): element is HTMLLinkElement {
+  return (
+    element instanceof HTMLAnchorElement &&
+    // it may be a <a> starting with a hashtag
+    (element.textContent?.[0] === '#' ||
+      // or a #<a>
+      element.previousSibling?.textContent?.[
+        element.previousSibling.textContent.length - 1
+      ] === '#')
+  );
+}
+
+/**
+ * Removes duplicates from an hashtag list, case-insensitive, keeping only the best one
+ * "Best" here is defined by the one with the more casing difference (ie, the most camel-cased one)
+ * @param hashtags The list of hashtags
+ * @returns The input hashtags, but with only 1 occurence of each (case-insensitive)
+ */
+function uniqueHashtagsWithCaseHandling(hashtags: string[]) {
+  const groups = groupBy(hashtags, (tag) =>
+    tag.normalize('NFKD').toLowerCase(),
+  );
+
+  return Object.values(groups).map((tags) => {
+    if (tags.length === 1) return tags[0];
+
+    // The best match is the one where we have the less difference between upper and lower case letter count
+    const best = minBy(tags, (tag) => {
+      const upperCase = Array.from(tag).reduce(
+        (acc, char) => (acc += char.toUpperCase() === char ? 1 : 0),
+        0,
+      );
+
+      const lowerCase = tag.length - upperCase;
+
+      return Math.abs(lowerCase - upperCase);
+    });
+
+    return best ?? tags[0];
+  });
+}
+
+// Create the collator once, this is much more efficient
+const collator = new Intl.Collator(undefined, {
+  sensitivity: 'base', // we use this to emulate the ASCII folding done on the server-side, hopefuly more efficiently
+});
+
+function localeAwareInclude(collection: string[], value: string) {
+  const normalizedValue = value.normalize('NFKC');
+
+  return !!collection.find(
+    (item) => collator.compare(item.normalize('NFKC'), normalizedValue) === 0,
+  );
+}
+
+// We use an intermediate function here to make it easier to test
+export function computeHashtagBarForStatus(status: StatusLike): {
+  statusContentProps: { statusContent: string };
+  hashtagsInBar: string[];
+} {
+  let statusContent = getStatusContent(status);
+
+  const tagNames = status
+    .get('tags')
+    .map((tag) => tag.get('name'))
+    .toJS();
+
+  // this is returned if we stop the processing early, it does not change what is displayed
+  const defaultResult = {
+    statusContentProps: { statusContent },
+    hashtagsInBar: [],
+  };
+
+  // return early if this status does not have any tags
+  if (tagNames.length === 0) return defaultResult;
+
+  const template = document.createElement('template');
+  template.innerHTML = statusContent.trim();
+
+  const lastChild = template.content.lastChild;
+
+  if (!lastChild || lastChild.nodeType === Node.TEXT_NODE) return defaultResult;
+
+  template.content.removeChild(lastChild);
+  const contentWithoutLastLine = template;
+
+  // First, try to parse
+  const contentHashtags = Array.from(
+    contentWithoutLastLine.content.querySelectorAll<HTMLLinkElement>('a[href]'),
+  ).reduce<string[]>((result, link) => {
+    if (isNodeLinkHashtag(link)) {
+      if (link.textContent) result.push(normalizeHashtag(link.textContent));
+    }
+    return result;
+  }, []);
+
+  // Now we parse the last line, and try to see if it only contains hashtags
+  const lastLineHashtags: string[] = [];
+  // try to see if the last line is only hashtags
+  let onlyHashtags = true;
+
+  const normalizedTagNames = tagNames.map((tag) => tag.normalize('NFKC'));
+
+  Array.from(lastChild.childNodes).forEach((node) => {
+    if (isNodeLinkHashtag(node) && node.textContent) {
+      const normalized = normalizeHashtag(node.textContent);
+
+      if (!localeAwareInclude(normalizedTagNames, normalized)) {
+        // stop here, this is not a real hashtag, so consider it as text
+        onlyHashtags = false;
+        return;
+      }
+
+      if (!localeAwareInclude(contentHashtags, normalized))
+        // only add it if it does not appear in the rest of the content
+        lastLineHashtags.push(normalized);
+    } else if (node.nodeType !== Node.TEXT_NODE || node.nodeValue?.trim()) {
+      // not a space
+      onlyHashtags = false;
+    }
+  });
+
+  const hashtagsInBar = tagNames.filter((tag) => {
+    const normalizedTag = tag.normalize('NFKC');
+    // the tag does not appear at all in the status content, it is an out-of-band tag
+    return (
+      !localeAwareInclude(contentHashtags, normalizedTag) &&
+      !localeAwareInclude(lastLineHashtags, normalizedTag)
+    );
+  });
+
+  const isOnlyOneLine = contentWithoutLastLine.content.childElementCount === 0;
+  const hasMedia = status.get('media_attachments').size > 0;
+  const hasSpoiler = !!status.get('spoiler_text');
+
+  // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- due to https://github.com/microsoft/TypeScript/issues/9998
+  if (onlyHashtags && ((hasMedia && !hasSpoiler) || !isOnlyOneLine)) {
+    // if the last line only contains hashtags, and we either:
+    // - have other content in the status
+    // - dont have other content, but a media and no CW. If it has a CW, then we do not remove the content to avoid having an empty content behind the CW button
+    statusContent = contentWithoutLastLine.innerHTML;
+    // and add the tags to the bar
+    hashtagsInBar.push(...lastLineHashtags);
+  }
+
+  return {
+    statusContentProps: { statusContent },
+    hashtagsInBar: uniqueHashtagsWithCaseHandling(hashtagsInBar),
+  };
+}
+
+/**
+ *  This function will process a status to, at the same time (avoiding parsing it twice):
+ * - build the HashtagBar for this status
+ * - remove the last-line hashtags from the status content
+ * @param status The status to process
+ * @returns Props to be passed to the <StatusContent> component, and the hashtagBar to render
+ */
+export function getHashtagBarForStatus(status: StatusLike) {
+  const { statusContentProps, hashtagsInBar } =
+    computeHashtagBarForStatus(status);
+
+  return {
+    statusContentProps,
+    hashtagBar: <HashtagBar hashtags={hashtagsInBar} />,
+  };
+}
+
+const HashtagBar: React.FC<{
+  hashtags: string[];
+}> = ({ hashtags }) => {
+  const [expanded, setExpanded] = useState(false);
+  const handleClick = useCallback(() => {
+    setExpanded(true);
+  }, []);
+
+  if (hashtags.length === 0) {
+    return null;
+  }
+
+  const revealedHashtags = expanded
+    ? hashtags
+    : hashtags.slice(0, VISIBLE_HASHTAGS);
+
+  return (
+    <div className='hashtag-bar'>
+      {revealedHashtags.map((hashtag) => (
+        <Link key={hashtag} to={`/tags/${hashtag}`}>
+          #<span>{hashtag}</span>
+        </Link>
+      ))}
+
+      {!expanded && hashtags.length > VISIBLE_HASHTAGS && (
+        <button className='link-button' onClick={handleClick}>
+          <FormattedMessage
+            id='hashtags.and_other'
+            defaultMessage='…and {count, plural, other {# more}}'
+            values={{ count: hashtags.length - VISIBLE_HASHTAGS }}
+          />
+        </button>
+      )}
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/icon.tsx b/app/javascript/flavours/blobfox/components/icon.tsx
new file mode 100644
index 00000000000000..3d091c7059e18c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/icon.tsx
@@ -0,0 +1,20 @@
+import classNames from 'classnames';
+
+interface Props extends React.HTMLAttributes<HTMLImageElement> {
+  id: string;
+  className?: string;
+  fixedWidth?: boolean;
+  children?: never;
+}
+
+export const Icon: React.FC<Props> = ({
+  id,
+  className,
+  fixedWidth,
+  ...other
+}) => (
+  <i
+    className={classNames('fa', `fa-${id}`, className, { 'fa-fw': fixedWidth })}
+    {...other}
+  />
+);
diff --git a/app/javascript/flavours/blobfox/components/icon_button.tsx b/app/javascript/flavours/blobfox/components/icon_button.tsx
new file mode 100644
index 00000000000000..8bca60fa971f78
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/icon_button.tsx
@@ -0,0 +1,180 @@
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import { AnimatedNumber } from './animated_number';
+import { Icon } from './icon';
+
+interface Props {
+  className?: string;
+  title: string;
+  icon: string;
+  onClick?: React.MouseEventHandler<HTMLButtonElement>;
+  onMouseDown?: React.MouseEventHandler<HTMLButtonElement>;
+  onKeyDown?: React.KeyboardEventHandler<HTMLButtonElement>;
+  onKeyPress?: React.KeyboardEventHandler<HTMLButtonElement>;
+  size: number;
+  active: boolean;
+  expanded?: boolean;
+  style?: React.CSSProperties;
+  activeStyle?: React.CSSProperties;
+  disabled: boolean;
+  inverted?: boolean;
+  animate: boolean;
+  overlay: boolean;
+  tabIndex: number;
+  label?: string;
+  counter?: number;
+  obfuscateCount?: boolean;
+  href?: string;
+  ariaHidden: boolean;
+}
+interface States {
+  activate: boolean;
+  deactivate: boolean;
+}
+export class IconButton extends PureComponent<Props, States> {
+  static defaultProps = {
+    size: 18,
+    active: false,
+    disabled: false,
+    animate: false,
+    overlay: false,
+    tabIndex: 0,
+    ariaHidden: false,
+  };
+
+  state = {
+    activate: false,
+    deactivate: false,
+  };
+
+  UNSAFE_componentWillReceiveProps(nextProps: Props) {
+    if (!nextProps.animate) return;
+
+    if (this.props.active && !nextProps.active) {
+      this.setState({ activate: false, deactivate: true });
+    } else if (!this.props.active && nextProps.active) {
+      this.setState({ activate: true, deactivate: false });
+    }
+  }
+
+  handleClick: React.MouseEventHandler<HTMLButtonElement> = (e) => {
+    e.preventDefault();
+
+    if (!this.props.disabled && this.props.onClick != null) {
+      this.props.onClick(e);
+    }
+  };
+
+  handleKeyPress: React.KeyboardEventHandler<HTMLButtonElement> = (e) => {
+    if (this.props.onKeyPress && !this.props.disabled) {
+      this.props.onKeyPress(e);
+    }
+  };
+
+  handleMouseDown: React.MouseEventHandler<HTMLButtonElement> = (e) => {
+    if (!this.props.disabled && this.props.onMouseDown) {
+      this.props.onMouseDown(e);
+    }
+  };
+
+  handleKeyDown: React.KeyboardEventHandler<HTMLButtonElement> = (e) => {
+    if (!this.props.disabled && this.props.onKeyDown) {
+      this.props.onKeyDown(e);
+    }
+  };
+
+  render() {
+    // Hack required for some icons which have an overriden size
+    let containerSize = '1.28571429em';
+    if (this.props.style?.fontSize) {
+      containerSize = `${this.props.size * 1.28571429}px`;
+    }
+
+    const style = {
+      fontSize: `${this.props.size}px`,
+      height: containerSize,
+      lineHeight: `${this.props.size}px`,
+      ...this.props.style,
+      ...(this.props.active ? this.props.activeStyle : {}),
+    };
+    if (!this.props.label) {
+      style.width = containerSize;
+    } else {
+      style.textAlign = 'left';
+    }
+
+    const {
+      active,
+      className,
+      disabled,
+      expanded,
+      icon,
+      inverted,
+      overlay,
+      tabIndex,
+      title,
+      counter,
+      obfuscateCount,
+      href,
+      ariaHidden,
+    } = this.props;
+
+    const { activate, deactivate } = this.state;
+
+    const classes = classNames(className, 'icon-button', {
+      active,
+      disabled,
+      inverted,
+      activate,
+      deactivate,
+      overlayed: overlay,
+      'icon-button--with-counter': typeof counter !== 'undefined',
+    });
+
+    if (typeof counter !== 'undefined') {
+      style.width = 'auto';
+    }
+
+    let contents = (
+      <>
+        <Icon id={icon} fixedWidth aria-hidden='true' />{' '}
+        {typeof counter !== 'undefined' && (
+          <span className='icon-button__counter'>
+            <AnimatedNumber value={counter} obfuscate={obfuscateCount} />
+          </span>
+        )}
+        {this.props.label}
+      </>
+    );
+
+    if (href != null) {
+      contents = (
+        <a href={href} target='_blank' rel='noopener noreferrer'>
+          {contents}
+        </a>
+      );
+    }
+
+    return (
+      <button
+        type='button'
+        aria-label={title}
+        aria-expanded={expanded}
+        aria-hidden={ariaHidden}
+        title={title}
+        className={classes}
+        onClick={this.handleClick}
+        onMouseDown={this.handleMouseDown}
+        onKeyDown={this.handleKeyDown}
+        onKeyPress={this.handleKeyPress}
+        style={style}
+        tabIndex={tabIndex}
+        disabled={disabled}
+      >
+        {contents}
+      </button>
+    );
+  }
+}
diff --git a/app/javascript/flavours/blobfox/components/icon_with_badge.tsx b/app/javascript/flavours/blobfox/components/icon_with_badge.tsx
new file mode 100644
index 00000000000000..8898f413299480
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/icon_with_badge.tsx
@@ -0,0 +1,24 @@
+import { Icon } from './icon';
+
+const formatNumber = (num: number): number | string => (num > 40 ? '40+' : num);
+
+interface Props {
+  id: string;
+  count: number;
+  issueBadge: boolean;
+  className: string;
+}
+export const IconWithBadge: React.FC<Props> = ({
+  id,
+  count,
+  issueBadge,
+  className,
+}) => (
+  <i className='icon-with-badge'>
+    <Icon id={id} fixedWidth className={className} />
+    {count > 0 && (
+      <i className='icon-with-badge__badge'>{formatNumber(count)}</i>
+    )}
+    {issueBadge && <i className='icon-with-badge__issue-badge' />}
+  </i>
+);
diff --git a/app/javascript/flavours/blobfox/components/inline_account.jsx b/app/javascript/flavours/blobfox/components/inline_account.jsx
new file mode 100644
index 00000000000000..dd913c5e81ccef
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/inline_account.jsx
@@ -0,0 +1,37 @@
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { makeGetAccount } from 'flavours/blobfox/selectors';
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { accountId }) => ({
+    account: getAccount(state, accountId),
+  });
+
+  return mapStateToProps;
+};
+
+class InlineAccount extends PureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+  };
+
+  render () {
+    const { account } = this.props;
+
+    return (
+      <span className='inline-account'>
+        <Avatar size={13} account={account} /> <strong>{account.get('username')}</strong>
+      </span>
+    );
+  }
+
+}
+
+export default connect(makeMapStateToProps)(InlineAccount);
diff --git a/app/javascript/flavours/blobfox/components/intersection_observer_article.jsx b/app/javascript/flavours/blobfox/components/intersection_observer_article.jsx
new file mode 100644
index 00000000000000..8efa969f9bff56
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/intersection_observer_article.jsx
@@ -0,0 +1,131 @@
+import PropTypes from 'prop-types';
+import { cloneElement, Component } from 'react';
+
+import getRectFromEntry from '../features/ui/util/get_rect_from_entry';
+import scheduleIdleTask from '../features/ui/util/schedule_idle_task';
+
+// Diff these props in the "unrendered" state
+const updateOnPropsForUnrendered = ['id', 'index', 'listLength', 'cachedHeight'];
+
+export default class IntersectionObserverArticle extends Component {
+
+  static propTypes = {
+    intersectionObserverWrapper: PropTypes.object.isRequired,
+    id: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+    index: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+    listLength: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+    saveHeightKey: PropTypes.string,
+    cachedHeight: PropTypes.number,
+    onHeightChange: PropTypes.func,
+    children: PropTypes.node,
+  };
+
+  state = {
+    isHidden: false, // set to true in requestIdleCallback to trigger un-render
+  };
+
+  shouldComponentUpdate (nextProps, nextState) {
+    const isUnrendered = !this.state.isIntersecting && (this.state.isHidden || this.props.cachedHeight);
+    const willBeUnrendered = !nextState.isIntersecting && (nextState.isHidden || nextProps.cachedHeight);
+    if (!!isUnrendered !== !!willBeUnrendered) {
+      // If we're going from rendered to unrendered (or vice versa) then update
+      return true;
+    }
+    // If we are and remain hidden, diff based on props
+    if (isUnrendered) {
+      return !updateOnPropsForUnrendered.every(prop => nextProps[prop] === this.props[prop]);
+    }
+    // Else, assume the children have changed
+    return true;
+  }
+
+  componentDidMount () {
+    const { intersectionObserverWrapper, id } = this.props;
+
+    intersectionObserverWrapper.observe(
+      id,
+      this.node,
+      this.handleIntersection,
+    );
+
+    this.componentMounted = true;
+  }
+
+  componentWillUnmount () {
+    const { intersectionObserverWrapper, id } = this.props;
+    intersectionObserverWrapper.unobserve(id, this.node);
+
+    this.componentMounted = false;
+  }
+
+  handleIntersection = (entry) => {
+    this.entry = entry;
+
+    scheduleIdleTask(this.calculateHeight);
+    this.setState(this.updateStateAfterIntersection);
+  };
+
+  updateStateAfterIntersection = (prevState) => {
+    if (prevState.isIntersecting !== false && !this.entry.isIntersecting) {
+      scheduleIdleTask(this.hideIfNotIntersecting);
+    }
+    return {
+      isIntersecting: this.entry.isIntersecting,
+      isHidden: false,
+    };
+  };
+
+  calculateHeight = () => {
+    const { onHeightChange, saveHeightKey, id } = this.props;
+    // save the height of the fully-rendered element (this is expensive
+    // on Chrome, where we need to fall back to getBoundingClientRect)
+    this.height = getRectFromEntry(this.entry).height;
+
+    if (onHeightChange && saveHeightKey) {
+      onHeightChange(saveHeightKey, id, this.height);
+    }
+  };
+
+  hideIfNotIntersecting = () => {
+    if (!this.componentMounted) {
+      return;
+    }
+
+    // When the browser gets a chance, test if we're still not intersecting,
+    // and if so, set our isHidden to true to trigger an unrender. The point of
+    // this is to save DOM nodes and avoid using up too much memory.
+    // See: https://github.com/mastodon/mastodon/issues/2900
+    this.setState((prevState) => ({ isHidden: !prevState.isIntersecting }));
+  };
+
+  handleRef = (node) => {
+    this.node = node;
+  };
+
+  render () {
+    const { children, id, index, listLength, cachedHeight } = this.props;
+    const { isIntersecting, isHidden } = this.state;
+
+    if (!isIntersecting && (isHidden || cachedHeight)) {
+      return (
+        <article
+          ref={this.handleRef}
+          aria-posinset={index + 1}
+          aria-setsize={listLength}
+          style={{ height: `${this.height || cachedHeight}px`, opacity: 0, overflow: 'hidden' }}
+          data-id={id}
+          tabIndex={-1}
+        >
+          {children && cloneElement(children, { hidden: true })}
+        </article>
+      );
+    }
+
+    return (
+      <article ref={this.handleRef} aria-posinset={index + 1} aria-setsize={listLength} data-id={id} tabIndex={-1}>
+        {children && cloneElement(children, { hidden: false })}
+      </article>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/link.jsx b/app/javascript/flavours/blobfox/components/link.jsx
new file mode 100644
index 00000000000000..5059922f6ab1e6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/link.jsx
@@ -0,0 +1,97 @@
+//  Inspired by <CommonLink> from Mastodon GO!
+//  ~ 😘 kibi!
+
+//  Package imports.
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+//  Utils.
+import { assignHandlers } from 'flavours/blobfox/utils/react_helpers';
+//  Handlers.
+const handlers = {
+
+  //  We don't handle clicks that are made with modifiers, since these
+  //  often have special browser meanings (eg, "open in new tab").
+  click (e) {
+    const { onClick } = this.props;
+    if (!onClick || e.button || e.ctrlKey || e.shiftKey || e.altKey || e.metaKey) {
+      return;
+    }
+    onClick(e);
+    e.preventDefault();  //  Prevents following of the link
+  },
+};
+
+//  The component.
+export default class Link extends PureComponent {
+
+  //  Constructor.
+  constructor (props) {
+    super(props);
+    assignHandlers(this, handlers);
+  }
+
+  //  Rendering.
+  render () {
+    const { click } = this.handlers;
+    const {
+      children,
+      className,
+      href,
+      onClick,
+      role,
+      title,
+      ...rest
+    } = this.props;
+    const computedClass = classNames('link', className, `role-${role}`);
+
+    //  We assume that our `onClick` is a routing function and give it
+    //  the qualities of a link even if no `href` is provided. However,
+    //  if we have neither an `onClick` or an `href`, our link is
+    //  purely presentational.
+    const conditionalProps = {};
+    if (href) {
+      conditionalProps.href = href;
+      conditionalProps.onClick = click;
+    } else if (onClick) {
+      conditionalProps.onClick = click;
+      conditionalProps.role = 'link';
+      conditionalProps.tabIndex = 0;
+    } else {
+      conditionalProps.role = 'presentation';
+    }
+
+    //  If we were provided a `role` it overwrites any that we may have
+    //  set above.  This can be used for "links" which are actually
+    //  buttons.
+    if (role) {
+      conditionalProps.role = role;
+    }
+
+    //  Rendering.  We set `rel='noopener'` for user privacy, and our
+    //  `target` as `'_blank'`.
+    return (
+      <a
+        className={computedClass}
+        {...conditionalProps}
+        rel='noopener'
+        target='_blank'
+        title={title}
+        {...rest}
+      >{children}</a>
+    );
+  }
+
+}
+
+//  Props.
+Link.propTypes = {
+  children: PropTypes.node,
+  className: PropTypes.string,
+  href: PropTypes.string,  //  The link destination
+  onClick: PropTypes.func,  //  A function to call instead of opening the link
+  role: PropTypes.string,  //  An ARIA role for the link
+  title: PropTypes.string,  //  A title for the link
+};
diff --git a/app/javascript/flavours/blobfox/components/load_gap.tsx b/app/javascript/flavours/blobfox/components/load_gap.tsx
new file mode 100644
index 00000000000000..565930bb6ba668
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/load_gap.tsx
@@ -0,0 +1,34 @@
+import { useCallback } from 'react';
+
+import { useIntl, defineMessages } from 'react-intl';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const messages = defineMessages({
+  load_more: { id: 'status.load_more', defaultMessage: 'Load more' },
+});
+
+interface Props {
+  disabled: boolean;
+  maxId: string;
+  onClick: (maxId: string) => void;
+}
+
+export const LoadGap: React.FC<Props> = ({ disabled, maxId, onClick }) => {
+  const intl = useIntl();
+
+  const handleClick = useCallback(() => {
+    onClick(maxId);
+  }, [maxId, onClick]);
+
+  return (
+    <button
+      className='load-more load-gap'
+      disabled={disabled}
+      onClick={handleClick}
+      aria-label={intl.formatMessage(messages.load_more)}
+    >
+      <Icon id='ellipsis-h' />
+    </button>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/load_more.tsx b/app/javascript/flavours/blobfox/components/load_more.tsx
new file mode 100644
index 00000000000000..8b5746ad30b34a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/load_more.tsx
@@ -0,0 +1,24 @@
+import { FormattedMessage } from 'react-intl';
+
+interface Props {
+  onClick: (event: React.MouseEvent) => void;
+  disabled?: boolean;
+  visible?: boolean;
+}
+export const LoadMore: React.FC<Props> = ({
+  onClick,
+  disabled,
+  visible = true,
+}) => {
+  return (
+    <button
+      type='button'
+      className='load-more'
+      disabled={disabled || !visible}
+      style={{ visibility: visible ? 'visible' : 'hidden' }}
+      onClick={onClick}
+    >
+      <FormattedMessage id='status.load_more' defaultMessage='Load more' />
+    </button>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/load_pending.tsx b/app/javascript/flavours/blobfox/components/load_pending.tsx
new file mode 100644
index 00000000000000..f7589622edb2d5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/load_pending.tsx
@@ -0,0 +1,18 @@
+import { FormattedMessage } from 'react-intl';
+
+interface Props {
+  onClick: (event: React.MouseEvent) => void;
+  count: number;
+}
+
+export const LoadPending: React.FC<Props> = ({ onClick, count }) => {
+  return (
+    <button className='load-more load-gap' onClick={onClick}>
+      <FormattedMessage
+        id='load_pending'
+        defaultMessage='{count, plural, one {# new item} other {# new items}}'
+        values={{ count }}
+      />
+    </button>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/loading_indicator.tsx b/app/javascript/flavours/blobfox/components/loading_indicator.tsx
new file mode 100644
index 00000000000000..6bc24a0d617ecf
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/loading_indicator.tsx
@@ -0,0 +1,7 @@
+import { CircularProgress } from './circular_progress';
+
+export const LoadingIndicator: React.FC = () => (
+  <div className='loading-indicator'>
+    <CircularProgress size={50} strokeWidth={6} />
+  </div>
+);
diff --git a/app/javascript/flavours/blobfox/components/logo.jsx b/app/javascript/flavours/blobfox/components/logo.jsx
new file mode 100644
index 00000000000000..16ca9f80fd0b04
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/logo.jsx
@@ -0,0 +1,14 @@
+import logo from 'mastodon/../images/logo.svg';
+
+export const WordmarkLogo = () => (
+  <svg viewBox='0 0 261 66' className='logo logo--wordmark' role='img'>
+    <title>Mastodon</title>
+    <use xlinkHref='#logo-symbol-wordmark' />
+  </svg>
+);
+
+export const SymbolLogo = () => (
+  <img src={logo} alt='Mastodon' className='logo logo--icon' />
+);
+
+export default WordmarkLogo;
diff --git a/app/javascript/flavours/blobfox/components/media_attachments.jsx b/app/javascript/flavours/blobfox/components/media_attachments.jsx
new file mode 100644
index 00000000000000..a10a1b26115f26
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/media_attachments.jsx
@@ -0,0 +1,128 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import noop from 'lodash/noop';
+
+import Bundle from 'flavours/blobfox/features/ui/components/bundle';
+import { MediaGallery, Video, Audio } from 'flavours/blobfox/features/ui/util/async-components';
+
+export default class MediaAttachments extends ImmutablePureComponent {
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    lang: PropTypes.string,
+    height: PropTypes.number,
+    width: PropTypes.number,
+    visible: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    height: 110,
+    width: 239,
+  };
+
+  updateOnProps = [
+    'status',
+  ];
+
+  renderLoadingMediaGallery = () => {
+    const { height, width } = this.props;
+
+    return (
+      <div className='media-gallery' style={{ height, width }} />
+    );
+  };
+
+  renderLoadingVideoPlayer = () => {
+    const { height, width } = this.props;
+
+    return (
+      <div className='video-player' style={{ height, width }} />
+    );
+  };
+
+  renderLoadingAudioPlayer = () => {
+    const { height, width } = this.props;
+
+    return (
+      <div className='audio-player' style={{ height, width }} />
+    );
+  };
+
+  render () {
+    const { status, width, height, visible } = this.props;
+    const mediaAttachments = status.get('media_attachments');
+    const language = status.getIn(['language', 'translation']) || status.get('language') || this.props.lang;
+
+    if (mediaAttachments.size === 0) {
+      return null;
+    }
+
+    if (mediaAttachments.getIn([0, 'type']) === 'audio') {
+      const audio = mediaAttachments.get(0);
+      const description = audio.getIn(['translation', 'description']) || audio.get('description');
+
+      return (
+        <Bundle fetchComponent={Audio} loading={this.renderLoadingAudioPlayer} >
+          {Component => (
+            <Component
+              src={audio.get('url')}
+              alt={description}
+              lang={language}
+              width={width}
+              height={height}
+              poster={audio.get('preview_url') || status.getIn(['account', 'avatar_static'])}
+              backgroundColor={audio.getIn(['meta', 'colors', 'background'])}
+              foregroundColor={audio.getIn(['meta', 'colors', 'foreground'])}
+              accentColor={audio.getIn(['meta', 'colors', 'accent'])}
+              duration={audio.getIn(['meta', 'original', 'duration'], 0)}
+            />
+          )}
+        </Bundle>
+      );
+    } else if (mediaAttachments.getIn([0, 'type']) === 'video') {
+      const video = mediaAttachments.get(0);
+      const description = video.getIn(['translation', 'description']) || video.get('description');
+
+      return (
+        <Bundle fetchComponent={Video} loading={this.renderLoadingVideoPlayer} >
+          {Component => (
+            <Component
+              preview={video.get('preview_url')}
+              frameRate={video.getIn(['meta', 'original', 'frame_rate'])}
+              blurhash={video.get('blurhash')}
+              src={video.get('url')}
+              alt={description}
+              lang={language}
+              width={width}
+              height={height}
+              inline
+              sensitive={status.get('sensitive')}
+              visible={visible}
+              onOpenVideo={noop}
+            />
+          )}
+        </Bundle>
+      );
+    } else {
+      return (
+        <Bundle fetchComponent={MediaGallery} loading={this.renderLoadingMediaGallery} >
+          {Component => (
+            <Component
+              media={mediaAttachments}
+              lang={language}
+              sensitive={status.get('sensitive')}
+              defaultWidth={width}
+              visible={visible}
+              height={height}
+              onOpenMedia={noop}
+            />
+          )}
+        </Bundle>
+      );
+    }
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/media_gallery.jsx b/app/javascript/flavours/blobfox/components/media_gallery.jsx
new file mode 100644
index 00000000000000..a2778c39873194
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/media_gallery.jsx
@@ -0,0 +1,395 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { is } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { debounce } from 'lodash';
+
+import { Blurhash } from 'flavours/blobfox/components/blurhash';
+
+import { autoPlayGif, displayMedia, useBlurhash } from '../initial_state';
+
+import { IconButton } from './icon_button';
+
+const messages = defineMessages({
+  hidden: {
+    defaultMessage: 'Media hidden',
+    id: 'status.media_hidden',
+  },
+  sensitive: {
+    defaultMessage: 'Sensitive',
+    id: 'media_gallery.sensitive',
+  },
+  toggle: {
+    defaultMessage: 'Click to view',
+    id: 'status.sensitive_toggle',
+  },
+  toggle_visible: {
+    defaultMessage: '{number, plural, one {Hide image} other {Hide images}}',
+    id: 'media_gallery.toggle_visible',
+  },
+  warning: {
+    defaultMessage: 'Sensitive content',
+    id: 'status.sensitive_warning',
+  },
+});
+
+class Item extends PureComponent {
+
+  static propTypes = {
+    attachment: ImmutablePropTypes.map.isRequired,
+    lang: PropTypes.string,
+    standalone: PropTypes.bool,
+    index: PropTypes.number.isRequired,
+    size: PropTypes.number.isRequired,
+    letterbox: PropTypes.bool,
+    onClick: PropTypes.func.isRequired,
+    displayWidth: PropTypes.number,
+    visible: PropTypes.bool.isRequired,
+    autoplay: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    standalone: false,
+    index: 0,
+    size: 1,
+  };
+
+  state = {
+    loaded: false,
+  };
+
+  handleMouseEnter = (e) => {
+    if (this.hoverToPlay()) {
+      e.target.play();
+    }
+  };
+
+  handleMouseLeave = (e) => {
+    if (this.hoverToPlay()) {
+      e.target.pause();
+      e.target.currentTime = 0;
+    }
+  };
+
+  getAutoPlay() {
+    return this.props.autoplay || autoPlayGif;
+  }
+
+  hoverToPlay () {
+    const { attachment } = this.props;
+    return !this.getAutoPlay() && attachment.get('type') === 'gifv';
+  }
+
+  handleClick = (e) => {
+    const { index, onClick } = this.props;
+
+    if (e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      if (this.hoverToPlay()) {
+        e.target.pause();
+        e.target.currentTime = 0;
+      }
+      e.preventDefault();
+      onClick(index);
+    }
+
+    e.stopPropagation();
+  };
+
+  handleImageLoad = () => {
+    this.setState({ loaded: true });
+  };
+
+  render () {
+    const { attachment, lang, index, size, standalone, letterbox, displayWidth, visible } = this.props;
+
+    let badges = [], thumbnail;
+
+    let width  = 50;
+    let height = 100;
+
+    if (size === 1) {
+      width = 100;
+    }
+
+    if (size === 4 || (size === 3 && index > 0)) {
+      height = 50;
+    }
+
+    if (attachment.get('description')?.length > 0) {
+      badges.push(<span key='alt' className='media-gallery__gifv__label'>ALT</span>);
+    }
+
+    const description = attachment.getIn(['translation', 'description']) || attachment.get('description');
+
+    if (attachment.get('type') === 'unknown') {
+      return (
+        <div className={classNames('media-gallery__item', { standalone, 'media-gallery__item--tall': height === 100, 'media-gallery__item--wide': width === 100 })} key={attachment.get('id')}>
+          <a className='media-gallery__item-thumbnail' href={attachment.get('remote_url') || attachment.get('url')} style={{ cursor: 'pointer' }} title={description} lang={lang} target='_blank' rel='noopener noreferrer'>
+            <Blurhash
+              hash={attachment.get('blurhash')}
+              className='media-gallery__preview'
+              dummy={!useBlurhash}
+            />
+          </a>
+        </div>
+      );
+    } else if (attachment.get('type') === 'image') {
+      const previewUrl   = attachment.get('preview_url');
+      const previewWidth = attachment.getIn(['meta', 'small', 'width']);
+
+      const originalUrl   = attachment.get('url');
+      const originalWidth = attachment.getIn(['meta', 'original', 'width']);
+
+      const hasSize = typeof originalWidth === 'number' && typeof previewWidth === 'number';
+
+      const srcSet = hasSize ? `${originalUrl} ${originalWidth}w, ${previewUrl} ${previewWidth}w` : null;
+      const sizes  = hasSize && (displayWidth > 0) ? `${displayWidth * (width / 100)}px` : null;
+
+      const focusX = attachment.getIn(['meta', 'focus', 'x']) || 0;
+      const focusY = attachment.getIn(['meta', 'focus', 'y']) || 0;
+      const x      = ((focusX /  2) + .5) * 100;
+      const y      = ((focusY / -2) + .5) * 100;
+
+      thumbnail = (
+        <a
+          className='media-gallery__item-thumbnail'
+          href={attachment.get('remote_url') || originalUrl}
+          onClick={this.handleClick}
+          target='_blank'
+          rel='noopener noreferrer'
+        >
+          <img
+            className={letterbox ? 'letterbox' : null}
+            src={previewUrl}
+            srcSet={srcSet}
+            sizes={sizes}
+            alt={description}
+            title={description}
+            lang={lang}
+            style={{ objectPosition: letterbox ? null : `${x}% ${y}%` }}
+            onLoad={this.handleImageLoad}
+          />
+        </a>
+      );
+    } else if (attachment.get('type') === 'gifv') {
+      const autoPlay = this.getAutoPlay();
+
+      badges.push(<span key='gif' className='media-gallery__gifv__label'>GIF</span>);
+
+      thumbnail = (
+        <div className={classNames('media-gallery__gifv', { autoplay: autoPlay })}>
+          <video
+            className={`media-gallery__item-gifv-thumbnail${letterbox ? ' letterbox' : ''}`}
+            aria-label={description}
+            title={description}
+            lang={lang}
+            role='application'
+            src={attachment.get('url')}
+            onClick={this.handleClick}
+            onMouseEnter={this.handleMouseEnter}
+            onMouseLeave={this.handleMouseLeave}
+            autoPlay={autoPlay}
+            playsInline
+            loop
+            muted
+          />
+        </div>
+      );
+    }
+
+    return (
+      <div className={classNames('media-gallery__item', { standalone, letterbox, 'media-gallery__item--tall': height === 100, 'media-gallery__item--wide': width === 100 })} key={attachment.get('id')}>
+        <Blurhash
+          hash={attachment.get('blurhash')}
+          dummy={!useBlurhash}
+          className={classNames('media-gallery__preview', {
+            'media-gallery__preview--hidden': visible && this.state.loaded,
+          })}
+        />
+
+        {visible && thumbnail}
+
+        {badges && (
+          <div className='media-gallery__item__badges'>
+            {badges}
+          </div>
+        )}
+      </div>
+    );
+  }
+
+}
+
+class MediaGallery extends PureComponent {
+
+  static propTypes = {
+    sensitive: PropTypes.bool,
+    standalone: PropTypes.bool,
+    letterbox: PropTypes.bool,
+    fullwidth: PropTypes.bool,
+    hidden: PropTypes.bool,
+    media: ImmutablePropTypes.list.isRequired,
+    lang: PropTypes.string,
+    size: PropTypes.object,
+    onOpenMedia: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    defaultWidth: PropTypes.number,
+    cacheWidth: PropTypes.func,
+    visible: PropTypes.bool,
+    autoplay: PropTypes.bool,
+    onToggleVisibility: PropTypes.func,
+  };
+
+  static defaultProps = {
+    standalone: false,
+  };
+
+  state = {
+    visible: this.props.visible !== undefined ? this.props.visible : (displayMedia !== 'hide_all' && !this.props.sensitive || displayMedia === 'show_all'),
+    width: this.props.defaultWidth,
+  };
+
+  componentDidMount () {
+    window.addEventListener('resize', this.handleResize, { passive: true });
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('resize', this.handleResize);
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (!is(nextProps.media, this.props.media) && nextProps.visible === undefined) {
+      this.setState({ visible: displayMedia !== 'hide_all' && !nextProps.sensitive || displayMedia === 'show_all' });
+    } else if (!is(nextProps.visible, this.props.visible) && nextProps.visible !== undefined) {
+      this.setState({ visible: nextProps.visible });
+    }
+  }
+
+  componentDidUpdate () {
+    if (this.node) {
+      this.handleResize();
+    }
+  }
+
+  handleResize = debounce(() => {
+    if (this.node) {
+      this._setDimensions();
+    }
+  }, 250, {
+    leading: true,
+    trailing: true,
+  });
+
+  handleOpen = () => {
+    if (this.props.onToggleVisibility) {
+      this.props.onToggleVisibility();
+    } else {
+      this.setState({ visible: !this.state.visible });
+    }
+  };
+
+  handleClick = (index) => {
+    this.props.onOpenMedia(this.props.media, index, this.props.lang);
+  };
+
+  handleRef = (node) => {
+    this.node = node;
+
+    if (this.node) {
+      this._setDimensions();
+    }
+  };
+
+  _setDimensions () {
+    const width = this.node.offsetWidth;
+
+    if (width && width !== this.state.width) {
+      // offsetWidth triggers a layout, so only calculate when we need to
+      if (this.props.cacheWidth) {
+        this.props.cacheWidth(width);
+      }
+
+      this.setState({
+        width: width,
+      });
+    }
+  }
+
+  isStandaloneEligible() {
+    const { media, standalone } = this.props;
+    return standalone && media.size === 1 && media.getIn([0, 'meta', 'small', 'aspect']);
+  }
+
+  render () {
+    const { media, lang, intl, sensitive, letterbox, fullwidth, defaultWidth, autoplay } = this.props;
+    const { visible } = this.state;
+    const size     = media.take(4).size;
+    const uncached = media.every(attachment => attachment.get('type') === 'unknown');
+
+    const width = this.state.width || defaultWidth;
+
+    let children, spoilerButton;
+
+    const style = {};
+
+    const computedClass = classNames('media-gallery', { 'full-width': fullwidth });
+
+    if (this.isStandaloneEligible()) { // TODO: cropImages setting
+      style.aspectRatio = `${this.props.media.getIn([0, 'meta', 'small', 'aspect'])}`;
+    } else {
+      style.aspectRatio = '16 / 9';
+    }
+
+    if (this.isStandaloneEligible()) {
+      children = <Item standalone autoplay={autoplay} onClick={this.handleClick} attachment={media.get(0)} lang={lang} displayWidth={width} visible={visible} />;
+    } else {
+      children = media.take(4).map((attachment, i) => <Item key={attachment.get('id')} autoplay={autoplay} onClick={this.handleClick} attachment={attachment} index={i} lang={lang} size={size} letterbox={letterbox} displayWidth={width} visible={visible || uncached} />);
+    }
+
+    if (uncached) {
+      spoilerButton = (
+        <button type='button' disabled className='spoiler-button__overlay'>
+          <span className='spoiler-button__overlay__label'>
+            <FormattedMessage id='status.uncached_media_warning' defaultMessage='Preview not available' />
+            <span className='spoiler-button__overlay__action'><FormattedMessage id='status.media.open' defaultMessage='Click to open' /></span>
+          </span>
+        </button>
+      );
+    } else if (visible) {
+      spoilerButton = <IconButton title={intl.formatMessage(messages.toggle_visible, { number: size })} icon='eye-slash' overlay onClick={this.handleOpen} ariaHidden />;
+    } else {
+      spoilerButton = (
+        <button type='button' onClick={this.handleOpen} className='spoiler-button__overlay'>
+          <span className='spoiler-button__overlay__label'>
+            {sensitive ? <FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' /> : <FormattedMessage id='status.media_hidden' defaultMessage='Media hidden' />}
+            <span className='spoiler-button__overlay__action'><FormattedMessage id='status.media.show' defaultMessage='Click to show' /></span>
+          </span>
+        </button>
+      );
+    }
+
+    return (
+      <div className={computedClass} style={style} ref={this.handleRef}>
+        <div className={classNames('spoiler-button', { 'spoiler-button--minified': visible && !uncached, 'spoiler-button--click-thru': uncached })}>
+          {spoilerButton}
+          {visible && sensitive && (
+            <span className='sensitive-marker'>
+              <FormattedMessage {...messages.sensitive} />
+            </span>
+          )}
+        </div>
+
+        {children}
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(MediaGallery);
diff --git a/app/javascript/flavours/blobfox/components/modal_root.jsx b/app/javascript/flavours/blobfox/components/modal_root.jsx
new file mode 100644
index 00000000000000..d5723c58cde47a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/modal_root.jsx
@@ -0,0 +1,165 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import 'wicg-inert';
+
+import { multiply } from 'color-blend';
+import { createBrowserHistory } from 'history';
+
+import { WithOptionalRouterPropTypes, withOptionalRouter } from 'flavours/blobfox/utils/react_router';
+
+class ModalRoot extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node,
+    onClose: PropTypes.func.isRequired,
+    backgroundColor: PropTypes.shape({
+      r: PropTypes.number,
+      g: PropTypes.number,
+      b: PropTypes.number,
+    }),
+    noEsc: PropTypes.bool,
+    ignoreFocus: PropTypes.bool,
+    ...WithOptionalRouterPropTypes,
+  };
+
+  activeElement = this.props.children ? document.activeElement : null;
+
+  handleKeyUp = (e) => {
+    if ((e.key === 'Escape' || e.key === 'Esc' || e.keyCode === 27)
+         && !!this.props.children && !this.props.noEsc) {
+      this.props.onClose();
+    }
+  };
+
+  handleKeyDown = (e) => {
+    if (e.key === 'Tab') {
+      const focusable = Array.from(this.node.querySelectorAll('button:not([disabled]), [href], input:not([disabled]), select:not([disabled]), textarea:not([disabled]), [tabindex]:not([tabindex="-1"])')).filter((x) => window.getComputedStyle(x).display !== 'none');
+      const index = focusable.indexOf(e.target);
+
+      let element;
+
+      if (e.shiftKey) {
+        element = focusable[index - 1] || focusable[focusable.length - 1];
+      } else {
+        element = focusable[index + 1] || focusable[0];
+      }
+
+      if (element) {
+        element.focus();
+        e.stopPropagation();
+        e.preventDefault();
+      }
+    }
+  };
+
+  componentDidMount () {
+    window.addEventListener('keyup', this.handleKeyUp, false);
+    window.addEventListener('keydown', this.handleKeyDown, false);
+    this.history = this.props.history || createBrowserHistory();
+
+    if (this.props.children) {
+      this._handleModalOpen();
+    }
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (!!nextProps.children && !this.props.children) {
+      this.activeElement = document.activeElement;
+
+      this.getSiblings().forEach(sibling => sibling.setAttribute('inert', true));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    if (!this.props.children && !!prevProps.children) {
+      this.getSiblings().forEach(sibling => sibling.removeAttribute('inert'));
+
+      // Because of the wicg-inert polyfill, the activeElement may not be
+      // immediately selectable, we have to wait for observers to run, as
+      // described in https://github.com/WICG/inert#performance-and-gotchas
+      Promise.resolve().then(() => {
+        if (!this.props.ignoreFocus) {
+          this.activeElement.focus({ preventScroll: true });
+        }
+        this.activeElement = null;
+      }).catch(console.error);
+
+      this._handleModalClose();
+    }
+    if (this.props.children && !prevProps.children) {
+      this._handleModalOpen();
+    }
+    if (this.props.children) {
+      this._ensureHistoryBuffer();
+    }
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('keyup', this.handleKeyUp);
+    window.removeEventListener('keydown', this.handleKeyDown);
+  }
+
+  _handleModalOpen () {
+    this._modalHistoryKey = Date.now();
+    this.unlistenHistory = this.history.listen((_, action) => {
+      if (action === 'POP') {
+        this.props.onClose();
+      }
+    });
+  }
+
+  _handleModalClose () {
+    if (this.unlistenHistory) {
+      this.unlistenHistory();
+    }
+    const { state } = this.history.location;
+    if (state && state.mastodonModalKey === this._modalHistoryKey) {
+      this.history.goBack();
+    }
+  }
+
+  _ensureHistoryBuffer () {
+    const { pathname, state } = this.history.location;
+    if (!state || state.mastodonModalKey !== this._modalHistoryKey) {
+      this.history.push(pathname, { ...state, mastodonModalKey: this._modalHistoryKey });
+    }
+  }
+
+  getSiblings = () => {
+    return Array(...this.node.parentElement.childNodes).filter(node => node !== this.node);
+  };
+
+  setRef = ref => {
+    this.node = ref;
+  };
+
+  render () {
+    const { children, onClose } = this.props;
+    const visible = !!children;
+
+    if (!visible) {
+      return (
+        <div className='modal-root' ref={this.setRef} style={{ opacity: 0 }} />
+      );
+    }
+
+    let backgroundColor = null;
+
+    if (this.props.backgroundColor) {
+      backgroundColor = multiply({ ...this.props.backgroundColor, a: 1 }, { r: 0, g: 0, b: 0, a: 0.7 });
+    }
+
+    return (
+      <div className='modal-root' ref={this.setRef}>
+        <div style={{ pointerEvents: visible ? 'auto' : 'none' }}>
+          <div role='presentation' className='modal-root__overlay' onClick={onClose} style={{ backgroundColor: backgroundColor ? `rgba(${backgroundColor.r}, ${backgroundColor.g}, ${backgroundColor.b}, 0.7)` : null }} />
+          <div role='dialog' className='modal-root__container'>{children}</div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default withOptionalRouter(ModalRoot);
diff --git a/app/javascript/flavours/blobfox/components/navigation_portal.tsx b/app/javascript/flavours/blobfox/components/navigation_portal.tsx
new file mode 100644
index 00000000000000..b1856f4509d6a7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/navigation_portal.tsx
@@ -0,0 +1,25 @@
+import { Switch, Route } from 'react-router-dom';
+
+import AccountNavigation from 'flavours/blobfox/features/account/navigation';
+import Trends from 'flavours/blobfox/features/getting_started/containers/trends_container';
+import { showTrends } from 'flavours/blobfox/initial_state';
+
+const DefaultNavigation: React.FC = () =>
+  showTrends ? (
+    <>
+      <div className='flex-spacer' />
+      <Trends />
+    </>
+  ) : null;
+
+export const NavigationPortal: React.FC = () => (
+  <Switch>
+    <Route path='/@:acct' exact component={AccountNavigation} />
+    <Route path='/@:acct/tagged/:tagged?' exact component={AccountNavigation} />
+    <Route path='/@:acct/with_replies' exact component={AccountNavigation} />
+    <Route path='/@:acct/followers' exact component={AccountNavigation} />
+    <Route path='/@:acct/following' exact component={AccountNavigation} />
+    <Route path='/@:acct/media' exact component={AccountNavigation} />
+    <Route component={DefaultNavigation} />
+  </Switch>
+);
diff --git a/app/javascript/flavours/blobfox/components/not_signed_in_indicator.tsx b/app/javascript/flavours/blobfox/components/not_signed_in_indicator.tsx
new file mode 100644
index 00000000000000..015f74dcaeabed
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/not_signed_in_indicator.tsx
@@ -0,0 +1,12 @@
+import { FormattedMessage } from 'react-intl';
+
+export const NotSignedInIndicator: React.FC = () => (
+  <div className='scrollable scrollable--flex'>
+    <div className='empty-column-indicator'>
+      <FormattedMessage
+        id='not_signed_in_indicator.not_signed_in'
+        defaultMessage='You need to login to access this resource.'
+      />
+    </div>
+  </div>
+);
diff --git a/app/javascript/flavours/blobfox/components/notification_purge_buttons.jsx b/app/javascript/flavours/blobfox/components/notification_purge_buttons.jsx
new file mode 100644
index 00000000000000..2500d082a5c503
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/notification_purge_buttons.jsx
@@ -0,0 +1,64 @@
+/**
+ * Buttons widget for controlling the notification clearing mode.
+ * In idle state, the cleaning mode button is shown. When the mode is active,
+ * a Confirm and Abort buttons are shown in its place.
+ */
+
+
+//  Package imports  //
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const messages = defineMessages({
+  btnAll : { id: 'notification_purge.btn_all', defaultMessage: 'Select\nall' },
+  btnNone : { id: 'notification_purge.btn_none', defaultMessage: 'Select\nnone' },
+  btnInvert : { id: 'notification_purge.btn_invert', defaultMessage: 'Invert\nselection' },
+  btnApply : { id: 'notification_purge.btn_apply', defaultMessage: 'Clear\nselected' },
+});
+
+class NotificationPurgeButtons extends ImmutablePureComponent {
+
+  static propTypes = {
+    onDeleteMarked : PropTypes.func.isRequired,
+    onMarkAll : PropTypes.func.isRequired,
+    onMarkNone : PropTypes.func.isRequired,
+    onInvert : PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    markNewForDelete: PropTypes.bool,
+  };
+
+  render () {
+    const { intl, markNewForDelete } = this.props;
+
+    //className='active'
+    return (
+      <div className='column-header__notif-cleaning-buttons'>
+        <button onClick={this.props.onMarkAll} className={classNames('column-header__button', { active: markNewForDelete })}>
+          <b>∀</b><br />{intl.formatMessage(messages.btnAll)}
+        </button>
+
+        <button onClick={this.props.onMarkNone} className={classNames('column-header__button', { active: !markNewForDelete })}>
+          <b>∅</b><br />{intl.formatMessage(messages.btnNone)}
+        </button>
+
+        <button onClick={this.props.onInvert} className='column-header__button'>
+          <b>¬</b><br />{intl.formatMessage(messages.btnInvert)}
+        </button>
+
+        <button onClick={this.props.onDeleteMarked} className='column-header__button'>
+          <Icon id='trash' /><br />{intl.formatMessage(messages.btnApply)}
+        </button>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(NotificationPurgeButtons);
diff --git a/app/javascript/flavours/blobfox/components/permalink.jsx b/app/javascript/flavours/blobfox/components/permalink.jsx
new file mode 100644
index 00000000000000..d3e77efd3890ea
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/permalink.jsx
@@ -0,0 +1,50 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { withOptionalRouter, WithOptionalRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+class Permalink extends PureComponent {
+
+  static propTypes = {
+    className: PropTypes.string,
+    href: PropTypes.string.isRequired,
+    to: PropTypes.string.isRequired,
+    children: PropTypes.node,
+    onInterceptClick: PropTypes.func,
+    ...WithOptionalRouterPropTypes,
+  };
+
+  handleClick = (e) => {
+    if (e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      if (this.props.onInterceptClick && this.props.onInterceptClick()) {
+        e.preventDefault();
+        return;
+      }
+
+      if (this.props.history) {
+        e.preventDefault();
+        this.props.history.push(this.props.to);
+      }
+    }
+  };
+
+  render () {
+    const {
+      children,
+      className,
+      href,
+      to,
+      onInterceptClick,
+      ...other
+    } = this.props;
+
+    return (
+      <a target='_blank' href={href} onClick={this.handleClick} {...other} className={`permalink${className ? ' ' + className : ''}`}>
+        {children}
+      </a>
+    );
+  }
+
+}
+
+export default withOptionalRouter(Permalink);
diff --git a/app/javascript/flavours/blobfox/components/picture_in_picture_placeholder.jsx b/app/javascript/flavours/blobfox/components/picture_in_picture_placeholder.jsx
new file mode 100644
index 00000000000000..159719b4a12e68
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/picture_in_picture_placeholder.jsx
@@ -0,0 +1,33 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { removePictureInPicture } from 'flavours/blobfox/actions/picture_in_picture';
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+class PictureInPicturePlaceholder extends PureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+  };
+
+  handleClick = () => {
+    const { dispatch } = this.props;
+    dispatch(removePictureInPicture());
+  };
+
+  render () {
+    return (
+      <div className='picture-in-picture-placeholder' role='button' tabIndex={0} onClick={this.handleClick}>
+        <Icon id='window-restore' />
+        <FormattedMessage id='picture_in_picture.restore' defaultMessage='Put it back' />
+      </div>
+    );
+  }
+
+}
+
+export default connect()(PictureInPicturePlaceholder);
diff --git a/app/javascript/flavours/blobfox/components/poll.jsx b/app/javascript/flavours/blobfox/components/poll.jsx
new file mode 100644
index 00000000000000..db3e49073b8415
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/poll.jsx
@@ -0,0 +1,249 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import escapeTextContentForBrowser from 'escape-html';
+import spring from 'react-motion/lib/spring';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import emojify from 'flavours/blobfox/features/emoji/emoji';
+import Motion from 'flavours/blobfox/features/ui/util/optional_motion';
+
+import { RelativeTimestamp } from './relative_timestamp';
+
+const messages = defineMessages({
+  closed: {
+    id: 'poll.closed',
+    defaultMessage: 'Closed',
+  },
+  voted: {
+    id: 'poll.voted',
+    defaultMessage: 'You voted for this answer',
+  },
+  votes: {
+    id: 'poll.votes',
+    defaultMessage: '{votes, plural, one {# vote} other {# votes}}',
+  },
+});
+
+const makeEmojiMap = record => record.get('emojis').reduce((obj, emoji) => {
+  obj[`:${emoji.get('shortcode')}:`] = emoji.toJS();
+  return obj;
+}, {});
+
+class Poll extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    poll: ImmutablePropTypes.map,
+    lang: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    disabled: PropTypes.bool,
+    refresh: PropTypes.func,
+    onVote: PropTypes.func,
+  };
+
+  state = {
+    selected: {},
+    expired: null,
+  };
+
+  static getDerivedStateFromProps (props, state) {
+    const { poll } = props;
+    const expires_at = poll.get('expires_at');
+    const expired = poll.get('expired') || expires_at !== null && (new Date(expires_at)).getTime() < Date.now();
+    return (expired === state.expired) ? null : { expired };
+  }
+
+  componentDidMount () {
+    this._setupTimer();
+  }
+
+  componentDidUpdate () {
+    this._setupTimer();
+  }
+
+  componentWillUnmount () {
+    clearTimeout(this._timer);
+  }
+
+  _setupTimer () {
+    const { poll } = this.props;
+    clearTimeout(this._timer);
+    if (!this.state.expired) {
+      const delay = (new Date(poll.get('expires_at'))).getTime() - Date.now();
+      this._timer = setTimeout(() => {
+        this.setState({ expired: true });
+      }, delay);
+    }
+  }
+
+  _toggleOption = value => {
+    if (this.props.poll.get('multiple')) {
+      const tmp = { ...this.state.selected };
+      if (tmp[value]) {
+        delete tmp[value];
+      } else {
+        tmp[value] = true;
+      }
+      this.setState({ selected: tmp });
+    } else {
+      const tmp = {};
+      tmp[value] = true;
+      this.setState({ selected: tmp });
+    }
+  };
+
+  handleOptionChange = ({ target: { value } }) => {
+    this._toggleOption(value);
+  };
+
+  handleOptionKeyPress = (e) => {
+    if (e.key === 'Enter' || e.key === ' ') {
+      this._toggleOption(e.target.getAttribute('data-index'));
+      e.stopPropagation();
+      e.preventDefault();
+    }
+  };
+
+  handleVote = () => {
+    if (this.props.disabled) {
+      return;
+    }
+
+    this.props.onVote(Object.keys(this.state.selected));
+  };
+
+  handleRefresh = () => {
+    if (this.props.disabled) {
+      return;
+    }
+
+    this.props.refresh();
+  };
+
+  handleReveal = () => {
+    this.setState({ revealed: true });
+  };
+
+  renderOption (option, optionIndex, showResults) {
+    const { poll, lang, disabled, intl } = this.props;
+    const pollVotesCount  = poll.get('voters_count') || poll.get('votes_count');
+    const percent         = pollVotesCount === 0 ? 0 : (option.get('votes_count') / pollVotesCount) * 100;
+    const leading         = poll.get('options').filterNot(other => other.get('title') === option.get('title')).every(other => option.get('votes_count') >= other.get('votes_count'));
+    const active          = !!this.state.selected[`${optionIndex}`];
+    const voted           = option.get('voted') || (poll.get('own_votes') && poll.get('own_votes').includes(optionIndex));
+
+    const title = option.getIn(['translation', 'title']) || option.get('title');
+    let titleHtml = option.getIn(['translation', 'titleHtml']) || option.get('titleHtml');
+
+    if (!titleHtml) {
+      const emojiMap = makeEmojiMap(poll);
+      titleHtml = emojify(escapeTextContentForBrowser(title), emojiMap);
+    }
+
+    return (
+      <li key={option.get('title')}>
+        <label className={classNames('poll__option', { selectable: !showResults })}>
+          <input
+            name='vote-options'
+            type={poll.get('multiple') ? 'checkbox' : 'radio'}
+            value={optionIndex}
+            checked={active}
+            onChange={this.handleOptionChange}
+            disabled={disabled}
+          />
+
+          {!showResults && (
+            <span
+              className={classNames('poll__input', { checkbox: poll.get('multiple'), active })}
+              tabIndex={0}
+              role={poll.get('multiple') ? 'checkbox' : 'radio'}
+              onKeyPress={this.handleOptionKeyPress}
+              aria-checked={active}
+              aria-label={title}
+              lang={lang}
+              data-index={optionIndex}
+            />
+          )}
+          {showResults && (
+            <span
+              className='poll__number'
+              title={intl.formatMessage(messages.votes, {
+                votes: option.get('votes_count'),
+              })}
+            >
+              {Math.round(percent)}%
+            </span>
+          )}
+
+          <span
+            className='poll__option__text translate'
+            lang={lang}
+            dangerouslySetInnerHTML={{ __html: titleHtml }}
+          />
+
+          {!!voted && <span className='poll__voted'>
+            <Icon id='check' className='poll__voted__mark' title={intl.formatMessage(messages.voted)} />
+          </span>}
+        </label>
+
+        {showResults && (
+          <Motion defaultStyle={{ width: 0 }} style={{ width: spring(percent, { stiffness: 180, damping: 12 }) }}>
+            {({ width }) =>
+              <span className={classNames('poll__chart', { leading })} style={{ width: `${width}%` }} />
+            }
+          </Motion>
+        )}
+      </li>
+    );
+  }
+
+  render () {
+    const { poll, intl } = this.props;
+    const { revealed, expired } = this.state;
+
+    if (!poll) {
+      return null;
+    }
+
+    const timeRemaining = expired ? intl.formatMessage(messages.closed) : <RelativeTimestamp timestamp={poll.get('expires_at')} futureDate />;
+    const showResults   = poll.get('voted') || revealed || expired;
+    const disabled      = this.props.disabled || Object.entries(this.state.selected).every(item => !item);
+
+    let votesCount = null;
+
+    if (poll.get('voters_count') !== null && poll.get('voters_count') !== undefined) {
+      votesCount = <FormattedMessage id='poll.total_people' defaultMessage='{count, plural, one {# person} other {# people}}' values={{ count: poll.get('voters_count') }} />;
+    } else {
+      votesCount = <FormattedMessage id='poll.total_votes' defaultMessage='{count, plural, one {# vote} other {# votes}}' values={{ count: poll.get('votes_count') }} />;
+    }
+
+    return (
+      <div className='poll'>
+        <ul>
+          {poll.get('options').map((option, i) => this.renderOption(option, i, showResults))}
+        </ul>
+
+        <div className='poll__footer'>
+          {!showResults && <button className='button button-secondary' disabled={disabled || !this.context.identity.signedIn} onClick={this.handleVote}><FormattedMessage id='poll.vote' defaultMessage='Vote' /></button>}
+          {!showResults && <><button className='poll__link' onClick={this.handleReveal}><FormattedMessage id='poll.reveal' defaultMessage='See results' /></button> · </>}
+          {showResults && !this.props.disabled && <><button className='poll__link' onClick={this.handleRefresh}><FormattedMessage id='poll.refresh' defaultMessage='Refresh' /></button> · </>}
+          {votesCount}
+          {poll.get('expires_at') && <> · {timeRemaining}</>}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Poll);
diff --git a/app/javascript/flavours/blobfox/components/radio_button.tsx b/app/javascript/flavours/blobfox/components/radio_button.tsx
new file mode 100644
index 00000000000000..d0a565b9e65aa3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/radio_button.tsx
@@ -0,0 +1,33 @@
+import classNames from 'classnames';
+
+interface Props {
+  value: string;
+  checked: boolean;
+  name: string;
+  onChange: (event: React.ChangeEvent<HTMLInputElement>) => void;
+  label: React.ReactNode;
+}
+
+export const RadioButton: React.FC<Props> = ({
+  name,
+  value,
+  checked,
+  onChange,
+  label,
+}) => {
+  return (
+    <label className='radio-button'>
+      <input
+        name={name}
+        type='radio'
+        value={value}
+        checked={checked}
+        onChange={onChange}
+      />
+
+      <span className={classNames('radio-button__input', { checked })} />
+
+      <span>{label}</span>
+    </label>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/regeneration_indicator.jsx b/app/javascript/flavours/blobfox/components/regeneration_indicator.jsx
new file mode 100644
index 00000000000000..caaa67ca1dbbf6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/regeneration_indicator.jsx
@@ -0,0 +1,18 @@
+import { FormattedMessage } from 'react-intl';
+
+import illustration from 'flavours/blobfox/images/elephant_ui_working.svg';
+
+const RegenerationIndicator = () => (
+  <div className='regeneration-indicator'>
+    <div className='regeneration-indicator__figure'>
+      <img src={illustration} alt='' />
+    </div>
+
+    <div className='regeneration-indicator__label'>
+      <FormattedMessage id='regeneration_indicator.label' tagName='strong' defaultMessage='Loading&hellip;' />
+      <FormattedMessage id='regeneration_indicator.sublabel' defaultMessage='Your home feed is being prepared!' />
+    </div>
+  </div>
+);
+
+export default RegenerationIndicator;
diff --git a/app/javascript/flavours/blobfox/components/relative_timestamp.tsx b/app/javascript/flavours/blobfox/components/relative_timestamp.tsx
new file mode 100644
index 00000000000000..ac3ab0fb4d48f6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/relative_timestamp.tsx
@@ -0,0 +1,282 @@
+import { Component } from 'react';
+
+import type { IntlShape } from 'react-intl';
+import { injectIntl, defineMessages } from 'react-intl';
+
+const messages = defineMessages({
+  today: { id: 'relative_time.today', defaultMessage: 'today' },
+  just_now: { id: 'relative_time.just_now', defaultMessage: 'now' },
+  just_now_full: {
+    id: 'relative_time.full.just_now',
+    defaultMessage: 'just now',
+  },
+  seconds: { id: 'relative_time.seconds', defaultMessage: '{number}s' },
+  seconds_full: {
+    id: 'relative_time.full.seconds',
+    defaultMessage: '{number, plural, one {# second} other {# seconds}} ago',
+  },
+  minutes: { id: 'relative_time.minutes', defaultMessage: '{number}m' },
+  minutes_full: {
+    id: 'relative_time.full.minutes',
+    defaultMessage: '{number, plural, one {# minute} other {# minutes}} ago',
+  },
+  hours: { id: 'relative_time.hours', defaultMessage: '{number}h' },
+  hours_full: {
+    id: 'relative_time.full.hours',
+    defaultMessage: '{number, plural, one {# hour} other {# hours}} ago',
+  },
+  days: { id: 'relative_time.days', defaultMessage: '{number}d' },
+  days_full: {
+    id: 'relative_time.full.days',
+    defaultMessage: '{number, plural, one {# day} other {# days}} ago',
+  },
+  moments_remaining: {
+    id: 'time_remaining.moments',
+    defaultMessage: 'Moments remaining',
+  },
+  seconds_remaining: {
+    id: 'time_remaining.seconds',
+    defaultMessage: '{number, plural, one {# second} other {# seconds}} left',
+  },
+  minutes_remaining: {
+    id: 'time_remaining.minutes',
+    defaultMessage: '{number, plural, one {# minute} other {# minutes}} left',
+  },
+  hours_remaining: {
+    id: 'time_remaining.hours',
+    defaultMessage: '{number, plural, one {# hour} other {# hours}} left',
+  },
+  days_remaining: {
+    id: 'time_remaining.days',
+    defaultMessage: '{number, plural, one {# day} other {# days}} left',
+  },
+});
+
+const dateFormatOptions = {
+  hour12: false,
+  year: 'numeric',
+  month: 'short',
+  day: '2-digit',
+  hour: '2-digit',
+  minute: '2-digit',
+} as const;
+
+const shortDateFormatOptions = {
+  month: 'short',
+  day: 'numeric',
+} as const;
+
+const SECOND = 1000;
+const MINUTE = 1000 * 60;
+const HOUR = 1000 * 60 * 60;
+const DAY = 1000 * 60 * 60 * 24;
+
+const MAX_DELAY = 2147483647;
+
+const selectUnits = (delta: number) => {
+  const absDelta = Math.abs(delta);
+
+  if (absDelta < MINUTE) {
+    return 'second';
+  } else if (absDelta < HOUR) {
+    return 'minute';
+  } else if (absDelta < DAY) {
+    return 'hour';
+  }
+
+  return 'day';
+};
+
+const getUnitDelay = (units: string) => {
+  switch (units) {
+    case 'second':
+      return SECOND;
+    case 'minute':
+      return MINUTE;
+    case 'hour':
+      return HOUR;
+    case 'day':
+      return DAY;
+    default:
+      return MAX_DELAY;
+  }
+};
+
+export const timeAgoString = (
+  intl: IntlShape,
+  date: Date,
+  now: number,
+  year: number,
+  timeGiven: boolean,
+  short?: boolean,
+) => {
+  const delta = now - date.getTime();
+
+  let relativeTime;
+
+  if (delta < DAY && !timeGiven) {
+    relativeTime = intl.formatMessage(messages.today);
+  } else if (delta < 10 * SECOND) {
+    relativeTime = intl.formatMessage(
+      short ? messages.just_now : messages.just_now_full,
+    );
+  } else if (delta < 7 * DAY) {
+    if (delta < MINUTE) {
+      relativeTime = intl.formatMessage(
+        short ? messages.seconds : messages.seconds_full,
+        { number: Math.floor(delta / SECOND) },
+      );
+    } else if (delta < HOUR) {
+      relativeTime = intl.formatMessage(
+        short ? messages.minutes : messages.minutes_full,
+        { number: Math.floor(delta / MINUTE) },
+      );
+    } else if (delta < DAY) {
+      relativeTime = intl.formatMessage(
+        short ? messages.hours : messages.hours_full,
+        { number: Math.floor(delta / HOUR) },
+      );
+    } else {
+      relativeTime = intl.formatMessage(
+        short ? messages.days : messages.days_full,
+        { number: Math.floor(delta / DAY) },
+      );
+    }
+  } else if (date.getFullYear() === year) {
+    relativeTime = intl.formatDate(date, shortDateFormatOptions);
+  } else {
+    relativeTime = intl.formatDate(date, {
+      ...shortDateFormatOptions,
+      year: 'numeric',
+    });
+  }
+
+  return relativeTime;
+};
+
+const timeRemainingString = (
+  intl: IntlShape,
+  date: Date,
+  now: number,
+  timeGiven = true,
+) => {
+  const delta = date.getTime() - now;
+
+  let relativeTime;
+
+  if (delta < DAY && !timeGiven) {
+    relativeTime = intl.formatMessage(messages.today);
+  } else if (delta < 10 * SECOND) {
+    relativeTime = intl.formatMessage(messages.moments_remaining);
+  } else if (delta < MINUTE) {
+    relativeTime = intl.formatMessage(messages.seconds_remaining, {
+      number: Math.floor(delta / SECOND),
+    });
+  } else if (delta < HOUR) {
+    relativeTime = intl.formatMessage(messages.minutes_remaining, {
+      number: Math.floor(delta / MINUTE),
+    });
+  } else if (delta < DAY) {
+    relativeTime = intl.formatMessage(messages.hours_remaining, {
+      number: Math.floor(delta / HOUR),
+    });
+  } else {
+    relativeTime = intl.formatMessage(messages.days_remaining, {
+      number: Math.floor(delta / DAY),
+    });
+  }
+
+  return relativeTime;
+};
+
+interface Props {
+  intl: IntlShape;
+  timestamp: string;
+  year: number;
+  futureDate?: boolean;
+  short?: boolean;
+}
+interface States {
+  now: number;
+}
+class RelativeTimestamp extends Component<Props, States> {
+  state = {
+    now: Date.now(),
+  };
+
+  static defaultProps = {
+    year: new Date().getFullYear(),
+    short: true,
+  };
+
+  _timer: number | undefined;
+
+  shouldComponentUpdate(nextProps: Props, nextState: States) {
+    // As of right now the locale doesn't change without a new page load,
+    // but we might as well check in case that ever changes.
+    return (
+      this.props.timestamp !== nextProps.timestamp ||
+      this.props.intl.locale !== nextProps.intl.locale ||
+      this.state.now !== nextState.now
+    );
+  }
+
+  UNSAFE_componentWillReceiveProps(nextProps: Props) {
+    if (this.props.timestamp !== nextProps.timestamp) {
+      this.setState({ now: Date.now() });
+    }
+  }
+
+  componentDidMount() {
+    this._scheduleNextUpdate(this.props, this.state);
+  }
+
+  UNSAFE_componentWillUpdate(nextProps: Props, nextState: States) {
+    this._scheduleNextUpdate(nextProps, nextState);
+  }
+
+  componentWillUnmount() {
+    window.clearTimeout(this._timer);
+  }
+
+  _scheduleNextUpdate(props: Props, state: States) {
+    window.clearTimeout(this._timer);
+
+    const { timestamp } = props;
+    const delta = new Date(timestamp).getTime() - state.now;
+    const unitDelay = getUnitDelay(selectUnits(delta));
+    const unitRemainder = Math.abs(delta % unitDelay);
+    const updateInterval = 1000 * 10;
+    const delay =
+      delta < 0
+        ? Math.max(updateInterval, unitDelay - unitRemainder)
+        : Math.max(updateInterval, unitRemainder);
+
+    this._timer = window.setTimeout(() => {
+      this.setState({ now: Date.now() });
+    }, delay);
+  }
+
+  render() {
+    const { timestamp, intl, year, futureDate, short } = this.props;
+
+    const timeGiven = timestamp.includes('T');
+    const date = new Date(timestamp);
+    const relativeTime = futureDate
+      ? timeRemainingString(intl, date, this.state.now, timeGiven)
+      : timeAgoString(intl, date, this.state.now, year, timeGiven, short);
+
+    return (
+      <time
+        dateTime={timestamp}
+        title={intl.formatDate(date, dateFormatOptions)}
+      >
+        {relativeTime}
+      </time>
+    );
+  }
+}
+
+const RelativeTimestampWithIntl = injectIntl(RelativeTimestamp);
+
+export { RelativeTimestampWithIntl as RelativeTimestamp };
diff --git a/app/javascript/flavours/blobfox/components/router.tsx b/app/javascript/flavours/blobfox/components/router.tsx
new file mode 100644
index 00000000000000..188f3b483865cb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/router.tsx
@@ -0,0 +1,80 @@
+import type { PropsWithChildren } from 'react';
+import React from 'react';
+
+import { Router as OriginalRouter } from 'react-router';
+
+import type {
+  LocationDescriptor,
+  LocationDescriptorObject,
+  Path,
+} from 'history';
+import { createBrowserHistory } from 'history';
+
+import { layoutFromWindow } from 'flavours/blobfox/is_mobile';
+
+interface MastodonLocationState {
+  fromMastodon?: boolean;
+  mastodonModalKey?: string;
+}
+type HistoryPath = Path | LocationDescriptor<MastodonLocationState>;
+
+const browserHistory = createBrowserHistory<
+  MastodonLocationState | undefined
+>();
+const originalPush = browserHistory.push.bind(browserHistory);
+const originalReplace = browserHistory.replace.bind(browserHistory);
+
+function normalizePath(
+  path: HistoryPath,
+  state?: MastodonLocationState,
+): LocationDescriptorObject<MastodonLocationState> {
+  const location = typeof path === 'string' ? { pathname: path } : { ...path };
+
+  if (location.state === undefined && state !== undefined) {
+    location.state = state;
+  } else if (
+    location.state !== undefined &&
+    state !== undefined &&
+    process.env.NODE_ENV === 'development'
+  ) {
+    // eslint-disable-next-line no-console
+    console.log(
+      'You should avoid providing a 2nd state argument to push when the 1st argument is a location-like object that already has state; it is ignored',
+    );
+  }
+
+  if (
+    layoutFromWindow() === 'multi-column' &&
+    !location.pathname?.startsWith('/deck')
+  ) {
+    location.pathname = `/deck${location.pathname}`;
+  }
+
+  return location;
+}
+
+browserHistory.push = (path: HistoryPath, state?: MastodonLocationState) => {
+  const location = normalizePath(path, state);
+
+  location.state = location.state ?? {};
+  location.state.fromMastodon = true;
+
+  originalPush(location);
+};
+
+browserHistory.replace = (path: HistoryPath, state?: MastodonLocationState) => {
+  const location = normalizePath(path, state);
+
+  if (!location.pathname) return;
+
+  if (browserHistory.location.state?.fromMastodon) {
+    location.state = location.state ?? {};
+    location.state.fromMastodon = true;
+  }
+
+  originalReplace(location);
+};
+
+export const Router: React.FC<PropsWithChildren> = ({ children }) => {
+  return <OriginalRouter history={browserHistory}>{children}</OriginalRouter>;
+};
diff --git a/app/javascript/flavours/blobfox/components/scrollable_list.jsx b/app/javascript/flavours/blobfox/components/scrollable_list.jsx
new file mode 100644
index 00000000000000..f8061aad96b9c8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/scrollable_list.jsx
@@ -0,0 +1,405 @@
+import PropTypes from 'prop-types';
+import { Children, cloneElement, PureComponent } from 'react';
+
+import classNames from 'classnames';
+import { useLocation } from 'react-router-dom';
+
+import { List as ImmutableList } from 'immutable';
+import { connect } from 'react-redux';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+import { throttle } from 'lodash';
+
+import ScrollContainer from 'flavours/blobfox/containers/scroll_container';
+
+import IntersectionObserverArticleContainer from '../containers/intersection_observer_article_container';
+import { attachFullscreenListener, detachFullscreenListener, isFullscreen } from '../features/ui/util/fullscreen';
+import IntersectionObserverWrapper from '../features/ui/util/intersection_observer_wrapper';
+
+import { LoadMore } from './load_more';
+import { LoadPending } from './load_pending';
+import { LoadingIndicator } from './loading_indicator';
+
+const MOUSE_IDLE_DELAY = 300;
+
+const listenerOptions = supportsPassiveEvents ? { passive: true } : false;
+
+/**
+ *
+ * @param {import('flavours/blobfox/store').RootState} state
+ * @param {*} props
+ */
+const mapStateToProps = (state, { scrollKey }) => {
+  return {
+    preventScroll: scrollKey === state.dropdownMenu.scrollKey,
+  };
+};
+
+// This component only exists to be able to call useLocation()
+const IOArticleContainerWrapper = ({id, index, listLength, intersectionObserverWrapper, trackScroll, scrollKey, children}) => {
+  const location = useLocation();
+
+  return (<IntersectionObserverArticleContainer
+    id={id}
+    index={index}
+    listLength={listLength}
+    intersectionObserverWrapper={intersectionObserverWrapper}
+    saveHeightKey={trackScroll ? `${location.key}:${scrollKey}` : null}
+  >
+    {children}
+  </IntersectionObserverArticleContainer>);
+};
+
+IOArticleContainerWrapper.propTypes =  {
+  id: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+  index: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+  listLength: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
+  scrollKey: PropTypes.string.isRequired,
+  intersectionObserverWrapper: PropTypes.object.isRequired,
+  trackScroll: PropTypes.bool.isRequired,
+  children: PropTypes.node,
+};
+
+class ScrollableList extends PureComponent {
+
+  static propTypes = {
+    scrollKey: PropTypes.string.isRequired,
+    onLoadMore: PropTypes.func,
+    onLoadPending: PropTypes.func,
+    onScrollToTop: PropTypes.func,
+    onScroll: PropTypes.func,
+    trackScroll: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    showLoading: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    numPending: PropTypes.number,
+    prepend: PropTypes.node,
+    append: PropTypes.node,
+    alwaysPrepend: PropTypes.bool,
+    emptyMessage: PropTypes.node,
+    children: PropTypes.node,
+    bindToDocument: PropTypes.bool,
+    preventScroll: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    trackScroll: true,
+  };
+
+  state = {
+    fullscreen: null,
+    cachedMediaWidth: 300,
+  };
+
+  intersectionObserverWrapper = new IntersectionObserverWrapper();
+
+  handleScroll = throttle(() => {
+    if (this.node) {
+      const scrollTop = this.getScrollTop();
+      const scrollHeight = this.getScrollHeight();
+      const clientHeight = this.getClientHeight();
+      const offset = scrollHeight - scrollTop - clientHeight;
+
+      if (scrollTop > 0 && offset < 400 && this.props.onLoadMore && this.props.hasMore && !this.props.isLoading) {
+        this.props.onLoadMore();
+      }
+
+      if (scrollTop < 100 && this.props.onScrollToTop) {
+        this.props.onScrollToTop();
+      } else if (this.props.onScroll) {
+        this.props.onScroll();
+      }
+
+      if (!this.lastScrollWasSynthetic) {
+        // If the last scroll wasn't caused by setScrollTop(), assume it was
+        // intentional and cancel any pending scroll reset on mouse idle
+        this.scrollToTopOnMouseIdle = false;
+      }
+      this.lastScrollWasSynthetic = false;
+    }
+  }, 150, {
+    trailing: true,
+  });
+
+  mouseIdleTimer = null;
+  mouseMovedRecently = false;
+  lastScrollWasSynthetic = false;
+  scrollToTopOnMouseIdle = false;
+
+  _getScrollingElement = () => {
+    if (this.props.bindToDocument) {
+      return (document.scrollingElement || document.body);
+    } else {
+      return this.node;
+    }
+  };
+
+  setScrollTop = newScrollTop => {
+    if (this.getScrollTop() !== newScrollTop) {
+      this.lastScrollWasSynthetic = true;
+
+      this._getScrollingElement().scrollTop = newScrollTop;
+    }
+  };
+
+  clearMouseIdleTimer = () => {
+    if (this.mouseIdleTimer === null) {
+      return;
+    }
+
+    clearTimeout(this.mouseIdleTimer);
+    this.mouseIdleTimer = null;
+  };
+
+  handleMouseMove = throttle(() => {
+    // As long as the mouse keeps moving, clear and restart the idle timer.
+    this.clearMouseIdleTimer();
+    this.mouseIdleTimer = setTimeout(this.handleMouseIdle, MOUSE_IDLE_DELAY);
+
+    if (!this.mouseMovedRecently && this.getScrollTop() === 0) {
+      // Only set if we just started moving and are scrolled to the top.
+      this.scrollToTopOnMouseIdle = true;
+    }
+
+    // Save setting this flag for last, so we can do the comparison above.
+    this.mouseMovedRecently = true;
+  }, MOUSE_IDLE_DELAY / 2);
+
+  handleWheel = throttle(() => {
+    this.scrollToTopOnMouseIdle = false;
+  }, 150, {
+    trailing: true,
+  });
+
+  handleMouseIdle = () => {
+    if (this.scrollToTopOnMouseIdle && !this.props.preventScroll) {
+      this.setScrollTop(0);
+    }
+
+    this.mouseMovedRecently = false;
+    this.scrollToTopOnMouseIdle = false;
+  };
+
+  componentDidMount () {
+    this.attachScrollListener();
+    this.attachIntersectionObserver();
+
+    attachFullscreenListener(this.onFullScreenChange);
+
+    // Handle initial scroll position
+    this.handleScroll();
+  }
+
+  getScrollPosition = () => {
+    if (this.node && (this.getScrollTop() > 0 || this.mouseMovedRecently)) {
+      return { height: this.getScrollHeight(), top: this.getScrollTop() };
+    } else {
+      return null;
+    }
+  };
+
+  getScrollTop = () => {
+    return this._getScrollingElement().scrollTop;
+  };
+
+  getScrollHeight = () => {
+    return this._getScrollingElement().scrollHeight;
+  };
+
+  getClientHeight = () => {
+    return this._getScrollingElement().clientHeight;
+  };
+
+  updateScrollBottom = (snapshot) => {
+    const newScrollTop = this.getScrollHeight() - snapshot;
+
+    this.setScrollTop(newScrollTop);
+  };
+
+  getSnapshotBeforeUpdate (prevProps) {
+    const someItemInserted = Children.count(prevProps.children) > 0 &&
+      Children.count(prevProps.children) < Children.count(this.props.children) &&
+      this.getFirstChildKey(prevProps) !== this.getFirstChildKey(this.props);
+    const pendingChanged = (prevProps.numPending > 0) !== (this.props.numPending > 0);
+
+    if (pendingChanged || someItemInserted && (this.getScrollTop() > 0 || this.mouseMovedRecently || this.props.preventScroll)) {
+      return this.getScrollHeight() - this.getScrollTop();
+    } else {
+      return null;
+    }
+  }
+
+  componentDidUpdate (prevProps, prevState, snapshot) {
+    // Reset the scroll position when a new child comes in in order not to
+    // jerk the scrollbar around if you're already scrolled down the page.
+    if (snapshot !== null) {
+      this.updateScrollBottom(snapshot);
+    }
+  }
+
+  cacheMediaWidth = (width) => {
+    if (width && this.state.cachedMediaWidth !== width) {
+      this.setState({ cachedMediaWidth: width });
+    }
+  };
+
+  componentWillUnmount () {
+    this.clearMouseIdleTimer();
+    this.detachScrollListener();
+    this.detachIntersectionObserver();
+
+    detachFullscreenListener(this.onFullScreenChange);
+  }
+
+  onFullScreenChange = () => {
+    this.setState({ fullscreen: isFullscreen() });
+  };
+
+  attachIntersectionObserver () {
+    let nodeOptions = {
+      root: this.node,
+      rootMargin: '300% 0px',
+    };
+
+    this.intersectionObserverWrapper
+      .connect(this.props.bindToDocument ? {} : nodeOptions);
+  }
+
+  detachIntersectionObserver () {
+    this.intersectionObserverWrapper.disconnect();
+  }
+
+  attachScrollListener () {
+    if (this.props.bindToDocument) {
+      document.addEventListener('scroll', this.handleScroll);
+      document.addEventListener('wheel', this.handleWheel,  listenerOptions);
+    } else {
+      this.node.addEventListener('scroll', this.handleScroll);
+      this.node.addEventListener('wheel', this.handleWheel, listenerOptions);
+    }
+  }
+
+  detachScrollListener () {
+    if (this.props.bindToDocument) {
+      document.removeEventListener('scroll', this.handleScroll);
+      document.removeEventListener('wheel', this.handleWheel, listenerOptions);
+    } else {
+      this.node.removeEventListener('scroll', this.handleScroll);
+      this.node.removeEventListener('wheel', this.handleWheel, listenerOptions);
+    }
+  }
+
+  getFirstChildKey (props) {
+    const { children } = props;
+    let firstChild     = children;
+
+    if (children instanceof ImmutableList) {
+      firstChild = children.get(0);
+    } else if (Array.isArray(children)) {
+      firstChild = children[0];
+    }
+
+    return firstChild && firstChild.key;
+  }
+
+  setRef = (c) => {
+    this.node = c;
+  };
+
+  handleLoadMore = e => {
+    e.preventDefault();
+    this.props.onLoadMore();
+  };
+
+  handleLoadPending = e => {
+    e.preventDefault();
+    this.props.onLoadPending();
+    // Prevent the weird scroll-jumping behavior, as we explicitly don't want to
+    // scroll to top, and we know the scroll height is going to change
+    this.scrollToTopOnMouseIdle = false;
+    this.lastScrollWasSynthetic = false;
+    this.clearMouseIdleTimer();
+    this.mouseIdleTimer = setTimeout(this.handleMouseIdle, MOUSE_IDLE_DELAY);
+    this.mouseMovedRecently = true;
+  };
+
+  render () {
+    const { children, scrollKey, trackScroll, showLoading, isLoading, hasMore, numPending, prepend, alwaysPrepend, append, emptyMessage, onLoadMore } = this.props;
+    const { fullscreen } = this.state;
+    const childrenCount = Children.count(children);
+
+    const loadMore     = (hasMore && onLoadMore) ? <LoadMore visible={!isLoading} onClick={this.handleLoadMore} /> : null;
+    const loadPending  = (numPending > 0) ? <LoadPending count={numPending} onClick={this.handleLoadPending} /> : null;
+    let scrollableArea = null;
+
+    if (showLoading) {
+      scrollableArea = (
+        <div className='scrollable scrollable--flex' ref={this.setRef}>
+          <div role='feed' className='item-list'>
+            {prepend}
+          </div>
+
+          <div className='scrollable__append'>
+            <LoadingIndicator />
+          </div>
+        </div>
+      );
+    } else if (isLoading || childrenCount > 0 || hasMore || !emptyMessage) {
+      scrollableArea = (
+        <div className={classNames('scrollable', { fullscreen })} ref={this.setRef} onMouseMove={this.handleMouseMove}>
+          <div role='feed' className='item-list'>
+            {prepend}
+
+            {loadPending}
+
+            {Children.map(this.props.children, (child, index) => (
+              <IOArticleContainerWrapper
+                key={child.key}
+                id={child.key}
+                index={index}
+                listLength={childrenCount}
+                intersectionObserverWrapper={this.intersectionObserverWrapper}
+                trackScroll={trackScroll}
+                scrollKey={scrollKey}
+              >
+                {cloneElement(child, {
+                  getScrollPosition: this.getScrollPosition,
+                  updateScrollBottom: this.updateScrollBottom,
+                  cachedMediaWidth: this.state.cachedMediaWidth,
+                  cacheMediaWidth: this.cacheMediaWidth,
+                })}
+              </IOArticleContainerWrapper>
+            ))}
+
+            {loadMore}
+
+            {!hasMore && append}
+          </div>
+        </div>
+      );
+    } else {
+      scrollableArea = (
+        <div className={classNames('scrollable scrollable--flex', { fullscreen })} ref={this.setRef}>
+          {alwaysPrepend && prepend}
+
+          <div className='empty-column-indicator'>
+            {emptyMessage}
+          </div>
+        </div>
+      );
+    }
+
+    if (trackScroll) {
+      return (
+        <ScrollContainer scrollKey={scrollKey}>
+          {scrollableArea}
+        </ScrollContainer>
+      );
+    } else {
+      return scrollableArea;
+    }
+  }
+
+}
+
+export default connect(mapStateToProps, null, null, { forwardRef: true })(ScrollableList);
diff --git a/app/javascript/flavours/blobfox/components/server_banner.jsx b/app/javascript/flavours/blobfox/components/server_banner.jsx
new file mode 100644
index 00000000000000..396ea67016b3c9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/server_banner.jsx
@@ -0,0 +1,97 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import { connect } from 'react-redux';
+
+import { fetchServer } from 'flavours/blobfox/actions/server';
+import { ServerHeroImage } from 'flavours/blobfox/components/server_hero_image';
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+import Account from 'flavours/blobfox/containers/account_container';
+import { domain } from 'flavours/blobfox/initial_state';
+
+const messages = defineMessages({
+  aboutActiveUsers: { id: 'server_banner.about_active_users', defaultMessage: 'People using this server during the last 30 days (Monthly Active Users)' },
+});
+
+const mapStateToProps = state => ({
+  server: state.getIn(['server', 'server']),
+});
+
+class ServerBanner extends PureComponent {
+
+  static propTypes = {
+    server: PropTypes.object,
+    dispatch: PropTypes.func,
+    intl: PropTypes.object,
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(fetchServer());
+  }
+
+  render () {
+    const { server, intl } = this.props;
+    const isLoading = server.get('isLoading');
+
+    return (
+      <div className='server-banner'>
+        <div className='server-banner__introduction'>
+          <FormattedMessage id='server_banner.introduction' defaultMessage='{domain} is part of the decentralized social network powered by {mastodon}.' values={{ domain: <strong>{domain}</strong>, mastodon: <a href='https://joinmastodon.org' target='_blank'>Mastodon</a> }} />
+        </div>
+
+        <ServerHeroImage blurhash={server.getIn(['thumbnail', 'blurhash'])} src={server.getIn(['thumbnail', 'url'])} className='server-banner__hero' />
+
+        <div className='server-banner__description'>
+          {isLoading ? (
+            <>
+              <Skeleton width='100%' />
+              <br />
+              <Skeleton width='100%' />
+              <br />
+              <Skeleton width='70%' />
+            </>
+          ) : server.get('description')}
+        </div>
+
+        <div className='server-banner__meta'>
+          <div className='server-banner__meta__column'>
+            <h4><FormattedMessage id='server_banner.administered_by' defaultMessage='Administered by:' /></h4>
+
+            <Account id={server.getIn(['contact', 'account', 'id'])} size={36} minimal />
+          </div>
+
+          <div className='server-banner__meta__column'>
+            <h4><FormattedMessage id='server_banner.server_stats' defaultMessage='Server stats:' /></h4>
+
+            {isLoading ? (
+              <>
+                <strong className='server-banner__number'><Skeleton width='10ch' /></strong>
+                <br />
+                <span className='server-banner__number-label'><Skeleton width='5ch' /></span>
+              </>
+            ) : (
+              <>
+                <strong className='server-banner__number'><ShortNumber value={server.getIn(['usage', 'users', 'active_month'])} /></strong>
+                <br />
+                <span className='server-banner__number-label' title={intl.formatMessage(messages.aboutActiveUsers)}><FormattedMessage id='server_banner.active_users' defaultMessage='active users' /></span>
+              </>
+            )}
+          </div>
+        </div>
+
+        <hr className='spacer' />
+
+        <Link className='button button--block button-secondary' to='/about'><FormattedMessage id='server_banner.learn_more' defaultMessage='Learn more' /></Link>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(ServerBanner));
diff --git a/app/javascript/flavours/blobfox/components/server_hero_image.tsx b/app/javascript/flavours/blobfox/components/server_hero_image.tsx
new file mode 100644
index 00000000000000..68b7f03df39744
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/server_hero_image.tsx
@@ -0,0 +1,35 @@
+import { useCallback, useState } from 'react';
+
+import classNames from 'classnames';
+
+import { Blurhash } from './blurhash';
+
+interface Props {
+  src: string;
+  srcSet?: string;
+  blurhash?: string;
+  className?: string;
+}
+
+export const ServerHeroImage: React.FC<Props> = ({
+  src,
+  srcSet,
+  blurhash,
+  className,
+}) => {
+  const [loaded, setLoaded] = useState(false);
+
+  const handleLoad = useCallback(() => {
+    setLoaded(true);
+  }, [setLoaded]);
+
+  return (
+    <div
+      className={classNames('image', { loaded }, className)}
+      role='presentation'
+    >
+      {blurhash && <Blurhash hash={blurhash} className='image__preview' />}
+      <img src={src} srcSet={srcSet} alt='' onLoad={handleLoad} />
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/components/setting_text.jsx b/app/javascript/flavours/blobfox/components/setting_text.jsx
new file mode 100644
index 00000000000000..79d4bf8ea315e2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/setting_text.jsx
@@ -0,0 +1,35 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+export default class SettingText extends PureComponent {
+
+  static propTypes = {
+    settings: ImmutablePropTypes.map.isRequired,
+    settingPath: PropTypes.array.isRequired,
+    label: PropTypes.string.isRequired,
+    onChange: PropTypes.func.isRequired,
+  };
+
+  handleChange = (e) => {
+    this.props.onChange(this.props.settingPath, e.target.value);
+  };
+
+  render () {
+    const { settings, settingPath, label } = this.props;
+
+    return (
+      <label>
+        <span style={{ display: 'none' }}>{label}</span>
+        <input
+          className='setting-text'
+          value={settings.getIn(settingPath)}
+          onChange={this.handleChange}
+          placeholder={label}
+        />
+      </label>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/short_number.tsx b/app/javascript/flavours/blobfox/components/short_number.tsx
new file mode 100644
index 00000000000000..74c3c5d75e71cf
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/short_number.tsx
@@ -0,0 +1,90 @@
+import { memo } from 'react';
+
+import { FormattedMessage, FormattedNumber } from 'react-intl';
+
+import { toShortNumber, pluralReady, DECIMAL_UNITS } from '../utils/numbers';
+
+type ShortNumberRenderer = (
+  displayNumber: JSX.Element,
+  pluralReady: number,
+) => JSX.Element;
+
+interface ShortNumberProps {
+  value: number;
+  renderer?: ShortNumberRenderer;
+  children?: ShortNumberRenderer;
+}
+
+export const ShortNumberRenderer: React.FC<ShortNumberProps> = ({
+  value,
+  renderer,
+  children,
+}) => {
+  const shortNumber = toShortNumber(value);
+  const [, division] = shortNumber;
+
+  if (children && renderer) {
+    console.warn(
+      'Both renderer prop and renderer as a child provided. This is a mistake and you really should fix that. Only renderer passed as a child will be used.',
+    );
+  }
+
+  const customRenderer = children ?? renderer ?? null;
+
+  const displayNumber = <ShortNumberCounter value={shortNumber} />;
+
+  return (
+    customRenderer?.(displayNumber, pluralReady(value, division)) ??
+    displayNumber
+  );
+};
+export const ShortNumber = memo(ShortNumberRenderer);
+
+interface ShortNumberCounterProps {
+  value: number[];
+}
+const ShortNumberCounter: React.FC<ShortNumberCounterProps> = ({ value }) => {
+  const [rawNumber, unit, maxFractionDigits = 0] = value;
+
+  const count = (
+    <FormattedNumber
+      value={rawNumber}
+      maximumFractionDigits={maxFractionDigits}
+    />
+  );
+
+  const values = { count, rawNumber };
+
+  switch (unit) {
+    case DECIMAL_UNITS.THOUSAND: {
+      return (
+        <FormattedMessage
+          id='units.short.thousand'
+          defaultMessage='{count}K'
+          values={values}
+        />
+      );
+    }
+    case DECIMAL_UNITS.MILLION: {
+      return (
+        <FormattedMessage
+          id='units.short.million'
+          defaultMessage='{count}M'
+          values={values}
+        />
+      );
+    }
+    case DECIMAL_UNITS.BILLION: {
+      return (
+        <FormattedMessage
+          id='units.short.billion'
+          defaultMessage='{count}B'
+          values={values}
+        />
+      );
+    }
+    // Not sure if we should go farther - @Sasha-Sorokin
+    default:
+      return count;
+  }
+};
diff --git a/app/javascript/flavours/blobfox/components/skeleton.tsx b/app/javascript/flavours/blobfox/components/skeleton.tsx
new file mode 100644
index 00000000000000..d6f1aed7238f87
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/skeleton.tsx
@@ -0,0 +1,10 @@
+interface Props {
+  width?: number | string;
+  height?: number | string;
+}
+
+export const Skeleton: React.FC<Props> = ({ width, height }) => (
+  <span className='skeleton' style={{ width, height }}>
+    &zwnj;
+  </span>
+);
diff --git a/app/javascript/flavours/blobfox/components/status.jsx b/app/javascript/flavours/blobfox/components/status.jsx
new file mode 100644
index 00000000000000..f4c4d0d5e387c8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status.jsx
@@ -0,0 +1,879 @@
+import PropTypes from 'prop-types';
+
+import { injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { HotKeys } from 'react-hotkeys';
+
+import PictureInPicturePlaceholder from 'flavours/blobfox/components/picture_in_picture_placeholder';
+import PollContainer from 'flavours/blobfox/containers/poll_container';
+import NotificationOverlayContainer from 'flavours/blobfox/features/notifications/containers/overlay_container';
+import { autoUnfoldCW } from 'flavours/blobfox/utils/content_warning';
+import { withOptionalRouter, WithOptionalRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import Card from '../features/status/components/card';
+// We use the component (and not the container) since we do not want
+// to use the progress bar to show download progress
+import Bundle from '../features/ui/components/bundle';
+import { MediaGallery, Video, Audio } from '../features/ui/util/async-components';
+import { displayMedia, visibleReactions } from '../initial_state';
+
+import AttachmentList from './attachment_list';
+import { getHashtagBarForStatus } from './hashtag_bar';
+import StatusActionBar from './status_action_bar';
+import StatusContent from './status_content';
+import StatusHeader from './status_header';
+import StatusIcons from './status_icons';
+import StatusPrepend from './status_prepend';
+import StatusReactions from './status_reactions';
+
+const domParser = new DOMParser();
+
+export const textForScreenReader = (intl, status, rebloggedByText = false, expanded = false) => {
+  const displayName = status.getIn(['account', 'display_name']);
+
+  const spoilerText = status.getIn(['translation', 'spoiler_text']) || status.get('spoiler_text');
+  const contentHtml = status.getIn(['translation', 'contentHtml']) || status.get('contentHtml');
+  const contentText = domParser.parseFromString(contentHtml, 'text/html').documentElement.textContent;
+
+  const values = [
+    displayName.length === 0 ? status.getIn(['account', 'acct']).split('@')[0] : displayName,
+    spoilerText && !expanded ? spoilerText : contentText,
+    intl.formatDate(status.get('created_at'), { hour: '2-digit', minute: '2-digit', month: 'short', day: 'numeric' }),
+    status.getIn(['account', 'acct']),
+  ];
+
+  if (rebloggedByText) {
+    values.push(rebloggedByText);
+  }
+
+  return values.join(', ');
+};
+
+export const defaultMediaVisibility = (status, settings) => {
+  if (!status) {
+    return undefined;
+  }
+
+  if (status.get('reblog', null) !== null && typeof status.get('reblog') === 'object') {
+    status = status.get('reblog');
+  }
+
+  if (settings.getIn(['media', 'reveal_behind_cw']) && !!status.get('spoiler_text')) {
+    return true;
+  }
+
+  return (displayMedia !== 'hide_all' && !status.get('sensitive') || displayMedia === 'show_all');
+};
+
+class Status extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    containerId: PropTypes.string,
+    id: PropTypes.string,
+    status: ImmutablePropTypes.map,
+    account: ImmutablePropTypes.record,
+    previousId: PropTypes.string,
+    nextInReplyToId: PropTypes.string,
+    rootId: PropTypes.string,
+    onClick: PropTypes.func,
+    onReply: PropTypes.func,
+    onFavourite: PropTypes.func,
+    onReblog: PropTypes.func,
+    onBookmark: PropTypes.func,
+    onDelete: PropTypes.func,
+    onDirect: PropTypes.func,
+    onMention: PropTypes.func,
+    onReactionAdd: PropTypes.func,
+    onReactionRemove: PropTypes.func,
+    onPin: PropTypes.func,
+    onOpenMedia: PropTypes.func,
+    onOpenVideo: PropTypes.func,
+    onBlock: PropTypes.func,
+    onAddFilter: PropTypes.func,
+    onEmbed: PropTypes.func,
+    onHeightChange: PropTypes.func,
+    onToggleHidden: PropTypes.func,
+    onTranslate: PropTypes.func,
+    onInteractionModal: PropTypes.func,
+    muted: PropTypes.bool,
+    hidden: PropTypes.bool,
+    unread: PropTypes.bool,
+    prepend: PropTypes.string,
+    withDismiss: PropTypes.bool,
+    onMoveUp: PropTypes.func,
+    onMoveDown: PropTypes.func,
+    getScrollPosition: PropTypes.func,
+    updateScrollBottom: PropTypes.func,
+    expanded: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    cacheMediaWidth: PropTypes.func,
+    cachedMediaWidth: PropTypes.number,
+    scrollKey: PropTypes.string,
+    deployPictureInPicture: PropTypes.func,
+    settings: ImmutablePropTypes.map.isRequired,
+    pictureInPicture: ImmutablePropTypes.contains({
+      inUse: PropTypes.bool,
+      available: PropTypes.bool,
+    }),
+    ...WithOptionalRouterPropTypes,
+  };
+
+  state = {
+    isCollapsed: false,
+    autoCollapsed: false,
+    isExpanded: undefined,
+    showMedia: undefined,
+    statusId: undefined,
+    revealBehindCW: undefined,
+    showCard: false,
+    forceFilter: undefined,
+  };
+
+  // Avoid checking props that are functions (and whose equality will always
+  // evaluate to false. See react-immutable-pure-component for usage.
+  updateOnProps = [
+    'status',
+    'account',
+    'settings',
+    'prepend',
+    'muted',
+    'notification',
+    'hidden',
+    'expanded',
+    'unread',
+    'pictureInPicture',
+    'previousId',
+    'nextInReplyToId',
+    'rootId',
+  ];
+
+  updateOnStates = [
+    'isExpanded',
+    'isCollapsed',
+    'showMedia',
+    'forceFilter',
+  ];
+
+  //  If our settings have changed to disable collapsed statuses, then we
+  //  need to make sure that we uncollapse every one. We do that by watching
+  //  for changes to `settings.collapsed.enabled` in
+  //  `getderivedStateFromProps()`.
+
+  //  We also need to watch for changes on the `collapse` prop---if this
+  //  changes to anything other than `undefined`, then we need to collapse or
+  //  uncollapse our status accordingly.
+  static getDerivedStateFromProps(nextProps, prevState) {
+    let update = {};
+    let updated = false;
+
+    // Make sure the state mirrors props we track…
+    if (nextProps.expanded !== prevState.expandedProp) {
+      update.expandedProp = nextProps.expanded;
+      updated = true;
+    }
+    if (nextProps.status?.get('hidden') !== prevState.statusPropHidden) {
+      update.statusPropHidden = nextProps.status?.get('hidden');
+      updated = true;
+    }
+
+    // Update state based on new props
+    if (!nextProps.settings.getIn(['collapsed', 'enabled'])) {
+      if (prevState.isCollapsed) {
+        update.isCollapsed = false;
+        updated = true;
+      }
+    }
+
+    // Handle uncollapsing toots when the shared CW state is expanded
+    if (nextProps.settings.getIn(['content_warnings', 'shared_state']) &&
+      nextProps.status?.get('spoiler_text')?.length && nextProps.status?.get('hidden') === false &&
+      prevState.statusPropHidden !== false && prevState.isCollapsed
+    ) {
+      update.isCollapsed = false;
+      updated = true;
+    }
+
+    // The “expanded” prop is used to one-off change the local state.
+    // It's used in the thread view when unfolding/re-folding all CWs at once.
+    if (nextProps.expanded !== prevState.expandedProp &&
+      nextProps.expanded !== undefined
+    ) {
+      update.isExpanded = nextProps.expanded;
+      if (nextProps.expanded) update.isCollapsed = false;
+      updated = true;
+    }
+
+    if (prevState.isExpanded === undefined && update.isExpanded === undefined) {
+      update.isExpanded = autoUnfoldCW(nextProps.settings, nextProps.status);
+      updated = true;
+    }
+
+    if (nextProps.status && nextProps.status.get('id') !== prevState.statusId) {
+      update.showMedia = defaultMediaVisibility(nextProps.status, nextProps.settings);
+      update.statusId = nextProps.status.get('id');
+      updated = true;
+    }
+
+    if (nextProps.settings.getIn(['media', 'reveal_behind_cw']) !== prevState.revealBehindCW) {
+      update.revealBehindCW = nextProps.settings.getIn(['media', 'reveal_behind_cw']);
+      if (update.revealBehindCW) {
+        update.showMedia = defaultMediaVisibility(nextProps.status, nextProps.settings);
+      }
+      updated = true;
+    }
+
+    return updated ? update : null;
+  }
+
+  //  When mounting, we just check to see if our status should be collapsed,
+  //  and collapse it if so. We don't need to worry about whether collapsing
+  //  is enabled here, because `setCollapsed()` already takes that into
+  //  account.
+
+  //  The cases where a status should be collapsed are:
+  //
+  //   -  The `collapse` prop has been set to `true`
+  //   -  The user has decided in local settings to collapse all statuses.
+  //   -  The user has decided to collapse all notifications ('muted'
+  //      statuses).
+  //   -  The user has decided to collapse long statuses and the status is
+  //      over the user set value (default 400 without media, or 610px with).
+  //   -  The status is a reply and the user has decided to collapse all
+  //      replies.
+  //   -  The status contains media and the user has decided to collapse all
+  //      statuses with media.
+  //   -  The status is a reblog the user has decided to collapse all
+  //      statuses which are reblogs.
+  componentDidMount () {
+    const { node } = this;
+    const {
+      status,
+      settings,
+      collapse,
+      muted,
+      prepend,
+    } = this.props;
+
+    // Prevent a crash when node is undefined. Not completely sure why this
+    // happens, might be because status === null.
+    if (node === undefined) return;
+
+    const autoCollapseSettings = settings.getIn(['collapsed', 'auto']);
+
+    // Don't autocollapse if CW state is shared and status is explicitly revealed,
+    // as it could cause surprising changes when receiving notifications
+    if (settings.getIn(['content_warnings', 'shared_state']) && status.get('spoiler_text').length && !status.get('hidden')) return;
+
+    let autoCollapseHeight = parseInt(autoCollapseSettings.get('height'));
+    if (status.get('media_attachments').size && !muted) {
+      autoCollapseHeight += 210;
+    }
+
+    if (collapse ||
+      autoCollapseSettings.get('all') ||
+      (autoCollapseSettings.get('notifications') && muted) ||
+      (autoCollapseSettings.get('lengthy') && node.clientHeight > autoCollapseHeight) ||
+      (autoCollapseSettings.get('reblogs') && prepend === 'reblogged_by') ||
+      (autoCollapseSettings.get('replies') && status.get('in_reply_to_id', null) !== null) ||
+      (autoCollapseSettings.get('media') && !(status.get('spoiler_text').length) && status.get('media_attachments').size > 0)
+    ) {
+      this.setCollapsed(true);
+      // Hack to fix timeline jumps on second rendering when auto-collapsing
+      this.setState({ autoCollapsed: true });
+    }
+
+    // Hack to fix timeline jumps when a preview card is fetched
+    this.setState({
+      showCard: !this.props.muted && !this.props.hidden && this.props.status && this.props.status.get('card') && this.props.settings.get('inline_preview_cards'),
+    });
+  }
+
+  //  Hack to fix timeline jumps on second rendering when auto-collapsing
+  //  or on subsequent rendering when a preview card has been fetched
+  getSnapshotBeforeUpdate() {
+    if (!this.props.getScrollPosition) return null;
+
+    const { muted, hidden, status, settings } = this.props;
+
+    const doShowCard = !muted && !hidden && status && status.get('card') && settings.get('inline_preview_cards');
+    if (this.state.autoCollapsed || (doShowCard && !this.state.showCard)) {
+      if (doShowCard) this.setState({ showCard: true });
+      if (this.state.autoCollapsed) this.setState({ autoCollapsed: false });
+      return this.props.getScrollPosition();
+    } else {
+      return null;
+    }
+  }
+
+  componentDidUpdate(prevProps, prevState, snapshot) {
+    if (snapshot !== null && this.props.updateScrollBottom && this.node.offsetTop < snapshot.top) {
+      this.props.updateScrollBottom(snapshot.height - snapshot.top);
+    }
+  }
+
+  componentWillUnmount() {
+    if (this.node && this.props.getScrollPosition) {
+      const position = this.props.getScrollPosition();
+      if (position !== null && this.node.offsetTop < position.top) {
+        requestAnimationFrame(() => {
+          this.props.updateScrollBottom(position.height - position.top);
+        });
+      }
+    }
+  }
+
+  //  `setCollapsed()` sets the value of `isCollapsed` in our state, that is,
+  //  whether the toot is collapsed or not.
+
+  //  `setCollapsed()` automatically checks for us whether toot collapsing
+  //  is enabled, so we don't have to.
+  setCollapsed = (value) => {
+    if (this.props.settings.getIn(['collapsed', 'enabled'])) {
+      if (value) {
+        this.setExpansion(false);
+      }
+      this.setState({ isCollapsed: value });
+    } else {
+      this.setState({ isCollapsed: false });
+    }
+  };
+
+  setExpansion = (value) => {
+    if (this.props.settings.getIn(['content_warnings', 'shared_state']) && this.props.status.get('hidden') === value) {
+      this.props.onToggleHidden(this.props.status);
+    }
+
+    this.setState({ isExpanded: value });
+    if (value) {
+      this.setCollapsed(false);
+    }
+  };
+
+  //  `parseClick()` takes a click event and responds appropriately.
+  //  If our status is collapsed, then clicking on it should uncollapse it.
+  //  If `Shift` is held, then clicking on it should collapse it.
+  //  Otherwise, we open the url handed to us in `destination`, if
+  //  applicable.
+  parseClick = (e, destination) => {
+    const { status, history } = this.props;
+    const { isCollapsed } = this.state;
+    if (!history) return;
+
+    if (e.button === 0 && !(e.ctrlKey || e.altKey || e.metaKey)) {
+      if (isCollapsed) this.setCollapsed(false);
+      else if (e.shiftKey) {
+        this.setCollapsed(true);
+        document.getSelection().removeAllRanges();
+      } else if (this.props.onClick) {
+        this.props.onClick();
+        return;
+      } else {
+        if (destination === undefined) {
+          destination = `/@${
+            status.getIn(['reblog', 'account', 'acct'], status.getIn(['account', 'acct']))
+          }/${
+            status.getIn(['reblog', 'id'], status.get('id'))
+          }`;
+        }
+        history.push(destination);
+      }
+      e.preventDefault();
+    }
+  };
+
+  handleToggleMediaVisibility = () => {
+    this.setState({ showMedia: !this.state.showMedia });
+  };
+
+  handleExpandedToggle = () => {
+    if (this.props.settings.getIn(['content_warnings', 'shared_state'])) {
+      this.props.onToggleHidden(this.props.status);
+    } else if (this.props.status.get('spoiler_text')) {
+      this.setExpansion(!this.state.isExpanded);
+    }
+  };
+
+  handleOpenVideo = (options) => {
+    const { status } = this.props;
+    const lang = status.getIn(['translation', 'language']) || status.get('language');
+    this.props.onOpenVideo(status.get('id'), status.getIn(['media_attachments', 0]), lang, options);
+  };
+
+  handleOpenMedia = (media, index) => {
+    const { status } = this.props;
+    const lang = status.getIn(['translation', 'language']) || status.get('language');
+    this.props.onOpenMedia(status.get('id'), media, index, lang);
+  };
+
+  handleHotkeyOpenMedia = e => {
+    const { status, onOpenMedia, onOpenVideo } = this.props;
+    const statusId = status.get('id');
+
+    e.preventDefault();
+
+    if (status.get('media_attachments').size > 0) {
+      const lang = status.getIn(['translation', 'language']) || status.get('language');
+      if (status.getIn(['media_attachments', 0, 'type']) === 'video') {
+        onOpenVideo(statusId, status.getIn(['media_attachments', 0]), lang, { startTime: 0 });
+      } else {
+        onOpenMedia(statusId, status.get('media_attachments'), 0, lang);
+      }
+    }
+  };
+
+  handleDeployPictureInPicture = (type, mediaProps) => {
+    const { deployPictureInPicture, status } = this.props;
+
+    deployPictureInPicture(status, type, mediaProps);
+  };
+
+  handleHotkeyReply = e => {
+    e.preventDefault();
+    this.props.onReply(this.props.status, this.props.history);
+  };
+
+  handleHotkeyFavourite = (e) => {
+    this.props.onFavourite(this.props.status, e);
+  };
+
+  handleHotkeyBoost = e => {
+    this.props.onReblog(this.props.status, e);
+  };
+
+  handleHotkeyBookmark = e => {
+    this.props.onBookmark(this.props.status, e);
+  };
+
+  handleHotkeyMention = e => {
+    e.preventDefault();
+    this.props.onMention(this.props.status.get('account'), this.props.history);
+  };
+
+  handleHotkeyOpen = () => {
+    const status = this.props.status;
+    this.props.history.push(`/@${status.getIn(['account', 'acct'])}/${status.get('id')}`);
+  };
+
+  handleHotkeyOpenProfile = () => {
+    this.props.history.push(`/@${this.props.status.getIn(['account', 'acct'])}`);
+  };
+
+  handleHotkeyMoveUp = e => {
+    this.props.onMoveUp(this.props.containerId || this.props.id, e.target.getAttribute('data-featured'));
+  };
+
+  handleHotkeyMoveDown = e => {
+    this.props.onMoveDown(this.props.containerId || this.props.id, e.target.getAttribute('data-featured'));
+  };
+
+  handleHotkeyCollapse = () => {
+    if (!this.props.settings.getIn(['collapsed', 'enabled']))
+      return;
+
+    this.setCollapsed(!this.state.isCollapsed);
+  };
+
+  handleHotkeyToggleSensitive = () => {
+    this.handleToggleMediaVisibility();
+  };
+
+  handleUnfilterClick = e => {
+    this.setState({ forceFilter: false });
+    e.preventDefault();
+  };
+
+  handleFilterClick = () => {
+    this.setState({ forceFilter: true });
+  };
+
+  handleRef = c => {
+    this.node = c;
+  };
+
+  handleTranslate = () => {
+    this.props.onTranslate(this.props.status);
+  };
+
+  renderLoadingMediaGallery () {
+    return <div className='media-gallery' style={{ height: '110px' }} />;
+  }
+
+  renderLoadingVideoPlayer () {
+    return <div className='video-player' style={{ height: '110px' }} />;
+  }
+
+  renderLoadingAudioPlayer () {
+    return <div className='audio-player' style={{ height: '110px' }} />;
+  }
+
+  render () {
+    const { signedIn } = this.context.identity;
+    const {
+      handleRef,
+      parseClick,
+      setCollapsed,
+    } = this;
+    const {
+      intl,
+      status,
+      account,
+      settings,
+      collapsed,
+      muted,
+      intersectionObserverWrapper,
+      onOpenVideo,
+      onOpenMedia,
+      notification,
+      hidden,
+      unread,
+      featured,
+      pictureInPicture,
+      previousId,
+      nextInReplyToId,
+      rootId,
+      history,
+      ...other
+    } = this.props;
+    const { isCollapsed } = this.state;
+    let background = null;
+    let attachments = null;
+
+    //  Depending on user settings, some media are considered as parts of the
+    //  contents (affected by CW) while other will be displayed outside of the
+    //  CW.
+    let contentMedia = [];
+    let contentMediaIcons = [];
+    let extraMedia = [];
+    let extraMediaIcons = [];
+    let media = contentMedia;
+    let mediaIcons = contentMediaIcons;
+
+    if (settings.getIn(['content_warnings', 'media_outside'])) {
+      media = extraMedia;
+      mediaIcons = extraMediaIcons;
+    }
+
+    if (status === null) {
+      return null;
+    }
+
+    const isExpanded = settings.getIn(['content_warnings', 'shared_state']) ? !status.get('hidden') : this.state.isExpanded;
+
+    const handlers = {
+      reply: this.handleHotkeyReply,
+      favourite: this.handleHotkeyFavourite,
+      boost: this.handleHotkeyBoost,
+      mention: this.handleHotkeyMention,
+      open: this.handleHotkeyOpen,
+      openProfile: this.handleHotkeyOpenProfile,
+      moveUp: this.handleHotkeyMoveUp,
+      moveDown: this.handleHotkeyMoveDown,
+      toggleHidden: this.handleExpandedToggle,
+      bookmark: this.handleHotkeyBookmark,
+      toggleCollapse: this.handleHotkeyCollapse,
+      toggleSensitive: this.handleHotkeyToggleSensitive,
+      openMedia: this.handleHotkeyOpenMedia,
+    };
+
+    let prepend, rebloggedByText;
+
+    if (hidden) {
+      return (
+        <HotKeys handlers={handlers}>
+          <div ref={this.handleRef} className='status focusable' tabIndex={0}>
+            <span>{status.getIn(['account', 'display_name']) || status.getIn(['account', 'username'])}</span>
+            <span>{status.get('content')}</span>
+          </div>
+        </HotKeys>
+      );
+    }
+
+    const connectUp = previousId && previousId === status.get('in_reply_to_id');
+    const connectToRoot = rootId && rootId === status.get('in_reply_to_id');
+    const connectReply = nextInReplyToId && nextInReplyToId === status.get('id');
+    const matchedFilters = status.get('matched_filters');
+
+    if (this.state.forceFilter === undefined ? matchedFilters : this.state.forceFilter) {
+      const minHandlers = this.props.muted ? {} : {
+        moveUp: this.handleHotkeyMoveUp,
+        moveDown: this.handleHotkeyMoveDown,
+      };
+
+      return (
+        <HotKeys handlers={minHandlers}>
+          <div className='status__wrapper status__wrapper--filtered focusable' tabIndex={0} ref={this.handleRef}>
+            <FormattedMessage id='status.filtered' defaultMessage='Filtered' />: {matchedFilters.join(', ')}.
+            {' '}
+            <button className='status__wrapper--filtered__button' onClick={this.handleUnfilterClick}>
+              <FormattedMessage id='status.show_filter_reason' defaultMessage='Show anyway' />
+            </button>
+          </div>
+        </HotKeys>
+      );
+    }
+
+    //  If user backgrounds for collapsed statuses are enabled, then we
+    //  initialize our background accordingly. This will only be rendered if
+    //  the status is collapsed.
+    if (settings.getIn(['collapsed', 'backgrounds', 'user_backgrounds'])) {
+      background = status.getIn(['account', 'header']);
+    }
+
+    //  This handles our media attachments.
+    //  If a media file is of unknwon type or if the status is muted
+    //  (notification), we show a list of links instead of embedded media.
+
+    //  After we have generated our appropriate media element and stored it in
+    //  `media`, we snatch the thumbnail to use as our `background` if media
+    //  backgrounds for collapsed statuses are enabled.
+
+    attachments = status.get('media_attachments');
+
+    if (pictureInPicture.get('inUse')) {
+      media.push(<PictureInPicturePlaceholder />);
+      mediaIcons.push('video-camera');
+    } else if (attachments.size > 0) {
+      const language = status.getIn(['translation', 'language']) || status.get('language');
+
+      if (muted || attachments.some(item => item.get('type') === 'unknown')) {
+        media.push(
+          <AttachmentList
+            compact
+            media={status.get('media_attachments')}
+          />,
+        );
+      } else if (attachments.getIn([0, 'type']) === 'audio') {
+        const attachment = status.getIn(['media_attachments', 0]);
+        const description = attachment.getIn(['translation', 'description']) || attachment.get('description');
+
+        media.push(
+          <Bundle fetchComponent={Audio} loading={this.renderLoadingAudioPlayer} >
+            {Component => (
+              <Component
+                src={attachment.get('url')}
+                alt={description}
+                lang={language}
+                poster={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
+                backgroundColor={attachment.getIn(['meta', 'colors', 'background'])}
+                foregroundColor={attachment.getIn(['meta', 'colors', 'foreground'])}
+                accentColor={attachment.getIn(['meta', 'colors', 'accent'])}
+                duration={attachment.getIn(['meta', 'original', 'duration'], 0)}
+                width={this.props.cachedMediaWidth}
+                height={110}
+                cacheWidth={this.props.cacheMediaWidth}
+                deployPictureInPicture={pictureInPicture.get('available') ? this.handleDeployPictureInPicture : undefined}
+                sensitive={status.get('sensitive')}
+                blurhash={attachment.get('blurhash')}
+                visible={this.state.showMedia}
+                onToggleVisibility={this.handleToggleMediaVisibility}
+              />
+            )}
+          </Bundle>,
+        );
+        mediaIcons.push('music');
+      } else if (attachments.getIn([0, 'type']) === 'video') {
+        const attachment = status.getIn(['media_attachments', 0]);
+        const description = attachment.getIn(['translation', 'description']) || attachment.get('description');
+
+        media.push(
+          <Bundle fetchComponent={Video} loading={this.renderLoadingVideoPlayer} >
+            {Component => (<Component
+              preview={attachment.get('preview_url')}
+              frameRate={attachment.getIn(['meta', 'original', 'frame_rate'])}
+              blurhash={attachment.get('blurhash')}
+              src={attachment.get('url')}
+              alt={description}
+              lang={language}
+              inline
+              sensitive={status.get('sensitive')}
+              letterbox={settings.getIn(['media', 'letterbox'])}
+              fullwidth={!rootId && settings.getIn(['media', 'fullwidth'])}
+              preventPlayback={isCollapsed || !isExpanded}
+              onOpenVideo={this.handleOpenVideo}
+              deployPictureInPicture={pictureInPicture.get('available') ? this.handleDeployPictureInPicture : undefined}
+              visible={this.state.showMedia}
+              onToggleVisibility={this.handleToggleMediaVisibility}
+            />)}
+          </Bundle>,
+        );
+        mediaIcons.push('video-camera');
+      } else {  //  Media type is 'image' or 'gifv'
+        media.push(
+          <Bundle fetchComponent={MediaGallery} loading={this.renderLoadingMediaGallery}>
+            {Component => (
+              <Component
+                media={attachments}
+                lang={language}
+                sensitive={status.get('sensitive')}
+                letterbox={settings.getIn(['media', 'letterbox'])}
+                fullwidth={!rootId && settings.getIn(['media', 'fullwidth'])}
+                hidden={isCollapsed || !isExpanded}
+                onOpenMedia={this.handleOpenMedia}
+                cacheWidth={this.props.cacheMediaWidth}
+                defaultWidth={this.props.cachedMediaWidth}
+                visible={this.state.showMedia}
+                onToggleVisibility={this.handleToggleMediaVisibility}
+              />
+            )}
+          </Bundle>,
+        );
+        mediaIcons.push('picture-o');
+      }
+
+      if (!status.get('sensitive') && !(status.get('spoiler_text').length > 0) && settings.getIn(['collapsed', 'backgrounds', 'preview_images'])) {
+        background = attachments.getIn([0, 'preview_url']);
+      }
+    } else if (status.get('card') && settings.get('inline_preview_cards') && !this.props.muted) {
+      media.push(
+        <Card
+          onOpenMedia={this.handleOpenMedia}
+          card={status.get('card')}
+          compact
+          sensitive={status.get('sensitive')}
+        />,
+      );
+      mediaIcons.push('link');
+    }
+
+    if (status.get('poll')) {
+      const language = status.getIn(['translation', 'language']) || status.get('language');
+      contentMedia.push(<PollContainer pollId={status.get('poll')} lang={language} />);
+      contentMediaIcons.push('tasks');
+    }
+
+    //  Here we prepare extra data-* attributes for CSS selectors.
+    //  Users can use those for theming, hiding avatars etc via UserStyle
+    const selectorAttribs = {
+      'data-status-by': `@${status.getIn(['account', 'acct'])}`,
+    };
+
+    if (this.props.prepend && account) {
+      const notifKind = {
+        favourite: 'favourited',
+        reaction: 'reacted',
+        reblog: 'boosted',
+        reblogged_by: 'boosted',
+        status: 'posted',
+      }[this.props.prepend];
+
+      selectorAttribs[`data-${notifKind}-by`] = `@${account.get('acct')}`;
+
+      prepend = (
+        <StatusPrepend
+          type={this.props.prepend}
+          account={account}
+          parseClick={parseClick}
+          notificationId={this.props.notificationId}
+        />
+      );
+    }
+
+    if (this.props.prepend === 'reblog') {
+      rebloggedByText = intl.formatMessage({ id: 'status.reblogged_by', defaultMessage: '{name} boosted' }, { name: account.get('acct') });
+    }
+
+    const computedClass = classNames('status', `status-${status.get('visibility')}`, {
+      collapsed: isCollapsed,
+      'has-background': isCollapsed && background,
+      'status__wrapper-reply': !!status.get('in_reply_to_id'),
+      'status--in-thread': !!rootId,
+      'status--first-in-thread': previousId && (!connectUp || connectToRoot),
+      unread,
+      muted,
+    }, 'focusable');
+
+    const {statusContentProps, hashtagBar} = getHashtagBarForStatus(status);
+    contentMedia.push(hashtagBar);
+
+    return (
+      <HotKeys handlers={handlers}>
+        <div
+          className={computedClass}
+          style={isCollapsed && background ? { backgroundImage: `url(${background})` } : null}
+          {...selectorAttribs}
+          ref={handleRef}
+          tabIndex={0}
+          data-featured={featured ? 'true' : null}
+          aria-label={textForScreenReader(intl, status, rebloggedByText, !status.get('hidden'))}
+          data-nosnippet={status.getIn(['account', 'noindex'], true) || undefined}
+        >
+          {!muted && prepend}
+
+          {(connectReply || connectUp || connectToRoot) && <div className={classNames('status__line', { 'status__line--full': connectReply, 'status__line--first': !status.get('in_reply_to_id') && !connectToRoot })} />}
+
+          <header className='status__info'>
+            <span>
+              {muted && prepend}
+              {!muted || !isCollapsed ? (
+                <StatusHeader
+                  status={status}
+                  friend={account}
+                  collapsed={isCollapsed}
+                  parseClick={parseClick}
+                />
+              ) : null}
+            </span>
+            <StatusIcons
+              status={status}
+              mediaIcons={contentMediaIcons.concat(extraMediaIcons)}
+              collapsible={settings.getIn(['collapsed', 'enabled'])}
+              collapsed={isCollapsed}
+              setCollapsed={setCollapsed}
+              settings={settings.get('status_icons')}
+            />
+          </header>
+          <StatusContent
+            status={status}
+            media={contentMedia}
+            extraMedia={extraMedia}
+            mediaIcons={contentMediaIcons}
+            expanded={isExpanded}
+            onExpandedToggle={this.handleExpandedToggle}
+            onTranslate={this.handleTranslate}
+            parseClick={parseClick}
+            disabled={!history}
+            tagLinks={settings.get('tag_misleading_links')}
+            rewriteMentions={settings.get('rewrite_mentions')}
+            {...statusContentProps}
+          />
+
+          <StatusReactions
+            statusId={status.get('id')}
+            reactions={status.get('reactions')}
+            numVisible={visibleReactions}
+            addReaction={this.props.onReactionAdd}
+            removeReaction={this.props.onReactionRemove}
+            canReact={this.context.identity.signedIn}
+          />
+
+          {!isCollapsed || !(muted || !settings.getIn(['collapsed', 'show_action_bar'])) ? (
+            <StatusActionBar
+              status={status}
+              account={status.get('account')}
+              showReplyCount={settings.get('show_reply_count')}
+              onFilter={matchedFilters ? this.handleFilterClick : null}
+              {...other}
+            />
+          ) : null}
+          {notification ? (
+            <NotificationOverlayContainer
+              notification={notification}
+            />
+          ) : null}
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default withOptionalRouter(injectIntl(Status));
diff --git a/app/javascript/flavours/blobfox/components/status_action_bar.jsx b/app/javascript/flavours/blobfox/components/status_action_bar.jsx
new file mode 100644
index 00000000000000..e5f79c84ca3a90
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_action_bar.jsx
@@ -0,0 +1,371 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { PERMISSION_MANAGE_USERS, PERMISSION_MANAGE_FEDERATION } from 'flavours/blobfox/permissions';
+import { accountAdminLink, statusAdminLink } from 'flavours/blobfox/utils/backend_links';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import DropdownMenuContainer from '../containers/dropdown_menu_container';
+import EmojiPickerDropdown from '../features/compose/containers/emoji_picker_dropdown_container';
+import { me, maxReactions } from '../initial_state';
+
+import { IconButton } from './icon_button';
+import { RelativeTimestamp } from './relative_timestamp';
+
+const messages = defineMessages({
+  delete: { id: 'status.delete', defaultMessage: 'Delete' },
+  redraft: { id: 'status.redraft', defaultMessage: 'Delete & re-draft' },
+  edit: { id: 'status.edit', defaultMessage: 'Edit' },
+  direct: { id: 'status.direct', defaultMessage: 'Privately mention @{name}' },
+  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
+  mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' },
+  block: { id: 'account.block', defaultMessage: 'Block @{name}' },
+  reply: { id: 'status.reply', defaultMessage: 'Reply' },
+  share: { id: 'status.share', defaultMessage: 'Share' },
+  more: { id: 'status.more', defaultMessage: 'More' },
+  replyAll: { id: 'status.replyAll', defaultMessage: 'Reply to thread' },
+  reblog: { id: 'status.reblog', defaultMessage: 'Boost' },
+  reblog_private: { id: 'status.reblog_private', defaultMessage: 'Boost with original visibility' },
+  cancel_reblog_private: { id: 'status.cancel_reblog_private', defaultMessage: 'Unboost' },
+  cannot_reblog: { id: 'status.cannot_reblog', defaultMessage: 'This post cannot be boosted' },
+  favourite: { id: 'status.favourite', defaultMessage: 'Favorite' },
+  react: { id: 'status.react', defaultMessage: 'React' },
+  bookmark: { id: 'status.bookmark', defaultMessage: 'Bookmark' },
+  open: { id: 'status.open', defaultMessage: 'Expand this status' },
+  report: { id: 'status.report', defaultMessage: 'Report @{name}' },
+  muteConversation: { id: 'status.mute_conversation', defaultMessage: 'Mute conversation' },
+  unmuteConversation: { id: 'status.unmute_conversation', defaultMessage: 'Unmute conversation' },
+  pin: { id: 'status.pin', defaultMessage: 'Pin on profile' },
+  unpin: { id: 'status.unpin', defaultMessage: 'Unpin from profile' },
+  embed: { id: 'status.embed', defaultMessage: 'Embed' },
+  admin_account: { id: 'status.admin_account', defaultMessage: 'Open moderation interface for @{name}' },
+  admin_status: { id: 'status.admin_status', defaultMessage: 'Open this post in the moderation interface' },
+  admin_domain: { id: 'status.admin_domain', defaultMessage: 'Open moderation interface for {domain}' },
+  copy: { id: 'status.copy', defaultMessage: 'Copy link to post' },
+  hide: { id: 'status.hide', defaultMessage: 'Hide post' },
+  edited: { id: 'status.edited', defaultMessage: 'Edited {date}' },
+  filter: { id: 'status.filter', defaultMessage: 'Filter this post' },
+  openOriginalPage: { id: 'account.open_original_page', defaultMessage: 'Open original page' },
+});
+
+class StatusActionBar extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    onReply: PropTypes.func,
+    onFavourite: PropTypes.func,
+    onReactionAdd: PropTypes.func,
+    onReblog: PropTypes.func,
+    onDelete: PropTypes.func,
+    onDirect: PropTypes.func,
+    onMention: PropTypes.func,
+    onMute: PropTypes.func,
+    onBlock: PropTypes.func,
+    onReport: PropTypes.func,
+    onEmbed: PropTypes.func,
+    onMuteConversation: PropTypes.func,
+    onPin: PropTypes.func,
+    onBookmark: PropTypes.func,
+    onFilter: PropTypes.func,
+    onAddFilter: PropTypes.func,
+    onInteractionModal: PropTypes.func,
+    withDismiss: PropTypes.bool,
+    withCounters: PropTypes.bool,
+    showReplyCount: PropTypes.bool,
+    scrollKey: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  // Avoid checking props that are functions (and whose equality will always
+  // evaluate to false. See react-immutable-pure-component for usage.
+  updateOnProps = [
+    'status',
+    'showReplyCount',
+    'withCounters',
+    'withDismiss',
+  ];
+
+  handleReplyClick = () => {
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      this.props.onReply(this.props.status, this.props.history);
+    } else {
+      this.props.onInteractionModal('reply', this.props.status);
+    }
+  };
+
+  handleShareClick = () => {
+    navigator.share({
+      url: this.props.status.get('url'),
+    });
+  };
+
+  handleFavouriteClick = (e) => {
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      this.props.onFavourite(this.props.status, e);
+    } else {
+      this.props.onInteractionModal('favourite', this.props.status);
+    }
+  };
+
+  handleEmojiPick = data => {
+    this.props.onReactionAdd(this.props.status.get('id'), data.native.replace(/:/g, ''), data.imageUrl);
+  };
+
+  handleReblogClick = e => {
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      this.props.onReblog(this.props.status, e);
+    } else {
+      this.props.onInteractionModal('reblog', this.props.status);
+    }
+  };
+
+  handleBookmarkClick = (e) => {
+    this.props.onBookmark(this.props.status, e);
+  };
+
+  handleDeleteClick = () => {
+    this.props.onDelete(this.props.status, this.props.history);
+  };
+
+  handleRedraftClick = () => {
+    this.props.onDelete(this.props.status, this.props.history, true);
+  };
+
+  handleEditClick = () => {
+    this.props.onEdit(this.props.status, this.props.history);
+  };
+
+  handlePinClick = () => {
+    this.props.onPin(this.props.status);
+  };
+
+  handleMentionClick = () => {
+    this.props.onMention(this.props.status.get('account'), this.props.history);
+  };
+
+  handleDirectClick = () => {
+    this.props.onDirect(this.props.status.get('account'), this.props.history);
+  };
+
+  handleMuteClick = () => {
+    this.props.onMute(this.props.status.get('account'));
+  };
+
+  handleBlockClick = () => {
+    this.props.onBlock(this.props.status);
+  };
+
+  handleOpen = () => {
+    this.props.history.push(`/@${this.props.status.getIn(['account', 'acct'])}/${this.props.status.get('id')}`);
+  };
+
+  handleEmbed = () => {
+    this.props.onEmbed(this.props.status);
+  };
+
+  handleReport = () => {
+    this.props.onReport(this.props.status);
+  };
+
+  handleConversationMuteClick = () => {
+    this.props.onMuteConversation(this.props.status);
+  };
+
+  handleCopy = () => {
+    const url = this.props.status.get('url');
+    navigator.clipboard.writeText(url);
+  };
+
+  handleHideClick = () => {
+    this.props.onFilter();
+  };
+
+  handleFilterClick = () => {
+    this.props.onAddFilter(this.props.status);
+  };
+
+  handleNoOp = () => {}; // hack for reaction add button
+
+  render () {
+    const { status, intl, withDismiss, withCounters, showReplyCount, scrollKey } = this.props;
+    const { permissions, signedIn } = this.context.identity;
+
+    const mutingConversation = status.get('muted');
+    const publicStatus       = ['public', 'unlisted'].includes(status.get('visibility'));
+    const pinnableStatus     = ['public', 'unlisted', 'private'].includes(status.get('visibility'));
+    const writtenByMe        = status.getIn(['account', 'id']) === me;
+    const isRemote           = status.getIn(['account', 'username']) !== status.getIn(['account', 'acct']);
+
+    let menu = [];
+    let reblogIcon = 'retweet';
+    let replyIcon;
+    let replyTitle;
+
+    menu.push({ text: intl.formatMessage(messages.open), action: this.handleOpen });
+
+    if (publicStatus && isRemote) {
+      menu.push({ text: intl.formatMessage(messages.openOriginalPage), href: status.get('url') });
+    }
+
+    menu.push({ text: intl.formatMessage(messages.copy), action: this.handleCopy });
+
+    if (publicStatus && 'share' in navigator) {
+      menu.push({ text: intl.formatMessage(messages.share), action: this.handleShareClick });
+    }
+
+    if (publicStatus && (signedIn || !isRemote)) {
+      menu.push({ text: intl.formatMessage(messages.embed), action: this.handleEmbed });
+    }
+
+    if (signedIn) {
+      menu.push(null);
+
+      if (writtenByMe && pinnableStatus) {
+        menu.push({ text: intl.formatMessage(status.get('pinned') ? messages.unpin : messages.pin), action: this.handlePinClick });
+        menu.push(null);
+      }
+
+      if (writtenByMe || withDismiss) {
+        menu.push({ text: intl.formatMessage(mutingConversation ? messages.unmuteConversation : messages.muteConversation), action: this.handleConversationMuteClick });
+        menu.push(null);
+      }
+
+      if (writtenByMe) {
+        menu.push({ text: intl.formatMessage(messages.edit), action: this.handleEditClick });
+        menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick, dangerous: true });
+        menu.push({ text: intl.formatMessage(messages.redraft), action: this.handleRedraftClick, dangerous: true });
+      } else {
+        menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
+        menu.push({ text: intl.formatMessage(messages.direct, { name: status.getIn(['account', 'username']) }), action: this.handleDirectClick });
+        menu.push(null);
+
+        if (!this.props.onFilter) {
+          menu.push({ text: intl.formatMessage(messages.filter), action: this.handleFilterClick, dangerous: true });
+          menu.push(null);
+        }
+
+        menu.push({ text: intl.formatMessage(messages.mute, { name: status.getIn(['account', 'username']) }), action: this.handleMuteClick, dangerous: true });
+        menu.push({ text: intl.formatMessage(messages.block, { name: status.getIn(['account', 'username']) }), action: this.handleBlockClick, dangerous: true });
+        menu.push({ text: intl.formatMessage(messages.report, { name: status.getIn(['account', 'username']) }), action: this.handleReport, dangerous: true });
+
+        if (((permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS && (accountAdminLink || statusAdminLink)) || (isRemote && (permissions & PERMISSION_MANAGE_FEDERATION) === PERMISSION_MANAGE_FEDERATION)) {
+          menu.push(null);
+          if ((permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS) {
+            if (accountAdminLink !== undefined) {
+              menu.push({ text: intl.formatMessage(messages.admin_account, { name: status.getIn(['account', 'username']) }), href: accountAdminLink(status.getIn(['account', 'id'])) });
+            }
+            if (statusAdminLink !== undefined) {
+              menu.push({ text: intl.formatMessage(messages.admin_status), href: statusAdminLink(status.getIn(['account', 'id']), status.get('id')) });
+            }
+          }
+          if (isRemote && (permissions & PERMISSION_MANAGE_FEDERATION) === PERMISSION_MANAGE_FEDERATION) {
+            const domain = status.getIn(['account', 'acct']).split('@')[1];
+            menu.push({ text: intl.formatMessage(messages.admin_domain, { domain: domain }), href: `/admin/instances/${domain}` });
+          }
+        }
+      }
+    }
+
+    if (status.get('in_reply_to_id', null) === null) {
+      replyIcon = 'reply';
+      replyTitle = intl.formatMessage(messages.reply);
+    } else {
+      replyIcon = 'reply-all';
+      replyTitle = intl.formatMessage(messages.replyAll);
+    }
+
+    const reblogPrivate = status.getIn(['account', 'id']) === me && status.get('visibility') === 'private';
+
+    let reblogTitle = '';
+    if (status.get('reblogged')) {
+      reblogTitle = intl.formatMessage(messages.cancel_reblog_private);
+    } else if (publicStatus) {
+      reblogTitle = intl.formatMessage(messages.reblog);
+    } else if (reblogPrivate) {
+      reblogTitle = intl.formatMessage(messages.reblog_private);
+    } else {
+      reblogTitle = intl.formatMessage(messages.cannot_reblog);
+    }
+
+    const filterButton = this.props.onFilter && (
+      <IconButton className='status__action-bar-button' title={intl.formatMessage(messages.hide)} icon='eye' onClick={this.handleHideClick} />
+    );
+
+    const canReact = permissions && status.get('reactions').filter(r => r.get('count') > 0 && r.get('me')).size < maxReactions;
+    const reactButton = (
+      <IconButton
+        className='status__action-bar-button'
+        onClick={this.handleNoOp} // EmojiPickerDropdown handles that
+        title={intl.formatMessage(messages.react)}
+        disabled={!canReact}
+        icon='plus'
+      />
+    );
+
+    return (
+      <div className='status__action-bar'>
+        <IconButton
+          className='status__action-bar-button'
+          title={replyTitle}
+          icon={replyIcon}
+          onClick={this.handleReplyClick}
+          counter={showReplyCount ? status.get('replies_count') : undefined}
+          obfuscateCount
+        />
+        <IconButton className={classNames('status__action-bar-button', { reblogPrivate })} disabled={!publicStatus && !reblogPrivate} active={status.get('reblogged')} title={reblogTitle} icon={reblogIcon} onClick={this.handleReblogClick} counter={withCounters ? status.get('reblogs_count') : undefined} />
+        <IconButton className='status__action-bar-button star-icon' animate active={status.get('favourited')} title={intl.formatMessage(messages.favourite)} icon='star' onClick={this.handleFavouriteClick} counter={withCounters ? status.get('favourites_count') : undefined} />
+        {
+          permissions
+            ? <EmojiPickerDropdown className='status__action-bar-button' onPickEmoji={this.handleEmojiPick} button={reactButton} disabled={!canReact} />
+            : reactButton
+        }
+        <IconButton className='status__action-bar-button bookmark-icon' disabled={!signedIn} active={status.get('bookmarked')} title={intl.formatMessage(messages.bookmark)} icon='bookmark' onClick={this.handleBookmarkClick} />
+        {
+          permissions
+            ? <EmojiPickerDropdown className='status__action-bar-button' onPickEmoji={this.handleEmojiPick} button={reactButton} disabled={!canReact} />
+            : reactButton
+        }
+
+        {filterButton}
+
+        <div className='status__action-bar-dropdown'>
+          <DropdownMenuContainer
+            scrollKey={scrollKey}
+            status={status}
+            items={menu}
+            icon='ellipsis-h'
+            size={18}
+            direction='right'
+            ariaLabel={intl.formatMessage(messages.more)}
+          />
+        </div>
+
+        <div className='status__action-bar-spacer' />
+        <a href={status.get('url')} className='status__relative-time' target='_blank' rel='noopener'>
+          <RelativeTimestamp timestamp={status.get('created_at')} />{status.get('edited_at') && <abbr title={intl.formatMessage(messages.edited, { date: intl.formatDate(status.get('edited_at'), { hour12: false, year: 'numeric', month: 'short', day: '2-digit', hour: '2-digit', minute: '2-digit' }) })}> *</abbr>}
+        </a>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(StatusActionBar));
diff --git a/app/javascript/flavours/blobfox/components/status_content.jsx b/app/javascript/flavours/blobfox/components/status_content.jsx
new file mode 100644
index 00000000000000..85df1b58385828
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_content.jsx
@@ -0,0 +1,491 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, injectIntl } from 'react-intl';
+
+import classnames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import { autoPlayGif, languages as preloadedLanguages } from 'flavours/blobfox/initial_state';
+import { decode as decodeIDNA } from 'flavours/blobfox/utils/idna';
+
+import Permalink from './permalink';
+
+const textMatchesTarget = (text, origin, host) => {
+  return (text === origin || text === host
+          || text.startsWith(origin + '/') || text.startsWith(host + '/')
+          || 'www.' + text === host || ('www.' + text).startsWith(host + '/'));
+};
+
+const isLinkMisleading = (link) => {
+  let linkTextParts = [];
+
+  // Reconstruct visible text, as we do not have much control over how links
+  // from remote software look, and we can't rely on `innerText` because the
+  // `invisible` class does not set `display` to `none`.
+
+  const walk = (node) => {
+    switch (node.nodeType) {
+    case Node.TEXT_NODE:
+      linkTextParts.push(node.textContent);
+      break;
+    case Node.ELEMENT_NODE:
+      if (node.classList.contains('invisible')) return;
+      const children = node.childNodes;
+      for (let i = 0; i < children.length; i++) {
+        walk(children[i]);
+      }
+      break;
+    }
+  };
+
+  walk(link);
+
+  const linkText = linkTextParts.join('');
+  const targetURL = new URL(link.href);
+
+  if (targetURL.protocol === 'magnet:') {
+    return !linkText.startsWith('magnet:');
+  }
+
+  if (targetURL.protocol === 'xmpp:') {
+    return !(linkText === targetURL.href || 'xmpp:' + linkText === targetURL.href);
+  }
+
+  // The following may not work with international domain names
+  if (textMatchesTarget(linkText, targetURL.origin, targetURL.host) || textMatchesTarget(linkText.toLowerCase(), targetURL.origin, targetURL.host)) {
+    return false;
+  }
+
+  // The link hasn't been recognized, maybe it features an international domain name
+  const hostname = decodeIDNA(targetURL.hostname).normalize('NFKC');
+  const host = targetURL.host.replace(targetURL.hostname, hostname);
+  const origin = targetURL.origin.replace(targetURL.host, host);
+  const text = linkText.normalize('NFKC');
+  return !(textMatchesTarget(text, origin, host) || textMatchesTarget(text.toLowerCase(), origin, host));
+};
+
+/**
+ *
+ * @param {any} status
+ * @returns {string}
+ */
+export function getStatusContent(status) {
+  return status.getIn(['translation', 'contentHtml']) || status.get('contentHtml');
+}
+
+class TranslateButton extends PureComponent {
+
+  static propTypes = {
+    translation: ImmutablePropTypes.map,
+    onClick: PropTypes.func,
+  };
+
+  render () {
+    const { translation, onClick } = this.props;
+
+    if (translation) {
+      const language     = preloadedLanguages.find(lang => lang[0] === translation.get('detected_source_language'));
+      const languageName = language ? language[2] : translation.get('detected_source_language');
+      const provider     = translation.get('provider');
+
+      return (
+        <div className='translate-button'>
+          <div className='translate-button__meta'>
+            <FormattedMessage id='status.translated_from_with' defaultMessage='Translated from {lang} using {provider}' values={{ lang: languageName, provider }} />
+          </div>
+
+          <button className='link-button' onClick={onClick}>
+            <FormattedMessage id='status.show_original' defaultMessage='Show original' />
+          </button>
+        </div>
+      );
+    }
+
+    return (
+      <button className='status__content__translate-button' onClick={onClick}>
+        <FormattedMessage id='status.translate' defaultMessage='Translate' />
+      </button>
+    );
+  }
+
+}
+
+const mapStateToProps = state => ({
+  languages: state.getIn(['server', 'translationLanguages', 'items']),
+});
+
+class StatusContent extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    statusContent: PropTypes.string,
+    expanded: PropTypes.bool,
+    collapsed: PropTypes.bool,
+    onExpandedToggle: PropTypes.func,
+    onTranslate: PropTypes.func,
+    media: PropTypes.node,
+    extraMedia: PropTypes.node,
+    mediaIcons: PropTypes.arrayOf(PropTypes.string),
+    parseClick: PropTypes.func,
+    disabled: PropTypes.bool,
+    onUpdate: PropTypes.func,
+    tagLinks: PropTypes.bool,
+    rewriteMentions: PropTypes.string,
+    languages: ImmutablePropTypes.map,
+    intl: PropTypes.object,
+    // from react-router
+    match: PropTypes.object.isRequired,
+    location: PropTypes.object.isRequired,
+    history: PropTypes.object.isRequired
+  };
+
+  static defaultProps = {
+    tagLinks: true,
+    rewriteMentions: 'no',
+  };
+
+  state = {
+    hidden: true,
+  };
+
+  _updateStatusLinks () {
+    const node = this.contentsNode;
+    const { tagLinks, rewriteMentions } = this.props;
+
+    if (!node) {
+      return;
+    }
+
+    const links = node.querySelectorAll('a');
+
+    for (var i = 0; i < links.length; ++i) {
+      let link = links[i];
+      if (link.classList.contains('status-link')) {
+        continue;
+      }
+      link.classList.add('status-link');
+
+      let mention = this.props.status.get('mentions').find(item => link.href === item.get('url'));
+
+      if (mention) {
+        link.addEventListener('click', this.onMentionClick.bind(this, mention), false);
+        link.setAttribute('title', `@${mention.get('acct')}`);
+        if (rewriteMentions !== 'no') {
+          while (link.firstChild) link.removeChild(link.firstChild);
+          link.appendChild(document.createTextNode('@'));
+          const acctSpan = document.createElement('span');
+          acctSpan.textContent = rewriteMentions === 'acct' ? mention.get('acct') : mention.get('username');
+          link.appendChild(acctSpan);
+        }
+      } else if (link.textContent[0] === '#' || (link.previousSibling && link.previousSibling.textContent && link.previousSibling.textContent[link.previousSibling.textContent.length - 1] === '#')) {
+        link.addEventListener('click', this.onHashtagClick.bind(this, link.text), false);
+      } else {
+        link.addEventListener('click', this.onLinkClick.bind(this), false);
+        link.setAttribute('title', link.href);
+        link.classList.add('unhandled-link');
+
+        link.setAttribute('target', '_blank');
+        link.setAttribute('rel', 'noopener nofollow noreferrer');
+
+        try {
+          if (tagLinks && isLinkMisleading(link)) {
+            // Add a tag besides the link to display its origin
+
+            const url = new URL(link.href);
+            const tag = document.createElement('span');
+            tag.classList.add('link-origin-tag');
+            switch (url.protocol) {
+            case 'xmpp:':
+              tag.textContent = `[${url.href}]`;
+              break;
+            case 'magnet:':
+              tag.textContent = '(magnet)';
+              break;
+            default:
+              tag.textContent = `[${url.host}]`;
+            }
+            link.insertAdjacentText('beforeend', ' ');
+            link.insertAdjacentElement('beforeend', tag);
+          }
+        } catch (e) {
+          // The URL is invalid, remove the href just to be safe
+          if (tagLinks && e instanceof TypeError) link.removeAttribute('href');
+        }
+      }
+    }
+  }
+
+  handleMouseEnter = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-original');
+    }
+  };
+
+  handleMouseLeave = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-static');
+    }
+  };
+
+  componentDidMount () {
+    this._updateStatusLinks();
+  }
+
+  componentDidUpdate () {
+    this._updateStatusLinks();
+    if (this.props.onUpdate) this.props.onUpdate();
+  }
+
+  onLinkClick = (e) => {
+    if (this.props.collapsed) {
+      if (this.props.parseClick) this.props.parseClick(e);
+    }
+  };
+
+  onMentionClick = (mention, e) => {
+    if (this.props.parseClick) {
+      this.props.parseClick(e, `/@${mention.get('acct')}`);
+    }
+  };
+
+  onHashtagClick = (hashtag, e) => {
+    hashtag = hashtag.replace(/^#/, '');
+
+    if (this.props.parseClick) {
+      this.props.parseClick(e, `/tags/${hashtag}`);
+    }
+  };
+
+  handleMouseDown = (e) => {
+    this.startXY = [e.clientX, e.clientY];
+  };
+
+  handleMouseUp = (e) => {
+    const { parseClick, disabled } = this.props;
+
+    if (disabled || !this.startXY) {
+      return;
+    }
+
+    const [ startX, startY ] = this.startXY;
+    const [ deltaX, deltaY ] = [Math.abs(e.clientX - startX), Math.abs(e.clientY - startY)];
+
+    let element = e.target;
+    while (element !== e.currentTarget) {
+      if (['button', 'video', 'a', 'label', 'canvas'].includes(element.localName) || element.getAttribute('role') === 'button') {
+        return;
+      }
+      element = element.parentNode;
+    }
+
+    if (deltaX + deltaY < 5 && e.button === 0 && parseClick) {
+      parseClick(e);
+    }
+
+    this.startXY = null;
+  };
+
+  handleSpoilerClick = (e) => {
+    e.preventDefault();
+
+    if (this.props.onExpandedToggle) {
+      this.props.onExpandedToggle();
+    } else {
+      this.setState({ hidden: !this.state.hidden });
+    }
+  };
+
+  handleTranslate = () => {
+    this.props.onTranslate();
+  };
+
+  setContentsRef = (c) => {
+    this.contentsNode = c;
+  };
+
+  render () {
+    const {
+      status,
+      media,
+      extraMedia,
+      mediaIcons,
+      parseClick,
+      disabled,
+      tagLinks,
+      rewriteMentions,
+      intl,
+      statusContent,
+    } = this.props;
+
+    const hidden = this.props.onExpandedToggle ? !this.props.expanded : this.state.hidden;
+    const contentLocale = intl.locale.replace(/[_-].*/, '');
+    const targetLanguages = this.props.languages?.get(status.get('language') || 'und');
+    const renderTranslate = this.props.onTranslate && this.context.identity.signedIn && ['public', 'unlisted'].includes(status.get('visibility')) && status.get('search_index').trim().length > 0 && targetLanguages?.includes(contentLocale);
+
+    const content = { __html: statusContent ?? getStatusContent(status) };
+    const spoilerContent = { __html: status.getIn(['translation', 'spoilerHtml']) || status.get('spoilerHtml') };
+    const language = status.getIn(['translation', 'language']) || status.get('language');
+    const classNames = classnames('status__content', {
+      'status__content--with-action': parseClick && !disabled,
+      'status__content--with-spoiler': status.get('spoiler_text').length > 0,
+    });
+
+    const translateButton = renderTranslate && (
+      <TranslateButton onClick={this.handleTranslate} translation={status.get('translation')} />
+    );
+
+    if (status.get('spoiler_text').length > 0) {
+      let mentionsPlaceholder = '';
+
+      const mentionLinks = status.get('mentions').map(item => (
+        <Permalink
+          to={`/@${item.get('acct')}`}
+          href={item.get('url')}
+          key={item.get('id')}
+          className='mention'
+        >
+          @<span>{item.get('username')}</span>
+        </Permalink>
+      )).reduce((aggregate, item) => [...aggregate, item, ' '], []);
+
+      let toggleText = null;
+      if (hidden) {
+        toggleText = [
+          <FormattedMessage
+            id='status.show_more'
+            defaultMessage='Show more'
+            key='0'
+          />,
+        ];
+        if (mediaIcons) {
+          mediaIcons.forEach((mediaIcon, idx) => {
+            toggleText.push(
+              <Icon
+                fixedWidth
+                className='status__content__spoiler-icon'
+                id={mediaIcon}
+                aria-hidden='true'
+                key={`icon-${idx}`}
+              />,
+            );
+          });
+        }
+      } else {
+        toggleText = (
+          <FormattedMessage
+            id='status.show_less'
+            defaultMessage='Show less'
+            key='0'
+          />
+        );
+      }
+
+      if (hidden) {
+        mentionsPlaceholder = <div>{mentionLinks}</div>;
+      }
+
+      return (
+        <div className={classNames} tabIndex={0} onMouseDown={this.handleMouseDown} onMouseUp={this.handleMouseUp}>
+          <p
+            style={{ marginBottom: hidden && status.get('mentions').isEmpty() ? '0px' : null }}
+          >
+            <span dangerouslySetInnerHTML={spoilerContent} className='translate' lang={language} />
+            {' '}
+            <button type='button' className='status__content__spoiler-link' onClick={this.handleSpoilerClick} aria-expanded={!hidden}>
+              {toggleText}
+            </button>
+          </p>
+
+          {mentionsPlaceholder}
+
+          <div className={`status__content__spoiler ${!hidden ? 'status__content__spoiler--visible' : ''}`}>
+            <div
+              ref={this.setContentsRef}
+              key={`contents-${tagLinks}`}
+              tabIndex={!hidden ? 0 : null}
+              dangerouslySetInnerHTML={content}
+              className='status__content__text translate'
+              onMouseEnter={this.handleMouseEnter}
+              onMouseLeave={this.handleMouseLeave}
+              lang={language}
+            />
+            {!hidden && translateButton}
+            {media}
+          </div>
+
+          {extraMedia}
+        </div>
+      );
+    } else if (parseClick) {
+      return (
+        <div
+          className={classNames}
+          onMouseDown={this.handleMouseDown}
+          onMouseUp={this.handleMouseUp}
+          tabIndex={0}
+        >
+          <div
+            ref={this.setContentsRef}
+            key={`contents-${tagLinks}-${rewriteMentions}`}
+            dangerouslySetInnerHTML={content}
+            className='status__content__text translate'
+            tabIndex={0}
+            onMouseEnter={this.handleMouseEnter}
+            onMouseLeave={this.handleMouseLeave}
+            lang={language}
+          />
+          {translateButton}
+          {media}
+          {extraMedia}
+        </div>
+      );
+    } else {
+      return (
+        <div
+          className='status__content'
+          tabIndex={0}
+        >
+          <div
+            ref={this.setContentsRef}
+            key={`contents-${tagLinks}`}
+            className='status__content__text translate'
+            dangerouslySetInnerHTML={content}
+            tabIndex={0}
+            onMouseEnter={this.handleMouseEnter}
+            onMouseLeave={this.handleMouseLeave}
+            lang={language}
+          />
+          {translateButton}
+          {media}
+          {extraMedia}
+        </div>
+      );
+    }
+  }
+
+}
+
+export default withRouter(connect(mapStateToProps)(injectIntl(StatusContent)));
diff --git a/app/javascript/flavours/blobfox/components/status_header.jsx b/app/javascript/flavours/blobfox/components/status_header.jsx
new file mode 100644
index 00000000000000..1c51707cefaa9c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_header.jsx
@@ -0,0 +1,71 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+//  Mastodon imports.
+import { Avatar } from './avatar';
+import { AvatarOverlay } from './avatar_overlay';
+import { DisplayName } from './display_name';
+
+export default class StatusHeader extends PureComponent {
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    friend: ImmutablePropTypes.map,
+    parseClick: PropTypes.func.isRequired,
+  };
+
+  //  Handles clicks on account name/image
+  handleClick = (acct, e) => {
+    const { parseClick } = this.props;
+    parseClick(e, `/@${acct}`);
+  };
+
+  handleAccountClick = (e) => {
+    const { status } = this.props;
+    this.handleClick(status.getIn(['account', 'acct']), e);
+  };
+
+  //  Rendering.
+  render () {
+    const {
+      status,
+      friend,
+    } = this.props;
+
+    const account = status.get('account');
+
+    let statusAvatar;
+    if (friend === undefined || friend === null) {
+      statusAvatar = <Avatar account={account} size={46} />;
+    } else {
+      statusAvatar = <AvatarOverlay account={account} friend={friend} />;
+    }
+
+    return (
+      <div className='status__info__account'>
+        <a
+          href={account.get('url')}
+          target='_blank'
+          className='status__avatar'
+          onClick={this.handleAccountClick}
+          rel='noopener noreferrer'
+        >
+          {statusAvatar}
+        </a>
+        <a
+          href={account.get('url')}
+          target='_blank'
+          className='status__display-name'
+          onClick={this.handleAccountClick}
+          rel='noopener noreferrer'
+        >
+          <DisplayName account={account} />
+        </a>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/status_icons.jsx b/app/javascript/flavours/blobfox/components/status_icons.jsx
new file mode 100644
index 00000000000000..34b4ad6e5885a4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_icons.jsx
@@ -0,0 +1,147 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+
+//  Mastodon imports.
+import { Icon } from 'flavours/blobfox/components/icon';
+import { languages } from 'flavours/blobfox/initial_state';
+
+import { IconButton } from './icon_button';
+import VisibilityIcon from './status_visibility_icon';
+
+//  Messages for use with internationalization stuff.
+const messages = defineMessages({
+  collapse: { id: 'status.collapse', defaultMessage: 'Collapse' },
+  uncollapse: { id: 'status.uncollapse', defaultMessage: 'Uncollapse' },
+  inReplyTo: { id: 'status.in_reply_to', defaultMessage: 'This toot is a reply' },
+  previewCard: { id: 'status.has_preview_card', defaultMessage: 'Features an attached preview card' },
+  pictures: { id: 'status.has_pictures', defaultMessage: 'Features attached pictures' },
+  poll: { id: 'status.is_poll', defaultMessage: 'This toot is a poll' },
+  video: { id: 'status.has_video', defaultMessage: 'Features attached videos' },
+  audio: { id: 'status.has_audio', defaultMessage: 'Features attached audio files' },
+  localOnly: { id: 'status.local_only', defaultMessage: 'Only visible from your instance' },
+});
+
+const LanguageIcon = ({ language }) => {
+  if (!languages) return null;
+
+  const lang = languages.find((lang) => lang[0] === language);
+  if (!lang) return null;
+
+  return (
+    <span className='text-icon' title={`${lang[2]} (${lang[1]})`} aria-hidden='true'>
+      {lang[0].toUpperCase()}
+    </span>
+  );
+};
+
+LanguageIcon.propTypes = {
+  language: PropTypes.string.isRequired,
+};
+
+class StatusIcons extends PureComponent {
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    mediaIcons: PropTypes.arrayOf(PropTypes.string),
+    collapsible: PropTypes.bool,
+    collapsed: PropTypes.bool,
+    setCollapsed: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    settings: ImmutablePropTypes.map.isRequired,
+  };
+
+  //  Handles clicks on collapsed button
+  handleCollapsedClick = (e) => {
+    const { collapsed, setCollapsed } = this.props;
+    if (e.button === 0) {
+      setCollapsed(!collapsed);
+      e.preventDefault();
+    }
+  };
+
+  mediaIconTitleText (mediaIcon) {
+    const { intl } = this.props;
+
+    const message = {
+      'link': messages.previewCard,
+      'picture-o': messages.pictures,
+      'tasks': messages.poll,
+      'video-camera': messages.video,
+      'music': messages.audio,
+    }[mediaIcon];
+
+    return message && intl.formatMessage(message);
+  }
+
+  renderIcon (mediaIcon) {
+    return (
+      <Icon
+        fixedWidth
+        className='status__media-icon'
+        key={`media-icon--${mediaIcon}`}
+        id={mediaIcon}
+        aria-hidden='true'
+        title={this.mediaIconTitleText(mediaIcon)}
+      />
+    );
+  }
+
+  //  Rendering.
+  render () {
+    const {
+      status,
+      mediaIcons,
+      collapsible,
+      collapsed,
+      settings,
+      intl,
+    } = this.props;
+
+    return (
+      <div className='status__info__icons'>
+        {settings.get('language') && status.get('language') && <LanguageIcon language={status.get('language')} />}
+        {settings.get('reply') && status.get('in_reply_to_id', null) !== null ? (
+          <Icon
+            className='status__reply-icon'
+            fixedWidth
+            id='comment'
+            aria-hidden='true'
+            title={intl.formatMessage(messages.inReplyTo)}
+          />
+        ) : null}
+        {settings.get('local_only') && status.get('local_only') &&
+          <Icon
+            fixedWidth
+            id='home'
+            aria-hidden='true'
+            title={intl.formatMessage(messages.localOnly)}
+          />}
+        {settings.get('media') && !!mediaIcons && mediaIcons.map(icon => this.renderIcon(icon))}
+        {settings.get('visibility') && <VisibilityIcon visibility={status.get('visibility')} />}
+        {collapsible && (
+          <IconButton
+            className='status__collapse-button'
+            animate
+            active={collapsed}
+            title={
+              collapsed ?
+                intl.formatMessage(messages.uncollapse) :
+                intl.formatMessage(messages.collapse)
+            }
+            icon='angle-double-up'
+            onClick={this.handleCollapsedClick}
+          />
+        )}
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(StatusIcons);
diff --git a/app/javascript/flavours/blobfox/components/status_list.jsx b/app/javascript/flavours/blobfox/components/status_list.jsx
new file mode 100644
index 00000000000000..3f103742f23037
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_list.jsx
@@ -0,0 +1,137 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { debounce } from 'lodash';
+
+import RegenerationIndicator from 'flavours/blobfox/components/regeneration_indicator';
+
+import StatusContainer from '../containers/status_container';
+
+import { LoadGap } from './load_gap';
+import ScrollableList from './scrollable_list';
+
+export default class StatusList extends ImmutablePureComponent {
+
+  static propTypes = {
+    scrollKey: PropTypes.string.isRequired,
+    statusIds: ImmutablePropTypes.list.isRequired,
+    featuredStatusIds: ImmutablePropTypes.list,
+    onLoadMore: PropTypes.func,
+    onScrollToTop: PropTypes.func,
+    onScroll: PropTypes.func,
+    trackScroll: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    isPartial: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    prepend: PropTypes.node,
+    emptyMessage: PropTypes.node,
+    alwaysPrepend: PropTypes.bool,
+    withCounters: PropTypes.bool,
+    timelineId: PropTypes.string.isRequired,
+    lastId: PropTypes.string,
+    regex: PropTypes.string,
+  };
+
+  static defaultProps = {
+    trackScroll: true,
+  };
+
+  getFeaturedStatusCount = () => {
+    return this.props.featuredStatusIds ? this.props.featuredStatusIds.size : 0;
+  };
+
+  getCurrentStatusIndex = (id, featured) => {
+    if (featured) {
+      return this.props.featuredStatusIds.indexOf(id);
+    } else {
+      return this.props.statusIds.indexOf(id) + this.getFeaturedStatusCount();
+    }
+  };
+
+  handleMoveUp = (id, featured) => {
+    const elementIndex = this.getCurrentStatusIndex(id, featured) - 1;
+    this._selectChild(elementIndex, true);
+  };
+
+  handleMoveDown = (id, featured) => {
+    const elementIndex = this.getCurrentStatusIndex(id, featured) + 1;
+    this._selectChild(elementIndex, false);
+  };
+
+  handleLoadOlder = debounce(() => {
+    const { statusIds, lastId, onLoadMore } = this.props;
+    onLoadMore(lastId || (statusIds.size > 0 ? statusIds.last() : undefined));
+  }, 300, { leading: true });
+
+  _selectChild (index, align_top) {
+    const container = this.node.node;
+    const element = container.querySelector(`article:nth-of-type(${index + 1}) .focusable`);
+
+    if (element) {
+      if (align_top && container.scrollTop > element.offsetTop) {
+        element.scrollIntoView(true);
+      } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) {
+        element.scrollIntoView(false);
+      }
+      element.focus();
+    }
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  render () {
+    const { statusIds, featuredStatusIds, onLoadMore, timelineId, ...other }  = this.props;
+    const { isLoading, isPartial } = other;
+
+    if (isPartial) {
+      return <RegenerationIndicator />;
+    }
+
+    let scrollableContent = (isLoading || statusIds.size > 0) ? (
+      statusIds.map((statusId, index) => statusId === null ? (
+        <LoadGap
+          key={'gap:' + statusIds.get(index + 1)}
+          disabled={isLoading}
+          maxId={index > 0 ? statusIds.get(index - 1) : null}
+          onClick={onLoadMore}
+        />
+      ) : (
+        <StatusContainer
+          key={statusId}
+          id={statusId}
+          onMoveUp={this.handleMoveUp}
+          onMoveDown={this.handleMoveDown}
+          contextType={timelineId}
+          scrollKey={this.props.scrollKey}
+          withCounters={this.props.withCounters}
+        />
+      ))
+    ) : null;
+
+    if (scrollableContent && featuredStatusIds) {
+      scrollableContent = featuredStatusIds.map(statusId => (
+        <StatusContainer
+          key={`f-${statusId}`}
+          id={statusId}
+          featured
+          onMoveUp={this.handleMoveUp}
+          onMoveDown={this.handleMoveDown}
+          contextType={timelineId}
+          scrollKey={this.props.scrollKey}
+          withCounters={this.props.withCounters}
+        />
+      )).concat(scrollableContent);
+    }
+
+    return (
+      <ScrollableList {...other} showLoading={isLoading && statusIds.size === 0} onLoadMore={onLoadMore && this.handleLoadOlder} ref={this.setRef}>
+        {scrollableContent}
+      </ScrollableList>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/status_prepend.jsx b/app/javascript/flavours/blobfox/components/status_prepend.jsx
new file mode 100644
index 00000000000000..017ca66a19bae7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_prepend.jsx
@@ -0,0 +1,158 @@
+//  Package imports  //
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import { me } from 'flavours/blobfox/initial_state';
+
+export default class StatusPrepend extends PureComponent {
+
+  static propTypes = {
+    type: PropTypes.string.isRequired,
+    account: ImmutablePropTypes.map.isRequired,
+    parseClick: PropTypes.func.isRequired,
+    notificationId: PropTypes.number,
+  };
+
+  handleClick = (e) => {
+    const { account, parseClick } = this.props;
+    parseClick(e, `/@${account.get('acct')}`);
+  };
+
+  Message = () => {
+    const { type, account } = this.props;
+    let link = (
+      <a
+        onClick={this.handleClick}
+        href={account.get('url')}
+        className='status__display-name'
+      >
+        <b
+          dangerouslySetInnerHTML={{
+            __html : account.get('display_name_html') || account.get('username'),
+          }}
+        />
+      </a>
+    );
+    switch (type) {
+    case 'featured':
+      return (
+        <FormattedMessage id='status.pinned' defaultMessage='Pinned post' />
+      );
+    case 'reblogged_by':
+      return (
+        <FormattedMessage
+          id='status.reblogged_by'
+          defaultMessage='{name} boosted'
+          values={{ name : link }}
+        />
+      );
+    case 'favourite':
+      return (
+        <FormattedMessage
+          id='notification.favourite'
+          defaultMessage='{name} favorited your status'
+          values={{ name : link }}
+        />
+      );
+    case 'reaction':
+      return (
+        <FormattedMessage
+          id='notification.reaction'
+          defaultMessage='{name} reacted to your status'
+          values={{ name: link }}
+        />
+      );
+    case 'reblog':
+      return (
+        <FormattedMessage
+          id='notification.reblog'
+          defaultMessage='{name} boosted your status'
+          values={{ name : link }}
+        />
+      );
+    case 'status':
+      return (
+        <FormattedMessage
+          id='notification.status'
+          defaultMessage='{name} just posted'
+          values={{ name: link }}
+        />
+      );
+    case 'poll':
+      if (me === account.get('id')) {
+        return (
+          <FormattedMessage
+            id='notification.own_poll'
+            defaultMessage='Your poll has ended'
+          />
+        );
+      } else {
+        return (
+          <FormattedMessage
+            id='notification.poll'
+            defaultMessage='A poll you have voted in has ended'
+          />
+        );
+      }
+    case 'update':
+      return (
+        <FormattedMessage
+          id='notification.update'
+          defaultMessage='{name} edited a post'
+          values={{ name: link }}
+        />
+      );
+    }
+    return null;
+  };
+
+  render () {
+    const { Message } = this;
+    const { type } = this.props;
+
+    let iconId;
+
+    switch(type) {
+    case 'favourite':
+      iconId = 'star';
+      break;
+    case 'reaction':
+      iconId = 'plus';
+      break;
+    case 'featured':
+      iconId = 'thumb-tack';
+      break;
+    case 'poll':
+      iconId = 'tasks';
+      break;
+    case 'reblog':
+    case 'reblogged_by':
+      iconId = 'retweet';
+      break;
+    case 'status':
+      iconId = 'bell';
+      break;
+    case 'update':
+      iconId = 'pencil';
+      break;
+    }
+
+    return !type ? null : (
+      <aside className={type === 'reblogged_by' || type === 'featured' ? 'status__prepend' : 'notification__message'}>
+        <div className={type === 'reblogged_by' || type === 'featured' ? 'status__prepend-icon-wrapper' : 'notification__favourite-icon-wrapper'}>
+          <Icon
+            className={`status__prepend-icon ${type === 'favourite' ? 'star-icon' : ''}`}
+            id={iconId}
+          />
+        </div>
+        <Message />
+      </aside>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/status_reactions.jsx b/app/javascript/flavours/blobfox/components/status_reactions.jsx
new file mode 100644
index 00000000000000..81443d20555e14
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_reactions.jsx
@@ -0,0 +1,175 @@
+import PropTypes from 'prop-types';
+import React from 'react';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import TransitionMotion from 'react-motion/lib/TransitionMotion';
+import spring from 'react-motion/lib/spring';
+
+import { unicodeMapping } from '../features/emoji/emoji_unicode_mapping_light';
+import { autoPlayGif, reduceMotion } from '../initial_state';
+import { assetHost } from '../utils/config';
+
+import { AnimatedNumber } from './animated_number';
+
+export default class StatusReactions extends ImmutablePureComponent {
+
+  static propTypes = {
+    statusId: PropTypes.string.isRequired,
+    reactions: ImmutablePropTypes.list.isRequired,
+    numVisible: PropTypes.number,
+    addReaction: PropTypes.func.isRequired,
+    canReact: PropTypes.bool.isRequired,
+    removeReaction: PropTypes.func.isRequired,
+  };
+
+  willEnter() {
+    return { scale: reduceMotion ? 1 : 0 };
+  }
+
+  willLeave() {
+    return { scale: reduceMotion ? 0 : spring(0, { stiffness: 170, damping: 26 }) };
+  }
+
+  render() {
+    const { reactions, numVisible } = this.props;
+    let visibleReactions = reactions
+      .filter(x => x.get('count') > 0)
+      .sort((a, b) => b.get('count') - a.get('count'));
+
+    if (numVisible >= 0) {
+      visibleReactions = visibleReactions.filter((_, i) => i < numVisible);
+    }
+
+    const styles = visibleReactions.map(reaction => ({
+      key: reaction.get('name'),
+      data: reaction,
+      style: { scale: reduceMotion ? 1 : spring(1, { stiffness: 150, damping: 13 }) },
+    })).toArray();
+
+    return (
+      <TransitionMotion styles={styles} willEnter={this.willEnter} willLeave={this.willLeave}>
+        {items => (
+          <div className={classNames('reactions-bar', { 'reactions-bar--empty': visibleReactions.isEmpty() })}>
+            {items.map(({ key, data, style }) => (
+              <Reaction
+                key={key}
+                statusId={this.props.statusId}
+                reaction={data}
+                style={{ transform: `scale(${style.scale})`, position: style.scale < 0.5 ? 'absolute' : 'static' }}
+                addReaction={this.props.addReaction}
+                removeReaction={this.props.removeReaction}
+                canReact={this.props.canReact}
+              />
+            ))}
+          </div>
+        )}
+      </TransitionMotion>
+    );
+  }
+
+}
+
+class Reaction extends ImmutablePureComponent {
+
+  static propTypes = {
+    statusId: PropTypes.string,
+    reaction: ImmutablePropTypes.map.isRequired,
+    addReaction: PropTypes.func.isRequired,
+    removeReaction: PropTypes.func.isRequired,
+    canReact: PropTypes.bool.isRequired,
+    style: PropTypes.object,
+  };
+
+  state = {
+    hovered: false,
+  };
+
+  handleClick = () => {
+    const { reaction, statusId, addReaction, removeReaction } = this.props;
+
+    if (reaction.get('me')) {
+      removeReaction(statusId, reaction.get('name'));
+    } else {
+      addReaction(statusId, reaction.get('name'));
+    }
+  };
+
+  handleMouseEnter = () => this.setState({ hovered: true });
+
+  handleMouseLeave = () => this.setState({ hovered: false });
+
+  render() {
+    const { reaction } = this.props;
+
+    return (
+      <button
+        className={classNames('reactions-bar__item', { active: reaction.get('me') })}
+        onClick={this.handleClick}
+        onMouseEnter={this.handleMouseEnter}
+        onMouseLeave={this.handleMouseLeave}
+        disabled={!this.props.canReact}
+        style={this.props.style}
+      >
+        <span className='reactions-bar__item__emoji'>
+          <Emoji
+            hovered={this.state.hovered}
+            emoji={reaction.get('name')}
+            url={reaction.get('url')}
+            staticUrl={reaction.get('static_url')}
+          />
+        </span>
+        <span className='reactions-bar__item__count'>
+          <AnimatedNumber value={reaction.get('count')} />
+        </span>
+      </button>
+    );
+  }
+
+}
+
+class Emoji extends React.PureComponent {
+
+  static propTypes = {
+    emoji: PropTypes.string.isRequired,
+    hovered: PropTypes.bool.isRequired,
+    url: PropTypes.string,
+    staticUrl: PropTypes.string,
+  };
+
+  render() {
+    const { emoji, hovered, url, staticUrl } = this.props;
+
+    if (unicodeMapping[emoji]) {
+      const { filename, shortCode } = unicodeMapping[this.props.emoji];
+      const title = shortCode ? `:${shortCode}:` : '';
+
+      return (
+        <img
+          draggable='false'
+          className='emojione'
+          alt={emoji}
+          title={title}
+          src={`${assetHost}/emoji/${filename}.svg`}
+        />
+      );
+    } else {
+      const filename = (autoPlayGif || hovered) ? url : staticUrl;
+      const shortCode = `:${emoji}:`;
+
+      return (
+        <img
+          draggable='false'
+          className='emojione custom-emoji'
+          alt={shortCode}
+          title={shortCode}
+          src={filename}
+        />
+      );
+    }
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/components/status_visibility_icon.jsx b/app/javascript/flavours/blobfox/components/status_visibility_icon.jsx
new file mode 100644
index 00000000000000..f2d7bd562b726d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/status_visibility_icon.jsx
@@ -0,0 +1,54 @@
+//  Package imports  //
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const messages = defineMessages({
+  public: { id: 'privacy.public.short', defaultMessage: 'Public' },
+  unlisted: { id: 'privacy.unlisted.short', defaultMessage: 'Unlisted' },
+  private: { id: 'privacy.private.short', defaultMessage: 'Followers only' },
+  direct: { id: 'privacy.direct.short', defaultMessage: 'Mentioned people only' },
+});
+
+class VisibilityIcon extends ImmutablePureComponent {
+
+  static propTypes = {
+    visibility: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    withLabel: PropTypes.bool,
+  };
+
+  render() {
+    const { withLabel, visibility, intl } = this.props;
+
+    const visibilityIcon = {
+      public: 'globe',
+      unlisted: 'unlock',
+      private: 'lock',
+      direct: 'envelope',
+    }[visibility];
+
+    const label = intl.formatMessage(messages[visibility]);
+
+    const icon = (<Icon
+      className='status__visibility-icon'
+      fixedWidth
+      id={visibilityIcon}
+      title={label}
+      aria-hidden='true'
+    />);
+
+    if (withLabel) {
+      return (<span style={{ whiteSpace: 'nowrap' }}>{icon} {label}</span>);
+    } else {
+      return icon;
+    }
+  }
+
+}
+
+export default injectIntl(VisibilityIcon);
diff --git a/app/javascript/flavours/blobfox/components/timeline_hint.tsx b/app/javascript/flavours/blobfox/components/timeline_hint.tsx
new file mode 100644
index 00000000000000..bf2a2d8bbaeda7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/timeline_hint.tsx
@@ -0,0 +1,25 @@
+import { FormattedMessage } from 'react-intl';
+
+interface Props {
+  resource: JSX.Element;
+  url: string;
+}
+
+export const TimelineHint: React.FC<Props> = ({ resource, url }) => (
+  <div className='timeline-hint'>
+    <strong>
+      <FormattedMessage
+        id='timeline_hint.remote_resource_not_displayed'
+        defaultMessage='{resource} from other servers are not displayed.'
+        values={{ resource }}
+      />
+    </strong>
+    <br />
+    <a href={url} target='_blank' rel='noopener noreferrer'>
+      <FormattedMessage
+        id='account.browse_more_on_origin_server'
+        defaultMessage='Browse more on the original profile'
+      />
+    </a>
+  </div>
+);
diff --git a/app/javascript/flavours/blobfox/components/verified_badge.tsx b/app/javascript/flavours/blobfox/components/verified_badge.tsx
new file mode 100644
index 00000000000000..9a6adcfa8601c9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/components/verified_badge.tsx
@@ -0,0 +1,27 @@
+import { Icon } from './icon';
+
+const domParser = new DOMParser();
+
+const stripRelMe = (html: string) => {
+  const document = domParser.parseFromString(html, 'text/html').documentElement;
+
+  document.querySelectorAll<HTMLAnchorElement>('a[rel]').forEach((link) => {
+    link.rel = link.rel
+      .split(' ')
+      .filter((x: string) => x !== 'me')
+      .join(' ');
+  });
+
+  const body = document.querySelector('body');
+  return body ? { __html: body.innerHTML } : undefined;
+};
+
+interface Props {
+  link: string;
+}
+export const VerifiedBadge: React.FC<Props> = ({ link }) => (
+  <span className='verified-badge'>
+    <Icon id='check' className='verified-badge__mark' />
+    <span dangerouslySetInnerHTML={stripRelMe(link)} />
+  </span>
+);
diff --git a/app/javascript/flavours/blobfox/containers/account_container.jsx b/app/javascript/flavours/blobfox/containers/account_container.jsx
new file mode 100644
index 00000000000000..a134452e77210f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/account_container.jsx
@@ -0,0 +1,76 @@
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import {
+  followAccount,
+  unfollowAccount,
+  blockAccount,
+  unblockAccount,
+  muteAccount,
+  unmuteAccount,
+} from '../actions/accounts';
+import { openModal } from '../actions/modal';
+import { initMuteModal } from '../actions/mutes';
+import Account from '../components/account';
+import { unfollowModal } from '../initial_state';
+import { makeGetAccount } from '../selectors';
+
+const messages = defineMessages({
+  unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow' },
+});
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, props) => ({
+    account: getAccount(state, props.id),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+
+  onFollow (account) {
+    if (account.getIn(['relationship', 'following']) || account.getIn(['relationship', 'requested'])) {
+      if (unfollowModal) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: <FormattedMessage id='confirmations.unfollow.message' defaultMessage='Are you sure you want to unfollow {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />,
+            confirm: intl.formatMessage(messages.unfollowConfirm),
+            onConfirm: () => dispatch(unfollowAccount(account.get('id'))),
+          },
+        }));
+      } else {
+        dispatch(unfollowAccount(account.get('id')));
+      }
+    } else {
+      dispatch(followAccount(account.get('id')));
+    }
+  },
+
+  onBlock (account) {
+    if (account.getIn(['relationship', 'blocking'])) {
+      dispatch(unblockAccount(account.get('id')));
+    } else {
+      dispatch(blockAccount(account.get('id')));
+    }
+  },
+
+  onMute (account) {
+    if (account.getIn(['relationship', 'muting'])) {
+      dispatch(unmuteAccount(account.get('id')));
+    } else {
+      dispatch(initMuteModal(account));
+    }
+  },
+
+
+  onMuteNotifications (account, notifications) {
+    dispatch(muteAccount(account.get('id'), notifications));
+  },
+});
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(Account));
diff --git a/app/javascript/flavours/blobfox/containers/admin_component.jsx b/app/javascript/flavours/blobfox/containers/admin_component.jsx
new file mode 100644
index 00000000000000..ee0bc2f316beb1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/admin_component.jsx
@@ -0,0 +1,22 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { IntlProvider } from 'flavours/blobfox/locales';
+
+export default class AdminComponent extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node.isRequired,
+  };
+
+  render () {
+    const { children } = this.props;
+
+    return (
+      <IntlProvider>
+        {children}
+      </IntlProvider>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/containers/compose_container.jsx b/app/javascript/flavours/blobfox/containers/compose_container.jsx
new file mode 100644
index 00000000000000..f76550678ed2e1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/compose_container.jsx
@@ -0,0 +1,31 @@
+import { PureComponent } from 'react';
+
+import { Provider } from 'react-redux';
+
+import { fetchCustomEmojis } from '../actions/custom_emojis';
+import { hydrateStore } from '../actions/store';
+import Compose from '../features/standalone/compose';
+import initialState from '../initial_state';
+import { IntlProvider } from '../locales';
+import { store } from '../store';
+
+
+if (initialState) {
+  store.dispatch(hydrateStore(initialState));
+}
+
+store.dispatch(fetchCustomEmojis());
+
+export default class ComposeContainer extends PureComponent {
+
+  render () {
+    return (
+      <IntlProvider>
+        <Provider store={store}>
+          <Compose />
+        </Provider>
+      </IntlProvider>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/containers/domain_container.jsx b/app/javascript/flavours/blobfox/containers/domain_container.jsx
new file mode 100644
index 00000000000000..c719a5775c7c16
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/domain_container.jsx
@@ -0,0 +1,36 @@
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { blockDomain, unblockDomain } from '../actions/domain_blocks';
+import { openModal } from '../actions/modal';
+import { Domain } from '../components/domain';
+
+const messages = defineMessages({
+  blockDomainConfirm: { id: 'confirmations.domain_block.confirm', defaultMessage: 'Block entire domain' },
+});
+
+const makeMapStateToProps = () => {
+  const mapStateToProps = () => ({});
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onBlockDomain (domain) {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: <FormattedMessage id='confirmations.domain_block.message' defaultMessage='Are you really, really sure you want to block the entire {domain}? In most cases a few targeted blocks or mutes are sufficient and preferable. You will not see content from that domain in any public timelines or your notifications. Your followers from that domain will be removed.' values={{ domain: <strong>{domain}</strong> }} />,
+        confirm: intl.formatMessage(messages.blockDomainConfirm),
+        onConfirm: () => dispatch(blockDomain(domain)),
+      },
+    }));
+  },
+
+  onUnblockDomain (domain) {
+    dispatch(unblockDomain(domain));
+  },
+});
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(Domain));
diff --git a/app/javascript/flavours/blobfox/containers/dropdown_menu_container.js b/app/javascript/flavours/blobfox/containers/dropdown_menu_container.js
new file mode 100644
index 00000000000000..501bafdc53c67f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/dropdown_menu_container.js
@@ -0,0 +1,37 @@
+import { connect } from 'react-redux';
+
+import { openDropdownMenu, closeDropdownMenu } from '../actions/dropdown_menu';
+import { openModal, closeModal } from '../actions/modal';
+import DropdownMenu from '../components/dropdown_menu';
+import { isUserTouching } from '../is_mobile';
+
+/**
+ * @param {import('flavours/blobfox/store').RootState} state
+ */
+const mapStateToProps = state => ({
+  openDropdownId: state.dropdownMenu.openId,
+  openedViaKeyboard: state.dropdownMenu.keyboard,
+});
+
+const mapDispatchToProps = (dispatch, { status, items, scrollKey }) => ({
+  onOpen(id, onItemClick, keyboard) {
+    dispatch(isUserTouching() ? openModal({
+      modalType: 'ACTIONS',
+      modalProps: {
+        status,
+        actions: items,
+        onClick: onItemClick,
+      },
+    }) : openDropdownMenu({ id, keyboard, scrollKey }));
+  },
+
+  onClose(id) {
+    dispatch(closeModal({
+      modalType: 'ACTIONS',
+      ignoreFocus: false,
+    }));
+    dispatch(closeDropdownMenu({ id }));
+  },
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(DropdownMenu);
diff --git a/app/javascript/flavours/blobfox/containers/intersection_observer_article_container.js b/app/javascript/flavours/blobfox/containers/intersection_observer_article_container.js
new file mode 100644
index 00000000000000..8d9bda6704326f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/intersection_observer_article_container.js
@@ -0,0 +1,18 @@
+import { connect } from 'react-redux';
+
+import { setHeight } from '../actions/height_cache';
+import IntersectionObserverArticle from '../components/intersection_observer_article';
+
+const makeMapStateToProps = (state, props) => ({
+  cachedHeight: state.getIn(['height_cache', props.saveHeightKey, props.id]),
+});
+
+const mapDispatchToProps = (dispatch) => ({
+
+  onHeightChange (key, id, height) {
+    dispatch(setHeight(key, id, height));
+  },
+
+});
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(IntersectionObserverArticle);
diff --git a/app/javascript/flavours/blobfox/containers/mastodon.jsx b/app/javascript/flavours/blobfox/containers/mastodon.jsx
new file mode 100644
index 00000000000000..fc0361808eee95
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/mastodon.jsx
@@ -0,0 +1,97 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { Helmet } from 'react-helmet';
+import { Route } from 'react-router-dom';
+
+import { Provider as ReduxProvider } from 'react-redux';
+
+import { ScrollContext } from 'react-router-scroll-4';
+
+import { fetchCustomEmojis } from 'flavours/blobfox/actions/custom_emojis';
+import { checkDeprecatedLocalSettings } from 'flavours/blobfox/actions/local_settings';
+import { hydrateStore } from 'flavours/blobfox/actions/store';
+import { connectUserStream } from 'flavours/blobfox/actions/streaming';
+import ErrorBoundary from 'flavours/blobfox/components/error_boundary';
+import { Router } from 'flavours/blobfox/components/router';
+import UI from 'flavours/blobfox/features/ui';
+import initialState, { title as siteTitle } from 'flavours/blobfox/initial_state';
+import { IntlProvider } from 'flavours/blobfox/locales';
+import { store } from 'flavours/blobfox/store';
+
+const title = process.env.NODE_ENV === 'production' ? siteTitle : `${siteTitle} (Dev)`;
+
+const hydrateAction = hydrateStore(initialState);
+
+store.dispatch(hydrateAction);
+
+// check for deprecated local settings
+store.dispatch(checkDeprecatedLocalSettings());
+
+if (initialState.meta.me) {
+  store.dispatch(fetchCustomEmojis());
+}
+
+const createIdentityContext = state => ({
+  signedIn: !!state.meta.me,
+  accountId: state.meta.me,
+  disabledAccountId: state.meta.disabled_account_id,
+  accessToken: state.meta.access_token,
+  permissions: state.role ? state.role.permissions : 0,
+});
+
+export default class Mastodon extends PureComponent {
+
+  static childContextTypes = {
+    identity: PropTypes.shape({
+      signedIn: PropTypes.bool.isRequired,
+      accountId: PropTypes.string,
+      disabledAccountId: PropTypes.string,
+      accessToken: PropTypes.string,
+    }).isRequired,
+  };
+
+  identity = createIdentityContext(initialState);
+
+  getChildContext() {
+    return {
+      identity: this.identity,
+    };
+  }
+
+  componentDidMount() {
+    if (this.identity.signedIn) {
+      this.disconnect = store.dispatch(connectUserStream());
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.disconnect) {
+      this.disconnect();
+      this.disconnect = null;
+    }
+  }
+
+  shouldUpdateScroll (prevRouterProps, { location }) {
+    return !(location.state?.mastodonModalKey && location.state?.mastodonModalKey !== prevRouterProps?.location?.state?.mastodonModalKey);
+  }
+
+  render () {
+    return (
+      <IntlProvider>
+        <ReduxProvider store={store}>
+          <ErrorBoundary>
+            <Router>
+              <ScrollContext shouldUpdateScroll={this.shouldUpdateScroll}>
+                <Route path='/' component={UI} />
+              </ScrollContext>
+            </Router>
+
+            <Helmet defaultTitle={title} titleTemplate={`%s - ${title}`} />
+          </ErrorBoundary>
+        </ReduxProvider>
+      </IntlProvider>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/containers/media_container.jsx b/app/javascript/flavours/blobfox/containers/media_container.jsx
new file mode 100644
index 00000000000000..84a9a2c9cd186b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/media_container.jsx
@@ -0,0 +1,127 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+import { createPortal } from 'react-dom';
+
+import { fromJS } from 'immutable';
+
+import { ImmutableHashtag as Hashtag } from 'flavours/blobfox/components/hashtag';
+import MediaGallery from 'flavours/blobfox/components/media_gallery';
+import ModalRoot from 'flavours/blobfox/components/modal_root';
+import Poll from 'flavours/blobfox/components/poll';
+import Audio from 'flavours/blobfox/features/audio';
+import Card from 'flavours/blobfox/features/status/components/card';
+import MediaModal from 'flavours/blobfox/features/ui/components/media_modal';
+import Video from 'flavours/blobfox/features/video';
+import { IntlProvider } from 'flavours/blobfox/locales';
+import { getScrollbarWidth } from 'flavours/blobfox/utils/scrollbar';
+
+const MEDIA_COMPONENTS = { MediaGallery, Video, Card, Poll, Hashtag, Audio };
+
+export default class MediaContainer extends PureComponent {
+
+  static propTypes = {
+    components: PropTypes.object.isRequired,
+  };
+
+  state = {
+    media: null,
+    index: null,
+    lang: null,
+    time: null,
+    backgroundColor: null,
+    options: null,
+  };
+
+  handleOpenMedia = (media, index, lang) => {
+    document.body.classList.add('with-modals--active');
+    document.documentElement.style.marginRight = `${getScrollbarWidth()}px`;
+
+    this.setState({ media, index, lang });
+  };
+
+  handleOpenVideo = (lang, options) => {
+    const { components } = this.props;
+    const { media } = JSON.parse(components[options.componentIndex].getAttribute('data-props'));
+    const mediaList = fromJS(media);
+
+    document.body.classList.add('with-modals--active');
+    document.documentElement.style.marginRight = `${getScrollbarWidth()}px`;
+
+    this.setState({ media: mediaList, lang, options });
+  };
+
+  handleCloseMedia = () => {
+    document.body.classList.remove('with-modals--active');
+    document.documentElement.style.marginRight = '0';
+
+    this.setState({
+      media: null,
+      index: null,
+      time: null,
+      backgroundColor: null,
+      options: null,
+    });
+  };
+
+  setBackgroundColor = color => {
+    this.setState({ backgroundColor: color });
+  };
+
+  render () {
+    const { components } = this.props;
+
+    let handleOpenVideo;
+
+    // Don't offer to expand the video in a lightbox if we're in a frame
+    if (window.self === window.top) {
+      handleOpenVideo = this.handleOpenVideo;
+    }
+
+    return (
+      <IntlProvider>
+        <>
+          {[].map.call(components, (component, i) => {
+            const componentName = component.getAttribute('data-component');
+            const Component = MEDIA_COMPONENTS[componentName];
+            const { media, card, poll, hashtag, ...props } = JSON.parse(component.getAttribute('data-props'));
+
+            Object.assign(props, {
+              ...(media   ? { media:   fromJS(media)   } : {}),
+              ...(card    ? { card:    fromJS(card)    } : {}),
+              ...(poll    ? { poll:    fromJS(poll)    } : {}),
+              ...(hashtag ? { hashtag: fromJS(hashtag) } : {}),
+
+              ...(componentName === 'Video' ? {
+                componentIndex: i,
+                onOpenVideo: handleOpenVideo,
+              } : {
+                onOpenMedia: this.handleOpenMedia,
+              }),
+            });
+
+            return createPortal(
+              <Component {...props} key={`media-${i}`} />,
+              component,
+            );
+          })}
+
+          <ModalRoot backgroundColor={this.state.backgroundColor} onClose={this.handleCloseMedia}>
+            {this.state.media && (
+              <MediaModal
+                media={this.state.media}
+                index={this.state.index || 0}
+                lang={this.state.lang}
+                currentTime={this.state.options?.startTime}
+                autoPlay={this.state.options?.autoPlay}
+                volume={this.state.options?.defaultVolume}
+                onClose={this.handleCloseMedia}
+                onChangeBackgroundColor={this.setBackgroundColor}
+              />
+            )}
+          </ModalRoot>
+        </>
+      </IntlProvider>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/containers/notification_purge_buttons_container.js b/app/javascript/flavours/blobfox/containers/notification_purge_buttons_container.js
new file mode 100644
index 00000000000000..22bf98f7f182a1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/notification_purge_buttons_container.js
@@ -0,0 +1,53 @@
+//  Package imports.
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+//  Our imports.
+import { openModal } from 'flavours/blobfox/actions/modal';
+import {
+  deleteMarkedNotifications,
+  enterNotificationClearingMode,
+  markAllNotifications,
+} from 'flavours/blobfox/actions/notifications';
+import NotificationPurgeButtons from 'flavours/blobfox/components/notification_purge_buttons';
+
+const messages = defineMessages({
+  clearMessage: { id: 'notifications.marked_clear_confirmation', defaultMessage: 'Are you sure you want to permanently clear all selected notifications?' },
+  clearConfirm: { id: 'notifications.marked_clear', defaultMessage: 'Clear selected notifications' },
+});
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onEnterCleaningMode(yes) {
+    dispatch(enterNotificationClearingMode(yes));
+  },
+
+  onDeleteMarked() {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.clearMessage),
+        confirm: intl.formatMessage(messages.clearConfirm),
+        onConfirm: () => dispatch(deleteMarkedNotifications()),
+      },
+    }));
+  },
+
+  onMarkAll() {
+    dispatch(markAllNotifications(true));
+  },
+
+  onMarkNone() {
+    dispatch(markAllNotifications(false));
+  },
+
+  onInvert() {
+    dispatch(markAllNotifications(null));
+  },
+});
+
+const mapStateToProps = state => ({
+  markNewForDelete: state.getIn(['notifications', 'markNewForDelete']),
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(NotificationPurgeButtons));
diff --git a/app/javascript/flavours/blobfox/containers/poll_container.js b/app/javascript/flavours/blobfox/containers/poll_container.js
new file mode 100644
index 00000000000000..a67055ec344177
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/poll_container.js
@@ -0,0 +1,26 @@
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchPoll, vote } from 'flavours/blobfox/actions/polls';
+import Poll from 'flavours/blobfox/components/poll';
+
+const mapDispatchToProps = (dispatch, { pollId }) => ({
+  refresh: debounce(
+    () => {
+      dispatch(fetchPoll(pollId));
+    },
+    1000,
+    { leading: true },
+  ),
+
+  onVote (choices) {
+    dispatch(vote(pollId, choices));
+  },
+});
+
+const mapStateToProps = (state, { pollId }) => ({
+  poll: state.getIn(['polls', pollId]),
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Poll);
diff --git a/app/javascript/flavours/blobfox/containers/scroll_container.js b/app/javascript/flavours/blobfox/containers/scroll_container.js
new file mode 100644
index 00000000000000..d21ff63687dbfd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/scroll_container.js
@@ -0,0 +1,18 @@
+import { ScrollContainer as OriginalScrollContainer } from 'react-router-scroll-4';
+
+// ScrollContainer is used to automatically scroll to the top when pushing a
+// new history state and remembering the scroll position when going back.
+// There are a few things we need to do differently, though.
+const defaultShouldUpdateScroll = (prevRouterProps, { location }) => {
+  // If the change is caused by opening a modal, do not scroll to top
+  return !(location.state?.mastodonModalKey && location.state?.mastodonModalKey !== prevRouterProps?.location?.state?.mastodonModalKey);
+};
+
+export default
+class ScrollContainer extends OriginalScrollContainer {
+
+  static defaultProps = {
+    shouldUpdateScroll: defaultShouldUpdateScroll,
+  };
+
+}
diff --git a/app/javascript/flavours/blobfox/containers/status_container.js b/app/javascript/flavours/blobfox/containers/status_container.js
new file mode 100644
index 00000000000000..f12d95c68e849a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/containers/status_container.js
@@ -0,0 +1,313 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { initBlockModal } from 'flavours/blobfox/actions/blocks';
+import { initBoostModal } from 'flavours/blobfox/actions/boosts';
+import {
+  replyCompose,
+  mentionCompose,
+  directCompose,
+} from 'flavours/blobfox/actions/compose';
+import {
+  initAddFilter,
+} from 'flavours/blobfox/actions/filters';
+import {
+  reblog,
+  favourite,
+  bookmark,
+  unreblog,
+  unfavourite,
+  unbookmark,
+  pin,
+  unpin,
+  addReaction,
+  removeReaction,
+} from 'flavours/blobfox/actions/interactions';
+import { changeLocalSetting } from 'flavours/blobfox/actions/local_settings';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { initMuteModal } from 'flavours/blobfox/actions/mutes';
+import { deployPictureInPicture } from 'flavours/blobfox/actions/picture_in_picture';
+import { initReport } from 'flavours/blobfox/actions/reports';
+import {
+  muteStatus,
+  unmuteStatus,
+  deleteStatus,
+  hideStatus,
+  revealStatus,
+  editStatus,
+  translateStatus,
+  undoStatusTranslation,
+} from 'flavours/blobfox/actions/statuses';
+import Status from 'flavours/blobfox/components/status';
+import { boostModal, favouriteModal, deleteModal } from 'flavours/blobfox/initial_state';
+import { makeGetStatus, makeGetPictureInPicture } from 'flavours/blobfox/selectors';
+
+import { showAlertForError } from '../actions/alerts';
+
+const messages = defineMessages({
+  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
+  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
+  redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' },
+  redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this status and re-draft it? Favorites and boosts will be lost, and replies to the original post will be orphaned.' },
+  replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' },
+  replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
+  editConfirm: { id: 'confirmations.edit.confirm', defaultMessage: 'Edit' },
+  editMessage: { id: 'confirmations.edit.message', defaultMessage: 'Editing now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
+  unfilterConfirm: { id: 'confirmations.unfilter.confirm', defaultMessage: 'Show' },
+  author: { id: 'confirmations.unfilter.author', defaultMessage: 'Author' },
+  matchingFilters: { id: 'confirmations.unfilter.filters', defaultMessage: 'Matching {count, plural, one {filter} other {filters}}' },
+  editFilter: { id: 'confirmations.unfilter.edit_filter', defaultMessage: 'Edit filter' },
+});
+
+const makeMapStateToProps = () => {
+  const getStatus = makeGetStatus();
+  const getPictureInPicture = makeGetPictureInPicture();
+
+  const mapStateToProps = (state, props) => {
+
+    let status = getStatus(state, props);
+    let reblogStatus = status ? status.get('reblog', null) : null;
+    let account = undefined;
+    let prepend = undefined;
+
+    if (props.featured && status) {
+      account = status.get('account');
+      prepend = 'featured';
+    } else if (reblogStatus !== null && typeof reblogStatus === 'object') {
+      account = status.get('account');
+      status = reblogStatus;
+      prepend = 'reblogged_by';
+    }
+
+    return {
+      containerId: props.containerId || props.id,  //  Should match reblogStatus's id for reblogs
+      status: status,
+      nextInReplyToId: props.nextId ? state.getIn(['statuses', props.nextId, 'in_reply_to_id']) : null,
+      account: account || props.account,
+      settings: state.get('local_settings'),
+      prepend: prepend || props.prepend,
+      pictureInPicture: getPictureInPicture(state, props),
+    };
+  };
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { intl, contextType }) => ({
+
+  onReply (status, router) {
+    dispatch((_, getState) => {
+      let state = getState();
+
+      if (state.getIn(['local_settings', 'confirm_before_clearing_draft']) && state.getIn(['compose', 'text']).trim().length !== 0) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: intl.formatMessage(messages.replyMessage),
+            confirm: intl.formatMessage(messages.replyConfirm),
+            onDoNotAsk: () => dispatch(changeLocalSetting(['confirm_before_clearing_draft'], false)),
+            onConfirm: () => dispatch(replyCompose(status, router)),
+          },
+        }));
+      } else {
+        dispatch(replyCompose(status, router));
+      }
+    });
+  },
+
+  onModalReblog (status, privacy) {
+    if (status.get('reblogged')) {
+      dispatch(unreblog(status));
+    } else {
+      dispatch(reblog(status, privacy));
+    }
+  },
+
+  onReblog (status, e) {
+    dispatch((_, getState) => {
+      let state = getState();
+      if (state.getIn(['local_settings', 'confirm_boost_missing_media_description']) && status.get('media_attachments').some(item => !item.get('description')) && !status.get('reblogged')) {
+        dispatch(initBoostModal({ status, onReblog: this.onModalReblog, missingMediaDescription: true }));
+      } else if (e.shiftKey || !boostModal) {
+        this.onModalReblog(status);
+      } else {
+        dispatch(initBoostModal({ status, onReblog: this.onModalReblog }));
+      }
+    });
+  },
+
+  onBookmark (status) {
+    if (status.get('bookmarked')) {
+      dispatch(unbookmark(status));
+    } else {
+      dispatch(bookmark(status));
+    }
+  },
+
+  onModalFavourite (status) {
+    dispatch(favourite(status));
+  },
+
+  onFavourite (status, e) {
+    if (status.get('favourited')) {
+      dispatch(unfavourite(status));
+    } else {
+      if (e.shiftKey || !favouriteModal) {
+        this.onModalFavourite(status);
+      } else {
+        dispatch(openModal({
+          modalType: 'FAVOURITE',
+          modalProps: {
+            status,
+            onFavourite: this.onModalFavourite,
+          },
+        }));
+      }
+    }
+  },
+
+  onPin (status) {
+    if (status.get('pinned')) {
+      dispatch(unpin(status));
+    } else {
+      dispatch(pin(status));
+    }
+  },
+
+  onReactionAdd (statusId, name, url) {
+    dispatch(addReaction(statusId, name, url));
+  },
+
+  onReactionRemove (statusId, name) {
+    dispatch(removeReaction(statusId, name));
+  },
+
+  onEmbed (status) {
+    dispatch(openModal({
+      modalType: 'EMBED',
+      modalProps: {
+        id: status.get('id'),
+        onError: error => dispatch(showAlertForError(error)),
+      },
+    }));
+  },
+
+  onDelete (status, history, withRedraft = false) {
+    if (!deleteModal) {
+      dispatch(deleteStatus(status.get('id'), history, withRedraft));
+    } else {
+      dispatch(openModal({
+        modalType: 'CONFIRM',
+        modalProps: {
+          message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage),
+          confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm),
+          onConfirm: () => dispatch(deleteStatus(status.get('id'), history, withRedraft)),
+        },
+      }));
+    }
+  },
+
+  onEdit (status, history) {
+    dispatch((_, getState) => {
+      let state = getState();
+      if (state.getIn(['compose', 'text']).trim().length !== 0) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: intl.formatMessage(messages.editMessage),
+            confirm: intl.formatMessage(messages.editConfirm),
+            onConfirm: () => dispatch(editStatus(status.get('id'), history)),
+          },
+        }));
+      } else {
+        dispatch(editStatus(status.get('id'), history));
+      }
+    });
+  },
+
+  onTranslate (status) {
+    if (status.get('translation')) {
+      dispatch(undoStatusTranslation(status.get('id'), status.get('poll')));
+    } else {
+      dispatch(translateStatus(status.get('id')));
+    }
+  },
+
+  onDirect (account, router) {
+    dispatch(directCompose(account, router));
+  },
+
+  onMention (account, router) {
+    dispatch(mentionCompose(account, router));
+  },
+
+  onOpenMedia (statusId, media, index, lang) {
+    dispatch(openModal({
+      modalType: 'MEDIA',
+      modalProps: { statusId, media, index, lang },
+    }));
+  },
+
+  onOpenVideo (statusId, media, lang, options) {
+    dispatch(openModal({
+      modalType: 'VIDEO',
+      modalProps: { statusId, media, lang, options },
+    }));
+  },
+
+  onBlock (status) {
+    const account = status.get('account');
+    dispatch(initBlockModal(account));
+  },
+
+  onReport (status) {
+    dispatch(initReport(status.get('account'), status));
+  },
+
+  onAddFilter (status) {
+    dispatch(initAddFilter(status, { contextType }));
+  },
+
+  onMute (account) {
+    dispatch(initMuteModal(account));
+  },
+
+  onMuteConversation (status) {
+    if (status.get('muted')) {
+      dispatch(unmuteStatus(status.get('id')));
+    } else {
+      dispatch(muteStatus(status.get('id')));
+    }
+  },
+
+  onToggleHidden (status) {
+    if (status.get('hidden')) {
+      dispatch(revealStatus(status.get('id')));
+    } else {
+      dispatch(hideStatus(status.get('id')));
+    }
+  },
+
+  deployPictureInPicture (status, type, mediaProps) {
+    dispatch((_, getState) => {
+      if (getState().getIn(['local_settings', 'media', 'pop_in_player'])) {
+        dispatch(deployPictureInPicture(status.get('id'), status.getIn(['account', 'id']), type, mediaProps));
+      }
+    });
+  },
+
+  onInteractionModal (type, status) {
+    dispatch(openModal({
+      modalType: 'INTERACTION',
+      modalProps: {
+        type,
+        accountId: status.getIn(['account', 'id']),
+        url: status.get('uri'),
+      },
+    }));
+  },
+
+});
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(Status));
diff --git a/app/javascript/flavours/blobfox/features/about/index.jsx b/app/javascript/flavours/blobfox/features/about/index.jsx
new file mode 100644
index 00000000000000..f15b0e45ef716d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/about/index.jsx
@@ -0,0 +1,225 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { fetchServer, fetchExtendedDescription, fetchDomainBlocks  } from 'flavours/blobfox/actions/server';
+import Column from 'flavours/blobfox/components/column';
+import { Icon  }  from 'flavours/blobfox/components/icon';
+import { ServerHeroImage } from 'flavours/blobfox/components/server_hero_image';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+import Account from 'flavours/blobfox/containers/account_container';
+import LinkFooter from 'flavours/blobfox/features/ui/components/link_footer';
+
+const messages = defineMessages({
+  title: { id: 'column.about', defaultMessage: 'About' },
+  rules: { id: 'about.rules', defaultMessage: 'Server rules' },
+  blocks: { id: 'about.blocks', defaultMessage: 'Moderated servers' },
+  silenced: { id: 'about.domain_blocks.silenced.title', defaultMessage: 'Limited' },
+  silencedExplanation: { id: 'about.domain_blocks.silenced.explanation', defaultMessage: 'You will generally not see profiles and content from this server, unless you explicitly look it up or opt into it by following.' },
+  suspended: { id: 'about.domain_blocks.suspended.title', defaultMessage: 'Suspended' },
+  suspendedExplanation: { id: 'about.domain_blocks.suspended.explanation', defaultMessage: 'No data from this server will be processed, stored or exchanged, making any interaction or communication with users from this server impossible.' },
+});
+
+const severityMessages = {
+  silence: {
+    title: messages.silenced,
+    explanation: messages.silencedExplanation,
+  },
+
+  suspend: {
+    title: messages.suspended,
+    explanation: messages.suspendedExplanation,
+  },
+};
+
+const mapStateToProps = state => ({
+  server: state.getIn(['server', 'server']),
+  extendedDescription: state.getIn(['server', 'extendedDescription']),
+  domainBlocks: state.getIn(['server', 'domainBlocks']),
+});
+
+class Section extends PureComponent {
+
+  static propTypes = {
+    title: PropTypes.string,
+    children: PropTypes.node,
+    open: PropTypes.bool,
+    onOpen: PropTypes.func,
+  };
+
+  state = {
+    collapsed: !this.props.open,
+  };
+
+  handleClick = () => {
+    const { onOpen } = this.props;
+    const { collapsed } = this.state;
+
+    this.setState({ collapsed: !collapsed }, () => onOpen && onOpen());
+  };
+
+  render () {
+    const { title, children } = this.props;
+    const { collapsed } = this.state;
+
+    return (
+      <div className={classNames('about__section', { active: !collapsed })}>
+        <div className='about__section__title' role='button' tabIndex={0} onClick={this.handleClick}>
+          <Icon id={collapsed ? 'chevron-right' : 'chevron-down'} fixedWidth /> {title}
+        </div>
+
+        {!collapsed && (
+          <div className='about__section__body'>{children}</div>
+        )}
+      </div>
+    );
+  }
+
+}
+
+class About extends PureComponent {
+
+  static propTypes = {
+    server: ImmutablePropTypes.map,
+    extendedDescription: ImmutablePropTypes.map,
+    domainBlocks: ImmutablePropTypes.contains({
+      isLoading: PropTypes.bool,
+      isAvailable: PropTypes.bool,
+      items: ImmutablePropTypes.list,
+    }),
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(fetchServer());
+    dispatch(fetchExtendedDescription());
+  }
+
+  handleDomainBlocksOpen = () => {
+    const { dispatch } = this.props;
+    dispatch(fetchDomainBlocks());
+  };
+
+  render () {
+    const { multiColumn, intl, server, extendedDescription, domainBlocks } = this.props;
+    const isLoading = server.get('isLoading');
+
+    return (
+      <Column bindToDocument={!multiColumn} label={intl.formatMessage(messages.title)}>
+        <div className='scrollable about'>
+          <div className='about__header'>
+            <ServerHeroImage blurhash={server.getIn(['thumbnail', 'blurhash'])} src={server.getIn(['thumbnail', 'url'])} srcSet={server.getIn(['thumbnail', 'versions'])?.map((value, key) => `${value} ${key.replace('@', '')}`).join(', ')} className='about__header__hero' />
+            <h1>{isLoading ? <Skeleton width='10ch' /> : server.get('domain')}</h1>
+            <p><FormattedMessage id='about.powered_by' defaultMessage='Decentralized social media powered by {mastodon}' values={{ mastodon: <a href='https://joinmastodon.org' className='about__mail' target='_blank'>Mastodon</a> }} /></p>
+          </div>
+
+          <div className='about__meta'>
+            <div className='about__meta__column'>
+              <h4><FormattedMessage id='server_banner.administered_by' defaultMessage='Administered by:' /></h4>
+
+              <Account id={server.getIn(['contact', 'account', 'id'])} size={36} minimal />
+            </div>
+
+            <hr className='about__meta__divider' />
+
+            <div className='about__meta__column'>
+              <h4><FormattedMessage id='about.contact' defaultMessage='Contact:' /></h4>
+
+              {isLoading ? <Skeleton width='10ch' /> : <a className='about__mail' href={`mailto:${server.getIn(['contact', 'email'])}`}>{server.getIn(['contact', 'email'])}</a>}
+            </div>
+          </div>
+
+          <Section open title={intl.formatMessage(messages.title)}>
+            {extendedDescription.get('isLoading') ? (
+              <>
+                <Skeleton width='100%' />
+                <br />
+                <Skeleton width='100%' />
+                <br />
+                <Skeleton width='100%' />
+                <br />
+                <Skeleton width='70%' />
+              </>
+            ) : (extendedDescription.get('content')?.length > 0 ? (
+              <div
+                className='prose'
+                dangerouslySetInnerHTML={{ __html: extendedDescription.get('content') }}
+              />
+            ) : (
+              <p><FormattedMessage id='about.not_available' defaultMessage='This information has not been made available on this server.' /></p>
+            ))}
+          </Section>
+
+          <Section title={intl.formatMessage(messages.rules)}>
+            {!isLoading && (server.get('rules', ImmutableList()).isEmpty() ? (
+              <p><FormattedMessage id='about.not_available' defaultMessage='This information has not been made available on this server.' /></p>
+            ) : (
+              <ol className='rules-list'>
+                {server.get('rules').map(rule => (
+                  <li key={rule.get('id')}>
+                    <span className='rules-list__text'>{rule.get('text')}</span>
+                  </li>
+                ))}
+              </ol>
+            ))}
+          </Section>
+
+          <Section title={intl.formatMessage(messages.blocks)} onOpen={this.handleDomainBlocksOpen}>
+            {domainBlocks.get('isLoading') ? (
+              <>
+                <Skeleton width='100%' />
+                <br />
+                <Skeleton width='70%' />
+              </>
+            ) : (domainBlocks.get('isAvailable') ? (
+              <>
+                <p><FormattedMessage id='about.domain_blocks.preamble' defaultMessage='Mastodon generally allows you to view content from and interact with users from any other server in the fediverse. These are the exceptions that have been made on this particular server.' /></p>
+
+                <div className='about__domain-blocks'>
+                  {domainBlocks.get('items').map(block => (
+                    <div className='about__domain-blocks__domain' key={block.get('domain')}>
+                      <div className='about__domain-blocks__domain__header'>
+                        <h6><span title={`SHA-256: ${block.get('digest')}`}>{block.get('domain')}</span></h6>
+                        <span className='about__domain-blocks__domain__type' title={intl.formatMessage(severityMessages[block.get('severity')].explanation)}>{intl.formatMessage(severityMessages[block.get('severity')].title)}</span>
+                      </div>
+
+                      <p>{(block.get('comment') || '').length > 0 ? block.get('comment') : <FormattedMessage id='about.domain_blocks.no_reason_available' defaultMessage='Reason not available' />}</p>
+                    </div>
+                  ))}
+                </div>
+              </>
+            ) : (
+              <p><FormattedMessage id='about.not_available' defaultMessage='This information has not been made available on this server.' /></p>
+            ))}
+          </Section>
+
+          <LinkFooter />
+
+          <div className='about__footer'>
+            <p><FormattedMessage id='about.fork_disclaimer' defaultMessage='blobfox-soc is free open source software forked from Mastodon.' /></p>
+            <p><FormattedMessage id='about.disclaimer' defaultMessage='Mastodon is free, open-source software, and a trademark of Mastodon gGmbH.' /></p>
+          </div>
+        </div>
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='all' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(About));
diff --git a/app/javascript/flavours/blobfox/features/account/components/account_note.jsx b/app/javascript/flavours/blobfox/features/account/components/account_note.jsx
new file mode 100644
index 00000000000000..272a4ee312c08d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/components/account_note.jsx
@@ -0,0 +1,174 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { is } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import Textarea from 'react-textarea-autosize';
+
+const messages = defineMessages({
+  placeholder: { id: 'account_note.placeholder', defaultMessage: 'Click to add a note' },
+});
+
+class InlineAlert extends PureComponent {
+
+  static propTypes = {
+    show: PropTypes.bool,
+  };
+
+  state = {
+    mountMessage: false,
+  };
+
+  static TRANSITION_DELAY = 200;
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (!this.props.show && nextProps.show) {
+      this.setState({ mountMessage: true });
+    } else if (this.props.show && !nextProps.show) {
+      setTimeout(() => this.setState({ mountMessage: false }), InlineAlert.TRANSITION_DELAY);
+    }
+  }
+
+  render () {
+    const { show } = this.props;
+    const { mountMessage } = this.state;
+
+    return (
+      <span aria-live='polite' role='status' className='inline-alert' style={{ opacity: show ? 1 : 0 }}>
+        {mountMessage && <FormattedMessage id='generic.saved' defaultMessage='Saved' />}
+      </span>
+    );
+  }
+
+}
+
+class AccountNote extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    value: PropTypes.string,
+    onSave: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    value: null,
+    saving: false,
+    saved: false,
+  };
+
+  UNSAFE_componentWillMount () {
+    this._reset();
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    const accountWillChange = !is(this.props.account, nextProps.account);
+    const newState = {};
+
+    if (accountWillChange && this._isDirty()) {
+      this._save(false);
+    }
+
+    if (accountWillChange || nextProps.value === this.state.value) {
+      newState.saving = false;
+    }
+
+    if (this.props.value !== nextProps.value) {
+      newState.value = nextProps.value;
+    }
+
+    this.setState(newState);
+  }
+
+  componentWillUnmount () {
+    if (this._isDirty()) {
+      this._save(false);
+    }
+  }
+
+  setTextareaRef = c => {
+    this.textarea = c;
+  };
+
+  handleChange = e => {
+    this.setState({ value: e.target.value, saving: false });
+  };
+
+  handleKeyDown = e => {
+    if (e.keyCode === 13 && (e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+
+      this._save();
+
+      if (this.textarea) {
+        this.textarea.blur();
+      }
+    } else if (e.keyCode === 27) {
+      e.preventDefault();
+
+      this._reset(() => {
+        if (this.textarea) {
+          this.textarea.blur();
+        }
+      });
+    }
+  };
+
+  handleBlur = () => {
+    if (this._isDirty()) {
+      this._save();
+    }
+  };
+
+  _save (showMessage = true) {
+    this.setState({ saving: true }, () => this.props.onSave(this.state.value));
+
+    if (showMessage) {
+      this.setState({ saved: true }, () => setTimeout(() => this.setState({ saved: false }), 2000));
+    }
+  }
+
+  _reset (callback) {
+    this.setState({ value: this.props.value }, callback);
+  }
+
+  _isDirty () {
+    return !this.state.saving && this.props.value !== null && this.state.value !== null && this.state.value !== this.props.value;
+  }
+
+  render () {
+    const { account, intl } = this.props;
+    const { value, saved } = this.state;
+
+    if (!account) {
+      return null;
+    }
+
+    return (
+      <div className='account__header__account-note'>
+        <label htmlFor={`account-note-${account.get('id')}`}>
+          <FormattedMessage id='account.account_note_header' defaultMessage='Note' /> <InlineAlert show={saved} />
+        </label>
+
+        <Textarea
+          id={`account-note-${account.get('id')}`}
+          className='account__header__account-note__content'
+          disabled={this.props.value === null || value === null}
+          placeholder={intl.formatMessage(messages.placeholder)}
+          value={value || ''}
+          onChange={this.handleChange}
+          onKeyDown={this.handleKeyDown}
+          onBlur={this.handleBlur}
+          ref={this.setTextareaRef}
+        />
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(AccountNote);
diff --git a/app/javascript/flavours/blobfox/features/account/components/action_bar.jsx b/app/javascript/flavours/blobfox/features/account/components/action_bar.jsx
new file mode 100644
index 00000000000000..8730f8fe2669f7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/components/action_bar.jsx
@@ -0,0 +1,85 @@
+import { PureComponent } from 'react';
+
+import { FormattedMessage, FormattedNumber } from 'react-intl';
+
+import { NavLink } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+class ActionBar extends PureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.map.isRequired,
+  };
+
+  isStatusesPageActive = (match, location) => {
+    if (!match) {
+      return false;
+    }
+    return !location.pathname.match(/\/(followers|following)\/?$/);
+  };
+
+  render () {
+    const { account } = this.props;
+
+    if (account.get('suspended')) {
+      return (
+        <div>
+          <div className='account__disclaimer'>
+            <Icon id='info-circle' fixedWidth /> <FormattedMessage
+              id='account.suspended_disclaimer_full'
+              defaultMessage='This user has been suspended by a moderator.'
+            />
+          </div>
+        </div>
+      );
+    }
+
+    let extraInfo = '';
+
+    if (account.get('acct') !== account.get('username')) {
+      extraInfo = (
+        <div className='account__disclaimer'>
+          <Icon id='info-circle' fixedWidth /> <FormattedMessage
+            id='account.disclaimer_full'
+            defaultMessage="Information below may reflect the user's profile incompletely."
+          />
+          {' '}
+          <a target='_blank' rel='noopener' href={account.get('url')}>
+            <FormattedMessage id='account.view_full_profile' defaultMessage='View full profile' />
+          </a>
+        </div>
+      );
+    }
+
+    return (
+      <div>
+        {extraInfo}
+
+        <div className='account__action-bar'>
+          <div className='account__action-bar-links'>
+            <NavLink isActive={this.isStatusesPageActive} activeClassName='active' className='account__action-bar__tab' to={`/@${account.get('acct')}`}>
+              <FormattedMessage id='account.posts' defaultMessage='Posts' />
+              <strong><FormattedNumber value={account.get('statuses_count')} /></strong>
+            </NavLink>
+
+            <NavLink exact activeClassName='active' className='account__action-bar__tab' to={`/@${account.get('acct')}/following`}>
+              <FormattedMessage id='account.follows' defaultMessage='Follows' />
+              <strong><FormattedNumber value={account.get('following_count')} /></strong>
+            </NavLink>
+
+            <NavLink exact activeClassName='active' className='account__action-bar__tab' to={`/@${account.get('acct')}/followers`}>
+              <FormattedMessage id='account.followers' defaultMessage='Followers' />
+              <strong>{ account.get('followers_count') < 0 ? '-' : <FormattedNumber value={account.get('followers_count')} /> }</strong>
+            </NavLink>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default ActionBar;
diff --git a/app/javascript/flavours/blobfox/features/account/components/featured_tags.jsx b/app/javascript/flavours/blobfox/features/account/components/featured_tags.jsx
new file mode 100644
index 00000000000000..4a1219cf984203
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/components/featured_tags.jsx
@@ -0,0 +1,52 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import Hashtag from 'flavours/blobfox/components/hashtag';
+
+const messages = defineMessages({
+  lastStatusAt: { id: 'account.featured_tags.last_status_at', defaultMessage: 'Last post on {date}' },
+  empty: { id: 'account.featured_tags.last_status_never', defaultMessage: 'No posts' },
+});
+
+class FeaturedTags extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record,
+    featuredTags: ImmutablePropTypes.list,
+    tagged: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { account, featuredTags, intl } = this.props;
+
+    if (!account || account.get('suspended') || featuredTags.isEmpty()) {
+      return null;
+    }
+
+    return (
+      <div className='getting-started__trends'>
+        <h4><FormattedMessage id='account.featured_tags.title' defaultMessage="{name}'s featured hashtags" values={{ name: <bdi dangerouslySetInnerHTML={{ __html: account.get('display_name_html') }} /> }} /></h4>
+
+        {featuredTags.take(3).map(featuredTag => (
+          <Hashtag
+            key={featuredTag.get('name')}
+            name={featuredTag.get('name')}
+            href={featuredTag.get('url')}
+            to={`/@${account.get('acct')}/tagged/${featuredTag.get('name')}`}
+            uses={featuredTag.get('statuses_count') * 1}
+            withGraph={false}
+            description={((featuredTag.get('statuses_count') * 1) > 0) ? intl.formatMessage(messages.lastStatusAt, { date: intl.formatDate(featuredTag.get('last_status_at'), { month: 'short', day: '2-digit' }) }) : intl.formatMessage(messages.empty)}
+          />
+        ))}
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(FeaturedTags);
diff --git a/app/javascript/flavours/blobfox/features/account/components/follow_request_note.jsx b/app/javascript/flavours/blobfox/features/account/components/follow_request_note.jsx
new file mode 100644
index 00000000000000..c830ca3d64eafd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/components/follow_request_note.jsx
@@ -0,0 +1,38 @@
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+export default class FollowRequestNote extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+  };
+
+  render () {
+    const { account, onAuthorize, onReject } = this.props;
+
+    return (
+      <div className='follow-request-banner'>
+        <div className='follow-request-banner__message'>
+          <FormattedMessage id='account.requested_follow' defaultMessage='{name} has requested to follow you' values={{ name: <bdi><strong dangerouslySetInnerHTML={{ __html: account.get('display_name_html') }} /></bdi> }} />
+        </div>
+
+        <div className='follow-request-banner__action'>
+          <button type='button' className='button button-tertiary button--confirmation' onClick={onAuthorize}>
+            <Icon id='check' fixedWidth />
+            <FormattedMessage id='follow_request.authorize' defaultMessage='Authorize' />
+          </button>
+
+          <button type='button' className='button button-tertiary button--destructive' onClick={onReject}>
+            <Icon id='times' fixedWidth />
+            <FormattedMessage id='follow_request.reject' defaultMessage='Reject' />
+          </button>
+        </div>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/account/components/header.jsx b/app/javascript/flavours/blobfox/features/account/components/header.jsx
new file mode 100644
index 00000000000000..35b3650825f855
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/components/header.jsx
@@ -0,0 +1,404 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { Button } from 'flavours/blobfox/components/button';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import DropdownMenuContainer from 'flavours/blobfox/containers/dropdown_menu_container';
+import { autoPlayGif, me, domain } from 'flavours/blobfox/initial_state';
+import { PERMISSION_MANAGE_USERS, PERMISSION_MANAGE_FEDERATION } from 'flavours/blobfox/permissions';
+import { preferencesLink, profileLink, accountAdminLink } from 'flavours/blobfox/utils/backend_links';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import AccountNoteContainer from '../containers/account_note_container';
+import FollowRequestNoteContainer from '../containers/follow_request_note_container';
+
+const messages = defineMessages({
+  unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' },
+  follow: { id: 'account.follow', defaultMessage: 'Follow' },
+  cancel_follow_request: { id: 'account.cancel_follow_request', defaultMessage: 'Withdraw follow request' },
+  requested: { id: 'account.requested', defaultMessage: 'Awaiting approval. Click to cancel follow request' },
+  unblock: { id: 'account.unblock', defaultMessage: 'Unblock @{name}' },
+  edit_profile: { id: 'account.edit_profile', defaultMessage: 'Edit profile' },
+  linkVerifiedOn: { id: 'account.link_verified_on', defaultMessage: 'Ownership of this link was checked on {date}' },
+  account_locked: { id: 'account.locked_info', defaultMessage: 'This account privacy status is set to locked. The owner manually reviews who can follow them.' },
+  mention: { id: 'account.mention', defaultMessage: 'Mention @{name}' },
+  direct: { id: 'account.direct', defaultMessage: 'Privately mention @{name}' },
+  unmute: { id: 'account.unmute', defaultMessage: 'Unmute @{name}' },
+  block: { id: 'account.block', defaultMessage: 'Block @{name}' },
+  mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' },
+  report: { id: 'account.report', defaultMessage: 'Report @{name}' },
+  share: { id: 'account.share', defaultMessage: 'Share @{name}\'s profile' },
+  media: { id: 'account.media', defaultMessage: 'Media' },
+  blockDomain: { id: 'account.block_domain', defaultMessage: 'Block domain {domain}' },
+  unblockDomain: { id: 'account.unblock_domain', defaultMessage: 'Unblock domain {domain}' },
+  hideReblogs: { id: 'account.hide_reblogs', defaultMessage: 'Hide boosts from @{name}' },
+  showReblogs: { id: 'account.show_reblogs', defaultMessage: 'Show boosts from @{name}' },
+  enableNotifications: { id: 'account.enable_notifications', defaultMessage: 'Notify me when @{name} posts' },
+  disableNotifications: { id: 'account.disable_notifications', defaultMessage: 'Stop notifying me when @{name} posts' },
+  pins: { id: 'navigation_bar.pins', defaultMessage: 'Pinned posts' },
+  preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
+  follow_requests: { id: 'navigation_bar.follow_requests', defaultMessage: 'Follow requests' },
+  favourites: { id: 'navigation_bar.favourites', defaultMessage: 'Favorites' },
+  lists: { id: 'navigation_bar.lists', defaultMessage: 'Lists' },
+  followed_tags: { id: 'navigation_bar.followed_tags', defaultMessage: 'Followed hashtags' },
+  blocks: { id: 'navigation_bar.blocks', defaultMessage: 'Blocked users' },
+  domain_blocks: { id: 'navigation_bar.domain_blocks', defaultMessage: 'Blocked domains' },
+  mutes: { id: 'navigation_bar.mutes', defaultMessage: 'Muted users' },
+  endorse: { id: 'account.endorse', defaultMessage: 'Feature on profile' },
+  unendorse: { id: 'account.unendorse', defaultMessage: 'Don\'t feature on profile' },
+  add_or_remove_from_list: { id: 'account.add_or_remove_from_list', defaultMessage: 'Add or Remove from lists' },
+  admin_account: { id: 'status.admin_account', defaultMessage: 'Open moderation interface for @{name}' },
+  admin_domain: { id: 'status.admin_domain', defaultMessage: 'Open moderation interface for {domain}' },
+  languages: { id: 'account.languages', defaultMessage: 'Change subscribed languages' },
+  openOriginalPage: { id: 'account.open_original_page', defaultMessage: 'Open original page' },
+});
+
+const titleFromAccount = account => {
+  const displayName = account.get('display_name');
+  const acct = account.get('acct') === account.get('username') ? `${account.get('username')}@${domain}` : account.get('acct');
+  const prefix = displayName.trim().length === 0 ? account.get('username') : displayName;
+
+  return `${prefix} (@${acct})`;
+};
+
+const dateFormatOptions = {
+  month: 'short',
+  day: 'numeric',
+  year: 'numeric',
+  hour12: false,
+  hour: '2-digit',
+  minute: '2-digit',
+};
+
+class Header extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record,
+    identity_props: ImmutablePropTypes.list,
+    onFollow: PropTypes.func.isRequired,
+    onBlock: PropTypes.func.isRequired,
+    onMention: PropTypes.func.isRequired,
+    onDirect: PropTypes.func.isRequired,
+    onReblogToggle: PropTypes.func.isRequired,
+    onNotifyToggle: PropTypes.func.isRequired,
+    onReport: PropTypes.func.isRequired,
+    onMute: PropTypes.func.isRequired,
+    onBlockDomain: PropTypes.func.isRequired,
+    onUnblockDomain: PropTypes.func.isRequired,
+    onEndorseToggle: PropTypes.func.isRequired,
+    onAddToList: PropTypes.func.isRequired,
+    onChangeLanguages: PropTypes.func.isRequired,
+    onInteractionModal: PropTypes.func.isRequired,
+    onOpenAvatar: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    domain: PropTypes.string.isRequired,
+    hidden: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  openEditProfile = () => {
+    window.open(profileLink, '_blank');
+  };
+
+  handleMouseEnter = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-original');
+    }
+  };
+
+  handleMouseLeave = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-static');
+    }
+  };
+
+  handleAvatarClick = e => {
+    if (e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+      this.props.onOpenAvatar();
+    }
+  };
+
+  handleShare = () => {
+    const { account } = this.props;
+
+    navigator.share({
+      url: account.get('url'),
+    }).catch((e) => {
+      if (e.name !== 'AbortError') console.error(e);
+    });
+  };
+
+  render () {
+    const { account, hidden, intl, domain } = this.props;
+    const { signedIn, permissions } = this.context.identity;
+
+    if (!account) {
+      return null;
+    }
+
+    const suspended    = account.get('suspended');
+    const isRemote     = account.get('acct') !== account.get('username');
+    const remoteDomain = isRemote ? account.get('acct').split('@')[1] : null;
+
+    let info        = [];
+    let actionBtn   = '';
+    let bellBtn     = '';
+    let lockedIcon  = '';
+    let menu        = [];
+
+    if (me !== account.get('id') && account.getIn(['relationship', 'followed_by'])) {
+      info.push(<span className='relationship-tag'><FormattedMessage id='account.follows_you' defaultMessage='Follows you' /></span>);
+    } else if (me !== account.get('id') && account.getIn(['relationship', 'blocking'])) {
+      info.push(<span className='relationship-tag'><FormattedMessage id='account.blocked' defaultMessage='Blocked' /></span>);
+    }
+
+    if (me !== account.get('id') && account.getIn(['relationship', 'muting'])) {
+      info.push(<span className='relationship-tag'><FormattedMessage id='account.muted' defaultMessage='Muted' /></span>);
+    } else if (me !== account.get('id') && account.getIn(['relationship', 'domain_blocking'])) {
+      info.push(<span className='relationship-tag'><FormattedMessage id='account.domain_blocked' defaultMessage='Domain blocked' /></span>);
+    }
+
+    if (account.getIn(['relationship', 'requested']) || account.getIn(['relationship', 'following'])) {
+      bellBtn = <IconButton icon={account.getIn(['relationship', 'notifying']) ? 'bell' : 'bell-o'} size={24} active={account.getIn(['relationship', 'notifying'])} title={intl.formatMessage(account.getIn(['relationship', 'notifying']) ? messages.disableNotifications : messages.enableNotifications, { name: account.get('username') })} onClick={this.props.onNotifyToggle} />;
+    }
+
+    if (me !== account.get('id')) {
+      if (signedIn && !account.get('relationship')) { // Wait until the relationship is loaded
+        actionBtn = '';
+      } else if (account.getIn(['relationship', 'requested'])) {
+        actionBtn = <Button text={intl.formatMessage(messages.cancel_follow_request)} title={intl.formatMessage(messages.requested)} onClick={this.props.onFollow} />;
+      } else if (!account.getIn(['relationship', 'blocking'])) {
+        actionBtn = <Button className={classNames({ 'button--destructive': account.getIn(['relationship', 'following']) })} text={intl.formatMessage(account.getIn(['relationship', 'following']) ? messages.unfollow : messages.follow)} onClick={signedIn ? this.props.onFollow : this.props.onInteractionModal} />;
+      } else if (account.getIn(['relationship', 'blocking'])) {
+        actionBtn = <Button text={intl.formatMessage(messages.unblock, { name: account.get('username') })} onClick={this.props.onBlock} />;
+      }
+    } else if (profileLink) {
+      actionBtn = <Button text={intl.formatMessage(messages.edit_profile)} onClick={this.openEditProfile} />;
+    }
+
+    if (account.get('moved') && !account.getIn(['relationship', 'following'])) {
+      actionBtn = '';
+    }
+
+    if (account.get('suspended') && !account.getIn(['relationship', 'following'])) {
+      actionBtn = '';
+    }
+
+    if (account.get('locked')) {
+      lockedIcon = <Icon id='lock' title={intl.formatMessage(messages.account_locked)} />;
+    }
+
+    if (signedIn && account.get('id') !== me && !account.get('suspended')) {
+      menu.push({ text: intl.formatMessage(messages.mention, { name: account.get('username') }), action: this.props.onMention });
+      menu.push({ text: intl.formatMessage(messages.direct, { name: account.get('username') }), action: this.props.onDirect });
+      menu.push(null);
+    }
+
+    if (isRemote) {
+      menu.push({ text: intl.formatMessage(messages.openOriginalPage), href: account.get('url') });
+      menu.push(null);
+    }
+
+    if ('share' in navigator && !account.get('suspended')) {
+      menu.push({ text: intl.formatMessage(messages.share, { name: account.get('username') }), action: this.handleShare });
+      menu.push(null);
+    }
+
+    if (account.get('id') === me) {
+      if (profileLink) menu.push({ text: intl.formatMessage(messages.edit_profile), href: profileLink });
+      if (preferencesLink) menu.push({ text: intl.formatMessage(messages.preferences), href: preferencesLink });
+      menu.push({ text: intl.formatMessage(messages.pins), to: '/pinned' });
+      menu.push(null);
+      menu.push({ text: intl.formatMessage(messages.follow_requests), to: '/follow_requests' });
+      menu.push({ text: intl.formatMessage(messages.favourites), to: '/favourites' });
+      menu.push({ text: intl.formatMessage(messages.lists), to: '/lists' });
+      menu.push({ text: intl.formatMessage(messages.followed_tags), to: '/followed_tags' });
+      menu.push(null);
+      menu.push({ text: intl.formatMessage(messages.mutes), to: '/mutes' });
+      menu.push({ text: intl.formatMessage(messages.blocks), to: '/blocks' });
+      menu.push({ text: intl.formatMessage(messages.domain_blocks), to: '/domain_blocks' });
+    } else if (signedIn) {
+      if (account.getIn(['relationship', 'following'])) {
+        if (!account.getIn(['relationship', 'muting'])) {
+          if (account.getIn(['relationship', 'showing_reblogs'])) {
+            menu.push({ text: intl.formatMessage(messages.hideReblogs, { name: account.get('username') }), action: this.props.onReblogToggle });
+          } else {
+            menu.push({ text: intl.formatMessage(messages.showReblogs, { name: account.get('username') }), action: this.props.onReblogToggle });
+          }
+
+          menu.push({ text: intl.formatMessage(messages.languages), action: this.props.onChangeLanguages });
+          menu.push(null);
+        }
+
+        menu.push({ text: intl.formatMessage(account.getIn(['relationship', 'endorsed']) ? messages.unendorse : messages.endorse), action: this.props.onEndorseToggle });
+        menu.push({ text: intl.formatMessage(messages.add_or_remove_from_list), action: this.props.onAddToList });
+        menu.push(null);
+      }
+
+      if (account.getIn(['relationship', 'muting'])) {
+        menu.push({ text: intl.formatMessage(messages.unmute, { name: account.get('username') }), action: this.props.onMute });
+      } else {
+        menu.push({ text: intl.formatMessage(messages.mute, { name: account.get('username') }), action: this.props.onMute, dangerous: true });
+      }
+
+      if (account.getIn(['relationship', 'blocking'])) {
+        menu.push({ text: intl.formatMessage(messages.unblock, { name: account.get('username') }), action: this.props.onBlock });
+      } else {
+        menu.push({ text: intl.formatMessage(messages.block, { name: account.get('username') }), action: this.props.onBlock, dangerous: true });
+      }
+
+      if (!account.get('suspended')) {
+        menu.push({ text: intl.formatMessage(messages.report, { name: account.get('username') }), action: this.props.onReport, dangerous: true });
+      }
+    }
+
+    if (signedIn && isRemote) {
+      menu.push(null);
+
+      if (account.getIn(['relationship', 'domain_blocking'])) {
+        menu.push({ text: intl.formatMessage(messages.unblockDomain, { domain: remoteDomain }), action: this.props.onUnblockDomain });
+      } else {
+        menu.push({ text: intl.formatMessage(messages.blockDomain, { domain: remoteDomain }), action: this.props.onBlockDomain, dangerous: true });
+      }
+    }
+
+    if (account.get('id') !== me && ((permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS && accountAdminLink) || (isRemote && (permissions & PERMISSION_MANAGE_FEDERATION) === PERMISSION_MANAGE_FEDERATION)) {
+      menu.push(null);
+      if ((permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS && accountAdminLink) {
+        menu.push({ text: intl.formatMessage(messages.admin_account, { name: account.get('username') }), href: accountAdminLink(account.get('id')) });
+      }
+      if (isRemote && (permissions & PERMISSION_MANAGE_FEDERATION) === PERMISSION_MANAGE_FEDERATION) {
+        menu.push({ text: intl.formatMessage(messages.admin_domain, { domain: remoteDomain }), href: `/admin/instances/${remoteDomain}` });
+      }
+    }
+
+    const content          = { __html: account.get('note_emojified') };
+    const displayNameHtml = { __html: account.get('display_name_html') };
+    const fields          = account.get('fields');
+    const isLocal         = account.get('acct').indexOf('@') === -1;
+    const acct            = isLocal && domain ? `${account.get('acct')}@${domain}` : account.get('acct');
+    const isIndexable     = !account.get('noindex');
+
+    let badge;
+
+    if (account.get('bot')) {
+      badge = (<div className='account-role bot'><FormattedMessage id='account.badges.bot' defaultMessage='Automated' /></div>);
+    } else if (account.get('group')) {
+      badge = (<div className='account-role group'><FormattedMessage id='account.badges.group' defaultMessage='Group' /></div>);
+    } else {
+      badge = null;
+    }
+
+    let role = null;
+    if (account.getIn(['roles', 0])) {
+      role = (<div key='role' className={`account-role user-role-${account.getIn(['roles', 0, 'id'])}`}>{account.getIn(['roles', 0, 'name'])}</div>);
+    }
+
+    return (
+      <div className={classNames('account__header', { inactive: !!account.get('moved') })} onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave}>
+        {!(suspended || hidden || account.get('moved')) && account.getIn(['relationship', 'requested_by']) && <FollowRequestNoteContainer account={account} />}
+
+        <div className='account__header__image'>
+          <div className='account__header__info'>
+            {info}
+          </div>
+
+          {!(suspended || hidden) && <img src={autoPlayGif ? account.get('header') : account.get('header_static')} alt='' className='parallax' />}
+        </div>
+
+        <div className='account__header__bar'>
+          <div className='account__header__tabs'>
+            <a className='avatar' href={account.get('avatar')} rel='noopener noreferrer' target='_blank' onClick={this.handleAvatarClick}>
+              <Avatar account={suspended || hidden ? undefined : account} size={90} />
+              {role}
+            </a>
+
+            <div className='account__header__tabs__buttons'>
+              {!hidden && (
+                <>
+                  {actionBtn}
+                  {bellBtn}
+                </>
+              )}
+
+              <DropdownMenuContainer disabled={menu.length === 0} items={menu} icon='ellipsis-v' size={24} direction='right' />
+            </div>
+          </div>
+
+          <div className='account__header__tabs__name'>
+            <h1>
+              <span dangerouslySetInnerHTML={displayNameHtml} /> {badge}
+              <small>
+                <span>@{acct}</span> {lockedIcon}
+              </small>
+            </h1>
+          </div>
+
+          {signedIn && <AccountNoteContainer account={account} />}
+
+          {!(suspended || hidden) && (
+            <div className='account__header__extra'>
+              <div className='account__header__bio'>
+                { fields.size > 0 && (
+                  <div className='account__header__fields'>
+                    {fields.map((pair, i) => (
+                      <dl key={i}>
+                        <dt dangerouslySetInnerHTML={{ __html: pair.get('name_emojified') }} title={pair.get('name')} />
+
+                        <dd className={pair.get('verified_at') && 'verified'} title={pair.get('value_plain')}>
+                          {pair.get('verified_at') && <span title={intl.formatMessage(messages.linkVerifiedOn, { date: intl.formatDate(pair.get('verified_at'), dateFormatOptions) })}><Icon id='check' className='verified__mark' /></span>} <span dangerouslySetInnerHTML={{ __html: pair.get('value_emojified') }} className='translate' />
+                        </dd>
+                      </dl>
+                    ))}
+                  </div>
+                )}
+
+                {account.get('note').length > 0 && account.get('note') !== '<p></p>' && <div className='account__header__content translate' dangerouslySetInnerHTML={content} />}
+
+                <div className='account__header__joined'><FormattedMessage id='account.joined' defaultMessage='Joined {date}' values={{ date: intl.formatDate(account.get('created_at'), { year: 'numeric', month: 'short', day: '2-digit' }) }} /></div>
+              </div>
+            </div>
+          )}
+        </div>
+
+        <Helmet>
+          <title>{titleFromAccount(account)}</title>
+          <meta name='robots' content={(isLocal && isIndexable) ? 'all' : 'noindex'} />
+          <link rel='canonical' href={account.get('url')} />
+        </Helmet>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(Header));
diff --git a/app/javascript/flavours/blobfox/features/account/components/profile_column_header.jsx b/app/javascript/flavours/blobfox/features/account/components/profile_column_header.jsx
new file mode 100644
index 00000000000000..2dc4216bdd1001
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/components/profile_column_header.jsx
@@ -0,0 +1,36 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import ColumnHeader from '../../../components/column_header';
+
+const messages = defineMessages({
+  profile: { id: 'column_header.profile', defaultMessage: 'Profile' },
+});
+
+class ProfileColumnHeader extends PureComponent {
+
+  static propTypes = {
+    onClick: PropTypes.func,
+    multiColumn: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render() {
+    const { onClick, intl, multiColumn } = this.props;
+
+    return (
+      <ColumnHeader
+        icon='user-circle'
+        title={intl.formatMessage(messages.profile)}
+        onClick={onClick}
+        showBackButton
+        multiColumn={multiColumn}
+      />
+    );
+  }
+
+}
+
+export default injectIntl(ProfileColumnHeader);
diff --git a/app/javascript/flavours/blobfox/features/account/containers/account_note_container.js b/app/javascript/flavours/blobfox/features/account/containers/account_note_container.js
new file mode 100644
index 00000000000000..3d24e158b34e77
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/containers/account_note_container.js
@@ -0,0 +1,19 @@
+import { connect } from 'react-redux';
+
+import { submitAccountNote } from 'flavours/blobfox/actions/account_notes';
+
+import AccountNote from '../components/account_note';
+
+const mapStateToProps = (state, { account }) => ({
+  value: account.getIn(['relationship', 'note']),
+});
+
+const mapDispatchToProps = (dispatch, { account }) => ({
+
+  onSave (value) {
+    dispatch(submitAccountNote({ id: account.get('id'), value}));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(AccountNote);
diff --git a/app/javascript/flavours/blobfox/features/account/containers/featured_tags_container.js b/app/javascript/flavours/blobfox/features/account/containers/featured_tags_container.js
new file mode 100644
index 00000000000000..5d4344086653bb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/containers/featured_tags_container.js
@@ -0,0 +1,17 @@
+import { List as ImmutableList } from 'immutable';
+import { connect } from 'react-redux';
+
+import { makeGetAccount } from 'flavours/blobfox/selectors';
+
+import FeaturedTags from '../components/featured_tags';
+
+const mapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  return (state, { accountId }) => ({
+    account: getAccount(state, accountId),
+    featuredTags: state.getIn(['user_lists', 'featured_tags', accountId, 'items'], ImmutableList()),
+  });
+};
+
+export default connect(mapStateToProps)(FeaturedTags);
diff --git a/app/javascript/flavours/blobfox/features/account/containers/follow_request_note_container.js b/app/javascript/flavours/blobfox/features/account/containers/follow_request_note_container.js
new file mode 100644
index 00000000000000..1771a78fecfae0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/containers/follow_request_note_container.js
@@ -0,0 +1,17 @@
+import { connect } from 'react-redux';
+
+import { authorizeFollowRequest, rejectFollowRequest } from 'flavours/blobfox/actions/accounts';
+
+import FollowRequestNote from '../components/follow_request_note';
+
+const mapDispatchToProps = (dispatch, { account }) => ({
+  onAuthorize () {
+    dispatch(authorizeFollowRequest(account.get('id')));
+  },
+
+  onReject () {
+    dispatch(rejectFollowRequest(account.get('id')));
+  },
+});
+
+export default connect(null, mapDispatchToProps)(FollowRequestNote);
diff --git a/app/javascript/flavours/blobfox/features/account/navigation.jsx b/app/javascript/flavours/blobfox/features/account/navigation.jsx
new file mode 100644
index 00000000000000..ddb8a835275c05
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account/navigation.jsx
@@ -0,0 +1,55 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { connect } from 'react-redux';
+
+import FeaturedTags from 'flavours/blobfox/features/account/containers/featured_tags_container';
+import { normalizeForLookup } from 'flavours/blobfox/reducers/accounts_map';
+
+const mapStateToProps = (state, { match: { params: { acct } } }) => {
+  const accountId = state.getIn(['accounts_map', normalizeForLookup(acct)]);
+
+  if (!accountId) {
+    return {
+      isLoading: true,
+    };
+  }
+
+  return {
+    accountId,
+    isLoading: false,
+  };
+};
+
+class AccountNavigation extends PureComponent {
+
+  static propTypes = {
+    match: PropTypes.shape({
+      params: PropTypes.shape({
+        acct: PropTypes.string,
+        tagged: PropTypes.string,
+      }).isRequired,
+    }).isRequired,
+
+    accountId: PropTypes.string,
+    isLoading: PropTypes.bool,
+  };
+
+  render () {
+    const { accountId, isLoading, match: { params: { tagged } } } = this.props;
+
+    if (isLoading) {
+      return null;
+    }
+
+    return (
+      <>
+        <div className='flex-spacer' />
+        <FeaturedTags accountId={accountId} tagged={tagged} />
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(AccountNavigation);
diff --git a/app/javascript/flavours/blobfox/features/account_gallery/components/media_item.jsx b/app/javascript/flavours/blobfox/features/account_gallery/components/media_item.jsx
new file mode 100644
index 00000000000000..cda7fd5e00be10
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_gallery/components/media_item.jsx
@@ -0,0 +1,155 @@
+import PropTypes from 'prop-types';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Blurhash } from 'flavours/blobfox/components/blurhash';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { autoPlayGif, displayMedia, useBlurhash } from 'flavours/blobfox/initial_state';
+
+export default class MediaItem extends ImmutablePureComponent {
+
+  static propTypes = {
+    attachment: ImmutablePropTypes.map.isRequired,
+    displayWidth: PropTypes.number.isRequired,
+    onOpenMedia: PropTypes.func.isRequired,
+  };
+
+  state = {
+    visible: displayMedia !== 'hide_all' && !this.props.attachment.getIn(['status', 'sensitive']) || displayMedia === 'show_all',
+    loaded: false,
+  };
+
+  handleImageLoad = () => {
+    this.setState({ loaded: true });
+  };
+
+  handleMouseEnter = e => {
+    if (this.hoverToPlay()) {
+      e.target.play();
+    }
+  };
+
+  handleMouseLeave = e => {
+    if (this.hoverToPlay()) {
+      e.target.pause();
+      e.target.currentTime = 0;
+    }
+  };
+
+  hoverToPlay () {
+    return !autoPlayGif && ['gifv', 'video'].indexOf(this.props.attachment.get('type')) !== -1;
+  }
+
+  handleClick = e => {
+    if (e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+
+      if (this.state.visible) {
+        this.props.onOpenMedia(this.props.attachment);
+      } else {
+        this.setState({ visible: true });
+      }
+    }
+  };
+
+  render () {
+    const { attachment, displayWidth } = this.props;
+    const { visible, loaded } = this.state;
+
+    const width  = `${Math.floor((displayWidth - 4) / 3) - 4}px`;
+    const height = width;
+    const status = attachment.get('status');
+    const title  = status.get('spoiler_text') || attachment.get('description');
+
+    let thumbnail, label, icon, content;
+
+    if (!visible) {
+      icon = (
+        <span className='account-gallery__item__icons'>
+          <Icon id='eye-slash' />
+        </span>
+      );
+    } else {
+      if (['audio', 'video'].includes(attachment.get('type'))) {
+        content = (
+          <img
+            src={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
+            alt={attachment.get('description')}
+            lang={status.get('language')}
+            onLoad={this.handleImageLoad}
+          />
+        );
+
+        if (attachment.get('type') === 'audio') {
+          label = <Icon id='music' />;
+        } else {
+          label = <Icon id='play' />;
+        }
+      } else if (attachment.get('type') === 'image') {
+        const focusX = attachment.getIn(['meta', 'focus', 'x']) || 0;
+        const focusY = attachment.getIn(['meta', 'focus', 'y']) || 0;
+        const x      = ((focusX /  2) + .5) * 100;
+        const y      = ((focusY / -2) + .5) * 100;
+
+        content = (
+          <img
+            src={attachment.get('preview_url')}
+            alt={attachment.get('description')}
+            lang={status.get('language')}
+            style={{ objectPosition: `${x}% ${y}%` }}
+            onLoad={this.handleImageLoad}
+          />
+        );
+      } else if (attachment.get('type') === 'gifv') {
+        content = (
+          <video
+            className='media-gallery__item-gifv-thumbnail'
+            aria-label={attachment.get('description')}
+            title={attachment.get('description')}
+            lang={status.get('language')}
+            role='application'
+            src={attachment.get('url')}
+            onMouseEnter={this.handleMouseEnter}
+            onMouseLeave={this.handleMouseLeave}
+            autoPlay={autoPlayGif}
+            playsInline
+            loop
+            muted
+          />
+        );
+
+        label = 'GIF';
+      }
+
+      thumbnail = (
+        <div className='media-gallery__gifv'>
+          {content}
+
+          {label && (
+            <div className='media-gallery__item__badges'>
+              <span className='media-gallery__gifv__label'>{label}</span>
+            </div>
+          )}
+        </div>
+      );
+    }
+
+    return (
+      <div className='account-gallery__item' style={{ width, height }}>
+        <a className='media-gallery__item-thumbnail' href={status.get('url')} onClick={this.handleClick} title={title} target='_blank' rel='noopener noreferrer'>
+          <Blurhash
+            hash={attachment.get('blurhash')}
+            className={classNames('media-gallery__preview', { 'media-gallery__preview--hidden': visible && loaded })}
+            dummy={!useBlurhash}
+          />
+
+          {visible ? thumbnail : icon}
+        </a>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/account_gallery/index.jsx b/app/javascript/flavours/blobfox/features/account_gallery/index.jsx
new file mode 100644
index 00000000000000..7ef3bc688c6d39
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_gallery/index.jsx
@@ -0,0 +1,239 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { lookupAccount, fetchAccount } from 'flavours/blobfox/actions/accounts';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { LoadMore } from 'flavours/blobfox/components/load_more';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import ScrollContainer from 'flavours/blobfox/containers/scroll_container';
+import ProfileColumnHeader from 'flavours/blobfox/features/account/components/profile_column_header';
+import BundleColumnError from 'flavours/blobfox/features/ui/components/bundle_column_error';
+import { normalizeForLookup } from 'flavours/blobfox/reducers/accounts_map';
+import { getAccountGallery } from 'flavours/blobfox/selectors';
+
+import { expandAccountMediaTimeline } from '../../actions/timelines';
+import HeaderContainer from '../account_timeline/containers/header_container';
+import Column from '../ui/components/column';
+
+import MediaItem from './components/media_item';
+
+const mapStateToProps = (state, { params: { acct, id } }) => {
+  const accountId = id || state.getIn(['accounts_map', normalizeForLookup(acct)]);
+
+  if (!accountId) {
+    return {
+      isLoading: true,
+    };
+  }
+
+  return {
+    accountId,
+    isAccount: !!state.getIn(['accounts', accountId]),
+    attachments: getAccountGallery(state, accountId),
+    isLoading: state.getIn(['timelines', `account:${accountId}:media`, 'isLoading']),
+    hasMore: state.getIn(['timelines', `account:${accountId}:media`, 'hasMore']),
+    suspended: state.getIn(['accounts', accountId, 'suspended'], false),
+  };
+};
+
+class LoadMoreMedia extends ImmutablePureComponent {
+
+  static propTypes = {
+    maxId: PropTypes.string,
+    onLoadMore: PropTypes.func.isRequired,
+  };
+
+  handleLoadMore = () => {
+    this.props.onLoadMore(this.props.maxId);
+  };
+
+  render () {
+    return (
+      <LoadMore
+        disabled={this.props.disabled}
+        onClick={this.handleLoadMore}
+      />
+    );
+  }
+
+}
+
+class AccountGallery extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.shape({
+      acct: PropTypes.string,
+      id: PropTypes.string,
+    }).isRequired,
+    accountId: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    attachments: ImmutablePropTypes.list.isRequired,
+    isLoading: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    isAccount: PropTypes.bool,
+    suspended: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+  };
+
+  state = {
+    width: 323,
+  };
+
+  _load () {
+    const { accountId, isAccount, dispatch } = this.props;
+
+    if (!isAccount) dispatch(fetchAccount(accountId));
+    dispatch(expandAccountMediaTimeline(accountId));
+  }
+
+  componentDidMount () {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (accountId) {
+      this._load();
+    } else {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (prevProps.accountId !== accountId && accountId) {
+      this._load();
+    } else if (prevProps.params.acct !== acct) {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  handleScrollToBottom = () => {
+    if (this.props.hasMore) {
+      this.handleLoadMore(this.props.attachments.size > 0 ? this.props.attachments.last().getIn(['status', 'id']) : undefined);
+    }
+  };
+
+  handleScroll = e => {
+    const { scrollTop, scrollHeight, clientHeight } = e.target;
+    const offset = scrollHeight - scrollTop - clientHeight;
+
+    if (150 > offset && !this.props.isLoading) {
+      this.handleScrollToBottom();
+    }
+  };
+
+  handleLoadMore = maxId => {
+    this.props.dispatch(expandAccountMediaTimeline(this.props.accountId, { maxId }));
+  };
+
+  handleLoadOlder = e => {
+    e.preventDefault();
+    this.handleScrollToBottom();
+  };
+
+  setColumnRef = c => {
+    this.column = c;
+  };
+
+  handleOpenMedia = attachment => {
+    const { dispatch } = this.props;
+    const statusId = attachment.getIn(['status', 'id']);
+    const lang = attachment.getIn(['status', 'language']);
+
+    if (attachment.get('type') === 'video') {
+      dispatch(openModal({
+        modalType: 'VIDEO',
+        modalProps: { media: attachment, statusId, lang, options: { autoPlay: true } },
+      }));
+    } else if (attachment.get('type') === 'audio') {
+      dispatch(openModal({
+        modalType: 'AUDIO',
+        modalProps: { media: attachment, statusId, lang, options: { autoPlay: true } },
+      }));
+    } else {
+      const media = attachment.getIn(['status', 'media_attachments']);
+      const index = media.findIndex(x => x.get('id') === attachment.get('id'));
+
+      dispatch(openModal({
+        modalType: 'MEDIA',
+        modalProps: { media, index, statusId, lang },
+      }));
+    }
+  };
+
+  handleRef = c => {
+    if (c) {
+      this.setState({ width: c.offsetWidth });
+    }
+  };
+
+  render () {
+    const { attachments, isLoading, hasMore, isAccount, multiColumn, suspended } = this.props;
+    const { width } = this.state;
+
+    if (!isAccount) {
+      return (
+        <BundleColumnError multiColumn={multiColumn} errorType='routing' />
+      );
+    }
+
+    if (!attachments && isLoading) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    let loadOlder = null;
+
+    if (hasMore && !(isLoading && attachments.size === 0)) {
+      loadOlder = <LoadMore visible={!isLoading} onClick={this.handleLoadOlder} />;
+    }
+
+    return (
+      <Column ref={this.setColumnRef}>
+        <ProfileColumnHeader onClick={this.handleHeaderClick} multiColumn={multiColumn} />
+
+        <ScrollContainer scrollKey='account_gallery'>
+          <div className='scrollable scrollable--flex' onScroll={this.handleScroll}>
+            <HeaderContainer accountId={this.props.accountId} />
+
+            {suspended ? (
+              <div className='empty-column-indicator'>
+                <FormattedMessage id='empty_column.account_suspended' defaultMessage='Account suspended' />
+              </div>
+            ) : (
+              <div role='feed' className='account-gallery__container' ref={this.handleRef}>
+                {attachments.map((attachment, index) => attachment === null ? (
+                  <LoadMoreMedia key={'more:' + attachments.getIn(index + 1, 'id')} maxId={index > 0 ? attachments.getIn(index - 1, 'id') : null} onLoadMore={this.handleLoadMore} />
+                ) : (
+                  <MediaItem key={attachment.get('id')} attachment={attachment} displayWidth={width} onOpenMedia={this.handleOpenMedia} />
+                ))}
+
+                {loadOlder}
+              </div>
+            )}
+
+            {isLoading && attachments.size === 0 && (
+              <div className='scrollable__append'>
+                <LoadingIndicator />
+              </div>
+            )}
+          </div>
+        </ScrollContainer>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(AccountGallery);
diff --git a/app/javascript/flavours/blobfox/features/account_timeline/components/header.jsx b/app/javascript/flavours/blobfox/features/account_timeline/components/header.jsx
new file mode 100644
index 00000000000000..cb9fc5b87f4650
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_timeline/components/header.jsx
@@ -0,0 +1,165 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import { NavLink, withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import ActionBar from '../../account/components/action_bar';
+import InnerHeader from '../../account/components/header';
+
+import MemorialNote from './memorial_note';
+import MovedNote from './moved_note';
+
+class Header extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record,
+    onFollow: PropTypes.func.isRequired,
+    onBlock: PropTypes.func.isRequired,
+    onMention: PropTypes.func.isRequired,
+    onDirect: PropTypes.func.isRequired,
+    onReblogToggle: PropTypes.func.isRequired,
+    onReport: PropTypes.func.isRequired,
+    onMute: PropTypes.func.isRequired,
+    onBlockDomain: PropTypes.func.isRequired,
+    onUnblockDomain: PropTypes.func.isRequired,
+    onEndorseToggle: PropTypes.func.isRequired,
+    onAddToList: PropTypes.func.isRequired,
+    onChangeLanguages: PropTypes.func.isRequired,
+    onInteractionModal: PropTypes.func.isRequired,
+    onOpenAvatar: PropTypes.func.isRequired,
+    hideTabs: PropTypes.bool,
+    domain: PropTypes.string.isRequired,
+    hidden: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  handleFollow = () => {
+    this.props.onFollow(this.props.account);
+  };
+
+  handleBlock = () => {
+    this.props.onBlock(this.props.account);
+  };
+
+  handleMention = () => {
+    this.props.onMention(this.props.account, this.props.history);
+  };
+
+  handleDirect = () => {
+    this.props.onDirect(this.props.account, this.props.history);
+  };
+
+  handleReport = () => {
+    this.props.onReport(this.props.account);
+  };
+
+  handleReblogToggle = () => {
+    this.props.onReblogToggle(this.props.account);
+  };
+
+  handleNotifyToggle = () => {
+    this.props.onNotifyToggle(this.props.account);
+  };
+
+  handleMute = () => {
+    this.props.onMute(this.props.account);
+  };
+
+  handleBlockDomain = () => {
+    const domain = this.props.account.get('acct').split('@')[1];
+
+    if (!domain) return;
+
+    this.props.onBlockDomain(domain);
+  };
+
+  handleUnblockDomain = () => {
+    const domain = this.props.account.get('acct').split('@')[1];
+
+    if (!domain) return;
+
+    this.props.onUnblockDomain(domain);
+  };
+
+  handleEndorseToggle = () => {
+    this.props.onEndorseToggle(this.props.account);
+  };
+
+  handleAddToList = () => {
+    this.props.onAddToList(this.props.account);
+  };
+
+  handleEditAccountNote = () => {
+    this.props.onEditAccountNote(this.props.account);
+  };
+
+  handleChangeLanguages = () => {
+    this.props.onChangeLanguages(this.props.account);
+  };
+
+  handleInteractionModal = () => {
+    this.props.onInteractionModal(this.props.account);
+  };
+
+  handleOpenAvatar = () => {
+    this.props.onOpenAvatar(this.props.account);
+  };
+
+  render () {
+    const { account, hidden, hideTabs } = this.props;
+
+    if (account === null) {
+      return null;
+    }
+
+    return (
+      <div className='account-timeline__header'>
+        {(!hidden && account.get('memorial')) && <MemorialNote />}
+        {(!hidden && account.get('moved')) && <MovedNote from={account} to={account.get('moved')} />}
+
+        <InnerHeader
+          account={account}
+          onFollow={this.handleFollow}
+          onBlock={this.handleBlock}
+          onMention={this.handleMention}
+          onDirect={this.handleDirect}
+          onReblogToggle={this.handleReblogToggle}
+          onNotifyToggle={this.handleNotifyToggle}
+          onReport={this.handleReport}
+          onMute={this.handleMute}
+          onBlockDomain={this.handleBlockDomain}
+          onUnblockDomain={this.handleUnblockDomain}
+          onEndorseToggle={this.handleEndorseToggle}
+          onAddToList={this.handleAddToList}
+          onEditAccountNote={this.handleEditAccountNote}
+          onChangeLanguages={this.handleChangeLanguages}
+          onInteractionModal={this.handleInteractionModal}
+          onOpenAvatar={this.handleOpenAvatar}
+          domain={this.props.domain}
+          hidden={hidden}
+        />
+
+        <ActionBar
+          account={account}
+        />
+
+        {!(hideTabs || hidden) && (
+          <div className='account__section-headline'>
+            <NavLink exact to={`/@${account.get('acct')}`}><FormattedMessage id='account.posts' defaultMessage='Posts' /></NavLink>
+            <NavLink exact to={`/@${account.get('acct')}/with_replies`}><FormattedMessage id='account.posts_with_replies' defaultMessage='Posts and replies' /></NavLink>
+            <NavLink exact to={`/@${account.get('acct')}/media`}><FormattedMessage id='account.media' defaultMessage='Media' /></NavLink>
+          </div>
+        )}
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(Header);
diff --git a/app/javascript/flavours/blobfox/features/account_timeline/components/limited_account_hint.tsx b/app/javascript/flavours/blobfox/features/account_timeline/components/limited_account_hint.tsx
new file mode 100644
index 00000000000000..8096dff4996146
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_timeline/components/limited_account_hint.tsx
@@ -0,0 +1,35 @@
+import { useCallback } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { revealAccount } from 'flavours/blobfox/actions/accounts_typed';
+import { Button } from 'flavours/blobfox/components/button';
+import { domain } from 'flavours/blobfox/initial_state';
+import { useAppDispatch } from 'flavours/blobfox/store';
+
+export const LimitedAccountHint: React.FC<{ accountId: string }> = ({
+  accountId,
+}) => {
+  const dispatch = useAppDispatch();
+  const reveal = useCallback(() => {
+    dispatch(revealAccount({ id: accountId }));
+  }, [dispatch, accountId]);
+
+  return (
+    <div className='limited-account-hint'>
+      <p>
+        <FormattedMessage
+          id='limited_account_hint.title'
+          defaultMessage='This profile has been hidden by the moderators of {domain}.'
+          values={{ domain }}
+        />
+      </p>
+      <Button onClick={reveal}>
+        <FormattedMessage
+          id='limited_account_hint.action'
+          defaultMessage='Show profile anyway'
+        />
+      </Button>
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/features/account_timeline/components/memorial_note.jsx b/app/javascript/flavours/blobfox/features/account_timeline/components/memorial_note.jsx
new file mode 100644
index 00000000000000..a04808f1caf737
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_timeline/components/memorial_note.jsx
@@ -0,0 +1,11 @@
+import { FormattedMessage } from 'react-intl';
+
+const MemorialNote = () => (
+  <div className='account-memorial-banner'>
+    <div className='account-memorial-banner__message'>
+      <FormattedMessage id='account.in_memoriam' defaultMessage='In Memoriam.' />
+    </div>
+  </div>
+);
+
+export default MemorialNote;
diff --git a/app/javascript/flavours/blobfox/features/account_timeline/components/moved_note.jsx b/app/javascript/flavours/blobfox/features/account_timeline/components/moved_note.jsx
new file mode 100644
index 00000000000000..aecb30827a9e32
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_timeline/components/moved_note.jsx
@@ -0,0 +1,52 @@
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { AvatarOverlay } from '../../../components/avatar_overlay';
+import { DisplayName } from '../../../components/display_name';
+
+class MovedNote extends ImmutablePureComponent {
+
+  static propTypes = {
+    from: ImmutablePropTypes.map.isRequired,
+    to: ImmutablePropTypes.map.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  handleAccountClick = e => {
+    if (e.button === 0) {
+      e.preventDefault();
+      this.props.history.push(`/@${this.props.to.get('acct')}`);
+    }
+
+    e.stopPropagation();
+  };
+
+  render () {
+    const { from, to } = this.props;
+    const displayNameHtml = { __html: from.get('display_name_html') };
+
+    return (
+      <div className='account__moved-note'>
+        <div className='account__moved-note__message'>
+          <div className='account__moved-note__icon-wrapper'><Icon id='suitcase' className='account__moved-note__icon' fixedWidth /></div>
+          <FormattedMessage id='account.moved_to' defaultMessage='{name} has indicated that their new account is now:' values={{ name: <bdi><strong dangerouslySetInnerHTML={displayNameHtml} /></bdi> }} />
+        </div>
+
+        <a href={to.get('url')} onClick={this.handleAccountClick} className='detailed-status__display-name'>
+          <div className='detailed-status__display-avatar'><AvatarOverlay account={to} friend={from} /></div>
+          <DisplayName account={to} />
+        </a>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(MovedNote);
diff --git a/app/javascript/flavours/blobfox/features/account_timeline/containers/header_container.jsx b/app/javascript/flavours/blobfox/features/account_timeline/containers/header_container.jsx
new file mode 100644
index 00000000000000..c3a3de71ed9c3b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_timeline/containers/header_container.jsx
@@ -0,0 +1,186 @@
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import {
+  followAccount,
+  unfollowAccount,
+  unblockAccount,
+  unmuteAccount,
+  pinAccount,
+  unpinAccount,
+} from '../../../actions/accounts';
+import { initBlockModal } from '../../../actions/blocks';
+import {
+  mentionCompose,
+  directCompose,
+} from '../../../actions/compose';
+import { blockDomain, unblockDomain } from '../../../actions/domain_blocks';
+import { openModal } from '../../../actions/modal';
+import { initMuteModal } from '../../../actions/mutes';
+import { initReport } from '../../../actions/reports';
+import { unfollowModal } from '../../../initial_state';
+import { makeGetAccount, getAccountHidden } from '../../../selectors';
+import Header from '../components/header';
+
+const messages = defineMessages({
+  cancelFollowRequestConfirm: { id: 'confirmations.cancel_follow_request.confirm', defaultMessage: 'Withdraw request' },
+  unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow' },
+  blockDomainConfirm: { id: 'confirmations.domain_block.confirm', defaultMessage: 'Block entire domain' },
+});
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { accountId }) => ({
+    account: getAccount(state, accountId),
+    domain: state.getIn(['meta', 'domain']),
+    hidden: getAccountHidden(state, accountId),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+
+  onFollow (account) {
+    if (account.getIn(['relationship', 'following'])) {
+      if (unfollowModal) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: <FormattedMessage id='confirmations.unfollow.message' defaultMessage='Are you sure you want to unfollow {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />,
+            confirm: intl.formatMessage(messages.unfollowConfirm),
+            onConfirm: () => dispatch(unfollowAccount(account.get('id'))),
+          },
+        }));
+      } else {
+        dispatch(unfollowAccount(account.get('id')));
+      }
+    } else if (account.getIn(['relationship', 'requested'])) {
+      if (unfollowModal) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: <FormattedMessage id='confirmations.cancel_follow_request.message' defaultMessage='Are you sure you want to withdraw your request to follow {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />,
+            confirm: intl.formatMessage(messages.cancelFollowRequestConfirm),
+            onConfirm: () => dispatch(unfollowAccount(account.get('id'))),
+          },
+        }));
+      } else {
+        dispatch(unfollowAccount(account.get('id')));
+      }
+    } else {
+      dispatch(followAccount(account.get('id')));
+    }
+  },
+
+  onInteractionModal (account) {
+    dispatch(openModal({
+      modalType: 'INTERACTION',
+      modalProps: {
+        type: 'follow',
+        accountId: account.get('id'),
+        url: account.get('uri'),
+      },
+    }));
+  },
+
+  onBlock (account) {
+    if (account.getIn(['relationship', 'blocking'])) {
+      dispatch(unblockAccount(account.get('id')));
+    } else {
+      dispatch(initBlockModal(account));
+    }
+  },
+
+  onMention (account, router) {
+    dispatch(mentionCompose(account, router));
+  },
+
+  onDirect (account, router) {
+    dispatch(directCompose(account, router));
+  },
+
+  onReblogToggle (account) {
+    if (account.getIn(['relationship', 'showing_reblogs'])) {
+      dispatch(followAccount(account.get('id'), { reblogs: false }));
+    } else {
+      dispatch(followAccount(account.get('id'), { reblogs: true }));
+    }
+  },
+
+  onEndorseToggle (account) {
+    if (account.getIn(['relationship', 'endorsed'])) {
+      dispatch(unpinAccount(account.get('id')));
+    } else {
+      dispatch(pinAccount(account.get('id')));
+    }
+  },
+
+  onNotifyToggle (account) {
+    if (account.getIn(['relationship', 'notifying'])) {
+      dispatch(followAccount(account.get('id'), { notify: false }));
+    } else {
+      dispatch(followAccount(account.get('id'), { notify: true }));
+    }
+  },
+
+  onReport (account) {
+    dispatch(initReport(account));
+  },
+
+  onMute (account) {
+    if (account.getIn(['relationship', 'muting'])) {
+      dispatch(unmuteAccount(account.get('id')));
+    } else {
+      dispatch(initMuteModal(account));
+    }
+  },
+
+  onBlockDomain (domain) {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: <FormattedMessage id='confirmations.domain_block.message' defaultMessage='Are you really, really sure you want to block the entire {domain}? In most cases a few targeted blocks or mutes are sufficient and preferable. You will not see content from that domain in any public timelines or your notifications. Your followers from that domain will be removed.' values={{ domain: <strong>{domain}</strong> }} />,
+        confirm: intl.formatMessage(messages.blockDomainConfirm),
+        onConfirm: () => dispatch(blockDomain(domain)),
+      },
+    }));
+  },
+
+  onUnblockDomain (domain) {
+    dispatch(unblockDomain(domain));
+  },
+
+  onAddToList (account) {
+    dispatch(openModal({
+      modalType: 'LIST_ADDER',
+      modalProps: {
+        accountId: account.get('id'),
+      },
+    }));
+  },
+
+  onChangeLanguages (account) {
+    dispatch(openModal({
+      modalType: 'SUBSCRIBED_LANGUAGES',
+      modalProps: {
+        accountId: account.get('id'),
+      },
+    }));
+  },
+
+  onOpenAvatar (account) {
+    dispatch(openModal({
+      modalType: 'IMAGE',
+      modalProps: {
+        src: account.get('avatar'),
+        alt: account.get('acct'),
+      },
+    }));
+  },
+
+});
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(Header));
diff --git a/app/javascript/flavours/blobfox/features/account_timeline/index.jsx b/app/javascript/flavours/blobfox/features/account_timeline/index.jsx
new file mode 100644
index 00000000000000..4ecd55dff43228
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/account_timeline/index.jsx
@@ -0,0 +1,210 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { TimelineHint } from 'flavours/blobfox/components/timeline_hint';
+import ProfileColumnHeader from 'flavours/blobfox/features/account/components/profile_column_header';
+import BundleColumnError from 'flavours/blobfox/features/ui/components/bundle_column_error';
+import { normalizeForLookup } from 'flavours/blobfox/reducers/accounts_map';
+import { getAccountHidden } from 'flavours/blobfox/selectors';
+
+import { lookupAccount, fetchAccount } from '../../actions/accounts';
+import { fetchFeaturedTags } from '../../actions/featured_tags';
+import { expandAccountFeaturedTimeline, expandAccountTimeline } from '../../actions/timelines';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import StatusList from '../../components/status_list';
+import Column from '../ui/components/column';
+
+import { LimitedAccountHint } from './components/limited_account_hint';
+import HeaderContainer from './containers/header_container';
+
+const emptyList = ImmutableList();
+
+const mapStateToProps = (state, { params: { acct, id, tagged }, withReplies = false }) => {
+  const accountId = id || state.getIn(['accounts_map', normalizeForLookup(acct)]);
+
+  if (accountId === null) {
+    return {
+      isLoading: false,
+      isAccount: false,
+      statusIds: emptyList,
+    };
+  } else if (!accountId) {
+    return {
+      isLoading: true,
+      statusIds: emptyList,
+    };
+  }
+
+  const path = withReplies ? `${accountId}:with_replies` : `${accountId}${tagged ? `:${tagged}` : ''}`;
+
+  return {
+    accountId,
+    remote: !!(state.getIn(['accounts', accountId, 'acct']) !== state.getIn(['accounts', accountId, 'username'])),
+    remoteUrl: state.getIn(['accounts', accountId, 'url']),
+    isAccount: !!state.getIn(['accounts', accountId]),
+    statusIds: state.getIn(['timelines', `account:${path}`, 'items'], ImmutableList()),
+    featuredStatusIds: withReplies ? ImmutableList() : state.getIn(['timelines', `account:${accountId}:pinned${tagged ? `:${tagged}` : ''}`, 'items'], ImmutableList()),
+    isLoading: state.getIn(['timelines', `account:${path}`, 'isLoading']),
+    hasMore:   state.getIn(['timelines', `account:${path}`, 'hasMore']),
+    suspended: state.getIn(['accounts', accountId, 'suspended'], false),
+    hidden: getAccountHidden(state, accountId),
+  };
+};
+
+const RemoteHint = ({ url }) => (
+  <TimelineHint url={url} resource={<FormattedMessage id='timeline_hint.resources.statuses' defaultMessage='Older posts' />} />
+);
+
+RemoteHint.propTypes = {
+  url: PropTypes.string.isRequired,
+};
+
+class AccountTimeline extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.shape({
+      acct: PropTypes.string,
+      id: PropTypes.string,
+      tagged: PropTypes.string,
+    }).isRequired,
+    accountId: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    statusIds: ImmutablePropTypes.list,
+    featuredStatusIds: ImmutablePropTypes.list,
+    isLoading: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    withReplies: PropTypes.bool,
+    isAccount: PropTypes.bool,
+    suspended: PropTypes.bool,
+    hidden: PropTypes.bool,
+    remote: PropTypes.bool,
+    remoteUrl: PropTypes.string,
+    multiColumn: PropTypes.bool,
+  };
+
+  _load () {
+    const { accountId, withReplies, params: { tagged }, dispatch } = this.props;
+
+    dispatch(fetchAccount(accountId));
+
+    if (!withReplies) {
+      dispatch(expandAccountFeaturedTimeline(accountId, { tagged }));
+    }
+
+    dispatch(fetchFeaturedTags(accountId));
+    dispatch(expandAccountTimeline(accountId, { withReplies, tagged }));
+  }
+
+  componentDidMount () {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (accountId) {
+      this._load();
+    } else {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { params: { acct, tagged }, accountId, withReplies, dispatch } = this.props;
+
+    if (prevProps.accountId !== accountId && accountId) {
+      this._load();
+    } else if (prevProps.params.acct !== acct) {
+      dispatch(lookupAccount(acct));
+    } else if (prevProps.params.tagged !== tagged) {
+      if (!withReplies) {
+        dispatch(expandAccountFeaturedTimeline(accountId, { tagged }));
+      }
+      dispatch(expandAccountTimeline(accountId, { withReplies, tagged }));
+    }
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    const { dispatch } = this.props;
+
+    if ((nextProps.params.accountId !== this.props.params.accountId && nextProps.params.accountId) || nextProps.withReplies !== this.props.withReplies) {
+      dispatch(fetchAccount(nextProps.params.accountId));
+
+      if (!nextProps.withReplies) {
+        dispatch(expandAccountFeaturedTimeline(nextProps.params.accountId));
+      }
+
+      dispatch(expandAccountTimeline(nextProps.params.accountId, { withReplies: nextProps.params.withReplies }));
+    }
+  }
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  handleLoadMore = maxId => {
+    this.props.dispatch(expandAccountTimeline(this.props.accountId, { maxId, withReplies: this.props.withReplies, tagged: this.props.params.tagged }));
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  render () {
+    const { accountId, statusIds, featuredStatusIds, isLoading, hasMore, suspended, isAccount, hidden, multiColumn, remote, remoteUrl } = this.props;
+
+    if (isLoading && statusIds.isEmpty()) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    } else if (!isLoading && !isAccount) {
+      return (
+        <BundleColumnError multiColumn={multiColumn} errorType='routing' />
+      );
+    }
+
+    let emptyMessage;
+
+    const forceEmptyState = suspended || hidden;
+
+    if (suspended) {
+      emptyMessage = <FormattedMessage id='empty_column.account_suspended' defaultMessage='Account suspended' />;
+    } else if (hidden) {
+      emptyMessage = <LimitedAccountHint accountId={accountId} />;
+    } else if (remote && statusIds.isEmpty()) {
+      emptyMessage = <RemoteHint url={remoteUrl} />;
+    } else {
+      emptyMessage = <FormattedMessage id='empty_column.account_timeline' defaultMessage='No posts found' />;
+    }
+
+    const remoteMessage = remote ? <RemoteHint url={remoteUrl} /> : null;
+
+    return (
+      <Column ref={this.setRef}>
+        <ProfileColumnHeader onClick={this.handleHeaderClick} multiColumn={multiColumn} />
+
+        <StatusList
+          prepend={<HeaderContainer accountId={this.props.accountId} hideTabs={forceEmptyState} tagged={this.props.params.tagged} />}
+          alwaysPrepend
+          append={remoteMessage}
+          scrollKey='account_timeline'
+          statusIds={forceEmptyState ? emptyList : statusIds}
+          featuredStatusIds={featuredStatusIds}
+          isLoading={isLoading}
+          hasMore={!forceEmptyState && hasMore}
+          onLoadMore={this.handleLoadMore}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+          timelineId='account'
+        />
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(AccountTimeline);
diff --git a/app/javascript/flavours/blobfox/features/audio/index.jsx b/app/javascript/flavours/blobfox/features/audio/index.jsx
new file mode 100644
index 00000000000000..199eda8a7eb633
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/audio/index.jsx
@@ -0,0 +1,600 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { is } from 'immutable';
+
+import { throttle, debounce } from 'lodash';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { formatTime, getPointerPosition, fileNameFromURL } from 'flavours/blobfox/features/video';
+
+import { Blurhash } from '../../components/blurhash';
+import { displayMedia, useBlurhash } from '../../initial_state';
+
+import Visualizer from './visualizer';
+
+const messages = defineMessages({
+  play: { id: 'video.play', defaultMessage: 'Play' },
+  pause: { id: 'video.pause', defaultMessage: 'Pause' },
+  mute: { id: 'video.mute', defaultMessage: 'Mute sound' },
+  unmute: { id: 'video.unmute', defaultMessage: 'Unmute sound' },
+  download: { id: 'video.download', defaultMessage: 'Download file' },
+  hide: { id: 'audio.hide', defaultMessage: 'Hide audio' },
+});
+
+const TICK_SIZE = 10;
+const PADDING   = 180;
+
+class Audio extends PureComponent {
+
+  static propTypes = {
+    src: PropTypes.string.isRequired,
+    alt: PropTypes.string,
+    lang: PropTypes.string,
+    poster: PropTypes.string,
+    duration: PropTypes.number,
+    width: PropTypes.number,
+    height: PropTypes.number,
+    sensitive: PropTypes.bool,
+    editable: PropTypes.bool,
+    fullscreen: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    blurhash: PropTypes.string,
+    cacheWidth: PropTypes.func,
+    visible: PropTypes.bool,
+    onToggleVisibility: PropTypes.func,
+    backgroundColor: PropTypes.string,
+    foregroundColor: PropTypes.string,
+    accentColor: PropTypes.string,
+    currentTime: PropTypes.number,
+    autoPlay: PropTypes.bool,
+    volume: PropTypes.number,
+    muted: PropTypes.bool,
+    deployPictureInPicture: PropTypes.func,
+  };
+
+  state = {
+    width: this.props.width,
+    currentTime: 0,
+    buffer: 0,
+    duration: null,
+    paused: true,
+    muted: false,
+    volume: 1,
+    dragging: false,
+    revealed: this.props.visible !== undefined ? this.props.visible : (displayMedia !== 'hide_all' && !this.props.sensitive || displayMedia === 'show_all'),
+  };
+
+  constructor (props) {
+    super(props);
+    this.visualizer = new Visualizer(TICK_SIZE);
+  }
+
+  setPlayerRef = c => {
+    this.player = c;
+
+    if (this.player) {
+      this._setDimensions();
+    }
+  };
+
+  _pack() {
+    return {
+      src: this.props.src,
+      volume: this.state.volume,
+      muted: this.state.muted,
+      currentTime: this.audio.currentTime,
+      poster: this.props.poster,
+      backgroundColor: this.props.backgroundColor,
+      foregroundColor: this.props.foregroundColor,
+      accentColor: this.props.accentColor,
+      sensitive: this.props.sensitive,
+      visible: this.props.visible,
+    };
+  }
+
+  _setDimensions () {
+    const width  = this.player.offsetWidth;
+    const height = this.props.fullscreen ? this.player.offsetHeight : (width / (16/9));
+
+    if (width && width !== this.state.containerWidth) {
+      if (this.props.cacheWidth) {
+        this.props.cacheWidth(width);
+      }
+
+      this.setState({ width, height });
+    }
+  }
+
+  setSeekRef = c => {
+    this.seek = c;
+  };
+
+  setVolumeRef = c => {
+    this.volume = c;
+  };
+
+  setAudioRef = c => {
+    this.audio = c;
+
+    if (this.audio) {
+      this.audio.volume = 1;
+      this.audio.muted = false;
+    }
+  };
+
+  setCanvasRef = c => {
+    this.canvas = c;
+
+    this.visualizer.setCanvas(c);
+  };
+
+  componentDidMount () {
+    window.addEventListener('scroll', this.handleScroll);
+    window.addEventListener('resize', this.handleResize, { passive: true });
+  }
+
+  componentDidUpdate (prevProps, prevState) {
+    if (this.player) {
+      this._setDimensions();
+    }
+
+    if (prevProps.src !== this.props.src || this.state.width !== prevState.width || this.state.height !== prevState.height || prevProps.accentColor !== this.props.accentColor) {
+      this._clear();
+      this._draw();
+    }
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (!is(nextProps.visible, this.props.visible) && nextProps.visible !== undefined) {
+      this.setState({ revealed: nextProps.visible });
+    }
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('scroll', this.handleScroll);
+    window.removeEventListener('resize', this.handleResize);
+
+    if (!this.state.paused && this.audio && this.props.deployPictureInPicture) {
+      this.props.deployPictureInPicture('audio', this._pack());
+    }
+  }
+
+  togglePlay = () => {
+    if (!this.audioContext) {
+      this._initAudioContext();
+    }
+
+    if (this.state.paused) {
+      this.setState({ paused: false }, () => this.audio.play());
+    } else {
+      this.setState({ paused: true }, () => this.audio.pause());
+    }
+  };
+
+  handleResize = debounce(() => {
+    if (this.player) {
+      this._setDimensions();
+    }
+  }, 250, {
+    trailing: true,
+  });
+
+  handlePlay = () => {
+    this.setState({ paused: false });
+
+    if (this.audioContext && this.audioContext.state === 'suspended') {
+      this.audioContext.resume();
+    }
+
+    this._renderCanvas();
+  };
+
+  handlePause = () => {
+    this.setState({ paused: true });
+
+    if (this.audioContext) {
+      this.audioContext.suspend();
+    }
+  };
+
+  handleProgress = () => {
+    const lastTimeRange = this.audio.buffered.length - 1;
+
+    if (lastTimeRange > -1) {
+      this.setState({ buffer: Math.ceil(this.audio.buffered.end(lastTimeRange) / this.audio.duration * 100) });
+    }
+  };
+
+  toggleMute = () => {
+    const muted = !(this.state.muted || this.state.volume === 0);
+
+    this.setState((state) => ({ muted, volume: Math.max(state.volume || 0.5, 0.05) }), () => {
+      if (this.gainNode) {
+        this.gainNode.gain.value = this.state.muted ? 0 : this.state.volume;
+      }
+    });
+  };
+
+  toggleReveal = () => {
+    if (this.props.onToggleVisibility) {
+      this.props.onToggleVisibility();
+    } else {
+      this.setState({ revealed: !this.state.revealed });
+    }
+  };
+
+  handleVolumeMouseDown = e => {
+    document.addEventListener('mousemove', this.handleMouseVolSlide, true);
+    document.addEventListener('mouseup', this.handleVolumeMouseUp, true);
+    document.addEventListener('touchmove', this.handleMouseVolSlide, true);
+    document.addEventListener('touchend', this.handleVolumeMouseUp, true);
+
+    this.handleMouseVolSlide(e);
+
+    e.preventDefault();
+    e.stopPropagation();
+  };
+
+  handleVolumeMouseUp = () => {
+    document.removeEventListener('mousemove', this.handleMouseVolSlide, true);
+    document.removeEventListener('mouseup', this.handleVolumeMouseUp, true);
+    document.removeEventListener('touchmove', this.handleMouseVolSlide, true);
+    document.removeEventListener('touchend', this.handleVolumeMouseUp, true);
+  };
+
+  handleMouseDown = e => {
+    document.addEventListener('mousemove', this.handleMouseMove, true);
+    document.addEventListener('mouseup', this.handleMouseUp, true);
+    document.addEventListener('touchmove', this.handleMouseMove, true);
+    document.addEventListener('touchend', this.handleMouseUp, true);
+
+    this.setState({ dragging: true });
+    this.audio.pause();
+    this.handleMouseMove(e);
+
+    e.preventDefault();
+    e.stopPropagation();
+  };
+
+  handleMouseUp = () => {
+    document.removeEventListener('mousemove', this.handleMouseMove, true);
+    document.removeEventListener('mouseup', this.handleMouseUp, true);
+    document.removeEventListener('touchmove', this.handleMouseMove, true);
+    document.removeEventListener('touchend', this.handleMouseUp, true);
+
+    this.setState({ dragging: false });
+    this.audio.play();
+  };
+
+  handleMouseMove = throttle(e => {
+    const { x } = getPointerPosition(this.seek, e);
+    const currentTime = this.audio.duration * x;
+
+    if (!isNaN(currentTime)) {
+      this.setState({ currentTime }, () => {
+        this.audio.currentTime = currentTime;
+      });
+    }
+  }, 15);
+
+  handleTimeUpdate = () => {
+    this.setState({
+      currentTime: this.audio.currentTime,
+      duration: this.audio.duration,
+    });
+  };
+
+  handleMouseVolSlide = throttle(e => {
+    const { x } = getPointerPosition(this.volume, e);
+
+    if(!isNaN(x)) {
+      this.setState((state) => ({ volume: x, muted: state.muted && x === 0 }), () => {
+        if (this.gainNode) {
+          this.gainNode.gain.value = this.state.muted ? 0 : x;
+        }
+      });
+    }
+  }, 15);
+
+  handleScroll = throttle(() => {
+    if (!this.canvas || !this.audio) {
+      return;
+    }
+
+    const { top, height } = this.canvas.getBoundingClientRect();
+    const inView = (top <= (window.innerHeight || document.documentElement.clientHeight)) && (top + height >= 0);
+
+    if (!this.state.paused && !inView) {
+      this.audio.pause();
+
+      if (this.props.deployPictureInPicture) {
+        this.props.deployPictureInPicture('audio', this._pack());
+      }
+
+      this.setState({ paused: true });
+    }
+  }, 150, { trailing: true });
+
+  handleMouseEnter = () => {
+    this.setState({ hovered: true });
+  };
+
+  handleMouseLeave = () => {
+    this.setState({ hovered: false });
+  };
+
+  handleLoadedData = () => {
+    const { autoPlay, currentTime } = this.props;
+
+    if (currentTime) {
+      this.audio.currentTime = currentTime;
+    }
+
+    if (autoPlay) {
+      this.togglePlay();
+    }
+  };
+
+  _initAudioContext () {
+    const AudioContext = window.AudioContext || window.webkitAudioContext;
+    const context      = new AudioContext();
+    const source       = context.createMediaElementSource(this.audio);
+    const gainNode     = context.createGain();
+
+    gainNode.gain.value = this.state.muted ? 0 : this.state.volume;
+
+    this.visualizer.setAudioContext(context, source);
+    source.connect(gainNode);
+    gainNode.connect(context.destination);
+
+    this.audioContext = context;
+    this.gainNode = gainNode;
+  }
+
+  handleDownload = () => {
+    fetch(this.props.src).then(res => res.blob()).then(blob => {
+      const element   = document.createElement('a');
+      const objectURL = URL.createObjectURL(blob);
+
+      element.setAttribute('href', objectURL);
+      element.setAttribute('download', fileNameFromURL(this.props.src));
+
+      document.body.appendChild(element);
+      element.click();
+      document.body.removeChild(element);
+
+      URL.revokeObjectURL(objectURL);
+    }).catch(err => {
+      console.error(err);
+    });
+  };
+
+  _renderCanvas () {
+    requestAnimationFrame(() => {
+      if (!this.audio) return;
+
+      this.handleTimeUpdate();
+      this._clear();
+      this._draw();
+
+      if (!this.state.paused) {
+        this._renderCanvas();
+      }
+    });
+  }
+
+  _clear() {
+    this.visualizer.clear(this.state.width, this.state.height);
+  }
+
+  _draw() {
+    this.visualizer.draw(this._getCX(), this._getCY(), this._getAccentColor(), this._getRadius(), this._getScaleCoefficient());
+  }
+
+  _getRadius () {
+    return parseInt((this.state.height || this.props.height) / 2 - PADDING * this._getScaleCoefficient());
+  }
+
+  _getScaleCoefficient () {
+    return (this.state.height || this.props.height) / 982;
+  }
+
+  _getCX() {
+    return Math.floor(this.state.width / 2);
+  }
+
+  _getCY() {
+    return Math.floor((this.state.height || this.props.height) / 2);
+  }
+
+  _getAccentColor () {
+    return this.props.accentColor || '#ffffff';
+  }
+
+  _getBackgroundColor () {
+    return this.props.backgroundColor || '#000000';
+  }
+
+  _getForegroundColor () {
+    return this.props.foregroundColor || '#ffffff';
+  }
+
+  seekBy (time) {
+    const currentTime = this.audio.currentTime + time;
+
+    if (!isNaN(currentTime)) {
+      this.setState({ currentTime }, () => {
+        this.audio.currentTime = currentTime;
+      });
+    }
+  }
+
+  handleAudioKeyDown = e => {
+    // On the audio element or the seek bar, we can safely use the space bar
+    // for playback control because there are no buttons to press
+
+    if (e.key === ' ') {
+      e.preventDefault();
+      e.stopPropagation();
+      this.togglePlay();
+    }
+  };
+
+  handleKeyDown = e => {
+    switch(e.key) {
+    case 'k':
+      e.preventDefault();
+      e.stopPropagation();
+      this.togglePlay();
+      break;
+    case 'm':
+      e.preventDefault();
+      e.stopPropagation();
+      this.toggleMute();
+      break;
+    case 'j':
+      e.preventDefault();
+      e.stopPropagation();
+      this.seekBy(-10);
+      break;
+    case 'l':
+      e.preventDefault();
+      e.stopPropagation();
+      this.seekBy(10);
+      break;
+    }
+  };
+
+  render () {
+    const { src, intl, alt, lang, editable, autoPlay, sensitive, blurhash } = this.props;
+    const { paused, volume, currentTime, duration, buffer, dragging, revealed } = this.state;
+    const progress = Math.min((currentTime / duration) * 100, 100);
+    const muted = this.state.muted || volume === 0;
+
+    let warning;
+
+    if (sensitive) {
+      warning = <FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' />;
+    } else {
+      warning = <FormattedMessage id='status.media_hidden' defaultMessage='Media hidden' />;
+    }
+
+    return (
+      <div className={classNames('audio-player', { editable, inactive: !revealed })} ref={this.setPlayerRef} style={{ backgroundColor: this._getBackgroundColor(), color: this._getForegroundColor(), aspectRatio: '16 / 9' }} onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave} tabIndex={0} onKeyDown={this.handleKeyDown}>
+
+        <Blurhash
+          hash={blurhash}
+          className={classNames('media-gallery__preview', {
+            'media-gallery__preview--hidden': revealed,
+          })}
+          dummy={!useBlurhash}
+        />
+
+        {(revealed || editable) && <audio
+          ref={this.setAudioRef}
+          preload={autoPlay ? 'auto' : 'none'}
+          onPlay={this.handlePlay}
+          onPause={this.handlePause}
+          onProgress={this.handleProgress}
+          onLoadedData={this.handleLoadedData}
+          crossOrigin='anonymous'
+        >
+          <source src={src.replace("/original/", "/opus/").replace(".mp3", ".webm")} type="audio/webm; codecs=opus"/>
+          <source src={src} type="audio/mpeg; codecs=mp3"/>
+        </audio>
+        }
+
+        <canvas
+          role='button'
+          tabIndex={0}
+          className='audio-player__canvas'
+          width={this.state.width}
+          height={this.state.height}
+          style={{ width: '100%', position: 'absolute', top: 0, left: 0 }}
+          ref={this.setCanvasRef}
+          onClick={this.togglePlay}
+          onKeyDown={this.handleAudioKeyDown}
+          title={alt}
+          aria-label={alt}
+          lang={lang}
+        />
+
+        <div className={classNames('spoiler-button', { 'spoiler-button--hidden': revealed || editable })}>
+          <button type='button' className='spoiler-button__overlay' onClick={this.toggleReveal}>
+            <span className='spoiler-button__overlay__label'>
+              {warning}
+              <span className='spoiler-button__overlay__action'><FormattedMessage id='status.media.show' defaultMessage='Click to show' /></span>
+            </span>
+          </button>
+        </div>
+
+        {(revealed || editable) && <img
+          src={this.props.poster}
+          alt=''
+          style={{
+            position: 'absolute',
+            left: '50%',
+            top: '50%',
+            height: `calc(${(100 - 2 * 100 * PADDING / 982)}% - ${TICK_SIZE * 2}px)`,
+            aspectRatio: '1',
+            transform: 'translate(-50%, -50%)',
+            borderRadius: '50%',
+            pointerEvents: 'none',
+          }}
+        />}
+
+        <div className='video-player__seek' onMouseDown={this.handleMouseDown} ref={this.setSeekRef}>
+          <div className='video-player__seek__buffer' style={{ width: `${buffer}%` }} />
+          <div className='video-player__seek__progress' style={{ width: `${progress}%`, backgroundColor: this._getAccentColor() }} />
+
+          <span
+            className={classNames('video-player__seek__handle', { active: dragging })}
+            tabIndex={0}
+            style={{ left: `${progress}%`, backgroundColor: this._getAccentColor() }}
+            onKeyDown={this.handleAudioKeyDown}
+          />
+        </div>
+
+        <div className='video-player__controls active'>
+          <div className='video-player__buttons-bar'>
+            <div className='video-player__buttons left'>
+              <button type='button' title={intl.formatMessage(paused ? messages.play : messages.pause)} aria-label={intl.formatMessage(paused ? messages.play : messages.pause)} className='player-button' onClick={this.togglePlay}><Icon id={paused ? 'play' : 'pause'} fixedWidth /></button>
+              <button type='button' title={intl.formatMessage(muted ? messages.unmute : messages.mute)} aria-label={intl.formatMessage(muted ? messages.unmute : messages.mute)} className='player-button' onClick={this.toggleMute}><Icon id={muted ? 'volume-off' : 'volume-up'} fixedWidth /></button>
+
+              <div className={classNames('video-player__volume', { active: this.state.hovered })} ref={this.setVolumeRef} onMouseDown={this.handleVolumeMouseDown}>
+                <div className='video-player__volume__current' style={{ width: `${muted ? 0 : volume * 100}%`, backgroundColor: this._getAccentColor() }} />
+
+                <span
+                  className='video-player__volume__handle'
+                  tabIndex={0}
+                  style={{ left: `${muted ? 0 : volume * 100}%`, backgroundColor: this._getAccentColor() }}
+                />
+              </div>
+
+              <span className='video-player__time'>
+                <span className='video-player__time-current'>{formatTime(Math.floor(currentTime))}</span>
+                <span className='video-player__time-sep'>/</span>
+                <span className='video-player__time-total'>{formatTime(Math.floor(this.state.duration || this.props.duration))}</span>
+              </span>
+            </div>
+
+            <div className='video-player__buttons right'>
+              {!editable && <button type='button' title={intl.formatMessage(messages.hide)} aria-label={intl.formatMessage(messages.hide)} className='player-button' onClick={this.toggleReveal}><Icon id='eye-slash' fixedWidth /></button>}
+              <a title={intl.formatMessage(messages.download)} aria-label={intl.formatMessage(messages.download)} className='video-player__download__icon player-button' href={this.props.src} download>
+                <Icon id={'download'} fixedWidth />
+              </a>
+            </div>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Audio);
diff --git a/app/javascript/flavours/blobfox/features/audio/visualizer.js b/app/javascript/flavours/blobfox/features/audio/visualizer.js
new file mode 100644
index 00000000000000..77d5b5a65cdce1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/audio/visualizer.js
@@ -0,0 +1,136 @@
+/*
+Copyright (c) 2020 by Alex Permyakov (https://codepen.io/alexdevp/pen/RNELPV)
+
+Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
+*/
+
+const hex2rgba = (hex, alpha = 1) => {
+  const [r, g, b] = hex.match(/\w\w/g).map(x => parseInt(x, 16));
+  return `rgba(${r}, ${g}, ${b}, ${alpha})`;
+};
+
+export default class Visualizer {
+
+  constructor (tickSize) {
+    this.tickSize = tickSize;
+  }
+
+  setCanvas(canvas) {
+    this.canvas = canvas;
+    if (canvas) {
+      this.context = canvas.getContext('2d');
+    }
+  }
+
+  setAudioContext(context, source) {
+    const analyser = context.createAnalyser();
+
+    analyser.smoothingTimeConstant = 0.6;
+    analyser.fftSize = 2048;
+
+    source.connect(analyser);
+
+    this.analyser = analyser;
+  }
+
+  getTickPoints (count) {
+    const coords = [];
+
+    for(let i = 0; i < count; i++) {
+      const rad = Math.PI * 2 * i / count;
+      coords.push({ x: Math.cos(rad), y: -Math.sin(rad) });
+    }
+
+    return coords;
+  }
+
+  drawTick (cx, cy, mainColor, x1, y1, x2, y2) {
+    const dx1 = Math.ceil(cx + x1);
+    const dy1 = Math.ceil(cy + y1);
+    const dx2 = Math.ceil(cx + x2);
+    const dy2 = Math.ceil(cy + y2);
+
+    const gradient = this.context.createLinearGradient(dx1, dy1, dx2, dy2);
+
+    const lastColor = hex2rgba(mainColor, 0);
+
+    gradient.addColorStop(0, mainColor);
+    gradient.addColorStop(0.6, mainColor);
+    gradient.addColorStop(1, lastColor);
+
+    this.context.beginPath();
+    this.context.strokeStyle = gradient;
+    this.context.lineWidth = 2;
+    this.context.moveTo(dx1, dy1);
+    this.context.lineTo(dx2, dy2);
+    this.context.stroke();
+  }
+
+  getTicks (count, size, radius, scaleCoefficient) {
+    const ticks = this.getTickPoints(count);
+    const lesser = 200;
+    const m = [];
+    const bufferLength = this.analyser ? this.analyser.frequencyBinCount : 0;
+    const frequencyData = new Uint8Array(bufferLength);
+    const allScales = [];
+
+    if (this.analyser) {
+      this.analyser.getByteFrequencyData(frequencyData);
+    }
+
+    ticks.forEach((tick, i) => {
+      const coef = 1 - i / (ticks.length * 2.5);
+
+      let delta = ((frequencyData[i] || 0) - lesser * coef) * scaleCoefficient;
+
+      if (delta < 0) {
+        delta = 0;
+      }
+
+      const k = radius / (radius - (size + delta));
+
+      const x1 = tick.x * (radius - size);
+      const y1 = tick.y * (radius - size);
+      const x2 = x1 * k;
+      const y2 = y1 * k;
+
+      m.push({ x1, y1, x2, y2 });
+
+      if (i < 20) {
+        let scale = delta / (200 * scaleCoefficient);
+        scale = scale < 1 ? 1 : scale;
+        allScales.push(scale);
+      }
+    });
+
+    const scale = allScales.reduce((pv, cv) => pv + cv, 0) / allScales.length;
+
+    return m.map(({ x1, y1, x2, y2 }) => ({
+      x1: x1,
+      y1: y1,
+      x2: x2 * scale,
+      y2: y2 * scale,
+    }));
+  }
+
+  clear (width, height) {
+    this.context.clearRect(0, 0, width, height);
+  }
+
+  draw (cx, cy, color, radius, coefficient) {
+    this.context.save();
+
+    const ticks = this.getTicks(parseInt(360 * coefficient), this.tickSize, radius, coefficient);
+
+    ticks.forEach(tick => {
+      this.drawTick(cx, cy, color, tick.x1, tick.y1, tick.x2, tick.y2);
+    });
+
+    this.context.restore();
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/blocks/index.jsx b/app/javascript/flavours/blobfox/features/blocks/index.jsx
new file mode 100644
index 00000000000000..d976174ce0c2b0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/blocks/index.jsx
@@ -0,0 +1,82 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchBlocks, expandBlocks } from '../../actions/blocks';
+import ColumnBackButtonSlim from '../../components/column_back_button_slim';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import ScrollableList from '../../components/scrollable_list';
+import AccountContainer from '../../containers/account_container';
+import Column from '../ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'column.blocks', defaultMessage: 'Blocked users' },
+});
+
+const mapStateToProps = state => ({
+  accountIds: state.getIn(['user_lists', 'blocks', 'items']),
+  hasMore: !!state.getIn(['user_lists', 'blocks', 'next']),
+  isLoading: state.getIn(['user_lists', 'blocks', 'isLoading'], true),
+});
+
+class Blocks extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    accountIds: ImmutablePropTypes.list,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchBlocks());
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandBlocks());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, accountIds, hasMore, multiColumn, isLoading } = this.props;
+
+    if (!accountIds) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    const emptyMessage = <FormattedMessage id='empty_column.blocks' defaultMessage="You haven't blocked any users yet." />;
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='ban' heading={intl.formatMessage(messages.heading)}>
+        <ColumnBackButtonSlim />
+        <ScrollableList
+          scrollKey='blocks'
+          onLoadMore={this.handleLoadMore}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {accountIds.map(id =>
+            <AccountContainer key={id} id={id} defaultAction='block' />,
+          )}
+        </ScrollableList>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Blocks));
diff --git a/app/javascript/flavours/blobfox/features/bookmarked_statuses/index.jsx b/app/javascript/flavours/blobfox/features/bookmarked_statuses/index.jsx
new file mode 100644
index 00000000000000..d61b97022df25f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/bookmarked_statuses/index.jsx
@@ -0,0 +1,113 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchBookmarkedStatuses, expandBookmarkedStatuses } from 'flavours/blobfox/actions/bookmarks';
+import { addColumn, removeColumn, moveColumn } from 'flavours/blobfox/actions/columns';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import StatusList from 'flavours/blobfox/components/status_list';
+import Column from 'flavours/blobfox/features/ui/components/column';
+import { getStatusList } from 'flavours/blobfox/selectors';
+
+const messages = defineMessages({
+  heading: { id: 'column.bookmarks', defaultMessage: 'Bookmarks' },
+});
+
+const mapStateToProps = state => ({
+  statusIds: getStatusList(state, 'bookmarks'),
+  isLoading: state.getIn(['status_lists', 'bookmarks', 'isLoading'], true),
+  hasMore: !!state.getIn(['status_lists', 'bookmarks', 'next']),
+});
+
+class Bookmarks extends ImmutablePureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    statusIds: ImmutablePropTypes.list.isRequired,
+    intl: PropTypes.object.isRequired,
+    columnId: PropTypes.string,
+    multiColumn: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchBookmarkedStatuses());
+  }
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('BOOKMARKS', {}));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandBookmarkedStatuses());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, statusIds, columnId, multiColumn, hasMore, isLoading } = this.props;
+    const pinned = !!columnId;
+
+    const emptyMessage = <FormattedMessage id='empty_column.bookmarked_statuses' defaultMessage="You don't have any bookmarked posts yet. When you bookmark one, it will show up here." />;
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef}>
+        <ColumnHeader
+          icon='bookmark'
+          title={intl.formatMessage(messages.heading)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+          showBackButton
+        />
+
+        <StatusList
+          trackScroll={!pinned}
+          statusIds={statusIds}
+          scrollKey={`bookmarked_statuses-${columnId}`}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          onLoadMore={this.handleLoadMore}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        />
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.heading)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Bookmarks));
diff --git a/app/javascript/flavours/blobfox/features/closed_registrations_modal/index.jsx b/app/javascript/flavours/blobfox/features/closed_registrations_modal/index.jsx
new file mode 100644
index 00000000000000..b3742a30fa638b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/closed_registrations_modal/index.jsx
@@ -0,0 +1,77 @@
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { fetchServer } from 'flavours/blobfox/actions/server';
+import { domain } from 'flavours/blobfox/initial_state';
+
+const mapStateToProps = state => ({
+  message: state.getIn(['server', 'server', 'registrations', 'message']),
+});
+
+class ClosedRegistrationsModal extends ImmutablePureComponent {
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(fetchServer());
+  }
+
+  render () {
+    let closedRegistrationsMessage;
+
+    if (this.props.message) {
+      closedRegistrationsMessage = (
+        <p
+          className='prose'
+          dangerouslySetInnerHTML={{ __html: this.props.message }}
+        />
+      );
+    } else {
+      closedRegistrationsMessage = (
+        <p className='prose'>
+          <FormattedMessage
+            id='closed_registrations_modal.description'
+            defaultMessage='Creating an account on {domain} is currently not possible, but please keep in mind that you do not need an account specifically on {domain} to use Mastodon.'
+            values={{ domain: <strong>{domain}</strong> }}
+          />
+        </p>
+      );
+    }
+
+    return (
+      <div className='modal-root__modal interaction-modal'>
+        <div className='interaction-modal__lead'>
+          <h3><FormattedMessage id='closed_registrations_modal.title' defaultMessage='Signing up on Mastodon' /></h3>
+          <p>
+            <FormattedMessage
+              id='closed_registrations_modal.preamble'
+              defaultMessage='Mastodon is decentralized, so no matter where you create your account, you will be able to follow and interact with anyone on this server. You can even self-host it!'
+            />
+          </p>
+        </div>
+
+        <div className='interaction-modal__choices'>
+          <div className='interaction-modal__choices__choice'>
+            <h3><FormattedMessage id='interaction_modal.on_this_server' defaultMessage='On this server' /></h3>
+            {closedRegistrationsMessage}
+          </div>
+
+          <div className='interaction-modal__choices__choice'>
+            <h3><FormattedMessage id='interaction_modal.on_another_server' defaultMessage='On a different server' /></h3>
+            <p className='prose'>
+              <FormattedMessage
+                id='closed_registrations.other_server_instructions'
+                defaultMessage='Since Mastodon is decentralized, you can create an account on another server and still interact with this one.'
+              />
+            </p>
+            <a href='https://joinmastodon.org/servers' className='button button--block'><FormattedMessage id='closed_registrations_modal.find_another_server' defaultMessage='Find another server' /></a>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(ClosedRegistrationsModal);
diff --git a/app/javascript/flavours/blobfox/features/community_timeline/components/column_settings.jsx b/app/javascript/flavours/blobfox/features/community_timeline/components/column_settings.jsx
new file mode 100644
index 00000000000000..b31d490ba2593c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/community_timeline/components/column_settings.jsx
@@ -0,0 +1,45 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import SettingText from 'flavours/blobfox/components/setting_text';
+import SettingToggle from 'flavours/blobfox/features/notifications/components/setting_toggle';
+
+const messages = defineMessages({
+  filter_regex: { id: 'home.column_settings.filter_regex', defaultMessage: 'Filter out by regular expressions' },
+  settings: { id: 'home.settings', defaultMessage: 'Column settings' },
+});
+
+class ColumnSettings extends PureComponent {
+
+  static propTypes = {
+    settings: ImmutablePropTypes.map.isRequired,
+    onChange: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    columnId: PropTypes.string,
+  };
+
+  render () {
+    const { settings, onChange, intl } = this.props;
+
+    return (
+      <div>
+        <div className='column-settings__row'>
+          <SettingToggle settings={settings} settingPath={['other', 'onlyMedia']} onChange={onChange} label={<FormattedMessage id='community.column_settings.media_only' defaultMessage='Media only' />} />
+        </div>
+
+        <span className='column-settings__section'><FormattedMessage id='home.column_settings.advanced' defaultMessage='Advanced' /></span>
+
+        <div className='column-settings__row'>
+          <SettingText settings={settings} settingPath={['regex', 'body']} onChange={onChange} label={intl.formatMessage(messages.filter_regex)} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/community_timeline/containers/column_settings_container.js b/app/javascript/flavours/blobfox/features/community_timeline/containers/column_settings_container.js
new file mode 100644
index 00000000000000..1e9f1213945a03
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/community_timeline/containers/column_settings_container.js
@@ -0,0 +1,29 @@
+import { connect } from 'react-redux';
+
+import { changeColumnParams } from '../../../actions/columns';
+import { changeSetting } from '../../../actions/settings';
+import ColumnSettings from '../components/column_settings';
+
+const mapStateToProps = (state, { columnId }) => {
+  const uuid = columnId;
+  const columns = state.getIn(['settings', 'columns']);
+  const index = columns.findIndex(c => c.get('uuid') === uuid);
+
+  return {
+    settings: (uuid && index >= 0) ? columns.get(index).get('params') : state.getIn(['settings', 'community']),
+  };
+};
+
+const mapDispatchToProps = (dispatch, { columnId }) => {
+  return {
+    onChange (key, checked) {
+      if (columnId) {
+        dispatch(changeColumnParams(columnId, key, checked));
+      } else {
+        dispatch(changeSetting(['community', ...key], checked));
+      }
+    },
+  };
+};
+
+export default connect(mapStateToProps, mapDispatchToProps)(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/community_timeline/index.jsx b/app/javascript/flavours/blobfox/features/community_timeline/index.jsx
new file mode 100644
index 00000000000000..3f715b644e6fd7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/community_timeline/index.jsx
@@ -0,0 +1,166 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import { connect } from 'react-redux';
+
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import { domain } from 'flavours/blobfox/initial_state';
+
+import { addColumn, removeColumn, moveColumn } from '../../actions/columns';
+import { connectCommunityStream } from '../../actions/streaming';
+import { expandCommunityTimeline } from '../../actions/timelines';
+import Column from '../../components/column';
+import ColumnHeader from '../../components/column_header';
+import StatusListContainer from '../ui/containers/status_list_container';
+
+import ColumnSettingsContainer from './containers/column_settings_container';
+
+const messages = defineMessages({
+  title: { id: 'column.community', defaultMessage: 'Local timeline' },
+});
+
+const mapStateToProps = (state, { columnId }) => {
+  const uuid = columnId;
+  const columns = state.getIn(['settings', 'columns']);
+  const index = columns.findIndex(c => c.get('uuid') === uuid);
+  const onlyMedia = (columnId && index >= 0) ? columns.get(index).getIn(['params', 'other', 'onlyMedia']) : state.getIn(['settings', 'community', 'other', 'onlyMedia']);
+  const regex = (columnId && index >= 0) ? columns.get(index).getIn(['params', 'regex', 'body']) : state.getIn(['settings', 'community', 'regex', 'body']);
+  const timelineState = state.getIn(['timelines', `community${onlyMedia ? ':media' : ''}`]);
+
+  return {
+    hasUnread: !!timelineState && timelineState.get('unread') > 0,
+    onlyMedia,
+    regex,
+  };
+};
+
+class CommunityTimeline extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static defaultProps = {
+    onlyMedia: false,
+  };
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    columnId: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    hasUnread: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    onlyMedia: PropTypes.bool,
+    regex: PropTypes.string,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch, onlyMedia } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('COMMUNITY', { other: { onlyMedia } }));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  componentDidMount () {
+    const { dispatch, onlyMedia } = this.props;
+    const { signedIn } = this.context.identity;
+
+    dispatch(expandCommunityTimeline({ onlyMedia }));
+
+    if (signedIn) {
+      this.disconnect = dispatch(connectCommunityStream({ onlyMedia }));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { signedIn } = this.context.identity;
+
+    if (prevProps.onlyMedia !== this.props.onlyMedia) {
+      const { dispatch, onlyMedia } = this.props;
+
+      if (this.disconnect) {
+        this.disconnect();
+      }
+
+      dispatch(expandCommunityTimeline({ onlyMedia }));
+
+      if (signedIn) {
+        this.disconnect = dispatch(connectCommunityStream({ onlyMedia }));
+      }
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.disconnect) {
+      this.disconnect();
+      this.disconnect = null;
+    }
+  }
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = maxId => {
+    const { dispatch, onlyMedia } = this.props;
+
+    dispatch(expandCommunityTimeline({ maxId, onlyMedia }));
+  };
+
+  render () {
+    const { intl, hasUnread, columnId, multiColumn, onlyMedia } = this.props;
+    const pinned = !!columnId;
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.title)}>
+        <ColumnHeader
+          icon='users'
+          active={hasUnread}
+          title={intl.formatMessage(messages.title)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+        >
+          <ColumnSettingsContainer columnId={columnId} />
+        </ColumnHeader>
+
+        <StatusListContainer
+          prepend={<DismissableBanner id='community_timeline'><FormattedMessage id='dismissable_banner.community_timeline' defaultMessage='These are the most recent public posts from people whose accounts are hosted by {domain}.' values={{ domain }} /></DismissableBanner>}
+          trackScroll={!pinned}
+          scrollKey={`community_timeline-${columnId}`}
+          timelineId={`community${onlyMedia ? ':media' : ''}`}
+          onLoadMore={this.handleLoadMore}
+          emptyMessage={<FormattedMessage id='empty_column.community' defaultMessage='The local timeline is empty. Write something publicly to get the ball rolling!' />}
+          bindToDocument={!multiColumn}
+          regex={this.props.regex}
+        />
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(CommunityTimeline));
diff --git a/app/javascript/flavours/blobfox/features/compose/components/action_bar.jsx b/app/javascript/flavours/blobfox/features/compose/components/action_bar.jsx
new file mode 100644
index 00000000000000..2da781b792df0b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/action_bar.jsx
@@ -0,0 +1,73 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { preferencesLink, profileLink } from 'flavours/blobfox/utils/backend_links';
+
+import DropdownMenuContainer from '../../../containers/dropdown_menu_container';
+
+const messages = defineMessages({
+  edit_profile: { id: 'account.edit_profile', defaultMessage: 'Edit profile' },
+  pins: { id: 'navigation_bar.pins', defaultMessage: 'Pinned posts' },
+  preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
+  follow_requests: { id: 'navigation_bar.follow_requests', defaultMessage: 'Follow requests' },
+  favourites: { id: 'navigation_bar.favourites', defaultMessage: 'Favorites' },
+  lists: { id: 'navigation_bar.lists', defaultMessage: 'Lists' },
+  followed_tags: { id: 'navigation_bar.followed_tags', defaultMessage: 'Followed hashtags' },
+  blocks: { id: 'navigation_bar.blocks', defaultMessage: 'Blocked users' },
+  domain_blocks: { id: 'navigation_bar.domain_blocks', defaultMessage: 'Blocked domains' },
+  mutes: { id: 'navigation_bar.mutes', defaultMessage: 'Muted users' },
+  filters: { id: 'navigation_bar.filters', defaultMessage: 'Muted words' },
+  logout: { id: 'navigation_bar.logout', defaultMessage: 'Logout' },
+  bookmarks: { id: 'navigation_bar.bookmarks', defaultMessage: 'Bookmarks' },
+});
+
+class ActionBar extends PureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    onLogout: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleLogout = () => {
+    this.props.onLogout();
+  };
+
+  render () {
+    const { intl } = this.props;
+
+    let menu = [];
+
+    menu.push({ text: intl.formatMessage(messages.edit_profile), href: profileLink });
+    menu.push({ text: intl.formatMessage(messages.preferences), href: preferencesLink });
+    menu.push({ text: intl.formatMessage(messages.pins), to: '/pinned' });
+    menu.push(null);
+    menu.push({ text: intl.formatMessage(messages.follow_requests), to: '/follow_requests' });
+    menu.push({ text: intl.formatMessage(messages.favourites), to: '/favourites' });
+    menu.push({ text: intl.formatMessage(messages.bookmarks), to: '/bookmarks' });
+    menu.push({ text: intl.formatMessage(messages.lists), to: '/lists' });
+    menu.push({ text: intl.formatMessage(messages.followed_tags), to: '/followed_tags' });
+    menu.push(null);
+    menu.push({ text: intl.formatMessage(messages.mutes), to: '/mutes' });
+    menu.push({ text: intl.formatMessage(messages.blocks), to: '/blocks' });
+    menu.push({ text: intl.formatMessage(messages.domain_blocks), to: '/domain_blocks' });
+    menu.push({ text: intl.formatMessage(messages.filters), href: '/filters' });
+    menu.push(null);
+    menu.push({ text: intl.formatMessage(messages.logout), action: this.handleLogout });
+
+    return (
+      <div className='compose__action-bar'>
+        <div className='compose__action-bar-dropdown'>
+          <DropdownMenuContainer items={menu} icon='bars' size={18} direction='right' />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ActionBar);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/autosuggest_account.jsx b/app/javascript/flavours/blobfox/features/compose/components/autosuggest_account.jsx
new file mode 100644
index 00000000000000..8ffc9062e5c9fd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/autosuggest_account.jsx
@@ -0,0 +1,24 @@
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Avatar } from '../../../components/avatar';
+import { DisplayName } from '../../../components/display_name';
+
+export default class AutosuggestAccount extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+  };
+
+  render () {
+    const { account } = this.props;
+
+    return (
+      <div className='autosuggest-account' title={account.get('acct')}>
+        <div className='autosuggest-account-icon'><Avatar account={account} size={24} /></div>
+        <DisplayName account={account} inline />
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/character_counter.jsx b/app/javascript/flavours/blobfox/features/compose/components/character_counter.jsx
new file mode 100644
index 00000000000000..42452b30f6af55
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/character_counter.jsx
@@ -0,0 +1,26 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { length } from 'stringz';
+
+export default class CharacterCounter extends PureComponent {
+
+  static propTypes = {
+    text: PropTypes.string.isRequired,
+    max: PropTypes.number.isRequired,
+  };
+
+  checkRemainingText (diff) {
+    if (diff < 0) {
+      return <span className='character-counter character-counter--over'>{diff}</span>;
+    }
+
+    return <span className='character-counter'>{diff}</span>;
+  }
+
+  render () {
+    const diff = this.props.max - length(this.props.text);
+    return this.checkRemainingText(diff);
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/compose_form.jsx b/app/javascript/flavours/blobfox/features/compose/components/compose_form.jsx
new file mode 100644
index 00000000000000..282a28ba711fa1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/compose_form.jsx
@@ -0,0 +1,356 @@
+import PropTypes from 'prop-types';
+import { createRef } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { length } from 'stringz';
+
+import { maxChars } from 'flavours/blobfox/initial_state';
+import { isMobile } from 'flavours/blobfox/is_mobile';
+import { WithOptionalRouterPropTypes, withOptionalRouter } from 'flavours/blobfox/utils/react_router';
+
+import AutosuggestInput from '../../../components/autosuggest_input';
+import AutosuggestTextarea from '../../../components/autosuggest_textarea';
+import EmojiPickerDropdown from '../containers/emoji_picker_dropdown_container';
+import OptionsContainer from '../containers/options_container';
+import PollFormContainer from '../containers/poll_form_container';
+import ReplyIndicatorContainer from '../containers/reply_indicator_container';
+import UploadFormContainer from '../containers/upload_form_container';
+import WarningContainer from '../containers/warning_container';
+import { countableText } from '../util/counter';
+
+import CharacterCounter from './character_counter';
+import Publisher from './publisher';
+import TextareaIcons from './textarea_icons';
+
+const messages = defineMessages({
+  placeholder: { id: 'compose_form.placeholder', defaultMessage: 'What is on your mind?' },
+  missingDescriptionMessage: {
+    id: 'confirmations.missing_media_description.message',
+    defaultMessage: 'At least one media attachment is lacking a description. Consider describing all media attachments for the visually impaired before sending your toot.',
+  },
+  missingDescriptionConfirm: {
+    id: 'confirmations.missing_media_description.confirm',
+    defaultMessage: 'Send anyway',
+  },
+  spoiler_placeholder: { id: 'compose_form.spoiler_placeholder', defaultMessage: 'Write your warning here' },
+});
+
+class ComposeForm extends ImmutablePureComponent {
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    text: PropTypes.string.isRequired,
+    suggestions: ImmutablePropTypes.list,
+    spoiler: PropTypes.bool,
+    privacy: PropTypes.string,
+    spoilerText: PropTypes.string,
+    focusDate: PropTypes.instanceOf(Date),
+    caretPosition: PropTypes.number,
+    preselectDate: PropTypes.instanceOf(Date),
+    isSubmitting: PropTypes.bool,
+    isChangingUpload: PropTypes.bool,
+    isEditing: PropTypes.bool,
+    isUploading: PropTypes.bool,
+    onChange: PropTypes.func.isRequired,
+    onSubmit: PropTypes.func.isRequired,
+    onClearSuggestions: PropTypes.func.isRequired,
+    onFetchSuggestions: PropTypes.func.isRequired,
+    onSuggestionSelected: PropTypes.func.isRequired,
+    onChangeSpoilerText: PropTypes.func.isRequired,
+    onPaste: PropTypes.func.isRequired,
+    onPickEmoji: PropTypes.func.isRequired,
+    showSearch: PropTypes.bool,
+    anyMedia: PropTypes.bool,
+    isInReply: PropTypes.bool,
+    singleColumn: PropTypes.bool,
+    lang: PropTypes.string,
+    advancedOptions: ImmutablePropTypes.map,
+    layout: PropTypes.string,
+    media: ImmutablePropTypes.list,
+    sideArm: PropTypes.string,
+    sensitive: PropTypes.bool,
+    spoilersAlwaysOn: PropTypes.bool,
+    mediaDescriptionConfirmation: PropTypes.bool,
+    preselectOnReply: PropTypes.bool,
+    onChangeSpoilerness: PropTypes.func.isRequired,
+    onChangeVisibility: PropTypes.func.isRequired,
+    onMediaDescriptionConfirm: PropTypes.func.isRequired,
+    ...WithOptionalRouterPropTypes
+  };
+
+  static defaultProps = {
+    showSearch: false,
+  };
+
+  state = {
+    highlighted: false,
+  };
+
+  constructor(props) {
+    super(props);
+    this.textareaRef = createRef(null);
+  }
+
+  handleChange = (e) => {
+    this.props.onChange(e.target.value);
+  };
+
+  handleKeyDown = (e) => {
+    if (e.keyCode === 13 && (e.ctrlKey || e.metaKey)) {
+      this.handleSubmit();
+    }
+
+    if (e.keyCode === 13 && e.altKey) {
+      this.handleSecondarySubmit();
+    }
+  };
+
+  getFulltextForCharacterCounting = () => {
+    return [
+      this.props.spoiler? this.props.spoilerText: '',
+      countableText(this.props.text),
+      this.props.advancedOptions && this.props.advancedOptions.get('do_not_federate') ? ' 👁️' : '',
+    ].join('');
+  };
+
+  canSubmit = () => {
+    const { isSubmitting, isChangingUpload, isUploading, anyMedia } = this.props;
+    const fulltext = this.getFulltextForCharacterCounting();
+
+    return !(isSubmitting || isUploading || isChangingUpload || length(fulltext) > maxChars || (!fulltext.trim().length && !anyMedia));
+  };
+
+  handleSubmit = (e, overriddenVisibility = null) => {
+    if (this.props.text !== this.textareaRef.current.value) {
+      // Something changed the text inside the textarea (e.g. browser extensions like Grammarly)
+      // Update the state to match the current text
+      this.props.onChange(this.textareaRef.current.value);
+    }
+
+    if (!this.canSubmit()) {
+      return;
+    }
+
+    if (e) {
+      e.preventDefault();
+    }
+
+    // Submit unless there are media with missing descriptions
+    if (this.props.mediaDescriptionConfirmation && this.props.media && this.props.media.some(item => !item.get('description'))) {
+      const firstWithoutDescription = this.props.media.find(item => !item.get('description'));
+      this.props.onMediaDescriptionConfirm(this.props.history || null, firstWithoutDescription.get('id'), overriddenVisibility);
+    } else {
+      if (overriddenVisibility) {
+        this.props.onChangeVisibility(overriddenVisibility);
+      }
+      this.props.onSubmit(this.props.history || null);
+    }
+  };
+
+  //  Handles the secondary submit button.
+  handleSecondarySubmit = () => {
+    const { sideArm } = this.props;
+    this.handleSubmit(null, sideArm === 'none' ? null : sideArm);
+  };
+
+  onSuggestionsClearRequested = () => {
+    this.props.onClearSuggestions();
+  };
+
+  onSuggestionsFetchRequested = (token) => {
+    this.props.onFetchSuggestions(token);
+  };
+
+  onSuggestionSelected = (tokenStart, token, value) => {
+    this.props.onSuggestionSelected(tokenStart, token, value, ['text']);
+  };
+
+  onSpoilerSuggestionSelected = (tokenStart, token, value) => {
+    this.props.onSuggestionSelected(tokenStart, token, value, ['spoiler_text']);
+  };
+
+  handleChangeSpoilerText = (e) => {
+    this.props.onChangeSpoilerText(e.target.value);
+  };
+
+  handleFocus = () => {
+    if (this.composeForm && !this.props.singleColumn) {
+      const { left, right } = this.composeForm.getBoundingClientRect();
+      if (left < 0 || right > (window.innerWidth || document.documentElement.clientWidth)) {
+        this.composeForm.scrollIntoView();
+      }
+    }
+  };
+
+  componentDidMount () {
+    this._updateFocusAndSelection({ });
+  }
+
+  componentWillUnmount () {
+    if (this.timeout) clearTimeout(this.timeout);
+  }
+
+  componentDidUpdate (prevProps) {
+    this._updateFocusAndSelection(prevProps);
+  }
+
+  _updateFocusAndSelection = (prevProps) => {
+    // This statement does several things:
+    // - If we're beginning a reply, and,
+    //     - Replying to zero or one users, places the cursor at the end of the textbox.
+    //     - Replying to more than one user, selects any usernames past the first;
+    //       this provides a convenient shortcut to drop everyone else from the conversation.
+    if (this.props.focusDate && this.props.focusDate !== prevProps.focusDate) {
+      let selectionEnd, selectionStart;
+
+      if (this.props.preselectDate !== prevProps.preselectDate && this.props.isInReply && this.props.preselectOnReply) {
+        selectionEnd   = this.props.text.length;
+        selectionStart = this.props.text.search(/\s/) + 1;
+      } else if (typeof this.props.caretPosition === 'number') {
+        selectionStart = this.props.caretPosition;
+        selectionEnd   = this.props.caretPosition;
+      } else {
+        selectionEnd   = this.props.text.length;
+        selectionStart = selectionEnd;
+      }
+
+      // Because of the wicg-inert polyfill, the activeElement may not be
+      // immediately selectable, we have to wait for observers to run, as
+      // described in https://github.com/WICG/inert#performance-and-gotchas
+      Promise.resolve().then(() => {
+        this.textareaRef.current.setSelectionRange(selectionStart, selectionEnd);
+        this.textareaRef.current.focus();
+        if (!this.props.singleColumn) this.textareaRef.current.scrollIntoView();
+        this.setState({ highlighted: true });
+        this.timeout = setTimeout(() => this.setState({ highlighted: false }), 700);
+      }).catch(console.error);
+    } else if(prevProps.isSubmitting && !this.props.isSubmitting) {
+      this.textareaRef.current.focus();
+    } else if (this.props.spoiler !== prevProps.spoiler) {
+      if (this.props.spoiler) {
+        this.spoilerText.input.focus();
+      } else if (prevProps.spoiler) {
+        this.textareaRef.current.focus();
+      }
+    }
+  };
+
+  setSpoilerText = (c) => {
+    this.spoilerText = c;
+  };
+
+  setRef = c => {
+    this.composeForm = c;
+  };
+
+  handleEmojiPick = (data) => {
+    const position = this.textareaRef.current.selectionStart;
+
+    this.props.onPickEmoji(position, data);
+  };
+
+  render () {
+    const {
+      intl,
+      advancedOptions,
+      isSubmitting,
+      layout,
+      onChangeSpoilerness,
+      onPaste,
+      privacy,
+      sensitive,
+      showSearch,
+      sideArm,
+      spoilersAlwaysOn,
+      isEditing,
+    } = this.props;
+    const { highlighted } = this.state;
+    const disabled = this.props.isSubmitting;
+
+    return (
+      <form className='compose-form' onSubmit={this.handleSubmit}>
+        <WarningContainer />
+
+        <ReplyIndicatorContainer />
+
+        <div className={`spoiler-input ${this.props.spoiler ? 'spoiler-input--visible' : ''}`} ref={this.setRef} aria-hidden={!this.props.spoiler}>
+          <AutosuggestInput
+            placeholder={intl.formatMessage(messages.spoiler_placeholder)}
+            value={this.props.spoilerText}
+            onChange={this.handleChangeSpoilerText}
+            onKeyDown={this.handleKeyDown}
+            disabled={!this.props.spoiler}
+            ref={this.setSpoilerText}
+            suggestions={this.props.suggestions}
+            onSuggestionsFetchRequested={this.onSuggestionsFetchRequested}
+            onSuggestionsClearRequested={this.onSuggestionsClearRequested}
+            onSuggestionSelected={this.onSpoilerSuggestionSelected}
+            searchTokens={[':']}
+            id='cw-spoiler-input'
+            className='spoiler-input__input'
+            lang={this.props.lang}
+            autoFocus={false}
+            spellCheck
+          />
+        </div>
+
+        <div className={classNames('compose-form__highlightable', { active: highlighted })}>
+          <AutosuggestTextarea
+            ref={this.textareaRef}
+            placeholder={intl.formatMessage(messages.placeholder)}
+            disabled={disabled}
+            value={this.props.text}
+            onChange={this.handleChange}
+            suggestions={this.props.suggestions}
+            onFocus={this.handleFocus}
+            onKeyDown={this.handleKeyDown}
+            onSuggestionsFetchRequested={this.onSuggestionsFetchRequested}
+            onSuggestionsClearRequested={this.onSuggestionsClearRequested}
+            onSuggestionSelected={this.onSuggestionSelected}
+            onPaste={onPaste}
+            autoFocus={!showSearch && !isMobile(window.innerWidth, layout)}
+            lang={this.props.lang}
+          >
+            <TextareaIcons advancedOptions={advancedOptions} />
+            <div className='compose-form__modifiers'>
+              <UploadFormContainer />
+              <PollFormContainer />
+            </div>
+          </AutosuggestTextarea>
+          <EmojiPickerDropdown onPickEmoji={this.handleEmojiPick} />
+
+          <div className='compose-form__buttons-wrapper'>
+            <OptionsContainer
+              advancedOptions={advancedOptions}
+              disabled={isSubmitting}
+              onToggleSpoiler={this.props.spoilersAlwaysOn ? null : onChangeSpoilerness}
+              onUpload={onPaste}
+              isEditing={isEditing}
+              sensitive={sensitive || (spoilersAlwaysOn && this.props.spoilerText && this.props.spoilerText.length > 0)}
+              spoiler={spoilersAlwaysOn ? (this.props.spoilerText && this.props.spoilerText.length > 0) : this.props.spoiler}
+            />
+            <div className='character-counter__wrapper'>
+              <CharacterCounter max={maxChars} text={this.getFulltextForCharacterCounting()} />
+            </div>
+          </div>
+        </div>
+
+        <Publisher
+          disabled={!this.canSubmit()}
+          isEditing={isEditing}
+          onSecondarySubmit={this.handleSecondarySubmit}
+          privacy={privacy}
+          sideArm={sideArm}
+        />
+      </form>
+    );
+  }
+
+}
+
+export default withOptionalRouter(injectIntl(ComposeForm));
diff --git a/app/javascript/flavours/blobfox/features/compose/components/dropdown.jsx b/app/javascript/flavours/blobfox/features/compose/components/dropdown.jsx
new file mode 100644
index 00000000000000..5f2de1189bfe25
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/dropdown.jsx
@@ -0,0 +1,243 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import Overlay from 'react-overlays/Overlay';
+
+//  Components.
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+import DropdownMenu from './dropdown_menu';
+
+//  The component.
+export default class ComposerOptionsDropdown extends PureComponent {
+
+  static propTypes = {
+    isUserTouching: PropTypes.func,
+    disabled: PropTypes.bool,
+    icon: PropTypes.string,
+    items: PropTypes.arrayOf(PropTypes.shape({
+      icon: PropTypes.string,
+      meta: PropTypes.string,
+      name: PropTypes.string.isRequired,
+      text: PropTypes.string,
+    })).isRequired,
+    onModalOpen: PropTypes.func,
+    onModalClose: PropTypes.func,
+    title: PropTypes.string,
+    value: PropTypes.string,
+    onChange: PropTypes.func,
+    container: PropTypes.func,
+    renderItemContents: PropTypes.func,
+    closeOnChange: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    closeOnChange: true,
+  };
+
+  state = {
+    open: false,
+    openedViaKeyboard: undefined,
+    placement: 'bottom',
+  };
+
+  //  Toggles opening and closing the dropdown.
+  handleToggle = ({ type }) => {
+    const { onModalOpen } = this.props;
+    const { open } = this.state;
+
+    if (this.props.isUserTouching && this.props.isUserTouching()) {
+      if (open) {
+        this.props.onModalClose();
+      } else {
+        const modal = this.handleMakeModal();
+        if (modal && onModalOpen) {
+          onModalOpen(modal);
+        }
+      }
+    } else {
+      if (open && this.activeElement) {
+        this.activeElement.focus({ preventScroll: true });
+      }
+      this.setState({ open: !open, openedViaKeyboard: type !== 'click' });
+    }
+  };
+
+  handleKeyDown = (e) => {
+    switch (e.key) {
+    case 'Escape':
+      this.handleClose();
+      break;
+    }
+  };
+
+  handleMouseDown = () => {
+    if (!this.state.open) {
+      this.activeElement = document.activeElement;
+    }
+  };
+
+  handleButtonKeyDown = (e) => {
+    switch(e.key) {
+    case ' ':
+    case 'Enter':
+      this.handleMouseDown();
+      break;
+    }
+  };
+
+  handleKeyPress = (e) => {
+    switch(e.key) {
+    case ' ':
+    case 'Enter':
+      this.handleToggle(e);
+      e.stopPropagation();
+      e.preventDefault();
+      break;
+    }
+  };
+
+  handleClose = () => {
+    if (this.state.open && this.activeElement) {
+      this.activeElement.focus({ preventScroll: true });
+    }
+    this.setState({ open: false });
+  };
+
+  handleItemClick = (e) => {
+    const {
+      items,
+      onChange,
+      onModalClose,
+      closeOnChange,
+    } = this.props;
+
+    const i = Number(e.currentTarget.getAttribute('data-index'));
+
+    const { name } = items[i];
+
+    e.preventDefault();  //  Prevents focus from changing
+    if (closeOnChange) onModalClose();
+    onChange(name);
+  };
+
+  //  Creates an action modal object.
+  handleMakeModal = () => {
+    const {
+      items,
+      onChange,
+      onModalOpen,
+      onModalClose,
+      value,
+    } = this.props;
+
+    //  Required props.
+    if (!(onChange && onModalOpen && onModalClose && items)) {
+      return null;
+    }
+
+    //  The object.
+    return {
+      renderItemContents: this.props.renderItemContents,
+      onClick: this.handleItemClick,
+      actions: items.map(
+        ({
+          name,
+          ...rest
+        }) => ({
+          ...rest,
+          active: value && name === value,
+          name,
+        }),
+      ),
+    };
+  };
+
+  setTargetRef = c => {
+    this.target = c;
+  };
+
+  findTarget = () => {
+    return this.target;
+  };
+
+  handleOverlayEnter = (state) => {
+    this.setState({ placement: state.placement });
+  };
+
+  //  Rendering.
+  render () {
+    const {
+      disabled,
+      title,
+      icon,
+      items,
+      onChange,
+      value,
+      container,
+      renderItemContents,
+      closeOnChange,
+    } = this.props;
+    const { open, placement } = this.state;
+
+    const active = value && items.findIndex(({ name }) => name === value) === (placement === 'bottom' ? 0 : (items.length - 1));
+
+    return (
+      <div
+        className={classNames('privacy-dropdown', placement, { active: open })}
+        onKeyDown={this.handleKeyDown}
+        ref={this.setTargetRef}
+      >
+        <div className={classNames('privacy-dropdown__value', { active })}>
+          <IconButton
+            active={open}
+            className='privacy-dropdown__value-icon'
+            disabled={disabled}
+            icon={icon}
+            inverted
+            onClick={this.handleToggle}
+            onMouseDown={this.handleMouseDown}
+            onKeyDown={this.handleButtonKeyDown}
+            onKeyPress={this.handleKeyPress}
+            size={18}
+            style={{
+              height: null,
+              lineHeight: '27px',
+            }}
+            title={title}
+          />
+        </div>
+
+        <Overlay
+          containerPadding={20}
+          placement={placement}
+          show={open}
+          flip
+          target={this.findTarget}
+          container={container}
+          popperConfig={{ strategy: 'fixed', onFirstUpdate: this.handleOverlayEnter }}
+        >
+          {({ props, placement }) => (
+            <div {...props}>
+              <div className={`dropdown-animation privacy-dropdown__dropdown ${placement}`}>
+                <DropdownMenu
+                  items={items}
+                  renderItemContents={renderItemContents}
+                  onChange={onChange}
+                  onClose={this.handleClose}
+                  value={value}
+                  openedViaKeyboard={this.state.openedViaKeyboard}
+                  closeOnChange={closeOnChange}
+                />
+              </div>
+            </div>
+          )}
+        </Overlay>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/dropdown_menu.jsx b/app/javascript/flavours/blobfox/features/compose/components/dropdown_menu.jsx
new file mode 100644
index 00000000000000..45b94dedee902c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/dropdown_menu.jsx
@@ -0,0 +1,200 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+
+//  Components.
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const listenerOptions = supportsPassiveEvents ? { passive: true, capture: true } : true;
+
+//  The component.
+export default class ComposerOptionsDropdownContent extends PureComponent {
+
+  static propTypes = {
+    items: PropTypes.arrayOf(PropTypes.shape({
+      icon: PropTypes.string,
+      meta: PropTypes.node,
+      name: PropTypes.string.isRequired,
+      text: PropTypes.node,
+    })),
+    onChange: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    style: PropTypes.object,
+    value: PropTypes.string,
+    renderItemContents: PropTypes.func,
+    openedViaKeyboard: PropTypes.bool,
+    closeOnChange: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    style: {},
+    closeOnChange: true,
+  };
+
+  state = {
+    value: this.props.openedViaKeyboard ? this.props.items[0].name : undefined,
+  };
+
+  //  When the document is clicked elsewhere, we close the dropdown.
+  handleDocumentClick = (e) => {
+    if (this.node && !this.node.contains(e.target)) {
+      this.props.onClose();
+      e.stopPropagation();
+    }
+  };
+
+  //  Stores our node in `this.node`.
+  setRef = (node) => {
+    this.node = node;
+  };
+
+  //  On mounting, we add our listeners.
+  componentDidMount () {
+    document.addEventListener('click', this.handleDocumentClick, { capture: true });
+    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
+    if (this.focusedItem) {
+      this.focusedItem.focus({ preventScroll: true });
+    } else {
+      this.node.firstChild.focus({ preventScroll: true });
+    }
+  }
+
+  //  On unmounting, we remove our listeners.
+  componentWillUnmount () {
+    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
+    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
+  }
+
+  handleClick = (e) => {
+    const i = Number(e.currentTarget.getAttribute('data-index'));
+
+    const {
+      onChange,
+      onClose,
+      closeOnChange,
+      items,
+    } = this.props;
+
+    const { name } = items[i];
+
+    e.preventDefault();  //  Prevents change in focus on click
+    if (closeOnChange) {
+      onClose();
+    }
+    onChange(name);
+  };
+
+  // Handle changes differently whether the dropdown is a list of options or actions
+  handleChange = (name) => {
+    if (this.props.value) {
+      this.props.onChange(name);
+    } else {
+      this.setState({ value: name });
+    }
+  };
+
+  handleKeyDown = (e) => {
+    const index = Number(e.currentTarget.getAttribute('data-index'));
+    const { items } = this.props;
+    let element = null;
+
+    switch(e.key) {
+    case 'Escape':
+      this.props.onClose();
+      break;
+    case 'Enter':
+    case ' ':
+      this.handleClick(e);
+      break;
+    case 'ArrowDown':
+      element = this.node.childNodes[index + 1] || this.node.firstChild;
+      break;
+    case 'ArrowUp':
+      element = this.node.childNodes[index - 1] || this.node.lastChild;
+      break;
+    case 'Tab':
+      if (e.shiftKey) {
+        element = this.node.childNodes[index - 1] || this.node.lastChild;
+      } else {
+        element = this.node.childNodes[index + 1] || this.node.firstChild;
+      }
+      break;
+    case 'Home':
+      element = this.node.firstChild;
+      break;
+    case 'End':
+      element = this.node.lastChild;
+      break;
+    }
+
+    if (element) {
+      element.focus();
+      this.handleChange(items[Number(element.getAttribute('data-index'))].name);
+      e.preventDefault();
+      e.stopPropagation();
+    }
+  };
+
+  setFocusRef = c => {
+    this.focusedItem = c;
+  };
+
+  renderItem = (item, i) => {
+    const { name, icon, meta, text } = item;
+
+    const active = (name === (this.props.value || this.state.value));
+
+    const computedClass = classNames('privacy-dropdown__option', { active });
+
+    let contents = this.props.renderItemContents && this.props.renderItemContents(item, i);
+
+    if (!contents) {
+      contents = (
+        <>
+          {icon && <Icon className='icon' fixedWidth id={icon} />}
+
+          <div className='privacy-dropdown__option__content'>
+            <strong>{text}</strong>
+            {meta}
+          </div>
+        </>
+      );
+    }
+
+    return (
+      <div
+        className={computedClass}
+        onClick={this.handleClick}
+        onKeyDown={this.handleKeyDown}
+        role='option'
+        aria-selected={active}
+        tabIndex={0}
+        key={name}
+        data-index={i}
+        ref={active ? this.setFocusRef : null}
+      >
+        {contents}
+      </div>
+    );
+  };
+
+  //  Rendering.
+  render () {
+    const {
+      items,
+      style,
+    } = this.props;
+
+    //  The result.
+    return (
+      <div style={{ ...style }} role='listbox' ref={this.setRef}>
+        {!!items && items.map((item, i) => this.renderItem(item, i))}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/emoji_picker_dropdown.jsx b/app/javascript/flavours/blobfox/features/compose/components/emoji_picker_dropdown.jsx
new file mode 100644
index 00000000000000..e228f9c205f922
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/emoji_picker_dropdown.jsx
@@ -0,0 +1,422 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+import Overlay from 'react-overlays/Overlay';
+
+import { useSystemEmojiFont } from 'flavours/blobfox/initial_state';
+import { assetHost } from 'flavours/blobfox/utils/config';
+
+import { buildCustomEmojis, categoriesFromEmojis } from '../../emoji/emoji';
+import { EmojiPicker as EmojiPickerAsync } from '../../ui/util/async-components';
+
+const messages = defineMessages({
+  emoji: { id: 'emoji_button.label', defaultMessage: 'Insert emoji' },
+  emoji_search: { id: 'emoji_button.search', defaultMessage: 'Search...' },
+  custom: { id: 'emoji_button.custom', defaultMessage: 'Custom' },
+  recent: { id: 'emoji_button.recent', defaultMessage: 'Frequently used' },
+  search_results: { id: 'emoji_button.search_results', defaultMessage: 'Search results' },
+  people: { id: 'emoji_button.people', defaultMessage: 'People' },
+  nature: { id: 'emoji_button.nature', defaultMessage: 'Nature' },
+  food: { id: 'emoji_button.food', defaultMessage: 'Food & Drink' },
+  activity: { id: 'emoji_button.activity', defaultMessage: 'Activity' },
+  travel: { id: 'emoji_button.travel', defaultMessage: 'Travel & Places' },
+  objects: { id: 'emoji_button.objects', defaultMessage: 'Objects' },
+  symbols: { id: 'emoji_button.symbols', defaultMessage: 'Symbols' },
+  flags: { id: 'emoji_button.flags', defaultMessage: 'Flags' },
+});
+
+let EmojiPicker, Emoji; // load asynchronously
+
+const listenerOptions = supportsPassiveEvents ? { passive: true, capture: true } : true;
+
+const backgroundImageFn = () => `${assetHost}/emoji/sheet_13.png`;
+
+const notFoundFn = () => (
+  <div className='emoji-mart-no-results'>
+    <Emoji
+      emoji='sleuth_or_spy'
+      set='twitter'
+      size={32}
+      sheetSize={32}
+      backgroundImageFn={backgroundImageFn}
+    />
+
+    <div className='emoji-mart-no-results-label'>
+      <FormattedMessage id='emoji_button.not_found' defaultMessage='No matching emojis found' />
+    </div>
+  </div>
+);
+
+class ModifierPickerMenu extends PureComponent {
+
+  static propTypes = {
+    active: PropTypes.bool,
+    onSelect: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+  };
+
+  handleClick = e => {
+    this.props.onSelect(e.currentTarget.getAttribute('data-index') * 1);
+  };
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (nextProps.active) {
+      this.attachListeners();
+    } else {
+      this.removeListeners();
+    }
+  }
+
+  componentWillUnmount () {
+    this.removeListeners();
+  }
+
+  handleDocumentClick = e => {
+    if (this.node && !this.node.contains(e.target)) {
+      this.props.onClose();
+    }
+  };
+
+  attachListeners () {
+    document.addEventListener('click', this.handleDocumentClick, { capture: true });
+    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
+  }
+
+  removeListeners () {
+    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
+    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  render () {
+    const { active } = this.props;
+
+    return (
+      <div className='emoji-picker-dropdown__modifiers__menu' style={{ display: active ? 'block' : 'none' }} ref={this.setRef}>
+        <button type='button' onClick={this.handleClick} data-index={1}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={1} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} /></button>
+        <button type='button' onClick={this.handleClick} data-index={2}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={2} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} /></button>
+        <button type='button' onClick={this.handleClick} data-index={3}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={3} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} /></button>
+        <button type='button' onClick={this.handleClick} data-index={4}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={4} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} /></button>
+        <button type='button' onClick={this.handleClick} data-index={5}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={5} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} /></button>
+        <button type='button' onClick={this.handleClick} data-index={6}><Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={6} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} /></button>
+      </div>
+    );
+  }
+
+}
+
+class ModifierPicker extends PureComponent {
+
+  static propTypes = {
+    active: PropTypes.bool,
+    modifier: PropTypes.number,
+    onChange: PropTypes.func,
+    onClose: PropTypes.func,
+    onOpen: PropTypes.func,
+  };
+
+  handleClick = () => {
+    if (this.props.active) {
+      this.props.onClose();
+    } else {
+      this.props.onOpen();
+    }
+  };
+
+  handleSelect = modifier => {
+    this.props.onChange(modifier);
+    this.props.onClose();
+  };
+
+  render () {
+    const { active, modifier } = this.props;
+
+    return (
+      <div className='emoji-picker-dropdown__modifiers'>
+        <Emoji emoji='fist' set='twitter' size={22} sheetSize={32} skin={modifier} onClick={this.handleClick} backgroundImageFn={backgroundImageFn} native={useSystemEmojiFont} />
+        <ModifierPickerMenu active={active} onSelect={this.handleSelect} onClose={this.props.onClose} />
+      </div>
+    );
+  }
+
+}
+
+class EmojiPickerMenuImpl extends PureComponent {
+
+  static propTypes = {
+    custom_emojis: ImmutablePropTypes.list,
+    frequentlyUsedEmojis: PropTypes.arrayOf(PropTypes.string),
+    loading: PropTypes.bool,
+    onClose: PropTypes.func.isRequired,
+    onPick: PropTypes.func.isRequired,
+    style: PropTypes.object,
+    intl: PropTypes.object.isRequired,
+    skinTone: PropTypes.number.isRequired,
+    onSkinTone: PropTypes.func.isRequired,
+  };
+
+  static defaultProps = {
+    style: {},
+    loading: true,
+    frequentlyUsedEmojis: [],
+  };
+
+  state = {
+    modifierOpen: false,
+    readyToFocus: false,
+  };
+
+  handleDocumentClick = e => {
+    if (this.node && !this.node.contains(e.target)) {
+      this.props.onClose();
+    }
+  };
+
+  componentDidMount () {
+    document.addEventListener('click', this.handleDocumentClick, { capture: true });
+    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
+
+    // Because of https://github.com/react-bootstrap/react-bootstrap/issues/2614 we need
+    // to wait for a frame before focusing
+    requestAnimationFrame(() => {
+      this.setState({ readyToFocus: true });
+      if (this.node) {
+        const element = this.node.querySelector('input[type="search"]');
+        if (element) element.focus();
+      }
+    });
+  }
+
+  componentWillUnmount () {
+    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
+    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  getI18n = () => {
+    const { intl } = this.props;
+
+    return {
+      search: intl.formatMessage(messages.emoji_search),
+      categories: {
+        search: intl.formatMessage(messages.search_results),
+        recent: intl.formatMessage(messages.recent),
+        people: intl.formatMessage(messages.people),
+        nature: intl.formatMessage(messages.nature),
+        foods: intl.formatMessage(messages.food),
+        activity: intl.formatMessage(messages.activity),
+        places: intl.formatMessage(messages.travel),
+        objects: intl.formatMessage(messages.objects),
+        symbols: intl.formatMessage(messages.symbols),
+        flags: intl.formatMessage(messages.flags),
+        custom: intl.formatMessage(messages.custom),
+      },
+    };
+  };
+
+  handleClick = (emoji, event) => {
+    if (!emoji.native) {
+      emoji.native = emoji.colons;
+    }
+    if (!(event.ctrlKey || event.metaKey)) {
+      this.props.onClose();
+    }
+    this.props.onPick(emoji);
+  };
+
+  handleModifierOpen = () => {
+    this.setState({ modifierOpen: true });
+  };
+
+  handleModifierClose = () => {
+    this.setState({ modifierOpen: false });
+  };
+
+  handleModifierChange = modifier => {
+    this.props.onSkinTone(modifier);
+  };
+
+  render () {
+    const { loading, style, intl, custom_emojis, skinTone, frequentlyUsedEmojis } = this.props;
+
+    if (loading) {
+      return <div style={{ width: 299 }} />;
+    }
+
+    const title = intl.formatMessage(messages.emoji);
+
+    const { modifierOpen } = this.state;
+
+    const categoriesSort = [
+      'recent',
+      'people',
+      'nature',
+      'foods',
+      'activity',
+      'places',
+      'objects',
+      'symbols',
+      'flags',
+    ];
+
+    categoriesSort.splice(1, 0, ...Array.from(categoriesFromEmojis(custom_emojis)).sort());
+
+    return (
+      <div className={classNames('emoji-picker-dropdown__menu', { selecting: modifierOpen })} style={style} ref={this.setRef}>
+        <EmojiPicker
+          perLine={8}
+          emojiSize={22}
+          sheetSize={32}
+          custom={buildCustomEmojis(custom_emojis)}
+          color=''
+          emoji=''
+          set='twitter'
+          title={title}
+          i18n={this.getI18n()}
+          onClick={this.handleClick}
+          include={categoriesSort}
+          recent={frequentlyUsedEmojis}
+          skin={skinTone}
+          showPreview={false}
+          showSkinTones={false}
+          backgroundImageFn={backgroundImageFn}
+          notFound={notFoundFn}
+          autoFocus={this.state.readyToFocus}
+          emojiTooltip
+          native={useSystemEmojiFont}
+        />
+
+        <ModifierPicker
+          active={modifierOpen}
+          modifier={skinTone}
+          onOpen={this.handleModifierOpen}
+          onClose={this.handleModifierClose}
+          onChange={this.handleModifierChange}
+        />
+      </div>
+    );
+  }
+
+}
+
+const EmojiPickerMenu = injectIntl(EmojiPickerMenuImpl);
+
+class EmojiPickerDropdown extends PureComponent {
+
+  static propTypes = {
+    custom_emojis: ImmutablePropTypes.list,
+    frequentlyUsedEmojis: PropTypes.arrayOf(PropTypes.string),
+    intl: PropTypes.object.isRequired,
+    onPickEmoji: PropTypes.func.isRequired,
+    onSkinTone: PropTypes.func.isRequired,
+    skinTone: PropTypes.number.isRequired,
+    button: PropTypes.node,
+    disabled: PropTypes.bool,
+  };
+
+  state = {
+    active: false,
+    loading: false,
+  };
+
+  setRef = (c) => {
+    this.dropdown = c;
+  };
+
+  onShowDropdown = () => {
+    this.setState({ active: true });
+
+    if (!EmojiPicker) {
+      this.setState({ loading: true });
+
+      EmojiPickerAsync().then(EmojiMart => {
+        EmojiPicker = EmojiMart.Picker;
+        Emoji       = EmojiMart.Emoji;
+
+        this.setState({ loading: false });
+      }).catch(() => {
+        this.setState({ loading: false, active: false });
+      });
+    }
+  };
+
+  onHideDropdown = () => {
+    this.setState({ active: false });
+  };
+
+  onToggle = (e) => {
+    if (!this.state.disabled && !this.state.loading && (!e.key || e.key === 'Enter')) {
+      if (this.state.active) {
+        this.onHideDropdown();
+      } else {
+        this.onShowDropdown(e);
+      }
+    }
+  };
+
+  handleKeyDown = e => {
+    if (e.key === 'Escape') {
+      this.onHideDropdown();
+    }
+  };
+
+  setTargetRef = c => {
+    this.target = c;
+  };
+
+  findTarget = () => {
+    return this.target;
+  };
+
+  render () {
+    const { intl, onPickEmoji, onSkinTone, skinTone, frequentlyUsedEmojis, button } = this.props;
+    const title = intl.formatMessage(messages.emoji);
+    const { active, loading } = this.state;
+
+    return (
+      <div className='emoji-picker-dropdown' onKeyDown={this.handleKeyDown}>
+        <div ref={this.setTargetRef} className='emoji-button' title={title} aria-label={title} aria-expanded={active} role='button' onClick={this.onToggle} onKeyDown={this.onToggle} tabIndex={0}>
+          {button || <img
+            className={classNames('emojione', { 'pulse-loading': active && loading })}
+            alt='🙂'
+            src={`${assetHost}/emoji/1f642.svg`}
+          />}
+        </div>
+
+        <Overlay show={active} placement={'bottom'} flip target={this.findTarget} popperConfig={{ strategy: 'fixed' }}>
+          {({ props, placement })=> (
+            <div {...props} style={{ ...props.style, width: 299 }}>
+              <div className={`dropdown-animation ${placement}`}>
+                <EmojiPickerMenu
+                  custom_emojis={this.props.custom_emojis}
+                  loading={loading}
+                  onClose={this.onHideDropdown}
+                  onPick={onPickEmoji}
+                  onSkinTone={onSkinTone}
+                  skinTone={skinTone}
+                  frequentlyUsedEmojis={frequentlyUsedEmojis}
+                />
+              </div>
+            </div>
+          )}
+        </Overlay>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(EmojiPickerDropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/header.jsx b/app/javascript/flavours/blobfox/features/compose/components/header.jsx
new file mode 100644
index 00000000000000..a8d3e080a04b7b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/header.jsx
@@ -0,0 +1,134 @@
+import PropTypes from 'prop-types';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import { signOutLink } from 'flavours/blobfox/utils/backend_links';
+import { conditionalRender } from 'flavours/blobfox/utils/react_helpers';
+
+const messages = defineMessages({
+  community: {
+    defaultMessage: 'Local timeline',
+    id: 'navigation_bar.community_timeline',
+  },
+  home_timeline: {
+    defaultMessage: 'Home',
+    id: 'tabs_bar.home',
+  },
+  logout: {
+    defaultMessage: 'Logout',
+    id: 'navigation_bar.logout',
+  },
+  notifications: {
+    defaultMessage: 'Notifications',
+    id: 'tabs_bar.notifications',
+  },
+  public: {
+    defaultMessage: 'Federated timeline',
+    id: 'navigation_bar.public_timeline',
+  },
+  settings: {
+    defaultMessage: 'App settings',
+    id: 'navigation_bar.app_settings',
+  },
+  start: {
+    defaultMessage: 'Getting started',
+    id: 'getting_started.heading',
+  },
+});
+
+class Header extends ImmutablePureComponent {
+
+  static propTypes = {
+    columns: ImmutablePropTypes.list,
+    unreadNotifications: PropTypes.number,
+    showNotificationsBadge: PropTypes.bool,
+    intl: PropTypes.object,
+    onSettingsClick: PropTypes.func,
+    onLogout: PropTypes.func.isRequired,
+  };
+
+  handleLogoutClick = e => {
+    e.preventDefault();
+    e.stopPropagation();
+
+    this.props.onLogout();
+
+    return false;
+  };
+
+  render () {
+    const { intl, columns, unreadNotifications, showNotificationsBadge, onSettingsClick } = this.props;
+
+    //  Only renders the component if the column isn't being shown.
+    const renderForColumn = conditionalRender.bind(null,
+      columnId => !columns || !columns.some(
+        column => column.get('id') === columnId,
+      ),
+    );
+
+    //  The result.
+    return (
+      <nav className='drawer__header'>
+        <Link
+          aria-label={intl.formatMessage(messages.start)}
+          title={intl.formatMessage(messages.start)}
+          to='/getting-started'
+        ><Icon id='asterisk' /></Link>
+        {renderForColumn('HOME', (
+          <Link
+            aria-label={intl.formatMessage(messages.home_timeline)}
+            title={intl.formatMessage(messages.home_timeline)}
+            to='/home'
+          ><Icon id='home' /></Link>
+        ))}
+        {renderForColumn('NOTIFICATIONS', (
+          <Link
+            aria-label={intl.formatMessage(messages.notifications)}
+            title={intl.formatMessage(messages.notifications)}
+            to='/notifications'
+          >
+            <span className='icon-badge-wrapper'>
+              <Icon id='bell' />
+              { showNotificationsBadge && unreadNotifications > 0 && <div className='icon-badge' />}
+            </span>
+          </Link>
+        ))}
+        {renderForColumn('COMMUNITY', (
+          <Link
+            aria-label={intl.formatMessage(messages.community)}
+            title={intl.formatMessage(messages.community)}
+            to='/public/local'
+          ><Icon id='users' /></Link>
+        ))}
+        {renderForColumn('PUBLIC', (
+          <Link
+            aria-label={intl.formatMessage(messages.public)}
+            title={intl.formatMessage(messages.public)}
+            to='/public'
+          ><Icon id='globe' /></Link>
+        ))}
+        <a
+          aria-label={intl.formatMessage(messages.settings)}
+          onClick={onSettingsClick}
+          href='/settings/preferences'
+          title={intl.formatMessage(messages.settings)}
+        ><Icon id='cogs' /></a>
+        <a
+          aria-label={intl.formatMessage(messages.logout)}
+          onClick={this.handleLogoutClick}
+          href={signOutLink}
+          title={intl.formatMessage(messages.logout)}
+        ><Icon id='sign-out' /></a>
+      </nav>
+    );
+  }
+
+}
+
+export default injectIntl(Header);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/language_dropdown.jsx b/app/javascript/flavours/blobfox/features/compose/components/language_dropdown.jsx
new file mode 100644
index 00000000000000..53ef6ebd63716b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/language_dropdown.jsx
@@ -0,0 +1,334 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+import fuzzysort from 'fuzzysort';
+import Overlay from 'react-overlays/Overlay';
+
+import { languages as preloadedLanguages } from 'flavours/blobfox/initial_state';
+import { loupeIcon, deleteIcon } from 'flavours/blobfox/utils/icons';
+
+import TextIconButton from './text_icon_button';
+
+const messages = defineMessages({
+  changeLanguage: { id: 'compose.language.change', defaultMessage: 'Change language' },
+  search: { id: 'compose.language.search', defaultMessage: 'Search languages...' },
+  clear: { id: 'emoji_button.clear', defaultMessage: 'Clear' },
+});
+
+const listenerOptions = supportsPassiveEvents ? { passive: true, capture: true } : true;
+
+class LanguageDropdownMenu extends PureComponent {
+
+  static propTypes = {
+    value: PropTypes.string.isRequired,
+    frequentlyUsedLanguages: PropTypes.arrayOf(PropTypes.string).isRequired,
+    onClose: PropTypes.func.isRequired,
+    onChange: PropTypes.func.isRequired,
+    languages: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.string)),
+    intl: PropTypes.object,
+  };
+
+  static defaultProps = {
+    languages: preloadedLanguages,
+  };
+
+  state = {
+    searchValue: '',
+  };
+
+  handleDocumentClick = e => {
+    if (this.node && !this.node.contains(e.target)) {
+      this.props.onClose();
+      e.stopPropagation();
+    }
+  };
+
+  componentDidMount () {
+    document.addEventListener('click', this.handleDocumentClick, { capture: true });
+    document.addEventListener('touchend', this.handleDocumentClick, listenerOptions);
+
+    // Because of https://github.com/react-bootstrap/react-bootstrap/issues/2614 we need
+    // to wait for a frame before focusing
+    requestAnimationFrame(() => {
+      if (this.node) {
+        const element = this.node.querySelector('input[type="search"]');
+        if (element) element.focus();
+      }
+    });
+  }
+
+  componentWillUnmount () {
+    document.removeEventListener('click', this.handleDocumentClick, { capture: true });
+    document.removeEventListener('touchend', this.handleDocumentClick, listenerOptions);
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  setListRef = c => {
+    this.listNode = c;
+  };
+
+  handleSearchChange = ({ target }) => {
+    this.setState({ searchValue: target.value });
+  };
+
+  search () {
+    const { languages, value, frequentlyUsedLanguages } = this.props;
+    const { searchValue } = this.state;
+
+    if (searchValue === '') {
+      return [...languages].sort((a, b) => {
+        // Push current selection to the top of the list
+
+        if (a[0] === value) {
+          return -1;
+        } else if (b[0] === value) {
+          return 1;
+        } else {
+          // Sort according to frequently used languages
+
+          const indexOfA = frequentlyUsedLanguages.indexOf(a[0]);
+          const indexOfB = frequentlyUsedLanguages.indexOf(b[0]);
+
+          return ((indexOfA > -1 ? indexOfA : Infinity) - (indexOfB > -1 ? indexOfB : Infinity));
+        }
+      });
+    }
+
+    return fuzzysort.go(searchValue, languages, {
+      keys: ['0', '1', '2'],
+      limit: 5,
+      threshold: -10000,
+    }).map(result => result.obj);
+  }
+
+  frequentlyUsed () {
+    const { languages, value } = this.props;
+    const current = languages.find(lang => lang[0] === value);
+    const results = [];
+
+    if (current) {
+      results.push(current);
+    }
+
+    return results;
+  }
+
+  handleClick = e => {
+    const value = e.currentTarget.getAttribute('data-index');
+
+    e.preventDefault();
+
+    this.props.onClose();
+    this.props.onChange(value);
+  };
+
+  handleKeyDown = e => {
+    const { onClose } = this.props;
+    const index = Array.from(this.listNode.childNodes).findIndex(node => node === e.currentTarget);
+
+    let element = null;
+
+    switch(e.key) {
+    case 'Escape':
+      onClose();
+      break;
+    case 'Enter':
+      this.handleClick(e);
+      break;
+    case 'ArrowDown':
+      element = this.listNode.childNodes[index + 1] || this.listNode.firstChild;
+      break;
+    case 'ArrowUp':
+      element = this.listNode.childNodes[index - 1] || this.listNode.lastChild;
+      break;
+    case 'Tab':
+      if (e.shiftKey) {
+        element = this.listNode.childNodes[index - 1] || this.listNode.lastChild;
+      } else {
+        element = this.listNode.childNodes[index + 1] || this.listNode.firstChild;
+      }
+      break;
+    case 'Home':
+      element = this.listNode.firstChild;
+      break;
+    case 'End':
+      element = this.listNode.lastChild;
+      break;
+    }
+
+    if (element) {
+      element.focus();
+      e.preventDefault();
+      e.stopPropagation();
+    }
+  };
+
+  handleSearchKeyDown = e => {
+    const { onChange, onClose } = this.props;
+    const { searchValue } = this.state;
+
+    let element = null;
+
+    switch(e.key) {
+    case 'Tab':
+    case 'ArrowDown':
+      element = this.listNode.firstChild;
+
+      if (element) {
+        element.focus();
+        e.preventDefault();
+        e.stopPropagation();
+      }
+
+      break;
+    case 'Enter':
+      element = this.listNode.firstChild;
+
+      if (element) {
+        onChange(element.getAttribute('data-index'));
+        onClose();
+      }
+      break;
+    case 'Escape':
+      if (searchValue !== '') {
+        e.preventDefault();
+        this.handleClear();
+      }
+
+      break;
+    }
+  };
+
+  handleClear = () => {
+    this.setState({ searchValue: '' });
+  };
+
+  renderItem = lang => {
+    const { value } = this.props;
+
+    return (
+      <div key={lang[0]} role='option' tabIndex={0} data-index={lang[0]} className={classNames('language-dropdown__dropdown__results__item', { active: lang[0] === value })} aria-selected={lang[0] === value} onClick={this.handleClick} onKeyDown={this.handleKeyDown}>
+        <span className='language-dropdown__dropdown__results__item__native-name' lang={lang[0]}>{lang[2]}</span> <span className='language-dropdown__dropdown__results__item__common-name'>({lang[1]})</span>
+      </div>
+    );
+  };
+
+  render () {
+    const { intl } = this.props;
+    const { searchValue } = this.state;
+    const isSearching = searchValue !== '';
+    const results = this.search();
+
+    return (
+      <div ref={this.setRef}>
+        <div className='emoji-mart-search'>
+          <input type='search' value={searchValue} onChange={this.handleSearchChange} onKeyDown={this.handleSearchKeyDown} placeholder={intl.formatMessage(messages.search)} />
+          <button type='button' className='emoji-mart-search-icon' disabled={!isSearching} aria-label={intl.formatMessage(messages.clear)} onClick={this.handleClear}>{!isSearching ? loupeIcon : deleteIcon}</button>
+        </div>
+
+        <div className='language-dropdown__dropdown__results emoji-mart-scroll' role='listbox' ref={this.setListRef}>
+          {results.map(this.renderItem)}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+class LanguageDropdown extends PureComponent {
+
+  static propTypes = {
+    value: PropTypes.string,
+    frequentlyUsedLanguages: PropTypes.arrayOf(PropTypes.string),
+    intl: PropTypes.object.isRequired,
+    onChange: PropTypes.func,
+    onClose: PropTypes.func,
+  };
+
+  state = {
+    open: false,
+    placement: 'bottom',
+  };
+
+  handleToggle = () => {
+    if (this.state.open && this.activeElement) {
+      this.activeElement.focus({ preventScroll: true });
+    }
+
+    this.setState({ open: !this.state.open });
+  };
+
+  handleClose = () => {
+    const { value, onClose } = this.props;
+
+    if (this.state.open && this.activeElement) {
+      this.activeElement.focus({ preventScroll: true });
+    }
+
+    this.setState({ open: false });
+    onClose(value);
+  };
+
+  handleChange = value => {
+    const { onChange } = this.props;
+    onChange(value);
+  };
+
+  setTargetRef = c => {
+    this.target = c;
+  };
+
+  findTarget = () => {
+    return this.target;
+  };
+
+  handleOverlayEnter = (state) => {
+    this.setState({ placement: state.placement });
+  };
+
+  render () {
+    const { value, intl, frequentlyUsedLanguages } = this.props;
+    const { open, placement } = this.state;
+
+    return (
+      <div className={classNames('privacy-dropdown', placement, { active: open })}>
+        <div className='privacy-dropdown__value' ref={this.setTargetRef} >
+          <TextIconButton
+            className='privacy-dropdown__value-icon'
+            label={value && value.toUpperCase()}
+            title={intl.formatMessage(messages.changeLanguage)}
+            active={open}
+            onClick={this.handleToggle}
+          />
+        </div>
+
+        <Overlay show={open} placement={'bottom'} flip target={this.findTarget} popperConfig={{ strategy: 'fixed', onFirstUpdate: this.handleOverlayEnter }}>
+          {({ props, placement }) => (
+            <div {...props}>
+              <div className={`dropdown-animation language-dropdown__dropdown ${placement}`} >
+                <LanguageDropdownMenu
+                  value={value}
+                  frequentlyUsedLanguages={frequentlyUsedLanguages}
+                  onClose={this.handleClose}
+                  onChange={this.handleChange}
+                  intl={intl}
+                />
+              </div>
+            </div>
+          )}
+        </Overlay>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(LanguageDropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/navigation_bar.jsx b/app/javascript/flavours/blobfox/features/compose/components/navigation_bar.jsx
new file mode 100644
index 00000000000000..b9601dd59ac49a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/navigation_bar.jsx
@@ -0,0 +1,54 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import Permalink from 'flavours/blobfox/components/permalink';
+import { profileLink } from 'flavours/blobfox/utils/backend_links';
+
+import { Avatar } from '../../../components/avatar';
+
+import ActionBar from './action_bar';
+
+export default class NavigationBar extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    onLogout: PropTypes.func.isRequired,
+    onClose: PropTypes.func,
+  };
+
+  render () {
+    const username = this.props.account.get('acct');
+    return (
+      <div className='navigation-bar'>
+        <Permalink className='avatar' href={this.props.account.get('url')} to={`/@${username}`}>
+          <span style={{ display: 'none' }}>{username}</span>
+          <Avatar account={this.props.account} size={46} />
+        </Permalink>
+
+        <div className='navigation-bar__profile'>
+          <span>
+            <Permalink className='acct' href={this.props.account.get('url')} to={`/@${username}`}>
+              <strong className='navigation-bar__profile-account'>@{username}</strong>
+            </Permalink>
+          </span>
+
+          { profileLink !== undefined && (
+            <a
+              className='edit'
+              href={profileLink}
+            ><FormattedMessage id='navigation_bar.edit_profile' defaultMessage='Edit profile' /></a>
+          )}
+        </div>
+
+        <div className='navigation-bar__actions'>
+          <ActionBar account={this.props.account} onLogout={this.props.onLogout} />
+        </div>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/options.jsx b/app/javascript/flavours/blobfox/features/compose/components/options.jsx
new file mode 100644
index 00000000000000..e3f46d97486b03
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/options.jsx
@@ -0,0 +1,311 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import Toggle from 'react-toggle';
+
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import { pollLimits } from 'flavours/blobfox/initial_state';
+
+import DropdownContainer from '../containers/dropdown_container';
+import LanguageDropdown from '../containers/language_dropdown_container';
+import PrivacyDropdownContainer from '../containers/privacy_dropdown_container';
+
+import TextIconButton from './text_icon_button';
+
+const messages = defineMessages({
+  advanced_options_icon_title: {
+    defaultMessage: 'Advanced options',
+    id: 'advanced_options.icon_title',
+  },
+  attach: {
+    defaultMessage: 'Attach...',
+    id: 'compose.attach',
+  },
+  content_type: {
+    defaultMessage: 'Content type',
+    id: 'content-type.change',
+  },
+  doodle: {
+    defaultMessage: 'Draw something',
+    id: 'compose.attach.doodle',
+  },
+  html: {
+    defaultMessage: 'HTML',
+    id: 'compose.content-type.html',
+  },
+  local_only_long: {
+    defaultMessage: 'Do not post to other instances',
+    id: 'advanced_options.local-only.long',
+  },
+  local_only_short: {
+    defaultMessage: 'Local-only',
+    id: 'advanced_options.local-only.short',
+  },
+  markdown: {
+    defaultMessage: 'Markdown',
+    id: 'compose.content-type.markdown',
+  },
+  plain: {
+    defaultMessage: 'Plain text',
+    id: 'compose.content-type.plain',
+  },
+  spoiler: {
+    defaultMessage: 'Hide text behind warning',
+    id: 'compose_form.spoiler',
+  },
+  threaded_mode_long: {
+    defaultMessage: 'Automatically opens a reply on posting',
+    id: 'advanced_options.threaded_mode.long',
+  },
+  threaded_mode_short: {
+    defaultMessage: 'Threaded mode',
+    id: 'advanced_options.threaded_mode.short',
+  },
+  upload: {
+    defaultMessage: 'Upload a file',
+    id: 'compose.attach.upload',
+  },
+  add_poll: {
+    defaultMessage: 'Add a poll',
+    id: 'poll_button.add_poll',
+  },
+  remove_poll: {
+    defaultMessage: 'Remove poll',
+    id: 'poll_button.remove_poll',
+  },
+});
+
+const mapStateToProps = (state, { name }) => ({
+  checked: state.getIn(['compose', 'advanced_options', name]),
+});
+
+class ToggleOptionImpl extends ImmutablePureComponent {
+
+  static propTypes = {
+    name: PropTypes.string.isRequired,
+    checked: PropTypes.bool,
+    onChangeAdvancedOption: PropTypes.func.isRequired,
+  };
+
+  handleChange = () => {
+    this.props.onChangeAdvancedOption(this.props.name);
+  };
+
+  render() {
+    const { meta, text, checked } = this.props;
+
+    return (
+      <>
+        <Toggle checked={checked} onChange={this.handleChange} />
+
+        <div className='privacy-dropdown__option__content'>
+          <strong>{text}</strong>
+          {meta}
+        </div>
+      </>
+    );
+  }
+
+}
+
+const ToggleOption = connect(mapStateToProps)(ToggleOptionImpl);
+
+class ComposerOptions extends ImmutablePureComponent {
+
+  static propTypes = {
+    acceptContentTypes: PropTypes.string,
+    advancedOptions: ImmutablePropTypes.map,
+    disabled: PropTypes.bool,
+    allowMedia: PropTypes.bool,
+    allowPoll: PropTypes.bool,
+    hasPoll: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    onChangeAdvancedOption: PropTypes.func.isRequired,
+    onChangeContentType: PropTypes.func.isRequired,
+    onTogglePoll: PropTypes.func.isRequired,
+    onDoodleOpen: PropTypes.func.isRequired,
+    onToggleSpoiler: PropTypes.func,
+    onUpload: PropTypes.func.isRequired,
+    contentType: PropTypes.string,
+    resetFileKey: PropTypes.number,
+    spoiler: PropTypes.bool,
+    showContentTypeChoice: PropTypes.bool,
+    isEditing: PropTypes.bool,
+  };
+
+  handleChangeFiles = ({ target: { files } }) => {
+    const { onUpload } = this.props;
+    if (files.length) {
+      onUpload(files);
+    }
+  };
+
+  handleClickAttach = (name) => {
+    const { fileElement } = this;
+    const { onDoodleOpen } = this.props;
+
+    switch (name) {
+    case 'upload':
+      if (fileElement) {
+        fileElement.click();
+      }
+      return;
+    case 'doodle':
+      onDoodleOpen();
+      return;
+    }
+  };
+
+  handleRefFileElement = (fileElement) => {
+    this.fileElement = fileElement;
+  };
+
+  renderToggleItemContents = (item) => {
+    const { onChangeAdvancedOption } = this.props;
+    const { name, meta, text } = item;
+
+    return <ToggleOption name={name} text={text} meta={meta} onChangeAdvancedOption={onChangeAdvancedOption} />;
+  };
+
+  render () {
+    const {
+      acceptContentTypes,
+      advancedOptions,
+      contentType,
+      disabled,
+      allowMedia,
+      allowPoll,
+      hasPoll,
+      onChangeAdvancedOption,
+      onChangeContentType,
+      onTogglePoll,
+      onToggleSpoiler,
+      resetFileKey,
+      spoiler,
+      showContentTypeChoice,
+      isEditing,
+      intl: { formatMessage },
+    } = this.props;
+
+    const contentTypeItems = {
+      plain: {
+        icon: 'file-text',
+        name: 'text/plain',
+        text: formatMessage(messages.plain),
+      },
+      html: {
+        icon: 'code',
+        name: 'text/html',
+        text: formatMessage(messages.html),
+      },
+      markdown: {
+        icon: 'arrow-circle-down',
+        name: 'text/markdown',
+        text: formatMessage(messages.markdown),
+      },
+    };
+
+    //  The result.
+    return (
+      <div className='compose-form__buttons'>
+        <input
+          accept={acceptContentTypes}
+          disabled={disabled || !allowMedia}
+          key={resetFileKey}
+          onChange={this.handleChangeFiles}
+          ref={this.handleRefFileElement}
+          type='file'
+          multiple
+          style={{ display: 'none' }}
+        />
+        <DropdownContainer
+          disabled={disabled || !allowMedia}
+          icon='paperclip'
+          items={[
+            {
+              icon: 'cloud-upload',
+              name: 'upload',
+              text: formatMessage(messages.upload),
+            },
+            {
+              icon: 'paint-brush',
+              name: 'doodle',
+              text: formatMessage(messages.doodle),
+            },
+          ]}
+          onChange={this.handleClickAttach}
+          title={formatMessage(messages.attach)}
+        />
+        {!!pollLimits && (
+          <IconButton
+            active={hasPoll}
+            disabled={disabled || !allowPoll}
+            icon='tasks'
+            inverted
+            onClick={onTogglePoll}
+            size={18}
+            style={{
+              height: null,
+              lineHeight: null,
+            }}
+            title={formatMessage(hasPoll ? messages.remove_poll : messages.add_poll)}
+          />
+        )}
+        <hr />
+        <PrivacyDropdownContainer disabled={disabled || isEditing} />
+        {showContentTypeChoice && (
+          <DropdownContainer
+            disabled={disabled}
+            icon={(contentTypeItems[contentType.split('/')[1]] || {}).icon}
+            items={[
+              contentTypeItems.plain,
+              contentTypeItems.html,
+              contentTypeItems.markdown,
+            ]}
+            onChange={onChangeContentType}
+            title={formatMessage(messages.content_type)}
+            value={contentType}
+          />
+        )}
+        {onToggleSpoiler && (
+          <TextIconButton
+            active={spoiler}
+            ariaControls='cw-spoiler-input'
+            label='CW'
+            onClick={onToggleSpoiler}
+            title={formatMessage(messages.spoiler)}
+          />
+        )}
+        <LanguageDropdown />
+        <DropdownContainer
+          disabled={disabled || isEditing}
+          icon='ellipsis-h'
+          items={advancedOptions ? [
+            {
+              meta: formatMessage(messages.local_only_long),
+              name: 'do_not_federate',
+              text: formatMessage(messages.local_only_short),
+            },
+            {
+              meta: formatMessage(messages.threaded_mode_long),
+              name: 'threaded_mode',
+              text: formatMessage(messages.threaded_mode_short),
+            },
+          ] : null}
+          onChange={onChangeAdvancedOption}
+          renderItemContents={this.renderToggleItemContents}
+          title={formatMessage(messages.advanced_options_icon_title)}
+          closeOnChange={false}
+        />
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ComposerOptions);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/poll_form.jsx b/app/javascript/flavours/blobfox/features/compose/components/poll_form.jsx
new file mode 100644
index 00000000000000..008888265091d2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/poll_form.jsx
@@ -0,0 +1,181 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import AutosuggestInput from 'flavours/blobfox/components/autosuggest_input';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import { pollLimits } from 'flavours/blobfox/initial_state';
+
+const messages = defineMessages({
+  option_placeholder: { id: 'compose_form.poll.option_placeholder', defaultMessage: 'Choice {number}' },
+  add_option: { id: 'compose_form.poll.add_option', defaultMessage: 'Add a choice' },
+  remove_option: { id: 'compose_form.poll.remove_option', defaultMessage: 'Remove this choice' },
+  poll_duration: { id: 'compose_form.poll.duration', defaultMessage: 'Poll duration' },
+  single_choice: { id: 'compose_form.poll.single_choice', defaultMessage: 'Allow one choice' },
+  multiple_choices: { id: 'compose_form.poll.multiple_choices', defaultMessage: 'Allow multiple choices' },
+  minutes: { id: 'intervals.full.minutes', defaultMessage: '{number, plural, one {# minute} other {# minutes}}' },
+  hours: { id: 'intervals.full.hours', defaultMessage: '{number, plural, one {# hour} other {# hours}}' },
+  days: { id: 'intervals.full.days', defaultMessage: '{number, plural, one {# day} other {# days}}' },
+});
+
+class OptionIntl extends PureComponent {
+
+  static propTypes = {
+    title: PropTypes.string.isRequired,
+    lang: PropTypes.string,
+    index: PropTypes.number.isRequired,
+    isPollMultiple: PropTypes.bool,
+    autoFocus: PropTypes.bool,
+    onChange: PropTypes.func.isRequired,
+    onRemove: PropTypes.func.isRequired,
+    suggestions: ImmutablePropTypes.list,
+    onClearSuggestions: PropTypes.func.isRequired,
+    onFetchSuggestions: PropTypes.func.isRequired,
+    onSuggestionSelected: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleOptionTitleChange = e => {
+    this.props.onChange(this.props.index, e.target.value);
+  };
+
+  handleOptionRemove = () => {
+    this.props.onRemove(this.props.index);
+  };
+
+  onSuggestionsClearRequested = () => {
+    this.props.onClearSuggestions();
+  };
+
+  onSuggestionsFetchRequested = (token) => {
+    this.props.onFetchSuggestions(token);
+  };
+
+  onSuggestionSelected = (tokenStart, token, value) => {
+    this.props.onSuggestionSelected(tokenStart, token, value, ['poll', 'options', this.props.index]);
+  };
+
+  render () {
+    const { isPollMultiple, title, lang, index, autoFocus, intl } = this.props;
+
+    return (
+      <li>
+        <label className='poll__option editable'>
+          <span className={classNames('poll__input', { checkbox: isPollMultiple })} />
+
+          <AutosuggestInput
+            placeholder={intl.formatMessage(messages.option_placeholder, { number: index + 1 })}
+            maxLength={pollLimits.max_option_chars}
+            value={title}
+            lang={lang}
+            spellCheck
+            onChange={this.handleOptionTitleChange}
+            suggestions={this.props.suggestions}
+            onSuggestionsFetchRequested={this.onSuggestionsFetchRequested}
+            onSuggestionsClearRequested={this.onSuggestionsClearRequested}
+            onSuggestionSelected={this.onSuggestionSelected}
+            searchTokens={[':']}
+            autoFocus={autoFocus}
+          />
+        </label>
+
+        <div className='poll__cancel'>
+          <IconButton disabled={index < 1} title={intl.formatMessage(messages.remove_option)} icon='times' onClick={this.handleOptionRemove} />
+        </div>
+      </li>
+    );
+  }
+
+}
+
+const Option = injectIntl(OptionIntl);
+
+class PollForm extends ImmutablePureComponent {
+
+  static propTypes = {
+    options: ImmutablePropTypes.list,
+    lang: PropTypes.string,
+    expiresIn: PropTypes.number,
+    isMultiple: PropTypes.bool,
+    onChangeOption: PropTypes.func.isRequired,
+    onAddOption: PropTypes.func.isRequired,
+    onRemoveOption: PropTypes.func.isRequired,
+    onChangeSettings: PropTypes.func.isRequired,
+    suggestions: ImmutablePropTypes.list,
+    onClearSuggestions: PropTypes.func.isRequired,
+    onFetchSuggestions: PropTypes.func.isRequired,
+    onSuggestionSelected: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleAddOption = () => {
+    this.props.onAddOption('');
+  };
+
+  handleSelectDuration = e => {
+    this.props.onChangeSettings(e.target.value, this.props.isMultiple);
+  };
+
+  handleSelectMultiple = e => {
+    this.props.onChangeSettings(this.props.expiresIn, e.target.value === 'true');
+  };
+
+  render () {
+    const { options, lang, expiresIn, isMultiple, onChangeOption, onRemoveOption, intl, ...other } = this.props;
+
+    if (!options) {
+      return null;
+    }
+
+    const autoFocusIndex = options.indexOf('');
+
+    return (
+      <div className='compose-form__poll-wrapper'>
+        <ul>
+          {options.map((title, i) => <Option title={title} lang={lang} key={i} index={i} onChange={onChangeOption} onRemove={onRemoveOption} isPollMultiple={isMultiple} autoFocus={i === autoFocusIndex} {...other} />)}
+          {options.size < pollLimits.max_options && (
+            <label className='poll__text editable'>
+              <span className={classNames('poll__input')} style={{ opacity: 0 }} />
+              <button className='button button-secondary' onClick={this.handleAddOption} type='button'><Icon id='plus' /> <FormattedMessage {...messages.add_option} /></button>
+            </label>
+          )}
+        </ul>
+
+        <div className='poll__footer'>
+          {/* eslint-disable-next-line jsx-a11y/no-onchange */}
+          <select value={isMultiple ? 'true' : 'false'} onChange={this.handleSelectMultiple}>
+            <option value='false'>{intl.formatMessage(messages.single_choice)}</option>
+            <option value='true'>{intl.formatMessage(messages.multiple_choices)}</option>
+          </select>
+
+          {/* eslint-disable-next-line jsx-a11y/no-onchange */}
+          <select value={expiresIn} onChange={this.handleSelectDuration}>
+            <option value={300}>{intl.formatMessage(messages.minutes, { number: 5 })}</option>
+            <option value={1800}>{intl.formatMessage(messages.minutes, { number: 30 })}</option>
+            <option value={3600}>{intl.formatMessage(messages.hours, { number: 1 })}</option>
+            <option value={21600}>{intl.formatMessage(messages.hours, { number: 6 })}</option>
+            <option value={43200}>{intl.formatMessage(messages.hours, { number: 12 })}</option>
+            <option value={86400}>{intl.formatMessage(messages.days, { number: 1 })}</option>
+            <option value={259200}>{intl.formatMessage(messages.days, { number: 3 })}</option>
+            <option value={604800}>{intl.formatMessage(messages.days, { number: 7 })}</option>
+            <option value={1209600}>{intl.formatMessage(messages.days, { number: 14 })}</option>
+            <option value={2629746}>{intl.formatMessage(messages.days, { number: 30 })}</option>
+            <option value={7889238}>{intl.formatMessage(messages.days, { number: 91 })}</option>
+            <option value={2629746}>{intl.formatMessage(messages.days, { number: 182 })}</option>
+            <option value={31536000}>{intl.formatMessage(messages.days, { number: 365 })}</option>
+          </select>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(PollForm);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/privacy_dropdown.jsx b/app/javascript/flavours/blobfox/features/compose/components/privacy_dropdown.jsx
new file mode 100644
index 00000000000000..06775230fe40d0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/privacy_dropdown.jsx
@@ -0,0 +1,90 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import Dropdown from './dropdown';
+
+const messages = defineMessages({
+  public_short: { id: 'privacy.public.short', defaultMessage: 'Public' },
+  public_long: { id: 'privacy.public.long', defaultMessage: 'Visible for all' },
+  unlisted_short: { id: 'privacy.unlisted.short', defaultMessage: 'Unlisted' },
+  unlisted_long: { id: 'privacy.unlisted.long', defaultMessage: 'Visible for all, but opted-out of discovery features' },
+  private_short: { id: 'privacy.private.short', defaultMessage: 'Followers only' },
+  private_long: { id: 'privacy.private.long', defaultMessage: 'Visible for followers only' },
+  direct_short: { id: 'privacy.direct.short', defaultMessage: 'Mentioned people only' },
+  direct_long: { id: 'privacy.direct.long', defaultMessage: 'Visible for mentioned users only' },
+  change_privacy: { id: 'privacy.change', defaultMessage: 'Adjust status privacy' },
+});
+
+class PrivacyDropdown extends PureComponent {
+
+  static propTypes = {
+    isUserTouching: PropTypes.func,
+    onModalOpen: PropTypes.func,
+    onModalClose: PropTypes.func,
+    value: PropTypes.string.isRequired,
+    onChange: PropTypes.func.isRequired,
+    noDirect: PropTypes.bool,
+    container: PropTypes.func,
+    disabled: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { value, onChange, onModalOpen, onModalClose, disabled, noDirect, container, isUserTouching, intl: { formatMessage } } = this.props;
+
+    //  We predefine our privacy items so that we can easily pick the
+    //  dropdown icon later.
+    const privacyItems = {
+      direct: {
+        icon: 'envelope',
+        meta: formatMessage(messages.direct_long),
+        name: 'direct',
+        text: formatMessage(messages.direct_short),
+      },
+      private: {
+        icon: 'lock',
+        meta: formatMessage(messages.private_long),
+        name: 'private',
+        text: formatMessage(messages.private_short),
+      },
+      public: {
+        icon: 'globe',
+        meta: formatMessage(messages.public_long),
+        name: 'public',
+        text: formatMessage(messages.public_short),
+      },
+      unlisted: {
+        icon: 'unlock',
+        meta: formatMessage(messages.unlisted_long),
+        name: 'unlisted',
+        text: formatMessage(messages.unlisted_short),
+      },
+    };
+
+    const items = [privacyItems.public, privacyItems.unlisted, privacyItems.private];
+
+    if (!noDirect) {
+      items.push(privacyItems.direct);
+    }
+
+    return (
+      <Dropdown
+        disabled={disabled}
+        icon={(privacyItems[value] || {}).icon}
+        items={items}
+        onChange={onChange}
+        isUserTouching={isUserTouching}
+        onModalClose={onModalClose}
+        onModalOpen={onModalOpen}
+        title={formatMessage(messages.change_privacy)}
+        container={container}
+        value={value}
+      />
+    );
+  }
+
+}
+
+export default injectIntl(PrivacyDropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/publisher.jsx b/app/javascript/flavours/blobfox/features/compose/components/publisher.jsx
new file mode 100644
index 00000000000000..df71df09818aa1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/publisher.jsx
@@ -0,0 +1,92 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Button } from 'flavours/blobfox/components/button';
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const messages = defineMessages({
+  publish: {
+    defaultMessage: 'Publish',
+    id: 'compose_form.publish',
+  },
+  publishLoud: {
+    defaultMessage: '{publish}!',
+    id: 'compose_form.publish_loud',
+  },
+  saveChanges: { id: 'compose_form.save_changes', defaultMessage: 'Save changes' },
+  public: { id: 'privacy.public.short', defaultMessage: 'Public' },
+  unlisted: { id: 'privacy.unlisted.short', defaultMessage: 'Unlisted' },
+  private: { id: 'privacy.private.short', defaultMessage: 'Followers only' },
+  direct: { id: 'privacy.direct.short', defaultMessage: 'Mentioned people only' },
+});
+
+class Publisher extends ImmutablePureComponent {
+
+  static propTypes = {
+    disabled: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    onSecondarySubmit: PropTypes.func,
+    privacy: PropTypes.oneOf(['direct', 'private', 'unlisted', 'public']),
+    sideArm: PropTypes.oneOf(['none', 'direct', 'private', 'unlisted', 'public']),
+    isEditing: PropTypes.bool,
+  };
+
+  render () {
+    const { disabled, intl, onSecondarySubmit, privacy, sideArm, isEditing } = this.props;
+
+    const privacyIcons = { direct: 'envelope', private: 'lock', public: 'globe', unlisted: 'unlock' };
+
+    let publishText;
+    if (isEditing) {
+      publishText = intl.formatMessage(messages.saveChanges);
+    } else if (privacy === 'private' || privacy === 'direct') {
+      const iconId = privacyIcons[privacy];
+      publishText = (
+        <span>
+          <Icon id={iconId} /> {intl.formatMessage(messages.publish)}
+        </span>
+      );
+    } else {
+      publishText = privacy !== 'unlisted' ? intl.formatMessage(messages.publishLoud, { publish: intl.formatMessage(messages.publish) }) : intl.formatMessage(messages.publish);
+    }
+
+    const privacyNames = {
+      public: messages.public,
+      unlisted: messages.unlisted,
+      private: messages.private,
+      direct: messages.direct,
+    };
+
+    return (
+      <div className='compose-form__publish'>
+        {sideArm && !isEditing && sideArm !== 'none' && (
+          <div className='compose-form__publish-button-wrapper'>
+            <Button
+              className='side_arm'
+              disabled={disabled}
+              onClick={onSecondarySubmit}
+              style={{ padding: null }}
+              text={<Icon id={privacyIcons[sideArm]} />}
+              title={`${intl.formatMessage(messages.publish)}: ${intl.formatMessage(privacyNames[sideArm])}`}
+            />
+          </div>
+        )}
+        <div className='compose-form__publish-button-wrapper'>
+          <Button
+            className='primary'
+            type='submit'
+            text={publishText}
+            title={`${intl.formatMessage(messages.publish)}: ${intl.formatMessage(privacyNames[privacy])}`}
+            disabled={disabled}
+          />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Publisher);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/reply_indicator.jsx b/app/javascript/flavours/blobfox/features/compose/components/reply_indicator.jsx
new file mode 100644
index 00000000000000..a3b42eea1c62bb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/reply_indicator.jsx
@@ -0,0 +1,70 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import AttachmentList from 'flavours/blobfox/components/attachment_list';
+
+import { IconButton } from '../../../components/icon_button';
+import AccountContainer from '../../../containers/account_container';
+
+const messages = defineMessages({
+  cancel: { id: 'reply_indicator.cancel', defaultMessage: 'Cancel' },
+});
+
+class ReplyIndicator extends ImmutablePureComponent {
+
+  static propTypes = {
+    status: ImmutablePropTypes.map,
+    onCancel: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleClick = () => {
+    const { onCancel } = this.props;
+    if (onCancel) {
+      onCancel();
+    }
+  };
+
+  render () {
+    const { status, intl } = this.props;
+
+    if (!status) {
+      return null;
+    }
+
+    const content = { __html: status.get('contentHtml') };
+
+    const account     = status.get('account');
+
+    return (
+      <div className='reply-indicator'>
+        <div className='reply-indicator__header'>
+          <div className='reply-indicator__cancel'><IconButton title={intl.formatMessage(messages.cancel)} icon='times' onClick={this.handleClick} inverted /></div>
+
+          {account && (
+            <AccountContainer
+              id={account}
+              small
+            />
+          )}
+        </div>
+
+        <div className='reply-indicator__content translate' dangerouslySetInnerHTML={content} />
+
+        {status.get('media_attachments').size > 0 && (
+          <AttachmentList
+            compact
+            media={status.get('media_attachments')}
+          />
+        )}
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ReplyIndicator);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/search.jsx b/app/javascript/flavours/blobfox/features/compose/components/search.jsx
new file mode 100644
index 00000000000000..76debfd7fb3b7c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/search.jsx
@@ -0,0 +1,400 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage, FormattedList } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { domain, searchEnabled } from 'flavours/blobfox/initial_state';
+import { HASHTAG_REGEX } from 'flavours/blobfox/utils/hashtags';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const messages = defineMessages({
+  placeholder: { id: 'search.placeholder', defaultMessage: 'Search' },
+  placeholderSignedIn: { id: 'search.search_or_paste', defaultMessage: 'Search or paste URL' },
+});
+
+const labelForRecentSearch = search => {
+  switch(search.get('type')) {
+  case 'account':
+    return `@${search.get('q')}`;
+  case 'hashtag':
+    return `#${search.get('q')}`;
+  default:
+    return search.get('q');
+  }
+};
+
+class Search extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object.isRequired,
+  };
+
+  static propTypes = {
+    value: PropTypes.string.isRequired,
+    recent: ImmutablePropTypes.orderedSet,
+    submitted: PropTypes.bool,
+    onChange: PropTypes.func.isRequired,
+    onSubmit: PropTypes.func.isRequired,
+    onOpenURL: PropTypes.func.isRequired,
+    onClickSearchResult: PropTypes.func.isRequired,
+    onForgetSearchResult: PropTypes.func.isRequired,
+    onClear: PropTypes.func.isRequired,
+    onShow: PropTypes.func.isRequired,
+    openInRoute: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    singleColumn: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  state = {
+    expanded: false,
+    selectedOption: -1,
+    options: [],
+  };
+
+  defaultOptions = [
+    { label: <><mark>has:</mark> <FormattedList type='disjunction' value={['media', 'poll', 'embed']} /></>, action: e => { e.preventDefault(); this._insertText('has:'); } },
+    { label: <><mark>is:</mark> <FormattedList type='disjunction' value={['reply', 'sensitive']} /></>, action: e => { e.preventDefault(); this._insertText('is:'); } },
+    { label: <><mark>language:</mark> <FormattedMessage id='search_popout.language_code' defaultMessage='ISO language code' /></>, action: e => { e.preventDefault(); this._insertText('language:'); } },
+    { label: <><mark>from:</mark> <FormattedMessage id='search_popout.user' defaultMessage='user' /></>, action: e => { e.preventDefault(); this._insertText('from:'); } },
+    { label: <><mark>before:</mark> <FormattedMessage id='search_popout.specific_date' defaultMessage='specific date' /></>, action: e => { e.preventDefault(); this._insertText('before:'); } },
+    { label: <><mark>during:</mark> <FormattedMessage id='search_popout.specific_date' defaultMessage='specific date' /></>, action: e => { e.preventDefault(); this._insertText('during:'); } },
+    { label: <><mark>after:</mark> <FormattedMessage id='search_popout.specific_date' defaultMessage='specific date' /></>, action: e => { e.preventDefault(); this._insertText('after:'); } },
+    { label: <><mark>in:</mark> <FormattedList type='disjunction' value={['all', 'library', 'public']} /></>, action: e => { e.preventDefault(); this._insertText('in:'); } }
+  ];
+
+  setRef = c => {
+    this.searchForm = c;
+  };
+
+  handleChange = ({ target }) => {
+    const { onChange } = this.props;
+
+    onChange(target.value);
+
+    this._calculateOptions(target.value);
+  };
+
+  handleClear = e => {
+    const { value, submitted, onClear } = this.props;
+
+    e.preventDefault();
+
+    if (value.length > 0 || submitted) {
+      onClear();
+      this.setState({ options: [], selectedOption: -1 });
+    }
+  };
+
+  handleKeyDown = (e) => {
+    const { selectedOption } = this.state;
+    const options = searchEnabled ? this._getOptions().concat(this.defaultOptions) : this._getOptions();
+
+    switch(e.key) {
+    case 'Escape':
+      e.preventDefault();
+      this._unfocus();
+
+      break;
+    case 'ArrowDown':
+      e.preventDefault();
+
+      if (options.length > 0) {
+        this.setState({ selectedOption: Math.min(selectedOption + 1, options.length - 1) });
+      }
+
+      break;
+    case 'ArrowUp':
+      e.preventDefault();
+
+      if (options.length > 0) {
+        this.setState({ selectedOption: Math.max(selectedOption - 1, -1) });
+      }
+
+      break;
+    case 'Enter':
+      e.preventDefault();
+
+      if (selectedOption === -1) {
+        this._submit();
+      } else if (options.length > 0) {
+        options[selectedOption].action(e);
+      }
+
+      break;
+    case 'Delete':
+      if (selectedOption > -1 && options.length > 0) {
+        const search = options[selectedOption];
+
+        if (typeof search.forget === 'function') {
+          e.preventDefault();
+          search.forget(e);
+        }
+      }
+
+      break;
+    }
+  };
+
+  handleFocus = () => {
+    const { onShow, singleColumn } = this.props;
+
+    this.setState({ expanded: true, selectedOption: -1 });
+    onShow();
+
+    if (this.searchForm && !singleColumn) {
+      const { left, right } = this.searchForm.getBoundingClientRect();
+
+      if (left < 0 || right > (window.innerWidth || document.documentElement.clientWidth)) {
+        this.searchForm.scrollIntoView();
+      }
+    }
+  };
+
+  handleBlur = () => {
+    this.setState({ expanded: false, selectedOption: -1 });
+  };
+
+  handleHashtagClick = () => {
+    const { value, onClickSearchResult, history } = this.props;
+
+    const query = value.trim().replace(/^#/, '');
+
+    history.push(`/tags/${query}`);
+    onClickSearchResult(query, 'hashtag');
+    this._unfocus();
+  };
+
+  handleAccountClick = () => {
+    const { value, onClickSearchResult, history } = this.props;
+
+    const query = value.trim().replace(/^@/, '');
+
+    history.push(`/@${query}`);
+    onClickSearchResult(query, 'account');
+    this._unfocus();
+  };
+
+  handleURLClick = () => {
+    const { onOpenURL, history } = this.props;
+
+    onOpenURL(history);
+    this._unfocus();
+  };
+
+  handleStatusSearch = () => {
+    this._submit('statuses');
+  };
+
+  handleAccountSearch = () => {
+    this._submit('accounts');
+  };
+
+  handleRecentSearchClick = search => {
+    const { onChange, history } = this.props;
+
+    if (search.get('type') === 'account') {
+      history.push(`/@${search.get('q')}`);
+    } else if (search.get('type') === 'hashtag') {
+      history.push(`/tags/${search.get('q')}`);
+    } else {
+      onChange(search.get('q'));
+      this._submit(search.get('type'));
+    }
+
+    this._unfocus();
+  };
+
+  handleForgetRecentSearchClick = search => {
+    const { onForgetSearchResult } = this.props;
+
+    onForgetSearchResult(search.get('q'));
+  };
+
+  _unfocus () {
+    document.querySelector('.ui').parentElement.focus();
+  }
+
+  _insertText (text) {
+    const { value, onChange } = this.props;
+
+    if (value === '') {
+      onChange(text);
+    } else if (value[value.length - 1] === ' ') {
+      onChange(`${value}${text}`);
+    } else {
+      onChange(`${value} ${text}`);
+    }
+  }
+
+  _submit (type) {
+    const { onSubmit, openInRoute, value, onClickSearchResult, history } = this.props;
+
+    onSubmit(type);
+
+    if (value) {
+      onClickSearchResult(value, type);
+    }
+
+    if (openInRoute) {
+      history.push('/search');
+    }
+
+    this._unfocus();
+  }
+
+  _getOptions () {
+    const { options } = this.state;
+
+    if (options.length > 0) {
+      return options;
+    }
+
+    const { recent } = this.props;
+
+    return recent.toArray().map(search => ({
+      label: labelForRecentSearch(search),
+
+      action: () => this.handleRecentSearchClick(search),
+
+      forget: e => {
+        e.stopPropagation();
+        this.handleForgetRecentSearchClick(search);
+      },
+    }));
+  }
+
+  _calculateOptions (value) {
+    const { signedIn } = this.context.identity;
+    const trimmedValue = value.trim();
+    const options = [];
+
+    if (trimmedValue.length > 0) {
+      const couldBeURL = trimmedValue.startsWith('https://') && !trimmedValue.includes(' ');
+
+      if (couldBeURL) {
+        options.push({ key: 'open-url', label: <FormattedMessage id='search.quick_action.open_url' defaultMessage='Open URL in Mastodon' />, action: this.handleURLClick });
+      }
+
+      const couldBeHashtag = (trimmedValue.startsWith('#') && trimmedValue.length > 1) || trimmedValue.match(HASHTAG_REGEX);
+
+      if (couldBeHashtag) {
+        options.push({ key: 'go-to-hashtag', label: <FormattedMessage id='search.quick_action.go_to_hashtag' defaultMessage='Go to hashtag {x}' values={{ x: <mark>#{trimmedValue.replace(/^#/, '')}</mark> }} />, action: this.handleHashtagClick });
+      }
+
+      const couldBeUsername = trimmedValue.match(/^@?[a-z0-9_-]+(@[^\s]+)?$/i);
+
+      if (couldBeUsername) {
+        options.push({ key: 'go-to-account', label: <FormattedMessage id='search.quick_action.go_to_account' defaultMessage='Go to profile {x}' values={{ x: <mark>@{trimmedValue.replace(/^@/, '')}</mark> }} />, action: this.handleAccountClick });
+      }
+
+      const couldBeStatusSearch = searchEnabled;
+
+      if (couldBeStatusSearch && signedIn) {
+        options.push({ key: 'status-search', label: <FormattedMessage id='search.quick_action.status_search' defaultMessage='Posts matching {x}' values={{ x: <mark>{trimmedValue}</mark> }} />, action: this.handleStatusSearch });
+      }
+
+      const couldBeUserSearch = true;
+
+      if (couldBeUserSearch) {
+        options.push({ key: 'account-search', label: <FormattedMessage id='search.quick_action.account_search' defaultMessage='Profiles matching {x}' values={{ x: <mark>{trimmedValue}</mark> }} />, action: this.handleAccountSearch });
+      }
+    }
+
+    this.setState({ options });
+  }
+
+  render () {
+    const { intl, value, submitted, recent } = this.props;
+    const { expanded, options, selectedOption } = this.state;
+    const { signedIn } = this.context.identity;
+
+    const hasValue = value.length > 0 || submitted;
+
+    return (
+      <div className={classNames('search', { active: expanded })}>
+        <input
+          ref={this.setRef}
+          className='search__input'
+          type='text'
+          placeholder={intl.formatMessage(signedIn ? messages.placeholderSignedIn : messages.placeholder)}
+          aria-label={intl.formatMessage(signedIn ? messages.placeholderSignedIn : messages.placeholder)}
+          value={value || ''}
+          onChange={this.handleChange}
+          onKeyDown={this.handleKeyDown}
+          onFocus={this.handleFocus}
+          onBlur={this.handleBlur}
+        />
+
+        <div role='button' tabIndex={0} className='search__icon' onClick={this.handleClear}>
+          <Icon id='search' className={hasValue ? '' : 'active'} />
+          <Icon id='times-circle' className={hasValue ? 'active' : ''} />
+        </div>
+
+        <div className='search__popout'>
+          {options.length === 0 && (
+            <>
+              <h4><FormattedMessage id='search_popout.recent' defaultMessage='Recent searches' /></h4>
+
+              <div className='search__popout__menu'>
+                {recent.size > 0 ? this._getOptions().map(({ label, action, forget }, i) => (
+                  <button key={label} onMouseDown={action} className={classNames('search__popout__menu__item search__popout__menu__item--flex', { selected: selectedOption === i })}>
+                    <span>{label}</span>
+                    <button className='icon-button' onMouseDown={forget}><Icon id='times' /></button>
+                  </button>
+                )) : (
+                  <div className='search__popout__menu__message'>
+                    <FormattedMessage id='search.no_recent_searches' defaultMessage='No recent searches' />
+                  </div>
+                )}
+              </div>
+            </>
+          )}
+
+          {options.length > 0 && (
+            <>
+              <h4><FormattedMessage id='search_popout.quick_actions' defaultMessage='Quick actions' /></h4>
+
+              <div className='search__popout__menu'>
+                {options.map(({ key, label, action }, i) => (
+                  <button key={key} onMouseDown={action} className={classNames('search__popout__menu__item', { selected: selectedOption === i })}>
+                    {label}
+                  </button>
+                ))}
+              </div>
+            </>
+          )}
+
+          <h4><FormattedMessage id='search_popout.options' defaultMessage='Search options' /></h4>
+
+          {searchEnabled && signedIn ? (
+            <div className='search__popout__menu'>
+              {this.defaultOptions.map(({ key, label, action }, i) => (
+                <button key={key} onMouseDown={action} className={classNames('search__popout__menu__item', { selected: selectedOption === ((options.length || recent.size) + i) })}>
+                  {label}
+                </button>
+              ))}
+            </div>
+          ) : (
+            <div className='search__popout__menu__message'>
+              {searchEnabled ? (
+                <FormattedMessage id='search_popout.full_text_search_logged_out_message' defaultMessage='Only available when logged in.' />
+              ) : (
+                <FormattedMessage id='search_popout.full_text_search_disabled_message' defaultMessage='Not available on {domain}.' values={{ domain }} />
+              )}
+            </div>
+          )}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(Search));
diff --git a/app/javascript/flavours/blobfox/features/compose/components/search_results.jsx b/app/javascript/flavours/blobfox/features/compose/components/search_results.jsx
new file mode 100644
index 00000000000000..dd383dec41d9c5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/search_results.jsx
@@ -0,0 +1,89 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { LoadMore } from 'flavours/blobfox/components/load_more';
+import { SearchSection } from 'flavours/blobfox/features/explore/components/search_section';
+
+import { ImmutableHashtag as Hashtag } from '../../../components/hashtag';
+import AccountContainer from '../../../containers/account_container';
+import StatusContainer from '../../../containers/status_container';
+
+const INITIAL_PAGE_LIMIT = 10;
+
+const withoutLastResult = list => {
+  if (list.size > INITIAL_PAGE_LIMIT && list.size % INITIAL_PAGE_LIMIT === 1) {
+    return list.skipLast(1);
+  } else {
+    return list;
+  }
+};
+
+class SearchResults extends ImmutablePureComponent {
+
+  static propTypes = {
+    results: ImmutablePropTypes.map.isRequired,
+    expandSearch: PropTypes.func.isRequired,
+    searchTerm: PropTypes.string,
+  };
+
+  handleLoadMoreAccounts = () => this.props.expandSearch('accounts');
+
+  handleLoadMoreStatuses = () => this.props.expandSearch('statuses');
+
+  handleLoadMoreHashtags = () => this.props.expandSearch('hashtags');
+
+  render () {
+    const { results } = this.props;
+
+    let accounts, statuses, hashtags;
+
+    if (results.get('accounts') && results.get('accounts').size > 0) {
+      accounts = (
+        <SearchSection title={<><Icon id='users' fixedWidth /><FormattedMessage id='search_results.accounts' defaultMessage='Profiles' /></>}>
+          {withoutLastResult(results.get('accounts')).map(accountId => <AccountContainer key={accountId} id={accountId} />)}
+          {(results.get('accounts').size > INITIAL_PAGE_LIMIT && results.get('accounts').size % INITIAL_PAGE_LIMIT === 1) && <LoadMore visible onClick={this.handleLoadMoreAccounts} />}
+        </SearchSection>
+      );
+    }
+
+    if (results.get('hashtags') && results.get('hashtags').size > 0) {
+      hashtags = (
+        <SearchSection title={<><Icon id='hashtag' fixedWidth /><FormattedMessage id='search_results.hashtags' defaultMessage='Hashtags' /></>}>
+          {withoutLastResult(results.get('hashtags')).map(hashtag => <Hashtag key={hashtag.get('name')} hashtag={hashtag} />)}
+          {(results.get('hashtags').size > INITIAL_PAGE_LIMIT && results.get('hashtags').size % INITIAL_PAGE_LIMIT === 1) && <LoadMore visible onClick={this.handleLoadMoreHashtags} />}
+        </SearchSection>
+      );
+    }
+
+    if (results.get('statuses') && results.get('statuses').size > 0) {
+      statuses = (
+        <SearchSection title={<><Icon id='quote-right' fixedWidth /><FormattedMessage id='search_results.statuses' defaultMessage='Posts' /></>}>
+          {withoutLastResult(results.get('statuses')).map(statusId => <StatusContainer key={statusId} id={statusId} />)}
+          {(results.get('statuses').size > INITIAL_PAGE_LIMIT && results.get('statuses').size % INITIAL_PAGE_LIMIT === 1) && <LoadMore visible onClick={this.handleLoadMoreStatuses} />}
+        </SearchSection>
+      );
+    }
+
+
+    return (
+      <div className='drawer--results'>
+        <header className='search-results__header'>
+          <Icon id='search' fixedWidth />
+          <FormattedMessage id='explore.search_results' defaultMessage='Search results' />
+        </header>
+
+        {accounts}
+        {hashtags}
+        {statuses}
+      </div>
+    );
+  }
+
+}
+
+export default SearchResults;
diff --git a/app/javascript/flavours/blobfox/features/compose/components/text_icon_button.jsx b/app/javascript/flavours/blobfox/features/compose/components/text_icon_button.jsx
new file mode 100644
index 00000000000000..166d022b88ac79
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/text_icon_button.jsx
@@ -0,0 +1,38 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+const iconStyle = {
+  height: null,
+  lineHeight: '27px',
+  minWidth: `${18 * 1.28571429}px`,
+};
+
+export default class TextIconButton extends PureComponent {
+
+  static propTypes = {
+    label: PropTypes.string.isRequired,
+    title: PropTypes.string,
+    active: PropTypes.bool,
+    onClick: PropTypes.func.isRequired,
+    ariaControls: PropTypes.string,
+  };
+
+  render () {
+    const { label, title, active, ariaControls } = this.props;
+
+    return (
+      <button
+        type='button'
+        title={title}
+        aria-label={title}
+        className={`text-icon-button ${active ? 'active' : ''}`}
+        aria-expanded={active}
+        onClick={this.props.onClick}
+        aria-controls={ariaControls} style={iconStyle}
+      >
+        {label}
+      </button>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/textarea_icons.jsx b/app/javascript/flavours/blobfox/features/compose/components/textarea_icons.jsx
new file mode 100644
index 00000000000000..b54f1256956da6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/textarea_icons.jsx
@@ -0,0 +1,61 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+//  Components.
+import { Icon } from 'flavours/blobfox/components/icon';
+//  Messages.
+const messages = defineMessages({
+  localOnly: {
+    defaultMessage: 'This post is local-only',
+    id: 'advanced_options.local-only.tooltip',
+  },
+  threadedMode: {
+    defaultMessage: 'Threaded mode enabled',
+    id: 'advanced_options.threaded_mode.tooltip',
+  },
+});
+
+//  We use an array of tuples here instead of an object because it
+//  preserves order.
+const iconMap = [
+  ['do_not_federate', 'home', messages.localOnly],
+  ['threaded_mode', 'comments', messages.threadedMode],
+];
+
+class TextareaIcons extends ImmutablePureComponent {
+
+  static propTypes = {
+    advancedOptions: ImmutablePropTypes.map,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { advancedOptions, intl } = this.props;
+    return (
+      <div className='compose-form__textarea-icons'>
+        {advancedOptions ? iconMap.map(
+          ([key, icon, message]) => advancedOptions.get(key) ? (
+            <span
+              className='textarea_icon'
+              key={key}
+              title={intl.formatMessage(message)}
+            >
+              <Icon
+                fixedWidth
+                id={icon}
+              />
+            </span>
+          ) : null,
+        ) : null}
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(TextareaIcons);
diff --git a/app/javascript/flavours/blobfox/features/compose/components/upload.jsx b/app/javascript/flavours/blobfox/features/compose/components/upload.jsx
new file mode 100644
index 00000000000000..5af24570baf806
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/upload.jsx
@@ -0,0 +1,66 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import spring from 'react-motion/lib/spring';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+import Motion from '../../ui/util/optional_motion';
+
+export default class Upload extends ImmutablePureComponent {
+
+  static propTypes = {
+    media: ImmutablePropTypes.map.isRequired,
+    onUndo: PropTypes.func.isRequired,
+    onOpenFocalPoint: PropTypes.func.isRequired,
+  };
+
+  handleUndoClick = e => {
+    e.stopPropagation();
+    this.props.onUndo(this.props.media.get('id'));
+  };
+
+  handleFocalPointClick = e => {
+    e.stopPropagation();
+    this.props.onOpenFocalPoint(this.props.media.get('id'));
+  };
+
+  render () {
+    const { media } = this.props;
+
+    if (!media) {
+      return null;
+    }
+
+    const focusX = media.getIn(['meta', 'focus', 'x']);
+    const focusY = media.getIn(['meta', 'focus', 'y']);
+    const x = ((focusX /  2) + .5) * 100;
+    const y = ((focusY / -2) + .5) * 100;
+
+    return (
+      <div className='compose-form__upload'>
+        <Motion defaultStyle={{ scale: 0.8 }} style={{ scale: spring(1, { stiffness: 180, damping: 12 }) }}>
+          {({ scale }) => (
+            <div className='compose-form__upload-thumbnail' style={{ transform: `scale(${scale})`, backgroundImage: `url(${media.get('preview_url')})`, backgroundPosition: `${x}% ${y}%` }}>
+              <div className='compose-form__upload__actions'>
+                <button type='button' className='icon-button' onClick={this.handleUndoClick}><Icon id='times' /> <FormattedMessage id='upload_form.undo' defaultMessage='Delete' /></button>
+                <button type='button' className='icon-button' onClick={this.handleFocalPointClick}><Icon id='pencil' /> <FormattedMessage id='upload_form.edit' defaultMessage='Edit' /></button>
+              </div>
+
+              {(media.get('description') || '').length === 0 && (
+                <div className='compose-form__upload__warning'>
+                  <button type='button' className='icon-button' onClick={this.handleFocalPointClick}><Icon id='info-circle' /> <FormattedMessage id='upload_form.description_missing' defaultMessage='No description added' /></button>
+                </div>
+              )}
+            </div>
+          )}
+        </Motion>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/upload_form.jsx b/app/javascript/flavours/blobfox/features/compose/components/upload_form.jsx
new file mode 100644
index 00000000000000..cf2e53ad905f0d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/upload_form.jsx
@@ -0,0 +1,32 @@
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import SensitiveButtonContainer from '../containers/sensitive_button_container';
+import UploadContainer from '../containers/upload_container';
+import UploadProgressContainer from '../containers/upload_progress_container';
+
+export default class UploadForm extends ImmutablePureComponent {
+
+  static propTypes = {
+    mediaIds: ImmutablePropTypes.list.isRequired,
+  };
+
+  render () {
+    const { mediaIds } = this.props;
+
+    return (
+      <div className='compose-form__upload-wrapper'>
+        <UploadProgressContainer />
+
+        <div className='compose-form__uploads-wrapper'>
+          {mediaIds.map(id => (
+            <UploadContainer id={id} key={id} />
+          ))}
+        </div>
+
+        {!mediaIds.isEmpty() && <SensitiveButtonContainer />}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/upload_progress.jsx b/app/javascript/flavours/blobfox/features/compose/components/upload_progress.jsx
new file mode 100644
index 00000000000000..906093d7821533
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/upload_progress.jsx
@@ -0,0 +1,56 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import spring from 'react-motion/lib/spring';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+import Motion from '../../ui/util/optional_motion';
+
+export default class UploadProgress extends PureComponent {
+
+  static propTypes = {
+    active: PropTypes.bool,
+    progress: PropTypes.number,
+    isProcessing: PropTypes.bool,
+  };
+
+  render () {
+    const { active, progress, isProcessing } = this.props;
+
+    if (!active) {
+      return null;
+    }
+
+    let message;
+
+    if (isProcessing) {
+      message = <FormattedMessage id='upload_progress.processing' defaultMessage='Processing…' />;
+    } else {
+      message = <FormattedMessage id='upload_progress.label' defaultMessage='Uploading…' />;
+    }
+
+    return (
+      <div className='upload-progress'>
+        <div className='upload-progress__icon'>
+          <Icon id='upload' />
+        </div>
+
+        <div className='upload-progress__message'>
+          {message}
+
+          <div className='upload-progress__backdrop'>
+            <Motion defaultStyle={{ width: 0 }} style={{ width: spring(progress) }}>
+              {({ width }) =>
+                <div className='upload-progress__tracker' style={{ width: `${width}%` }} />
+              }
+            </Motion>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/components/warning.jsx b/app/javascript/flavours/blobfox/features/compose/components/warning.jsx
new file mode 100644
index 00000000000000..c5babc30a5a1ad
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/components/warning.jsx
@@ -0,0 +1,28 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import spring from 'react-motion/lib/spring';
+
+import Motion from '../../ui/util/optional_motion';
+
+export default class Warning extends PureComponent {
+
+  static propTypes = {
+    message: PropTypes.node.isRequired,
+  };
+
+  render () {
+    const { message } = this.props;
+
+    return (
+      <Motion defaultStyle={{ opacity: 0, scaleX: 0.85, scaleY: 0.75 }} style={{ opacity: spring(1, { damping: 35, stiffness: 400 }), scaleX: spring(1, { damping: 35, stiffness: 400 }), scaleY: spring(1, { damping: 35, stiffness: 400 }) }}>
+        {({ opacity, scaleX, scaleY }) => (
+          <div className='compose-form__warning' style={{ opacity: opacity, transform: `scale(${scaleX}, ${scaleY})` }}>
+            {message}
+          </div>
+        )}
+      </Motion>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/autosuggest_account_container.js b/app/javascript/flavours/blobfox/features/compose/containers/autosuggest_account_container.js
new file mode 100644
index 00000000000000..f86f01bd97e38c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/autosuggest_account_container.js
@@ -0,0 +1,16 @@
+import { connect } from 'react-redux';
+
+import { makeGetAccount } from '../../../selectors';
+import AutosuggestAccount from '../components/autosuggest_account';
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { id }) => ({
+    account: getAccount(state, id),
+  });
+
+  return mapStateToProps;
+};
+
+export default connect(makeMapStateToProps)(AutosuggestAccount);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/compose_form_container.js b/app/javascript/flavours/blobfox/features/compose/containers/compose_form_container.js
new file mode 100644
index 00000000000000..e2713a22138661
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/compose_form_container.js
@@ -0,0 +1,150 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { privacyPreference } from 'flavours/blobfox/utils/privacy_preference';
+
+import {
+  changeCompose,
+  submitCompose,
+  clearComposeSuggestions,
+  fetchComposeSuggestions,
+  selectComposeSuggestion,
+  changeComposeSpoilerText,
+  changeComposeSpoilerness,
+  changeComposeVisibility,
+  insertEmojiCompose,
+  uploadCompose,
+} from '../../../actions/compose';
+import { changeLocalSetting } from '../../../actions/local_settings';
+import {
+  openModal,
+} from '../../../actions/modal';
+import ComposeForm from '../components/compose_form';
+
+const messages = defineMessages({
+  missingDescriptionMessage: {
+    id: 'confirmations.missing_media_description.message',
+    defaultMessage: 'At least one media attachment is lacking a description. Consider describing all media attachments for the visually impaired before sending your toot.',
+  },
+  missingDescriptionConfirm: {
+    id: 'confirmations.missing_media_description.confirm',
+    defaultMessage: 'Send anyway',
+  },
+  missingDescriptionEdit: {
+    id: 'confirmations.missing_media_description.edit',
+    defaultMessage: 'Edit media',
+  },
+});
+
+const sideArmPrivacy = state => {
+  const inReplyTo = state.getIn(['compose', 'in_reply_to']);
+  const replyPrivacy = inReplyTo ? state.getIn(['statuses', inReplyTo, 'visibility']) : null;
+  const sideArmBasePrivacy = state.getIn(['local_settings', 'side_arm']);
+  const sideArmRestrictedPrivacy = replyPrivacy ? privacyPreference(replyPrivacy, sideArmBasePrivacy) : null;
+  let sideArmPrivacy = null;
+  switch (state.getIn(['local_settings', 'side_arm_reply_mode'])) {
+  case 'copy':
+    sideArmPrivacy = replyPrivacy;
+    break;
+  case 'restrict':
+    sideArmPrivacy = sideArmRestrictedPrivacy;
+    break;
+  }
+  return sideArmPrivacy || sideArmBasePrivacy;
+};
+
+const mapStateToProps = state => ({
+  text: state.getIn(['compose', 'text']),
+  suggestions: state.getIn(['compose', 'suggestions']),
+  spoiler: state.getIn(['local_settings', 'always_show_spoilers_field']) || state.getIn(['compose', 'spoiler']),
+  spoilerText: state.getIn(['compose', 'spoiler_text']),
+  privacy: state.getIn(['compose', 'privacy']),
+  focusDate: state.getIn(['compose', 'focusDate']),
+  caretPosition: state.getIn(['compose', 'caretPosition']),
+  preselectDate: state.getIn(['compose', 'preselectDate']),
+  isSubmitting: state.getIn(['compose', 'is_submitting']),
+  isEditing: state.getIn(['compose', 'id']) !== null,
+  isChangingUpload: state.getIn(['compose', 'is_changing_upload']),
+  isUploading: state.getIn(['compose', 'is_uploading']),
+  anyMedia: state.getIn(['compose', 'media_attachments']).size > 0,
+  isInReply: state.getIn(['compose', 'in_reply_to']) !== null,
+  lang: state.getIn(['compose', 'language']),
+  advancedOptions: state.getIn(['compose', 'advanced_options']),
+  layout: state.getIn(['local_settings', 'layout']),
+  media: state.getIn(['compose', 'media_attachments']),
+  sideArm: sideArmPrivacy(state),
+  sensitive: state.getIn(['compose', 'sensitive']),
+  showSearch: state.getIn(['search', 'submitted']) && !state.getIn(['search', 'hidden']),
+  spoilersAlwaysOn: state.getIn(['local_settings', 'always_show_spoilers_field']),
+  mediaDescriptionConfirmation: state.getIn(['local_settings', 'confirm_missing_media_description']),
+  preselectOnReply: state.getIn(['local_settings', 'preselect_on_reply']),
+});
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+
+  onChange (text) {
+    dispatch(changeCompose(text));
+  },
+
+  onSubmit (router) {
+    dispatch(submitCompose(router));
+  },
+
+  onClearSuggestions () {
+    dispatch(clearComposeSuggestions());
+  },
+
+  onFetchSuggestions (token) {
+    dispatch(fetchComposeSuggestions(token));
+  },
+
+  onSuggestionSelected (position, token, suggestion, path) {
+    dispatch(selectComposeSuggestion(position, token, suggestion, path));
+  },
+
+  onChangeSpoilerText (text) {
+    dispatch(changeComposeSpoilerText(text));
+  },
+
+  onPaste (files) {
+    dispatch(uploadCompose(files));
+  },
+
+  onPickEmoji (position, emoji) {
+    dispatch(insertEmojiCompose(position, emoji));
+  },
+
+  onChangeSpoilerness() {
+    dispatch(changeComposeSpoilerness());
+  },
+
+  onChangeVisibility(value) {
+    dispatch(changeComposeVisibility(value));
+  },
+
+  onMediaDescriptionConfirm(routerHistory, mediaId, overriddenVisibility = null) {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.missingDescriptionMessage),
+        confirm: intl.formatMessage(messages.missingDescriptionConfirm),
+        onConfirm: () => {
+          if (overriddenVisibility) {
+            dispatch(changeComposeVisibility(overriddenVisibility));
+          }
+          dispatch(submitCompose(routerHistory));
+        },
+        secondary: intl.formatMessage(messages.missingDescriptionEdit),
+        onSecondary: () => dispatch(openModal({
+          modalType: 'FOCAL_POINT',
+          modalProps: { id: mediaId },
+        })),
+        onDoNotAsk: () => dispatch(changeLocalSetting(['confirm_missing_media_description'], false)),
+      },
+    }));
+  },
+
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(ComposeForm));
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/dropdown_container.js b/app/javascript/flavours/blobfox/features/compose/containers/dropdown_container.js
new file mode 100644
index 00000000000000..25d60d9328c8cb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/dropdown_container.js
@@ -0,0 +1,14 @@
+import { connect } from 'react-redux';
+
+import { openModal, closeModal } from 'flavours/blobfox/actions/modal';
+import { isUserTouching } from 'flavours/blobfox/is_mobile';
+
+import Dropdown from '../components/dropdown';
+
+const mapDispatchToProps = dispatch => ({
+  isUserTouching,
+  onModalOpen: props => dispatch(openModal({ modalType: 'ACTIONS', modalProps: props })),
+  onModalClose: () => dispatch(closeModal({ modalType: undefined, ignoreFocus: false })),
+});
+
+export default connect(null, mapDispatchToProps)(Dropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/emoji_picker_dropdown_container.js b/app/javascript/flavours/blobfox/features/compose/containers/emoji_picker_dropdown_container.js
new file mode 100644
index 00000000000000..a0e50029dfa303
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/emoji_picker_dropdown_container.js
@@ -0,0 +1,85 @@
+import { Map as ImmutableMap } from 'immutable';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { useEmoji } from '../../../actions/emojis';
+import { changeSetting } from '../../../actions/settings';
+import EmojiPickerDropdown from '../components/emoji_picker_dropdown';
+
+const perLine = 8;
+const lines   = 2;
+
+const DEFAULTS = [
+  '+1',
+  'grinning',
+  'kissing_heart',
+  'heart_eyes',
+  'laughing',
+  'stuck_out_tongue_winking_eye',
+  'sweat_smile',
+  'joy',
+  'yum',
+  'disappointed',
+  'thinking_face',
+  'weary',
+  'sob',
+  'sunglasses',
+  'heart',
+  'ok_hand',
+];
+
+const getFrequentlyUsedEmojis = createSelector([
+  state => state.getIn(['settings', 'frequentlyUsedEmojis'], ImmutableMap()),
+], emojiCounters => {
+  let emojis = emojiCounters
+    .keySeq()
+    .sort((a, b) => emojiCounters.get(a) - emojiCounters.get(b))
+    .reverse()
+    .slice(0, perLine * lines)
+    .toArray();
+
+  if (emojis.length < DEFAULTS.length) {
+    let uniqueDefaults = DEFAULTS.filter(emoji => !emojis.includes(emoji));
+    emojis = emojis.concat(uniqueDefaults.slice(0, DEFAULTS.length - emojis.length));
+  }
+
+  return emojis;
+});
+
+const getCustomEmojis = createSelector([
+  state => state.get('custom_emojis'),
+], emojis => emojis.filter(e => e.get('visible_in_picker')).sort((a, b) => {
+  const aShort = a.get('shortcode').toLowerCase();
+  const bShort = b.get('shortcode').toLowerCase();
+
+  if (aShort < bShort) {
+    return -1;
+  } else if (aShort > bShort ) {
+    return 1;
+  } else {
+    return 0;
+  }
+}));
+
+const mapStateToProps = state => ({
+  custom_emojis: getCustomEmojis(state),
+  skinTone: state.getIn(['settings', 'skinTone']),
+  frequentlyUsedEmojis: getFrequentlyUsedEmojis(state),
+});
+
+const mapDispatchToProps = (dispatch, { onPickEmoji }) => ({
+  onSkinTone: skinTone => {
+    dispatch(changeSetting(['skinTone'], skinTone));
+  },
+
+  onPickEmoji: emoji => {
+    // eslint-disable-next-line react-hooks/rules-of-hooks -- this is not a react hook
+    dispatch(useEmoji(emoji));
+
+    if (onPickEmoji) {
+      onPickEmoji(emoji);
+    }
+  },
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(EmojiPickerDropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/header_container.js b/app/javascript/flavours/blobfox/features/compose/containers/header_container.js
new file mode 100644
index 00000000000000..b7931ffa895d89
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/header_container.js
@@ -0,0 +1,42 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect }   from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { logOut } from 'flavours/blobfox/utils/log_out';
+
+import Header from '../components/header';
+
+const messages = defineMessages({
+  logoutMessage: { id: 'confirmations.logout.message', defaultMessage: 'Are you sure you want to log out?' },
+  logoutConfirm: { id: 'confirmations.logout.confirm', defaultMessage: 'Log out' },
+});
+
+const mapStateToProps = state => {
+  return {
+    columns: state.getIn(['settings', 'columns']),
+    unreadNotifications: state.getIn(['notifications', 'unread']),
+    showNotificationsBadge: state.getIn(['local_settings', 'notifications', 'tab_badge']),
+  };
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onSettingsClick (e) {
+    e.preventDefault();
+    e.stopPropagation();
+    dispatch(openModal({ modalType: 'SETTINGS', modalProps: {} }));
+  },
+  onLogout () {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.logoutMessage),
+        confirm: intl.formatMessage(messages.logoutConfirm),
+        closeWhenConfirm: false,
+        onConfirm: () => logOut(),
+      },
+    }));
+  },
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(Header));
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/language_dropdown_container.js b/app/javascript/flavours/blobfox/features/compose/containers/language_dropdown_container.js
new file mode 100644
index 00000000000000..7344097e229a2c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/language_dropdown_container.js
@@ -0,0 +1,37 @@
+import { Map as ImmutableMap } from 'immutable';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { changeComposeLanguage } from 'flavours/blobfox/actions/compose';
+import { useLanguage } from 'flavours/blobfox/actions/languages';
+
+import LanguageDropdown from '../components/language_dropdown';
+
+const getFrequentlyUsedLanguages = createSelector([
+  state => state.getIn(['settings', 'frequentlyUsedLanguages'], ImmutableMap()),
+], languageCounters => (
+  languageCounters.keySeq()
+    .sort((a, b) => languageCounters.get(a) - languageCounters.get(b))
+    .reverse()
+    .toArray()
+));
+
+const mapStateToProps = state => ({
+  frequentlyUsedLanguages: getFrequentlyUsedLanguages(state),
+  value: state.getIn(['compose', 'language']),
+});
+
+const mapDispatchToProps = dispatch => ({
+
+  onChange (value) {
+    dispatch(changeComposeLanguage(value));
+  },
+
+  onClose (value) {
+    // eslint-disable-next-line react-hooks/rules-of-hooks -- this is not a react hook
+    dispatch(useLanguage(value));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(LanguageDropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/navigation_container.js b/app/javascript/flavours/blobfox/features/compose/containers/navigation_container.js
new file mode 100644
index 00000000000000..8b8d2f57a2a7b8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/navigation_container.js
@@ -0,0 +1,36 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect }   from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { logOut } from 'flavours/blobfox/utils/log_out';
+
+import { me } from '../../../initial_state';
+import NavigationBar from '../components/navigation_bar';
+
+const messages = defineMessages({
+  logoutMessage: { id: 'confirmations.logout.message', defaultMessage: 'Are you sure you want to log out?' },
+  logoutConfirm: { id: 'confirmations.logout.confirm', defaultMessage: 'Log out' },
+});
+
+const mapStateToProps = state => {
+  return {
+    account: state.getIn(['accounts', me]),
+  };
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onLogout () {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.logoutMessage),
+        confirm: intl.formatMessage(messages.logoutConfirm),
+        closeWhenConfirm: false,
+        onConfirm: () => logOut(),
+      },
+    }));
+  },
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(NavigationBar));
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/options_container.js b/app/javascript/flavours/blobfox/features/compose/containers/options_container.js
new file mode 100644
index 00000000000000..12ded500f4c34c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/options_container.js
@@ -0,0 +1,56 @@
+import { connect } from 'react-redux';
+
+import {
+  changeComposeAdvancedOption,
+  changeComposeContentType,
+  addPoll,
+  removePoll,
+} from 'flavours/blobfox/actions/compose';
+import { openModal } from 'flavours/blobfox/actions/modal';
+
+import Options from '../components/options';
+
+function mapStateToProps (state) {
+  const poll = state.getIn(['compose', 'poll']);
+  const media = state.getIn(['compose', 'media_attachments']);
+  const pending_media = state.getIn(['compose', 'pending_media_attachments']);
+  return {
+    acceptContentTypes: state.getIn(['media_attachments', 'accept_content_types']).toArray().join(','),
+    resetFileKey: state.getIn(['compose', 'resetFileKey']),
+    hasPoll: !!poll,
+    allowMedia: !poll && (media ? media.size + pending_media < 4 && !media.some(item => ['video', 'audio'].includes(item.get('type'))) : pending_media < 4),
+    allowPoll: !(media && !!media.size),
+    showContentTypeChoice: state.getIn(['local_settings', 'show_content_type_choice']),
+    contentType: state.getIn(['compose', 'content_type']),
+  };
+}
+
+const mapDispatchToProps = (dispatch) => ({
+
+  onChangeAdvancedOption(option, value) {
+    dispatch(changeComposeAdvancedOption(option, value));
+  },
+
+  onChangeContentType(value) {
+    dispatch(changeComposeContentType(value));
+  },
+
+  onTogglePoll() {
+    dispatch((_, getState) => {
+      if (getState().getIn(['compose', 'poll'])) {
+        dispatch(removePoll());
+      } else {
+        dispatch(addPoll());
+      }
+    });
+  },
+
+  onDoodleOpen() {
+    dispatch(openModal({
+      modalType: 'DOODLE',
+      modalProps: { noEsc: true, noClose: true },
+    }));
+  },
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Options);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/poll_form_container.js b/app/javascript/flavours/blobfox/features/compose/containers/poll_form_container.js
new file mode 100644
index 00000000000000..177ffcea6acb4b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/poll_form_container.js
@@ -0,0 +1,53 @@
+import { connect } from 'react-redux';
+
+import {
+  addPollOption,
+  removePollOption,
+  changePollOption,
+  changePollSettings,
+  clearComposeSuggestions,
+  fetchComposeSuggestions,
+  selectComposeSuggestion,
+} from '../../../actions/compose';
+import PollForm from '../components/poll_form';
+
+const mapStateToProps = state => ({
+  suggestions: state.getIn(['compose', 'suggestions']),
+  options: state.getIn(['compose', 'poll', 'options']),
+  lang: state.getIn(['compose', 'language']),
+  expiresIn: state.getIn(['compose', 'poll', 'expires_in']),
+  isMultiple: state.getIn(['compose', 'poll', 'multiple']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onAddOption(title) {
+    dispatch(addPollOption(title));
+  },
+
+  onRemoveOption(index) {
+    dispatch(removePollOption(index));
+  },
+
+  onChangeOption(index, title) {
+    dispatch(changePollOption(index, title));
+  },
+
+  onChangeSettings(expiresIn, isMultiple) {
+    dispatch(changePollSettings(expiresIn, isMultiple));
+  },
+
+  onClearSuggestions () {
+    dispatch(clearComposeSuggestions());
+  },
+
+  onFetchSuggestions (token) {
+    dispatch(fetchComposeSuggestions(token));
+  },
+
+  onSuggestionSelected (position, token, accountId, path) {
+    dispatch(selectComposeSuggestion(position, token, accountId, path));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(PollForm);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/privacy_dropdown_container.js b/app/javascript/flavours/blobfox/features/compose/containers/privacy_dropdown_container.js
new file mode 100644
index 00000000000000..6d26abf4f6623b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/privacy_dropdown_container.js
@@ -0,0 +1,30 @@
+import { connect } from 'react-redux';
+
+import { changeComposeVisibility } from '../../../actions/compose';
+import { openModal, closeModal } from '../../../actions/modal';
+import { isUserTouching } from '../../../is_mobile';
+import PrivacyDropdown from '../components/privacy_dropdown';
+
+const mapStateToProps = state => ({
+  value: state.getIn(['compose', 'privacy']),
+});
+
+const mapDispatchToProps = dispatch => ({
+
+  onChange (value) {
+    dispatch(changeComposeVisibility(value));
+  },
+
+  isUserTouching,
+  onModalOpen: props => dispatch(openModal({
+    modalType: 'ACTIONS',
+    modalProps: props,
+  })),
+  onModalClose: () => dispatch(closeModal({
+    modalType: undefined,
+    ignoreFocus: false,
+  })),
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(PrivacyDropdown);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/reply_indicator_container.js b/app/javascript/flavours/blobfox/features/compose/containers/reply_indicator_container.js
new file mode 100644
index 00000000000000..678124b2a8d93c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/reply_indicator_container.js
@@ -0,0 +1,33 @@
+import { connect } from 'react-redux';
+
+import { cancelReplyCompose } from '../../../actions/compose';
+import ReplyIndicator from '../components/reply_indicator';
+
+const makeMapStateToProps = () => {
+  const mapStateToProps = state => {
+    let statusId = state.getIn(['compose', 'id'], null);
+    let editing  = true;
+
+    if (statusId === null) {
+      statusId = state.getIn(['compose', 'in_reply_to']);
+      editing  = false;
+    }
+
+    return {
+      status: state.getIn(['statuses', statusId]),
+      editing,
+    };
+  };
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = dispatch => ({
+
+  onCancel () {
+    dispatch(cancelReplyCompose());
+  },
+
+});
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(ReplyIndicator);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/search_container.js b/app/javascript/flavours/blobfox/features/compose/containers/search_container.js
new file mode 100644
index 00000000000000..a0217e3bd0febc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/search_container.js
@@ -0,0 +1,53 @@
+import { connect } from 'react-redux';
+
+import {
+  changeSearch,
+  clearSearch,
+  submitSearch,
+  showSearch,
+  openURL,
+  clickSearchResult,
+  forgetSearchResult,
+} from 'flavours/blobfox/actions/search';
+
+import Search from '../components/search';
+
+const mapStateToProps = state => ({
+  value: state.getIn(['search', 'value']),
+  submitted: state.getIn(['search', 'submitted']),
+  recent: state.getIn(['search', 'recent']).reverse(),
+});
+
+const mapDispatchToProps = dispatch => ({
+
+  onChange (value) {
+    dispatch(changeSearch(value));
+  },
+
+  onClear () {
+    dispatch(clearSearch());
+  },
+
+  onSubmit (type) {
+    dispatch(submitSearch(type));
+  },
+
+  onShow () {
+    dispatch(showSearch());
+  },
+
+  onOpenURL (routerHistory) {
+    dispatch(openURL(routerHistory));
+  },
+
+  onClickSearchResult (q, type) {
+    dispatch(clickSearchResult(q, type));
+  },
+
+  onForgetSearchResult (q) {
+    dispatch(forgetSearchResult(q));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Search);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/search_results_container.js b/app/javascript/flavours/blobfox/features/compose/containers/search_results_container.js
new file mode 100644
index 00000000000000..3906b08a0190f4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/search_results_container.js
@@ -0,0 +1,20 @@
+import { connect } from 'react-redux';
+
+import { expandSearch } from 'flavours/blobfox/actions/search';
+import { fetchSuggestions, dismissSuggestion } from 'flavours/blobfox/actions/suggestions';
+
+import SearchResults from '../components/search_results';
+
+const mapStateToProps = state => ({
+  results: state.getIn(['search', 'results']),
+  suggestions: state.getIn(['suggestions', 'items']),
+  searchTerm: state.getIn(['search', 'searchTerm']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  fetchSuggestions: () => dispatch(fetchSuggestions()),
+  expandSearch: type => dispatch(expandSearch(type)),
+  dismissSuggestion: account => dispatch(dismissSuggestion(account.get('id'))),
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(SearchResults);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/sensitive_button_container.jsx b/app/javascript/flavours/blobfox/features/compose/containers/sensitive_button_container.jsx
new file mode 100644
index 00000000000000..96bcdc9bad7aa9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/sensitive_button_container.jsx
@@ -0,0 +1,77 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { connect } from 'react-redux';
+
+import { changeComposeSensitivity } from 'flavours/blobfox/actions/compose';
+
+const messages = defineMessages({
+  marked: {
+    id: 'compose_form.sensitive.marked',
+    defaultMessage: '{count, plural, one {Media is marked as sensitive} other {Media is marked as sensitive}}',
+  },
+  unmarked: {
+    id: 'compose_form.sensitive.unmarked',
+    defaultMessage: '{count, plural, one {Media is not marked as sensitive} other {Media is not marked as sensitive}}',
+  },
+});
+
+const mapStateToProps = state => {
+  const spoilersAlwaysOn = state.getIn(['local_settings', 'always_show_spoilers_field']);
+  const spoilerText = state.getIn(['compose', 'spoiler_text']);
+  return {
+    active: state.getIn(['compose', 'sensitive']) || (spoilersAlwaysOn && spoilerText && spoilerText.length > 0),
+    disabled: state.getIn(['compose', 'spoiler']),
+    mediaCount: state.getIn(['compose', 'media_attachments']).size,
+  };
+};
+
+const mapDispatchToProps = dispatch => ({
+
+  onClick () {
+    dispatch(changeComposeSensitivity());
+  },
+
+});
+
+class SensitiveButton extends PureComponent {
+
+  static propTypes = {
+    active: PropTypes.bool,
+    disabled: PropTypes.bool,
+    mediaCount: PropTypes.number,
+    onClick: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { active, disabled, mediaCount, onClick, intl } = this.props;
+
+    return (
+      <div className='compose-form__sensitive-button'>
+        <label className={classNames('icon-button', { active })} title={intl.formatMessage(active ? messages.marked : messages.unmarked, { count: mediaCount })}>
+          <input
+            name='mark-sensitive'
+            type='checkbox'
+            checked={active}
+            onChange={onClick}
+            disabled={disabled}
+          />
+
+          <FormattedMessage
+            id='compose_form.sensitive.hide'
+            defaultMessage='{count, plural, one {Mark media as sensitive} other {Mark media as sensitive}}'
+            values={{ count: mediaCount }}
+          />
+        </label>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(SensitiveButton));
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/upload_container.js b/app/javascript/flavours/blobfox/features/compose/containers/upload_container.js
new file mode 100644
index 00000000000000..77bb90db87b91a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/upload_container.js
@@ -0,0 +1,26 @@
+import { connect } from 'react-redux';
+
+import { undoUploadCompose, initMediaEditModal, submitCompose } from '../../../actions/compose';
+import Upload from '../components/upload';
+
+const mapStateToProps = (state, { id }) => ({
+  media: state.getIn(['compose', 'media_attachments']).find(item => item.get('id') === id),
+});
+
+const mapDispatchToProps = dispatch => ({
+
+  onUndo: id => {
+    dispatch(undoUploadCompose(id));
+  },
+
+  onOpenFocalPoint: id => {
+    dispatch(initMediaEditModal(id));
+  },
+
+  onSubmit (router) {
+    dispatch(submitCompose(router));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Upload);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/upload_form_container.js b/app/javascript/flavours/blobfox/features/compose/containers/upload_form_container.js
new file mode 100644
index 00000000000000..336525cf5399d6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/upload_form_container.js
@@ -0,0 +1,9 @@
+import { connect } from 'react-redux';
+
+import UploadForm from '../components/upload_form';
+
+const mapStateToProps = state => ({
+  mediaIds: state.getIn(['compose', 'media_attachments']).map(item => item.get('id')),
+});
+
+export default connect(mapStateToProps)(UploadForm);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/upload_progress_container.js b/app/javascript/flavours/blobfox/features/compose/containers/upload_progress_container.js
new file mode 100644
index 00000000000000..ffff321c3fcdf1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/upload_progress_container.js
@@ -0,0 +1,11 @@
+import { connect } from 'react-redux';
+
+import UploadProgress from '../components/upload_progress';
+
+const mapStateToProps = state => ({
+  active: state.getIn(['compose', 'is_uploading']),
+  progress: state.getIn(['compose', 'progress']),
+  isProcessing: state.getIn(['compose', 'is_processing']),
+});
+
+export default connect(mapStateToProps)(UploadProgress);
diff --git a/app/javascript/flavours/blobfox/features/compose/containers/warning_container.jsx b/app/javascript/flavours/blobfox/features/compose/containers/warning_container.jsx
new file mode 100644
index 00000000000000..4908e0634e2bc0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/containers/warning_container.jsx
@@ -0,0 +1,47 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { me } from 'flavours/blobfox/initial_state';
+import { profileLink, privacyPolicyLink } from 'flavours/blobfox/utils/backend_links';
+import { HASHTAG_PATTERN_REGEX } from 'flavours/blobfox/utils/hashtags';
+
+import Warning from '../components/warning';
+
+const mapStateToProps = state => ({
+  needsLockWarning: state.getIn(['compose', 'privacy']) === 'private' && !state.getIn(['accounts', me, 'locked']),
+  hashtagWarning: state.getIn(['compose', 'privacy']) !== 'public' && HASHTAG_PATTERN_REGEX.test(state.getIn(['compose', 'text'])),
+  directMessageWarning: state.getIn(['compose', 'privacy']) === 'direct',
+});
+
+const WarningWrapper = ({ needsLockWarning, hashtagWarning, directMessageWarning }) => {
+  if (needsLockWarning) {
+    return <Warning message={<FormattedMessage id='compose_form.lock_disclaimer' defaultMessage='Your account is not {locked}. Anyone can follow you to view your follower-only posts.' values={{ locked: <a href={profileLink}><FormattedMessage id='compose_form.lock_disclaimer.lock' defaultMessage='locked' /></a> }} />} />;
+  }
+
+  if (hashtagWarning) {
+    return <Warning message={<FormattedMessage id='compose_form.hashtag_warning' defaultMessage="This post won't be listed under any hashtag as it is unlisted. Only public posts can be searched by hashtag." />} />;
+  }
+
+  if (directMessageWarning) {
+    const message = (
+      <span>
+        <FormattedMessage id='compose_form.encryption_warning' defaultMessage='Posts on Mastodon are not end-to-end encrypted. Do not share any dangerous information over Mastodon.' /> {!!privacyPolicyLink && <a href={privacyPolicyLink} target='_blank'><FormattedMessage id='compose_form.direct_message_warning_learn_more' defaultMessage='Learn more' /></a>}
+      </span>
+    );
+
+    return <Warning message={message} />;
+  }
+
+  return null;
+};
+
+WarningWrapper.propTypes = {
+  needsLockWarning: PropTypes.bool,
+  hashtagWarning: PropTypes.bool,
+  directMessageWarning: PropTypes.bool,
+};
+
+export default connect(mapStateToProps)(WarningWrapper);
diff --git a/app/javascript/flavours/blobfox/features/compose/index.jsx b/app/javascript/flavours/blobfox/features/compose/index.jsx
new file mode 100644
index 00000000000000..34da2b82676c1c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/index.jsx
@@ -0,0 +1,122 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+
+import { connect } from 'react-redux';
+
+import spring from 'react-motion/lib/spring';
+
+import { mountCompose, unmountCompose, cycleElefriendCompose } from 'flavours/blobfox/actions/compose';
+import Column from 'flavours/blobfox/components/column';
+
+import { mascot } from '../../initial_state';
+import Motion from '../ui/util/optional_motion';
+
+import ComposeFormContainer from './containers/compose_form_container';
+import HeaderContainer from './containers/header_container';
+import NavigationContainer from './containers/navigation_container';
+import SearchContainer from './containers/search_container';
+import SearchResultsContainer from './containers/search_results_container';
+
+const messages = defineMessages({
+  compose: { id: 'navigation_bar.compose', defaultMessage: 'Compose new post' },
+});
+
+const mapStateToProps = (state, ownProps) => ({
+  elefriend: state.getIn(['compose', 'elefriend']),
+  showSearch: ownProps.multiColumn ? state.getIn(['search', 'submitted']) && !state.getIn(['search', 'hidden']) : false,
+});
+
+const mapDispatchToProps = (dispatch) => ({
+  onClickElefriend () {
+    dispatch(cycleElefriendCompose());
+  },
+
+  onMount () {
+    dispatch(mountCompose());
+  },
+
+  onUnmount () {
+    dispatch(unmountCompose());
+  },
+});
+
+class Compose extends PureComponent {
+
+  static propTypes = {
+    multiColumn: PropTypes.bool,
+    showSearch: PropTypes.bool,
+    elefriend: PropTypes.number,
+    onClickElefriend: PropTypes.func,
+    onMount: PropTypes.func,
+    onUnmount: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+  };
+
+  componentDidMount () {
+    this.props.onMount();
+  }
+
+  componentWillUnmount () {
+    this.props.onUnmount();
+  }
+
+  render () {
+    const {
+      elefriend,
+      intl,
+      multiColumn,
+      onClickElefriend,
+      showSearch,
+    } = this.props;
+    const computedClass = classNames('drawer', `mbstobon-${elefriend}`);
+
+    if (multiColumn) {
+      return (
+        <div className={computedClass} role='region' aria-label={intl.formatMessage(messages.compose)}>
+          <HeaderContainer />
+
+          {multiColumn && <SearchContainer />}
+
+          <div className='drawer__pager'>
+            <div className='drawer__inner'>
+              <NavigationContainer />
+
+              <ComposeFormContainer />
+
+              <div className='drawer__inner__mastodon'>
+                {mascot ? <img alt='' draggable='false' src={mascot} /> : <button className='mastodon' onClick={onClickElefriend} />}
+              </div>
+            </div>
+
+            <Motion defaultStyle={{ x: -100 }} style={{ x: spring(showSearch ? 0 : -100, { stiffness: 210, damping: 20 }) }}>
+              {({ x }) => (
+                <div className='drawer__inner darker' style={{ transform: `translateX(${x}%)`, visibility: x === -100 ? 'hidden' : 'visible' }}>
+                  <SearchResultsContainer />
+                </div>
+              )}
+            </Motion>
+          </div>
+        </div>
+      );
+    }
+
+    return (
+      <Column>
+        <NavigationContainer />
+        <ComposeFormContainer />
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(Compose));
diff --git a/app/javascript/flavours/blobfox/features/compose/util/counter.js b/app/javascript/flavours/blobfox/features/compose/util/counter.js
new file mode 100644
index 00000000000000..ec2431096b5a46
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/util/counter.js
@@ -0,0 +1,9 @@
+import { urlRegex } from './url_regex';
+
+const urlPlaceholder = '$2xxxxxxxxxxxxxxxxxxxxxxx';
+
+export function countableText(inputText) {
+  return inputText
+    .replace(urlRegex, urlPlaceholder)
+    .replace(/(^|[^/\w])@(([a-z0-9_]+)@[a-z0-9.-]+[a-z0-9]+)/ig, '$1@$3');
+}
diff --git a/app/javascript/flavours/blobfox/features/compose/util/url_regex.js b/app/javascript/flavours/blobfox/features/compose/util/url_regex.js
new file mode 100644
index 00000000000000..887612ae306765
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/compose/util/url_regex.js
@@ -0,0 +1,30 @@
+import regexSupplant from 'twitter-text/dist/lib/regexSupplant';
+import validDomain from 'twitter-text/dist/regexp/validDomain';
+import validPortNumber from 'twitter-text/dist/regexp/validPortNumber';
+import validUrlPath from 'twitter-text/dist/regexp/validUrlPath';
+import validUrlPrecedingChars from 'twitter-text/dist/regexp/validUrlPrecedingChars';
+import validUrlQueryChars from 'twitter-text/dist/regexp/validUrlQueryChars';
+import validUrlQueryEndingChars from 'twitter-text/dist/regexp/validUrlQueryEndingChars';
+
+// The difference with twitter-text's extractURL is that the protocol isn't
+// optional.
+
+export const urlRegex = regexSupplant(
+  '('                                                          + // $1 URL
+    '(#{validUrlPrecedingChars})'                              + // $2
+    '(https?:\\/\\/)'                                          + // $3 Protocol
+    '(#{validDomain})'                                         + // $4 Domain(s)
+    '(?::(#{validPortNumber}))?'                               + // $5 Port number (optional)
+    '(\\/#{validUrlPath}*)?'                                   + // $6 URL Path
+    '(\\?#{validUrlQueryChars}*#{validUrlQueryEndingChars})?'  + // $7 Query String
+  ')',
+  {
+    validUrlPrecedingChars,
+    validDomain,
+    validPortNumber,
+    validUrlPath,
+    validUrlQueryChars,
+    validUrlQueryEndingChars,
+  },
+  'gi',
+);
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/components/column_settings.jsx b/app/javascript/flavours/blobfox/features/direct_timeline/components/column_settings.jsx
new file mode 100644
index 00000000000000..94168d1d920a90
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/components/column_settings.jsx
@@ -0,0 +1,47 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import SettingToggle from 'flavours/blobfox/features/notifications/components/setting_toggle';
+
+import SettingText from '../../../components/setting_text';
+
+const messages = defineMessages({
+  filter_regex: { id: 'home.column_settings.filter_regex', defaultMessage: 'Filter out by regular expressions' },
+  settings: { id: 'home.settings', defaultMessage: 'Column settings' },
+});
+
+class ColumnSettings extends PureComponent {
+
+  static propTypes = {
+    settings: ImmutablePropTypes.map.isRequired,
+    onChange: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { settings, onChange, intl } = this.props;
+
+    return (
+      <div>
+        <span className='column-settings__section'><FormattedMessage id='home.column_settings.basic' defaultMessage='Basic' /></span>
+
+        <div className='column-settings__row'>
+          <SettingToggle settings={settings} settingPath={['conversations']} onChange={onChange} label={<FormattedMessage id='direct.group_by_conversations' defaultMessage='Group by conversation' />} />
+        </div>
+
+        <span className='column-settings__section'><FormattedMessage id='home.column_settings.advanced' defaultMessage='Advanced' /></span>
+
+        <div className='column-settings__row'>
+          <SettingText settings={settings} settingPath={['regex', 'body']} onChange={onChange} label={intl.formatMessage(messages.filter_regex)} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/components/conversation.jsx b/app/javascript/flavours/blobfox/features/direct_timeline/components/conversation.jsx
new file mode 100644
index 00000000000000..cdac53539e8944
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/components/conversation.jsx
@@ -0,0 +1,233 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { HotKeys } from 'react-hotkeys';
+
+import AttachmentList from 'flavours/blobfox/components/attachment_list';
+import AvatarComposite from 'flavours/blobfox/components/avatar_composite';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Permalink from 'flavours/blobfox/components/permalink';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+import StatusContent from 'flavours/blobfox/components/status_content';
+import DropdownMenuContainer from 'flavours/blobfox/containers/dropdown_menu_container';
+import { autoPlayGif } from 'flavours/blobfox/initial_state';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const messages = defineMessages({
+  more: { id: 'status.more', defaultMessage: 'More' },
+  open: { id: 'conversation.open', defaultMessage: 'View conversation' },
+  reply: { id: 'status.reply', defaultMessage: 'Reply' },
+  markAsRead: { id: 'conversation.mark_as_read', defaultMessage: 'Mark as read' },
+  delete: { id: 'conversation.delete', defaultMessage: 'Delete conversation' },
+  muteConversation: { id: 'status.mute_conversation', defaultMessage: 'Mute conversation' },
+  unmuteConversation: { id: 'status.unmute_conversation', defaultMessage: 'Unmute conversation' },
+});
+
+class Conversation extends ImmutablePureComponent {
+
+  static propTypes = {
+    conversationId: PropTypes.string.isRequired,
+    accounts: ImmutablePropTypes.list.isRequired,
+    lastStatus: ImmutablePropTypes.map,
+    unread:PropTypes.bool.isRequired,
+    scrollKey: PropTypes.string,
+    onMoveUp: PropTypes.func,
+    onMoveDown: PropTypes.func,
+    markRead: PropTypes.func.isRequired,
+    delete: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  state = {
+    isExpanded: undefined,
+  };
+
+  parseClick = (e, destination) => {
+    const { history, lastStatus, unread, markRead } = this.props;
+    if (!history) return;
+
+    if (e.button === 0 && !(e.ctrlKey || e.altKey || e.metaKey)) {
+      if (destination === undefined) {
+        if (unread) {
+          markRead();
+        }
+        destination = `/statuses/${lastStatus.get('id')}`;
+      }
+      history.push(destination);
+      e.preventDefault();
+    }
+  };
+
+  handleMouseEnter = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-original');
+    }
+  };
+
+  handleMouseLeave = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-static');
+    }
+  };
+
+  handleClick = () => {
+    if (!this.props.history) {
+      return;
+    }
+
+    const { lastStatus, unread, markRead } = this.props;
+
+    if (unread) {
+      markRead();
+    }
+
+    this.props.history.push(`/@${lastStatus.getIn(['account', 'acct'])}/${lastStatus.get('id')}`);
+  };
+
+  handleMarkAsRead = () => {
+    this.props.markRead();
+  };
+
+  handleReply = () => {
+    this.props.reply(this.props.lastStatus, this.props.history);
+  };
+
+  handleDelete = () => {
+    this.props.delete();
+  };
+
+  handleHotkeyMoveUp = () => {
+    this.props.onMoveUp(this.props.conversationId);
+  };
+
+  handleHotkeyMoveDown = () => {
+    this.props.onMoveDown(this.props.conversationId);
+  };
+
+  handleConversationMute = () => {
+    this.props.onMute(this.props.lastStatus);
+  };
+
+  handleShowMore = () => {
+    this.props.onToggleHidden(this.props.lastStatus);
+
+    if (this.props.lastStatus.get('spoiler_text')) {
+      this.setExpansion(!this.state.isExpanded);
+    }
+  };
+
+  setExpansion = value => {
+    this.setState({ isExpanded: value });
+  };
+
+  render () {
+    const { accounts, lastStatus, unread, scrollKey, intl } = this.props;
+
+    if (lastStatus === null) {
+      return null;
+    }
+
+    const isExpanded = this.props.settings.getIn(['content_warnings', 'shared_state']) ? !lastStatus.get('hidden') : this.state.isExpanded;
+
+    const menu = [
+      { text: intl.formatMessage(messages.open), action: this.handleClick },
+      null,
+    ];
+
+    menu.push({ text: intl.formatMessage(lastStatus.get('muted') ? messages.unmuteConversation : messages.muteConversation), action: this.handleConversationMute });
+
+    if (unread) {
+      menu.push({ text: intl.formatMessage(messages.markAsRead), action: this.handleMarkAsRead });
+      menu.push(null);
+    }
+
+    menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDelete });
+
+    const names = accounts.map(a => <Permalink to={`/@${a.get('acct')}`} href={a.get('url')} key={a.get('id')} title={a.get('acct')}><bdi><strong className='display-name__html' dangerouslySetInnerHTML={{ __html: a.get('display_name_html') }} /></bdi></Permalink>).reduce((prev, cur) => [prev, ', ', cur]);
+
+    const handlers = {
+      reply: this.handleReply,
+      open: this.handleClick,
+      moveUp: this.handleHotkeyMoveUp,
+      moveDown: this.handleHotkeyMoveDown,
+      toggleHidden: this.handleShowMore,
+    };
+
+    let media = null;
+    if (lastStatus.get('media_attachments').size > 0) {
+      media = <AttachmentList compact media={lastStatus.get('media_attachments')} />;
+    }
+
+    return (
+      <HotKeys handlers={handlers}>
+        <div className={classNames('conversation focusable muted', { 'conversation--unread': unread })} tabIndex={0}>
+          <div className='conversation__avatar' onClick={this.handleClick} role='presentation'>
+            <AvatarComposite accounts={accounts} size={48} />
+          </div>
+
+          <div className='conversation__content'>
+            <div className='conversation__content__info'>
+              <div className='conversation__content__relative-time'>
+                {unread && <span className='conversation__unread' />} <RelativeTimestamp timestamp={lastStatus.get('created_at')} />
+              </div>
+
+              <div className='conversation__content__names' onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave}>
+                <FormattedMessage id='conversation.with' defaultMessage='With {names}' values={{ names: <span>{names}</span> }} />
+              </div>
+            </div>
+
+            <StatusContent
+              status={lastStatus}
+              parseClick={this.parseClick}
+              expanded={isExpanded}
+              onExpandedToggle={this.handleShowMore}
+              collapsible
+              media={media}
+            />
+
+            <div className='status__action-bar'>
+              <IconButton className='status__action-bar-button' title={intl.formatMessage(messages.reply)} icon='reply' onClick={this.handleReply} />
+
+              <div className='status__action-bar-dropdown'>
+                <DropdownMenuContainer
+                  scrollKey={scrollKey}
+                  status={lastStatus}
+                  items={menu}
+                  icon='ellipsis-h'
+                  size={18}
+                  direction='right'
+                  title={intl.formatMessage(messages.more)}
+                />
+              </div>
+            </div>
+          </div>
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(Conversation));
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/components/conversations_list.jsx b/app/javascript/flavours/blobfox/features/direct_timeline/components/conversations_list.jsx
new file mode 100644
index 00000000000000..8c12ea9e5f68a2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/components/conversations_list.jsx
@@ -0,0 +1,77 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { debounce } from 'lodash';
+
+import ScrollableList from '../../../components/scrollable_list';
+import ConversationContainer from '../containers/conversation_container';
+
+export default class ConversationsList extends ImmutablePureComponent {
+
+  static propTypes = {
+    conversations: ImmutablePropTypes.list.isRequired,
+    scrollKey: PropTypes.string.isRequired,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    onLoadMore: PropTypes.func,
+  };
+
+  getCurrentIndex = id => this.props.conversations.findIndex(x => x.get('id') === id);
+
+  handleMoveUp = id => {
+    const elementIndex = this.getCurrentIndex(id) - 1;
+    this._selectChild(elementIndex, true);
+  };
+
+  handleMoveDown = id => {
+    const elementIndex = this.getCurrentIndex(id) + 1;
+    this._selectChild(elementIndex, false);
+  };
+
+  _selectChild (index, align_top) {
+    const container = this.node.node;
+    const element = container.querySelector(`article:nth-of-type(${index + 1}) .focusable`);
+
+    if (element) {
+      if (align_top && container.scrollTop > element.offsetTop) {
+        element.scrollIntoView(true);
+      } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) {
+        element.scrollIntoView(false);
+      }
+      element.focus();
+    }
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  handleLoadOlder = debounce(() => {
+    const last = this.props.conversations.last();
+
+    if (last && last.get('last_status')) {
+      this.props.onLoadMore(last.get('last_status'));
+    }
+  }, 300, { leading: true });
+
+  render () {
+    const { conversations, isLoading, onLoadMore, ...other } = this.props;
+
+    return (
+      <ScrollableList {...other} isLoading={isLoading} showLoading={isLoading && conversations.isEmpty()} onLoadMore={onLoadMore && this.handleLoadOlder} ref={this.setRef}>
+        {conversations.map(item => (
+          <ConversationContainer
+            key={item.get('id')}
+            conversationId={item.get('id')}
+            onMoveUp={this.handleMoveUp}
+            onMoveDown={this.handleMoveDown}
+            scrollKey={this.props.scrollKey}
+          />
+        ))}
+      </ScrollableList>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/containers/column_settings_container.js b/app/javascript/flavours/blobfox/features/direct_timeline/containers/column_settings_container.js
new file mode 100644
index 00000000000000..e7cdbf60187b96
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/containers/column_settings_container.js
@@ -0,0 +1,19 @@
+import { connect } from 'react-redux';
+
+import { changeSetting } from 'flavours/blobfox/actions/settings';
+
+import ColumnSettings from '../components/column_settings';
+
+const mapStateToProps = state => ({
+  settings: state.getIn(['settings', 'direct']),
+});
+
+const mapDispatchToProps = dispatch => ({
+
+  onChange (path, checked) {
+    dispatch(changeSetting(['direct', ...path], checked));
+  },
+
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/containers/conversation_container.js b/app/javascript/flavours/blobfox/features/direct_timeline/containers/conversation_container.js
new file mode 100644
index 00000000000000..ab80d0629635d4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/containers/conversation_container.js
@@ -0,0 +1,81 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { replyCompose } from 'flavours/blobfox/actions/compose';
+import { markConversationRead, deleteConversation } from 'flavours/blobfox/actions/conversations';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { muteStatus, unmuteStatus, hideStatus, revealStatus } from 'flavours/blobfox/actions/statuses';
+import { makeGetStatus } from 'flavours/blobfox/selectors';
+
+import Conversation from '../components/conversation';
+
+const messages = defineMessages({
+  replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' },
+  replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
+});
+
+const mapStateToProps = () => {
+  const getStatus = makeGetStatus();
+
+  return (state, { conversationId }) => {
+    const conversation = state.getIn(['conversations', 'items']).find(x => x.get('id') === conversationId);
+    const lastStatusId = conversation.get('last_status', null);
+
+    return {
+      accounts: conversation.get('accounts').map(accountId => state.getIn(['accounts', accountId], null)),
+      unread: conversation.get('unread'),
+      lastStatus: lastStatusId && getStatus(state, { id: lastStatusId }),
+      settings: state.get('local_settings'),
+    };
+  };
+};
+
+const mapDispatchToProps = (dispatch, { intl, conversationId }) => ({
+
+  markRead () {
+    dispatch(markConversationRead(conversationId));
+  },
+
+  reply (status, router) {
+    dispatch((_, getState) => {
+      let state = getState();
+
+      if (state.getIn(['compose', 'text']).trim().length !== 0) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: intl.formatMessage(messages.replyMessage),
+            confirm: intl.formatMessage(messages.replyConfirm),
+            onConfirm: () => dispatch(replyCompose(status, router)),
+          },
+        }));
+      } else {
+        dispatch(replyCompose(status, router));
+      }
+    });
+  },
+
+  delete () {
+    dispatch(deleteConversation(conversationId));
+  },
+
+  onMute (status) {
+    if (status.get('muted')) {
+      dispatch(unmuteStatus(status.get('id')));
+    } else {
+      dispatch(muteStatus(status.get('id')));
+    }
+  },
+
+  onToggleHidden (status) {
+    if (status.get('hidden')) {
+      dispatch(revealStatus(status.get('id')));
+    } else {
+      dispatch(hideStatus(status.get('id')));
+    }
+  },
+
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(Conversation));
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/containers/conversations_list_container.js b/app/javascript/flavours/blobfox/features/direct_timeline/containers/conversations_list_container.js
new file mode 100644
index 00000000000000..1dcd3ec1bd4ad3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/containers/conversations_list_container.js
@@ -0,0 +1,16 @@
+import { connect } from 'react-redux';
+
+import { expandConversations } from '../../../actions/conversations';
+import ConversationsList from '../components/conversations_list';
+
+const mapStateToProps = state => ({
+  conversations: state.getIn(['conversations', 'items']),
+  isLoading: state.getIn(['conversations', 'isLoading'], true),
+  hasMore: state.getIn(['conversations', 'hasMore'], false),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onLoadMore: maxId => dispatch(expandConversations({ maxId })),
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(ConversationsList);
diff --git a/app/javascript/flavours/blobfox/features/direct_timeline/index.jsx b/app/javascript/flavours/blobfox/features/direct_timeline/index.jsx
new file mode 100644
index 00000000000000..d47464f0720d7e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/direct_timeline/index.jsx
@@ -0,0 +1,165 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import { connect } from 'react-redux';
+
+import { addColumn, removeColumn, moveColumn } from 'flavours/blobfox/actions/columns';
+import { mountConversations, unmountConversations, expandConversations } from 'flavours/blobfox/actions/conversations';
+import { connectDirectStream } from 'flavours/blobfox/actions/streaming';
+import { expandDirectTimeline } from 'flavours/blobfox/actions/timelines';
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import StatusListContainer from 'flavours/blobfox/features/ui/containers/status_list_container';
+
+import ColumnSettingsContainer from './containers/column_settings_container';
+import ConversationsListContainer from './containers/conversations_list_container';
+
+const messages = defineMessages({
+  title: { id: 'column.direct', defaultMessage: 'Private mentions' },
+});
+
+const mapStateToProps = state => ({
+  hasUnread: state.getIn(['timelines', 'direct', 'unread']) > 0,
+  conversationsMode: state.getIn(['settings', 'direct', 'conversations']),
+});
+
+class DirectTimeline extends PureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    columnId: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    hasUnread: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    conversationsMode: PropTypes.bool,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('DIRECT', {}));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  componentDidMount () {
+    const { dispatch, conversationsMode } = this.props;
+
+    dispatch(mountConversations());
+
+    if (conversationsMode) {
+      dispatch(expandConversations());
+    } else {
+      dispatch(expandDirectTimeline());
+    }
+
+    this.disconnect = dispatch(connectDirectStream());
+  }
+
+  componentDidUpdate(prevProps) {
+    const { dispatch, conversationsMode } = this.props;
+
+    if (prevProps.conversationsMode && !conversationsMode) {
+      dispatch(expandDirectTimeline());
+    } else if (!prevProps.conversationsMode && conversationsMode) {
+      dispatch(expandConversations());
+    }
+  }
+
+  componentWillUnmount () {
+    this.props.dispatch(unmountConversations());
+
+    if (this.disconnect) {
+      this.disconnect();
+      this.disconnect = null;
+    }
+  }
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMoreTimeline = maxId => {
+    this.props.dispatch(expandDirectTimeline({ maxId }));
+  };
+
+  handleLoadMoreConversations = maxId => {
+    this.props.dispatch(expandConversations({ maxId }));
+  };
+
+  render () {
+    const { intl, hasUnread, columnId, multiColumn, conversationsMode } = this.props;
+    const pinned = !!columnId;
+
+    let contents;
+    if (conversationsMode) {
+      contents = (
+        <ConversationsListContainer
+          trackScroll={!pinned}
+          scrollKey={`direct_timeline-${columnId}`}
+          timelineId='direct'
+          bindToDocument={!multiColumn}
+          onLoadMore={this.handleLoadMore}
+          prepend={<div className='follow_requests-unlocked_explanation'><span><FormattedMessage id='compose_form.encryption_warning' defaultMessage='Posts on Mastodon are not end-to-end encrypted. Do not share any dangerous information over Mastodon.' /> <a href='/terms' target='_blank'><FormattedMessage id='compose_form.direct_message_warning_learn_more' defaultMessage='Learn more' /></a></span></div>}
+          alwaysPrepend
+          emptyMessage={<FormattedMessage id='empty_column.direct' defaultMessage="You don't have any private mentions yet. When you send or receive one, it will show up here." />}
+        />
+      );
+    } else {
+      contents = (
+        <StatusListContainer
+          trackScroll={!pinned}
+          scrollKey={`direct_timeline-${columnId}`}
+          timelineId='direct'
+          bindToDocument={!multiColumn}
+          onLoadMore={this.handleLoadMoreTimeline}
+          prepend={<div className='follow_requests-unlocked_explanation'><span><FormattedMessage id='compose_form.encryption_warning' defaultMessage='Posts on Mastodon are not end-to-end encrypted. Do not share any dangerous information over Mastodon.' /> <a href='/terms' target='_blank'><FormattedMessage id='compose_form.direct_message_warning_learn_more' defaultMessage='Learn more' /></a></span></div>}
+          alwaysPrepend
+          emptyMessage={<FormattedMessage id='empty_column.direct' defaultMessage="You don't have any private mentions yet. When you send or receive one, it will show up here." />}
+        />
+      );
+    }
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.title)}>
+        <ColumnHeader
+          icon='envelope'
+          active={hasUnread}
+          title={intl.formatMessage(messages.title)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+        >
+          <ColumnSettingsContainer />
+        </ColumnHeader>
+
+        {contents}
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(DirectTimeline));
diff --git a/app/javascript/flavours/blobfox/features/directory/components/account_card.jsx b/app/javascript/flavours/blobfox/features/directory/components/account_card.jsx
new file mode 100644
index 00000000000000..cdf38e06a972cc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/directory/components/account_card.jsx
@@ -0,0 +1,255 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage, injectIntl, defineMessages } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import {
+  followAccount,
+  unfollowAccount,
+  unblockAccount,
+  unmuteAccount,
+} from 'flavours/blobfox/actions/accounts';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { Button } from 'flavours/blobfox/components/button';
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Permalink from 'flavours/blobfox/components/permalink';
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+import { autoPlayGif, me, unfollowModal } from 'flavours/blobfox/initial_state';
+import { makeGetAccount } from 'flavours/blobfox/selectors';
+
+const messages = defineMessages({
+  unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' },
+  follow: { id: 'account.follow', defaultMessage: 'Follow' },
+  cancel_follow_request: { id: 'account.cancel_follow_request', defaultMessage: 'Withdraw follow request' },
+  cancelFollowRequestConfirm: { id: 'confirmations.cancel_follow_request.confirm', defaultMessage: 'Withdraw request' },
+  requested: { id: 'account.requested', defaultMessage: 'Awaiting approval. Click to cancel follow request' },
+  unblock: { id: 'account.unblock_short', defaultMessage: 'Unblock' },
+  unmute: { id: 'account.unmute_short', defaultMessage: 'Unmute' },
+  unfollowConfirm: { id: 'confirmations.unfollow.confirm', defaultMessage: 'Unfollow' },
+  edit_profile: { id: 'account.edit_profile', defaultMessage: 'Edit profile' },
+  dismissSuggestion: { id: 'suggestions.dismiss', defaultMessage: 'Dismiss suggestion' },
+});
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { id }) => ({
+    account: getAccount(state, id),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onFollow(account) {
+    if (account.getIn(['relationship', 'following'])) {
+      if (unfollowModal) {
+        dispatch(
+          openModal({
+            modalType: 'CONFIRM',
+            modalProps: {
+              message: (
+                <FormattedMessage
+                  id='confirmations.unfollow.message'
+                  defaultMessage='Are you sure you want to unfollow {name}?'
+                  values={{ name: <strong>@{account.get('acct')}</strong> }}
+                />
+              ),
+              confirm: intl.formatMessage(messages.unfollowConfirm),
+              onConfirm: () => dispatch(unfollowAccount(account.get('id'))),
+            } }),
+        );
+      } else {
+        dispatch(unfollowAccount(account.get('id')));
+      }
+    } else if (account.getIn(['relationship', 'requested'])) {
+      if (unfollowModal) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: <FormattedMessage id='confirmations.cancel_follow_request.message' defaultMessage='Are you sure you want to withdraw your request to follow {name}?' values={{ name: <strong>@{account.get('acct')}</strong> }} />,
+            confirm: intl.formatMessage(messages.cancelFollowRequestConfirm),
+            onConfirm: () => dispatch(unfollowAccount(account.get('id'))),
+          },
+        }));
+      } else {
+        dispatch(unfollowAccount(account.get('id')));
+      }
+    } else {
+      dispatch(followAccount(account.get('id')));
+    }
+  },
+
+  onBlock(account) {
+    if (account.getIn(['relationship', 'blocking'])) {
+      dispatch(unblockAccount(account.get('id')));
+    }
+  },
+
+  onMute(account) {
+    if (account.getIn(['relationship', 'muting'])) {
+      dispatch(unmuteAccount(account.get('id')));
+    }
+  },
+
+});
+
+class AccountCard extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    intl: PropTypes.object.isRequired,
+    onFollow: PropTypes.func.isRequired,
+    onBlock: PropTypes.func.isRequired,
+    onMute: PropTypes.func.isRequired,
+    onDismiss: PropTypes.func,
+  };
+
+  handleMouseEnter = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-original');
+    }
+  };
+
+  handleMouseLeave = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-static');
+    }
+  };
+
+  handleFollow = () => {
+    this.props.onFollow(this.props.account);
+  };
+
+  handleBlock = () => {
+    this.props.onBlock(this.props.account);
+  };
+
+  handleMute = () => {
+    this.props.onMute(this.props.account);
+  };
+
+  handleEditProfile = () => {
+    window.open('/settings/profile', '_blank');
+  };
+
+  handleDismiss = (e) => {
+    const { account, onDismiss } = this.props;
+    onDismiss(account.get('id'));
+
+    e.preventDefault();
+    e.stopPropagation();
+  };
+
+  render() {
+    const { account, intl } = this.props;
+
+    let actionBtn;
+
+    if (me !== account.get('id')) {
+      if (!account.get('relationship')) { // Wait until the relationship is loaded
+        actionBtn = '';
+      } else if (account.getIn(['relationship', 'requested'])) {
+        actionBtn = <Button  text={intl.formatMessage(messages.cancel_follow_request)} title={intl.formatMessage(messages.requested)} onClick={this.handleFollow} />;
+      } else if (account.getIn(['relationship', 'muting'])) {
+        actionBtn = <Button  text={intl.formatMessage(messages.unmute)} onClick={this.handleMute} />;
+      } else if (!account.getIn(['relationship', 'blocking'])) {
+        actionBtn = <Button disabled={account.getIn(['relationship', 'blocked_by'])} className={classNames({ 'button--destructive': account.getIn(['relationship', 'following']) })} text={intl.formatMessage(account.getIn(['relationship', 'following']) ? messages.unfollow : messages.follow)} onClick={this.handleFollow} />;
+      } else if (account.getIn(['relationship', 'blocking'])) {
+        actionBtn = <Button  text={intl.formatMessage(messages.unblock)} onClick={this.handleBlock} />;
+      }
+    } else {
+      actionBtn = <Button  text={intl.formatMessage(messages.edit_profile)} onClick={this.handleEditProfile} />;
+    }
+
+    return (
+      <div className='account-card'>
+        <Permalink href={account.get('url')} to={`/@${account.get('acct')}`} className='account-card__permalink'>
+          <div className='account-card__header'>
+            {this.props.onDismiss && <IconButton className='media-modal__close' title={intl.formatMessage(messages.dismissSuggestion)} icon='times' onClick={this.handleDismiss} size={20} />}
+
+            <img
+              src={
+                autoPlayGif ? account.get('header') : account.get('header_static')
+              }
+              alt=''
+            />
+          </div>
+
+          <div className='account-card__title'>
+            <div className='account-card__title__avatar'><Avatar account={account} size={56} /></div>
+            <DisplayName account={account} />
+          </div>
+        </Permalink>
+
+        {account.get('note').length > 0 && (
+          <div
+            className='account-card__bio translate'
+            onMouseEnter={this.handleMouseEnter}
+            onMouseLeave={this.handleMouseLeave}
+            dangerouslySetInnerHTML={{ __html: account.get('note_emojified') }}
+          />
+        )}
+
+        <div className='account-card__actions'>
+          <div className='account-card__counters'>
+            <div className='account-card__counters__item'>
+              <ShortNumber value={account.get('statuses_count')} />
+              <small>
+                <FormattedMessage id='account.posts' defaultMessage='Posts' />
+              </small>
+            </div>
+
+            <div className='account-card__counters__item'>
+              {account.get('followers_count') < 0 ? '-' : <ShortNumber value={account.get('followers_count')} />}{' '}
+              <small>
+                <FormattedMessage
+                  id='account.followers'
+                  defaultMessage='Followers'
+                />
+              </small>
+            </div>
+
+            <div className='account-card__counters__item'>
+              <ShortNumber value={account.get('following_count')} />{' '}
+              <small>
+                <FormattedMessage
+                  id='account.following'
+                  defaultMessage='Following'
+                />
+              </small>
+            </div>
+          </div>
+
+          <div className='account-card__actions__button'>
+            {actionBtn}
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(AccountCard));
diff --git a/app/javascript/flavours/blobfox/features/directory/index.jsx b/app/javascript/flavours/blobfox/features/directory/index.jsx
new file mode 100644
index 00000000000000..95caaae1211629
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/directory/index.jsx
@@ -0,0 +1,179 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { addColumn, removeColumn, moveColumn, changeColumnParams } from 'flavours/blobfox/actions/columns';
+import { fetchDirectory, expandDirectory } from 'flavours/blobfox/actions/directory';
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import { LoadMore } from 'flavours/blobfox/components/load_more';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import { RadioButton } from 'flavours/blobfox/components/radio_button';
+import ScrollContainer from 'flavours/blobfox/containers/scroll_container';
+
+import AccountCard from './components/account_card';
+
+const messages = defineMessages({
+  title: { id: 'column.directory', defaultMessage: 'Browse profiles' },
+  recentlyActive: { id: 'directory.recently_active', defaultMessage: 'Recently active' },
+  newArrivals: { id: 'directory.new_arrivals', defaultMessage: 'New arrivals' },
+  local: { id: 'directory.local', defaultMessage: 'From {domain} only' },
+  federated: { id: 'directory.federated', defaultMessage: 'From known fediverse' },
+});
+
+const mapStateToProps = state => ({
+  accountIds: state.getIn(['user_lists', 'directory', 'items'], ImmutableList()),
+  isLoading: state.getIn(['user_lists', 'directory', 'isLoading'], true),
+  domain: state.getIn(['meta', 'domain']),
+});
+
+class Directory extends PureComponent {
+
+  static propTypes = {
+    isLoading: PropTypes.bool,
+    accountIds: ImmutablePropTypes.list.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    columnId: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+    domain: PropTypes.string.isRequired,
+    params: PropTypes.shape({
+      order: PropTypes.string,
+      local: PropTypes.bool,
+    }),
+  };
+
+  state = {
+    order: null,
+    local: null,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('DIRECTORY', this.getParams(this.props, this.state)));
+    }
+  };
+
+  getParams = (props, state) => ({
+    order: state.order === null ? (props.params.order || 'active') : state.order,
+    local: state.local === null ? (props.params.local || false) : state.local,
+  });
+
+  handleMove = dir => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(fetchDirectory(this.getParams(this.props, this.state)));
+  }
+
+  componentDidUpdate (prevProps, prevState) {
+    const { dispatch } = this.props;
+    const paramsOld = this.getParams(prevProps, prevState);
+    const paramsNew = this.getParams(this.props, this.state);
+
+    if (paramsOld.order !== paramsNew.order || paramsOld.local !== paramsNew.local) {
+      dispatch(fetchDirectory(paramsNew));
+    }
+  }
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleChangeOrder = e => {
+    const { dispatch, columnId } = this.props;
+
+    if (columnId) {
+      dispatch(changeColumnParams(columnId, ['order'], e.target.value));
+    } else {
+      this.setState({ order: e.target.value });
+    }
+  };
+
+  handleChangeLocal = e => {
+    const { dispatch, columnId } = this.props;
+
+    if (columnId) {
+      dispatch(changeColumnParams(columnId, ['local'], e.target.value === '1'));
+    } else {
+      this.setState({ local: e.target.value === '1' });
+    }
+  };
+
+  handleLoadMore = () => {
+    const { dispatch } = this.props;
+    dispatch(expandDirectory(this.getParams(this.props, this.state)));
+  };
+
+  render () {
+    const { isLoading, accountIds, intl, columnId, multiColumn, domain } = this.props;
+    const { order, local }  = this.getParams(this.props, this.state);
+    const pinned = !!columnId;
+
+    const scrollableArea = (
+      <div className='scrollable'>
+        <div className='filter-form'>
+          <div className='filter-form__column' role='group'>
+            <RadioButton name='order' value='active' label={intl.formatMessage(messages.recentlyActive)} checked={order === 'active'} onChange={this.handleChangeOrder} />
+            <RadioButton name='order' value='new' label={intl.formatMessage(messages.newArrivals)} checked={order === 'new'} onChange={this.handleChangeOrder} />
+          </div>
+
+          <div className='filter-form__column' role='group'>
+            <RadioButton name='local' value='1' label={intl.formatMessage(messages.local, { domain })} checked={local} onChange={this.handleChangeLocal} />
+            <RadioButton name='local' value='0' label={intl.formatMessage(messages.federated)} checked={!local} onChange={this.handleChangeLocal} />
+          </div>
+        </div>
+
+        <div className='directory__list'>
+          {isLoading ? <LoadingIndicator /> : accountIds.map(accountId => (
+            <AccountCard id={accountId} key={accountId} />
+          ))}
+        </div>
+
+        <LoadMore onClick={this.handleLoadMore} visible={!isLoading} />
+      </div>
+    );
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.title)}>
+        <ColumnHeader
+          icon='address-book-o'
+          title={intl.formatMessage(messages.title)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+        />
+
+        {multiColumn && !pinned ? <ScrollContainer scrollKey='directory'>{scrollableArea}</ScrollContainer> : scrollableArea}
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Directory));
diff --git a/app/javascript/flavours/blobfox/features/domain_blocks/index.jsx b/app/javascript/flavours/blobfox/features/domain_blocks/index.jsx
new file mode 100644
index 00000000000000..9e63b2f8170570
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/domain_blocks/index.jsx
@@ -0,0 +1,87 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchDomainBlocks, expandDomainBlocks } from '../../actions/domain_blocks';
+import ColumnBackButtonSlim from '../../components/column_back_button_slim';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import ScrollableList from '../../components/scrollable_list';
+import DomainContainer from '../../containers/domain_container';
+import Column from '../ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'column.domain_blocks', defaultMessage: 'Blocked domains' },
+  unblockDomain: { id: 'account.unblock_domain', defaultMessage: 'Unblock domain {domain}' },
+});
+
+const mapStateToProps = state => ({
+  domains: state.getIn(['domain_lists', 'blocks', 'items']),
+  hasMore: !!state.getIn(['domain_lists', 'blocks', 'next']),
+});
+
+class Blocks extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    hasMore: PropTypes.bool,
+    domains: ImmutablePropTypes.orderedSet,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchDomainBlocks());
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandDomainBlocks());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, domains, hasMore, multiColumn } = this.props;
+
+    if (!domains) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    const emptyMessage = <FormattedMessage id='empty_column.domain_blocks' defaultMessage='There are no blocked domains yet.' />;
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='minus-circle' heading={intl.formatMessage(messages.heading)}>
+        <ColumnBackButtonSlim />
+
+        <ScrollableList
+          scrollKey='domain_blocks'
+          onLoadMore={this.handleLoadMore}
+          hasMore={hasMore}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {domains.map(domain =>
+            <DomainContainer key={domain} domain={domain} />,
+          )}
+        </ScrollableList>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Blocks));
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji.js b/app/javascript/flavours/blobfox/features/emoji/emoji.js
new file mode 100644
index 00000000000000..d283871211f104
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji.js
@@ -0,0 +1,166 @@
+import Trie from 'substring-trie';
+
+import { assetHost } from 'flavours/blobfox/utils/config';
+
+import { autoPlayGif, useSystemEmojiFont } from '../../initial_state';
+
+import { unicodeMapping } from './emoji_unicode_mapping_light';
+
+const trie = new Trie(Object.keys(unicodeMapping));
+
+// Convert to file names from emojis. (For different variation selector emojis)
+const emojiFilenames = (emojis) => {
+  return emojis.map(v => unicodeMapping[v].filename);
+};
+
+// Emoji requiring extra borders depending on theme
+const darkEmoji = emojiFilenames(['🎱', '🐜', '⚫', '🖤', '⬛', '◼️', '◾', '◼️', '✒️', '▪️', '💣', '🎳', '📷', '📸', '♣️', '🕶️', '✴️', '🔌', '💂‍♀️', '📽️', '🍳', '🦍', '💂', '🔪', '🕳️', '🕹️', '🕋', '🖊️', '🖋️', '💂‍♂️', '🎤', '🎓', '🎥', '🎼', '♠️', '🎩', '🦃', '📼', '📹', '🎮', '🐃', '🏴', '🐞', '🕺', '📱', '📲', '🚲']);
+const lightEmoji = emojiFilenames(['👽', '⚾', '🐔', '☁️', '💨', '🕊️', '👀', '🍥', '👻', '🐐', '❕', '❔', '⛸️', '🌩️', '🔊', '🔇', '📃', '🌧️', '🐏', '🍚', '🍙', '🐓', '🐑', '💀', '☠️', '🌨️', '🔉', '🔈', '💬', '💭', '🏐', '🏳️', '⚪', '⬜', '◽', '◻️', '▫️']);
+
+const emojiFilename = (filename) => {
+  const borderedEmoji = (document.body && document.body.classList.contains('skin-mastodon-light')) ? lightEmoji : darkEmoji;
+  return borderedEmoji.includes(filename) ? (filename + '_border') : filename;
+};
+
+const emojifyTextNode = (node, customEmojis) => {
+  const VS15 = 0xFE0E;
+  const VS16 = 0xFE0F;
+
+  let str = node.textContent;
+
+  const fragment = new DocumentFragment();
+  let i = 0;
+
+  for (;;) {
+    let unicode_emoji;
+
+    // Skip to the next potential emoji to replace (either custom emoji or custom emoji :shortcode:)
+    if (customEmojis === null) {
+      while (i < str.length && (useSystemEmojiFont || !(unicode_emoji = trie.search(str.slice(i))))) {
+        i += str.codePointAt(i) < 65536 ? 1 : 2;
+      }
+    } else {
+      while (i < str.length && str[i] !== ':' && (useSystemEmojiFont || !(unicode_emoji = trie.search(str.slice(i))))) {
+        i += str.codePointAt(i) < 65536 ? 1 : 2;
+      }
+    }
+
+    // We reached the end of the string, nothing to replace
+    if (i === str.length) {
+      break;
+    }
+
+    let rend, replacement = null;
+    if (str[i] === ':') { // Potentially the start of a custom emoji :shortcode:
+      rend = str.indexOf(':', i + 1) + 1;
+
+      // no matching ending ':', skip
+      if (!rend) {
+        i++;
+        continue;
+      }
+
+      const shortcode = str.slice(i, rend);
+      const custom_emoji = customEmojis[shortcode];
+
+      // not a recognized shortcode, skip
+      if (!custom_emoji) {
+        i++;
+        continue;
+      }
+
+      // now got a replacee as ':shortcode:'
+      // if you want additional emoji handler, add statements below which set replacement and return true.
+      const filename = autoPlayGif ? custom_emoji.url : custom_emoji.static_url;
+      replacement = document.createElement('img');
+      replacement.setAttribute('draggable', 'false');
+      replacement.setAttribute('class', 'emojione custom-emoji');
+      replacement.setAttribute('alt', shortcode);
+      replacement.setAttribute('title', shortcode);
+      replacement.setAttribute('src', filename);
+      replacement.setAttribute('data-original', custom_emoji.url);
+      replacement.setAttribute('data-static', custom_emoji.static_url);
+    } else { // start of an unicode emoji
+      rend = i + unicode_emoji.length;
+
+      // If the matched character was followed by VS15 (for selecting text presentation), skip it.
+      if (str.codePointAt(rend - 1) !== VS16 && str.codePointAt(rend) === VS15) {
+        i = rend + 1;
+        continue;
+      }
+
+      const { filename, shortCode } = unicodeMapping[unicode_emoji];
+      const title = shortCode ? `:${shortCode}:` : '';
+
+      replacement = document.createElement('img');
+      replacement.setAttribute('draggable', 'false');
+      replacement.setAttribute('class', 'emojione');
+      replacement.setAttribute('alt', unicode_emoji);
+      replacement.setAttribute('title', title);
+      replacement.setAttribute('src', `${assetHost}/emoji/${emojiFilename(filename)}.svg`);
+    }
+
+    // Add the processed-up-to-now string and the emoji replacement
+    fragment.append(document.createTextNode(str.slice(0, i)));
+    fragment.append(replacement);
+    str = str.slice(rend);
+    i = 0;
+  }
+
+  fragment.append(document.createTextNode(str));
+  node.parentElement.replaceChild(fragment, node);
+};
+
+const emojifyNode = (node, customEmojis) => {
+  for (const child of node.childNodes) {
+    switch(child.nodeType) {
+    case Node.TEXT_NODE:
+      emojifyTextNode(child, customEmojis);
+      break;
+    case Node.ELEMENT_NODE:
+      if (!child.classList.contains('invisible'))
+        emojifyNode(child, customEmojis);
+      break;
+    }
+  }
+};
+
+const emojify = (str, customEmojis = {}) => {
+  const wrapper = document.createElement('div');
+  wrapper.innerHTML = str;
+
+  if (!Object.keys(customEmojis).length)
+    customEmojis = null;
+
+  emojifyNode(wrapper, customEmojis);
+
+  return wrapper.innerHTML;
+};
+
+export default emojify;
+
+export const buildCustomEmojis = (customEmojis) => {
+  const emojis = [];
+
+  customEmojis.forEach(emoji => {
+    const shortcode = emoji.get('shortcode');
+    const url       = autoPlayGif ? emoji.get('url') : emoji.get('static_url');
+    const name      = shortcode.replace(':', '');
+
+    emojis.push({
+      id: name,
+      name,
+      short_names: [name],
+      text: '',
+      emoticons: [],
+      keywords: [name],
+      imageUrl: url,
+      custom: true,
+      customCategory: emoji.get('category'),
+    });
+  });
+
+  return emojis;
+};
+
+export const categoriesFromEmojis = customEmojis => customEmojis.reduce((set, emoji) => set.add(emoji.get('category') ? `custom-${emoji.get('category')}` : 'custom'), new Set(['custom']));
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_compressed.d.ts b/app/javascript/flavours/blobfox/features/emoji/emoji_compressed.d.ts
new file mode 100644
index 00000000000000..3b627325a09c95
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_compressed.d.ts
@@ -0,0 +1,57 @@
+import type { BaseEmoji, EmojiData, NimbleEmojiIndex } from 'emoji-mart';
+import type { Category, Data, Emoji } from 'emoji-mart/dist-es/utils/data';
+
+/*
+ * The 'search' property, although not defined in the [`Emoji`]{@link node_modules/@types/emoji-mart/dist-es/utils/data.d.ts#Emoji} type,
+ * is used in the application.
+ * This could be due to an oversight by the library maintainer.
+ * The `search` property is defined and used [here]{@link node_modules/emoji-mart/dist/utils/data.js#uncompress}.
+ */
+export type Search = string;
+/*
+ * The 'skins' property does not exist in the application data.
+ * This could be a potential area of refactoring or error handling.
+ * The non-existence of 'skins' property is evident at [this location]{@link app/javascript/flavours/blobfox/features/emoji/emoji_compressed.js:121}.
+ */
+type Skins = null;
+
+type Filename = string;
+type UnicodeFilename = string;
+export type FilenameData = [
+  filename: Filename,
+  unicodeFilename?: UnicodeFilename,
+][];
+export type ShortCodesToEmojiDataKey =
+  | EmojiData['id']
+  | BaseEmoji['native']
+  | keyof NimbleEmojiIndex['emojis'];
+
+type SearchData = [
+  BaseEmoji['native'],
+  Emoji['short_names'],
+  Search,
+  Emoji['unified'],
+];
+
+export type ShortCodesToEmojiData = Record<
+  ShortCodesToEmojiDataKey,
+  [FilenameData, SearchData]
+>;
+type EmojisWithoutShortCodes = FilenameData;
+
+type EmojiCompressed = [
+  ShortCodesToEmojiData,
+  Skins,
+  Category[],
+  Data['aliases'],
+  EmojisWithoutShortCodes,
+];
+
+/*
+ * `emoji_compressed.js` uses `babel-plugin-preval`, which makes it difficult to convert to TypeScript.
+ * As a temporary solution, we are allowing a default export here to apply the TypeScript type `EmojiCompressed` to the JS file export.
+ * - {@link app/javascript/flavours/blobfox/features/emoji/emoji_compressed.js}
+ */
+declare const emojiCompressed: EmojiCompressed;
+
+export default emojiCompressed; // eslint-disable-line import/no-default-export
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_compressed.js b/app/javascript/flavours/blobfox/features/emoji/emoji_compressed.js
new file mode 100644
index 00000000000000..40cf707a03e51f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_compressed.js
@@ -0,0 +1,135 @@
+/* eslint-disable import/no-commonjs --
+   We need to use CommonJS here due to preval */
+// @preval
+// http://www.unicode.org/Public/emoji/5.0/emoji-test.txt
+// This file contains the compressed version of the emoji data from
+// both emoji_map.json and from emoji-mart's emojiIndex and data objects.
+// It's designed to be emitted in an array format to take up less space
+// over the wire.
+
+const { emojiIndex } = require('emoji-mart');
+let data = require('emoji-mart/data/all.json');
+const { uncompress: emojiMartUncompress } = require('emoji-mart/dist/utils/data');
+
+const emojiMap = require('./emoji_map.json');
+const { unicodeToFilename } = require('./unicode_to_filename');
+const { unicodeToUnifiedName } = require('./unicode_to_unified_name');
+
+
+if(data.compressed) {
+  data = emojiMartUncompress(data);
+}
+
+const emojiMartData = data;
+
+const excluded       = ['®', '©', '™'];
+const skinTones      = ['🏻', '🏼', '🏽', '🏾', '🏿'];
+const shortcodeMap   = {};
+
+const shortCodesToEmojiData = {};
+const emojisWithoutShortCodes = [];
+
+Object.keys(emojiIndex.emojis).forEach(key => {
+  let emoji = emojiIndex.emojis[key];
+
+  // Emojis with skin tone modifiers are stored like this
+  if (Object.prototype.hasOwnProperty.call(emoji, '1')) {
+    emoji = emoji['1'];
+  }
+
+  shortcodeMap[emoji.native] = emoji.id;
+});
+
+const stripModifiers = unicode => {
+  skinTones.forEach(tone => {
+    unicode = unicode.replace(tone, '');
+  });
+
+  return unicode;
+};
+
+Object.keys(emojiMap).forEach(key => {
+  if (excluded.includes(key)) {
+    delete emojiMap[key];
+    return;
+  }
+
+  const normalizedKey = stripModifiers(key);
+  let shortcode       = shortcodeMap[normalizedKey];
+
+  if (!shortcode) {
+    shortcode = shortcodeMap[normalizedKey + '\uFE0F'];
+  }
+
+  const filename = emojiMap[key];
+
+  const filenameData = [key];
+
+  if (unicodeToFilename(key) !== filename) {
+    // filename can't be derived using unicodeToFilename
+    filenameData.push(filename);
+  }
+
+  if (typeof shortcode === 'undefined') {
+    emojisWithoutShortCodes.push(filenameData);
+  } else {
+    if (!Array.isArray(shortCodesToEmojiData[shortcode])) {
+      shortCodesToEmojiData[shortcode] = [[]];
+    }
+
+    shortCodesToEmojiData[shortcode][0].push(filenameData);
+  }
+});
+
+Object.keys(emojiIndex.emojis).forEach(key => {
+  let emoji = emojiIndex.emojis[key];
+
+  // Emojis with skin tone modifiers are stored like this
+  if (Object.prototype.hasOwnProperty.call(emoji, '1')) {
+    emoji = emoji['1'];
+  }
+
+  const { native } = emoji;
+  let { short_names, search, unified } = emojiMartData.emojis[key];
+
+  if (short_names[0] !== key) {
+    throw new Error('The compressor expects the first short_code to be the ' +
+      'key. It may need to be rewritten if the emoji change such that this ' +
+      'is no longer the case.');
+  }
+
+  short_names = short_names.slice(1); // first short name can be inferred from the key
+
+  const searchData = [native, short_names, search];
+
+  if (unicodeToUnifiedName(native) !== unified) {
+    // unified name can't be derived from unicodeToUnifiedName
+    searchData.push(unified);
+  }
+
+  if (!Array.isArray(shortCodesToEmojiData[key])) {
+    shortCodesToEmojiData[key] = [[]];
+  }
+
+  shortCodesToEmojiData[key].push(searchData);
+});
+
+// JSON.parse/stringify is to emulate what @preval is doing and avoid any
+// inconsistent behavior in dev mode
+module.exports = JSON.parse(JSON.stringify([
+  shortCodesToEmojiData,
+  /*
+   * The property `skins` is not found in the current context.
+   * This could potentially lead to issues when interacting with modules or data structures
+   * that expect the presence of `skins` property.
+   * Currently, no definitions or references to `skins` property can be found in:
+   * - {@link node_modules/emoji-mart/dist/utils/data.js}
+   * - {@link node_modules/emoji-mart/data/all.json}
+   * - {@link app/javascript/flavours/blobfox/features/emoji/emoji_compressed.d.ts#Skins}
+   * Future refactorings or updates should consider adding definitions or handling for `skins` property.
+   */
+  emojiMartData.skins,
+  emojiMartData.categories,
+  emojiMartData.aliases,
+  emojisWithoutShortCodes,
+]));
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_map.json b/app/javascript/flavours/blobfox/features/emoji/emoji_map.json
new file mode 100644
index 00000000000000..64f6615b7932f4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_map.json
@@ -0,0 +1 @@
+{"😀":"1f600","😃":"1f603","😄":"1f604","😁":"1f601","😆":"1f606","😅":"1f605","🤣":"1f923","😂":"1f602","🙂":"1f642","🙃":"1f643","🫠":"1fae0","😉":"1f609","😊":"1f60a","😇":"1f607","🥰":"1f970","😍":"1f60d","🤩":"1f929","😘":"1f618","😗":"1f617","☺":"263a","😚":"1f61a","😙":"1f619","🥲":"1f972","😋":"1f60b","😛":"1f61b","😜":"1f61c","🤪":"1f92a","😝":"1f61d","🤑":"1f911","🤗":"1f917","🤭":"1f92d","🫢":"1fae2","🫣":"1fae3","🤫":"1f92b","🤔":"1f914","🫡":"1fae1","🤐":"1f910","🤨":"1f928","😐":"1f610","😑":"1f611","😶":"1f636","🫥":"1fae5","😏":"1f60f","😒":"1f612","🙄":"1f644","😬":"1f62c","🤥":"1f925","😌":"1f60c","😔":"1f614","😪":"1f62a","🤤":"1f924","😴":"1f634","😷":"1f637","🤒":"1f912","🤕":"1f915","🤢":"1f922","🤮":"1f92e","🤧":"1f927","🥵":"1f975","🥶":"1f976","🥴":"1f974","😵":"1f635","🤯":"1f92f","🤠":"1f920","🥳":"1f973","🥸":"1f978","😎":"1f60e","🤓":"1f913","🧐":"1f9d0","😕":"1f615","🫤":"1fae4","😟":"1f61f","🙁":"1f641","☹":"2639","😮":"1f62e","😯":"1f62f","😲":"1f632","😳":"1f633","🥺":"1f97a","🥹":"1f979","😦":"1f626","😧":"1f627","😨":"1f628","😰":"1f630","😥":"1f625","😢":"1f622","😭":"1f62d","😱":"1f631","😖":"1f616","😣":"1f623","😞":"1f61e","😓":"1f613","😩":"1f629","😫":"1f62b","🥱":"1f971","😤":"1f624","😡":"1f621","😠":"1f620","🤬":"1f92c","😈":"1f608","👿":"1f47f","💀":"1f480","☠":"2620","💩":"1f4a9","🤡":"1f921","👹":"1f479","👺":"1f47a","👻":"1f47b","👽":"1f47d","👾":"1f47e","🤖":"1f916","😺":"1f63a","😸":"1f638","😹":"1f639","😻":"1f63b","😼":"1f63c","😽":"1f63d","🙀":"1f640","😿":"1f63f","😾":"1f63e","🙈":"1f648","🙉":"1f649","🙊":"1f64a","💋":"1f48b","💌":"1f48c","💘":"1f498","💝":"1f49d","💖":"1f496","💗":"1f497","💓":"1f493","💞":"1f49e","💕":"1f495","💟":"1f49f","❣":"2763","💔":"1f494","❤":"2764","🧡":"1f9e1","💛":"1f49b","💚":"1f49a","💙":"1f499","💜":"1f49c","🤎":"1f90e","🖤":"1f5a4","🤍":"1f90d","💯":"1f4af","💢":"1f4a2","💥":"1f4a5","💫":"1f4ab","💦":"1f4a6","💨":"1f4a8","🕳":"1f573","💣":"1f4a3","💬":"1f4ac","🗨":"1f5e8","🗯":"1f5ef","💭":"1f4ad","💤":"1f4a4","👋":"1f44b","🤚":"1f91a","🖐":"1f590","✋":"270b","🖖":"1f596","🫱":"1faf1","🫲":"1faf2","🫳":"1faf3","🫴":"1faf4","👌":"1f44c","🤌":"1f90c","🤏":"1f90f","✌":"270c","🤞":"1f91e","🫰":"1faf0","🤟":"1f91f","🤘":"1f918","🤙":"1f919","👈":"1f448","👉":"1f449","👆":"1f446","🖕":"1f595","👇":"1f447","☝":"261d","🫵":"1faf5","👍":"1f44d","👎":"1f44e","✊":"270a","👊":"1f44a","🤛":"1f91b","🤜":"1f91c","👏":"1f44f","🙌":"1f64c","🫶":"1faf6","👐":"1f450","🤲":"1f932","🤝":"1f91d","🙏":"1f64f","✍":"270d","💅":"1f485","🤳":"1f933","💪":"1f4aa","🦾":"1f9be","🦿":"1f9bf","🦵":"1f9b5","🦶":"1f9b6","👂":"1f442","🦻":"1f9bb","👃":"1f443","🧠":"1f9e0","🫀":"1fac0","🫁":"1fac1","🦷":"1f9b7","🦴":"1f9b4","👀":"1f440","👁":"1f441","👅":"1f445","👄":"1f444","🫦":"1fae6","👶":"1f476","🧒":"1f9d2","👦":"1f466","👧":"1f467","🧑":"1f9d1","👱":"1f471","👨":"1f468","🧔":"1f9d4","👩":"1f469","🧓":"1f9d3","👴":"1f474","👵":"1f475","🙍":"1f64d","🙎":"1f64e","🙅":"1f645","🙆":"1f646","💁":"1f481","🙋":"1f64b","🧏":"1f9cf","🙇":"1f647","🤦":"1f926","🤷":"1f937","👮":"1f46e","🕵":"1f575","💂":"1f482","🥷":"1f977","👷":"1f477","🫅":"1fac5","🤴":"1f934","👸":"1f478","👳":"1f473","👲":"1f472","🧕":"1f9d5","🤵":"1f935","👰":"1f470","🤰":"1f930","🫃":"1fac3","🫄":"1fac4","🤱":"1f931","👼":"1f47c","🎅":"1f385","🤶":"1f936","🦸":"1f9b8","🦹":"1f9b9","🧙":"1f9d9","🧚":"1f9da","🧛":"1f9db","🧜":"1f9dc","🧝":"1f9dd","🧞":"1f9de","🧟":"1f9df","🧌":"1f9cc","💆":"1f486","💇":"1f487","🚶":"1f6b6","🧍":"1f9cd","🧎":"1f9ce","🏃":"1f3c3","💃":"1f483","🕺":"1f57a","🕴":"1f574","👯":"1f46f","🧖":"1f9d6","🧗":"1f9d7","🤺":"1f93a","🏇":"1f3c7","⛷":"26f7","🏂":"1f3c2","🏌":"1f3cc","🏄":"1f3c4","🚣":"1f6a3","🏊":"1f3ca","⛹":"26f9","🏋":"1f3cb","🚴":"1f6b4","🚵":"1f6b5","🤸":"1f938","🤼":"1f93c","🤽":"1f93d","🤾":"1f93e","🤹":"1f939","🧘":"1f9d8","🛀":"1f6c0","🛌":"1f6cc","👭":"1f46d","👫":"1f46b","👬":"1f46c","💏":"1f48f","💑":"1f491","👪":"1f46a","🗣":"1f5e3","👤":"1f464","👥":"1f465","🫂":"1fac2","👣":"1f463","🏻":"1f463","🏼":"1f463","🏽":"1f463","🏾":"1f463","🏿":"1f463","🦰":"1f463","🦱":"1f463","🦳":"1f463","🦲":"1f463","🐵":"1f435","🐒":"1f412","🦍":"1f98d","🦧":"1f9a7","🐶":"1f436","🐕":"1f415","🦮":"1f9ae","🐩":"1f429","🐺":"1f43a","🦊":"1f98a","🦝":"1f99d","🐱":"1f431","🐈":"1f408","🦁":"1f981","🐯":"1f42f","🐅":"1f405","🐆":"1f406","🐴":"1f434","🐎":"1f40e","🦄":"1f984","🦓":"1f993","🦌":"1f98c","🦬":"1f9ac","🐮":"1f42e","🐂":"1f402","🐃":"1f403","🐄":"1f404","🐷":"1f437","🐖":"1f416","🐗":"1f417","🐽":"1f43d","🐏":"1f40f","🐑":"1f411","🐐":"1f410","🐪":"1f42a","🐫":"1f42b","🦙":"1f999","🦒":"1f992","🐘":"1f418","🦣":"1f9a3","🦏":"1f98f","🦛":"1f99b","🐭":"1f42d","🐁":"1f401","🐀":"1f400","🐹":"1f439","🐰":"1f430","🐇":"1f407","🐿":"1f43f","🦫":"1f9ab","🦔":"1f994","🦇":"1f987","🐻":"1f43b","🐨":"1f428","🐼":"1f43c","🦥":"1f9a5","🦦":"1f9a6","🦨":"1f9a8","🦘":"1f998","🦡":"1f9a1","🐾":"1f43e","🦃":"1f983","🐔":"1f414","🐓":"1f413","🐣":"1f423","🐤":"1f424","🐥":"1f425","🐦":"1f426","🐧":"1f427","🕊":"1f54a","🦅":"1f985","🦆":"1f986","🦢":"1f9a2","🦉":"1f989","🦤":"1f9a4","🪶":"1fab6","🦩":"1f9a9","🦚":"1f99a","🦜":"1f99c","🐸":"1f438","🐊":"1f40a","🐢":"1f422","🦎":"1f98e","🐍":"1f40d","🐲":"1f432","🐉":"1f409","🦕":"1f995","🦖":"1f996","🐳":"1f433","🐋":"1f40b","🐬":"1f42c","🦭":"1f9ad","🐟":"1f41f","🐠":"1f420","🐡":"1f421","🦈":"1f988","🐙":"1f419","🐚":"1f41a","🪸":"1fab8","🐌":"1f40c","🦋":"1f98b","🐛":"1f41b","🐜":"1f41c","🐝":"1f41d","🪲":"1fab2","🐞":"1f41e","🦗":"1f997","🪳":"1fab3","🕷":"1f577","🕸":"1f578","🦂":"1f982","🦟":"1f99f","🪰":"1fab0","🪱":"1fab1","🦠":"1f9a0","💐":"1f490","🌸":"1f338","💮":"1f4ae","🪷":"1fab7","🏵":"1f3f5","🌹":"1f339","🥀":"1f940","🌺":"1f33a","🌻":"1f33b","🌼":"1f33c","🌷":"1f337","🌱":"1f331","🪴":"1fab4","🌲":"1f332","🌳":"1f333","🌴":"1f334","🌵":"1f335","🌾":"1f33e","🌿":"1f33f","☘":"2618","🍀":"1f340","🍁":"1f341","🍂":"1f342","🍃":"1f343","🪹":"1fab9","🪺":"1faba","🍇":"1f347","🍈":"1f348","🍉":"1f349","🍊":"1f34a","🍋":"1f34b","🍌":"1f34c","🍍":"1f34d","🥭":"1f96d","🍎":"1f34e","🍏":"1f34f","🍐":"1f350","🍑":"1f351","🍒":"1f352","🍓":"1f353","🫐":"1fad0","🥝":"1f95d","🍅":"1f345","🫒":"1fad2","🥥":"1f965","🥑":"1f951","🍆":"1f346","🥔":"1f954","🥕":"1f955","🌽":"1f33d","🌶":"1f336","🫑":"1fad1","🥒":"1f952","🥬":"1f96c","🥦":"1f966","🧄":"1f9c4","🧅":"1f9c5","🍄":"1f344","🥜":"1f95c","🫘":"1fad8","🌰":"1f330","🍞":"1f35e","🥐":"1f950","🥖":"1f956","🫓":"1fad3","🥨":"1f968","🥯":"1f96f","🥞":"1f95e","🧇":"1f9c7","🧀":"1f9c0","🍖":"1f356","🍗":"1f357","🥩":"1f969","🥓":"1f953","🍔":"1f354","🍟":"1f35f","🍕":"1f355","🌭":"1f32d","🥪":"1f96a","🌮":"1f32e","🌯":"1f32f","🫔":"1fad4","🥙":"1f959","🧆":"1f9c6","🥚":"1f95a","🍳":"1f373","🥘":"1f958","🍲":"1f372","🫕":"1fad5","🥣":"1f963","🥗":"1f957","🍿":"1f37f","🧈":"1f9c8","🧂":"1f9c2","🥫":"1f96b","🍱":"1f371","🍘":"1f358","🍙":"1f359","🍚":"1f35a","🍛":"1f35b","🍜":"1f35c","🍝":"1f35d","🍠":"1f360","🍢":"1f362","🍣":"1f363","🍤":"1f364","🍥":"1f365","🥮":"1f96e","🍡":"1f361","🥟":"1f95f","🥠":"1f960","🥡":"1f961","🦀":"1f980","🦞":"1f99e","🦐":"1f990","🦑":"1f991","🦪":"1f9aa","🍦":"1f366","🍧":"1f367","🍨":"1f368","🍩":"1f369","🍪":"1f36a","🎂":"1f382","🍰":"1f370","🧁":"1f9c1","🥧":"1f967","🍫":"1f36b","🍬":"1f36c","🍭":"1f36d","🍮":"1f36e","🍯":"1f36f","🍼":"1f37c","🥛":"1f95b","☕":"2615","🫖":"1fad6","🍵":"1f375","🍶":"1f376","🍾":"1f37e","🍷":"1f377","🍸":"1f378","🍹":"1f379","🍺":"1f37a","🍻":"1f37b","🥂":"1f942","🥃":"1f943","🫗":"1fad7","🥤":"1f964","🧋":"1f9cb","🧃":"1f9c3","🧉":"1f9c9","🧊":"1f9ca","🥢":"1f962","🍽":"1f37d","🍴":"1f374","🥄":"1f944","🔪":"1f52a","🫙":"1fad9","🏺":"1f3fa","🌍":"1f30d","🌎":"1f30e","🌏":"1f30f","🌐":"1f310","🗺":"1f5fa","🗾":"1f5fe","🧭":"1f9ed","🏔":"1f3d4","⛰":"26f0","🌋":"1f30b","🗻":"1f5fb","🏕":"1f3d5","🏖":"1f3d6","🏜":"1f3dc","🏝":"1f3dd","🏞":"1f3de","🏟":"1f3df","🏛":"1f3db","🏗":"1f3d7","🧱":"1f9f1","🪨":"1faa8","🪵":"1fab5","🛖":"1f6d6","🏘":"1f3d8","🏚":"1f3da","🏠":"1f3e0","🏡":"1f3e1","🏢":"1f3e2","🏣":"1f3e3","🏤":"1f3e4","🏥":"1f3e5","🏦":"1f3e6","🏨":"1f3e8","🏩":"1f3e9","🏪":"1f3ea","🏫":"1f3eb","🏬":"1f3ec","🏭":"1f3ed","🏯":"1f3ef","🏰":"1f3f0","💒":"1f492","🗼":"1f5fc","🗽":"1f5fd","⛪":"26ea","🕌":"1f54c","🛕":"1f6d5","🕍":"1f54d","⛩":"26e9","🕋":"1f54b","⛲":"26f2","⛺":"26fa","🌁":"1f301","🌃":"1f303","🏙":"1f3d9","🌄":"1f304","🌅":"1f305","🌆":"1f306","🌇":"1f307","🌉":"1f309","♨":"2668","🎠":"1f3a0","🛝":"1f6dd","🎡":"1f3a1","🎢":"1f3a2","💈":"1f488","🎪":"1f3aa","🚂":"1f682","🚃":"1f683","🚄":"1f684","🚅":"1f685","🚆":"1f686","🚇":"1f687","🚈":"1f688","🚉":"1f689","🚊":"1f68a","🚝":"1f69d","🚞":"1f69e","🚋":"1f68b","🚌":"1f68c","🚍":"1f68d","🚎":"1f68e","🚐":"1f690","🚑":"1f691","🚒":"1f692","🚓":"1f693","🚔":"1f694","🚕":"1f695","🚖":"1f696","🚗":"1f697","🚘":"1f698","🚙":"1f699","🛻":"1f6fb","🚚":"1f69a","🚛":"1f69b","🚜":"1f69c","🏎":"1f3ce","🏍":"1f3cd","🛵":"1f6f5","🦽":"1f9bd","🦼":"1f9bc","🛺":"1f6fa","🚲":"1f6b2","🛴":"1f6f4","🛹":"1f6f9","🛼":"1f6fc","🚏":"1f68f","🛣":"1f6e3","🛤":"1f6e4","🛢":"1f6e2","⛽":"26fd","🛞":"1f6de","🚨":"1f6a8","🚥":"1f6a5","🚦":"1f6a6","🛑":"1f6d1","🚧":"1f6a7","⚓":"2693","🛟":"1f6df","⛵":"26f5","🛶":"1f6f6","🚤":"1f6a4","🛳":"1f6f3","⛴":"26f4","🛥":"1f6e5","🚢":"1f6a2","✈":"2708","🛩":"1f6e9","🛫":"1f6eb","🛬":"1f6ec","🪂":"1fa82","💺":"1f4ba","🚁":"1f681","🚟":"1f69f","🚠":"1f6a0","🚡":"1f6a1","🛰":"1f6f0","🚀":"1f680","🛸":"1f6f8","🛎":"1f6ce","🧳":"1f9f3","⌛":"231b","⏳":"23f3","⌚":"231a","⏰":"23f0","⏱":"23f1","⏲":"23f2","🕰":"1f570","🕛":"1f55b","🕧":"1f567","🕐":"1f550","🕜":"1f55c","🕑":"1f551","🕝":"1f55d","🕒":"1f552","🕞":"1f55e","🕓":"1f553","🕟":"1f55f","🕔":"1f554","🕠":"1f560","🕕":"1f555","🕡":"1f561","🕖":"1f556","🕢":"1f562","🕗":"1f557","🕣":"1f563","🕘":"1f558","🕤":"1f564","🕙":"1f559","🕥":"1f565","🕚":"1f55a","🕦":"1f566","🌑":"1f311","🌒":"1f312","🌓":"1f313","🌔":"1f314","🌕":"1f315","🌖":"1f316","🌗":"1f317","🌘":"1f318","🌙":"1f319","🌚":"1f31a","🌛":"1f31b","🌜":"1f31c","🌡":"1f321","☀":"2600","🌝":"1f31d","🌞":"1f31e","🪐":"1fa90","⭐":"2b50","🌟":"1f31f","🌠":"1f320","🌌":"1f30c","☁":"2601","⛅":"26c5","⛈":"26c8","🌤":"1f324","🌥":"1f325","🌦":"1f326","🌧":"1f327","🌨":"1f328","🌩":"1f329","🌪":"1f32a","🌫":"1f32b","🌬":"1f32c","🌀":"1f300","🌈":"1f308","🌂":"1f302","☂":"2602","☔":"2614","⛱":"26f1","⚡":"26a1","❄":"2744","☃":"2603","⛄":"26c4","☄":"2604","🔥":"1f525","💧":"1f4a7","🌊":"1f30a","🎃":"1f383","🎄":"1f384","🎆":"1f386","🎇":"1f387","🧨":"1f9e8","✨":"2728","🎈":"1f388","🎉":"1f389","🎊":"1f38a","🎋":"1f38b","🎍":"1f38d","🎎":"1f38e","🎏":"1f38f","🎐":"1f390","🎑":"1f391","🧧":"1f9e7","🎀":"1f380","🎁":"1f381","🎗":"1f397","🎟":"1f39f","🎫":"1f3ab","🎖":"1f396","🏆":"1f3c6","🏅":"1f3c5","🥇":"1f947","🥈":"1f948","🥉":"1f949","⚽":"26bd","⚾":"26be","🥎":"1f94e","🏀":"1f3c0","🏐":"1f3d0","🏈":"1f3c8","🏉":"1f3c9","🎾":"1f3be","🥏":"1f94f","🎳":"1f3b3","🏏":"1f3cf","🏑":"1f3d1","🏒":"1f3d2","🥍":"1f94d","🏓":"1f3d3","🏸":"1f3f8","🥊":"1f94a","🥋":"1f94b","🥅":"1f945","⛳":"26f3","⛸":"26f8","🎣":"1f3a3","🤿":"1f93f","🎽":"1f3bd","🎿":"1f3bf","🛷":"1f6f7","🥌":"1f94c","🎯":"1f3af","🪀":"1fa80","🪁":"1fa81","🎱":"1f3b1","🔮":"1f52e","🪄":"1fa84","🧿":"1f9ff","🪬":"1faac","🎮":"1f3ae","🕹":"1f579","🎰":"1f3b0","🎲":"1f3b2","🧩":"1f9e9","🧸":"1f9f8","🪅":"1fa85","🪩":"1faa9","🪆":"1fa86","♠":"2660","♥":"2665","♦":"2666","♣":"2663","♟":"265f","🃏":"1f0cf","🀄":"1f004","🎴":"1f3b4","🎭":"1f3ad","🖼":"1f5bc","🎨":"1f3a8","🧵":"1f9f5","🪡":"1faa1","🧶":"1f9f6","🪢":"1faa2","👓":"1f453","🕶":"1f576","🥽":"1f97d","🥼":"1f97c","🦺":"1f9ba","👔":"1f454","👕":"1f455","👖":"1f456","🧣":"1f9e3","🧤":"1f9e4","🧥":"1f9e5","🧦":"1f9e6","👗":"1f457","👘":"1f458","🥻":"1f97b","🩱":"1fa71","🩲":"1fa72","🩳":"1fa73","👙":"1f459","👚":"1f45a","👛":"1f45b","👜":"1f45c","👝":"1f45d","🛍":"1f6cd","🎒":"1f392","🩴":"1fa74","👞":"1f45e","👟":"1f45f","🥾":"1f97e","🥿":"1f97f","👠":"1f460","👡":"1f461","🩰":"1fa70","👢":"1f462","👑":"1f451","👒":"1f452","🎩":"1f3a9","🎓":"1f393","🧢":"1f9e2","🪖":"1fa96","⛑":"26d1","📿":"1f4ff","💄":"1f484","💍":"1f48d","💎":"1f48e","🔇":"1f507","🔈":"1f508","🔉":"1f509","🔊":"1f50a","📢":"1f4e2","📣":"1f4e3","📯":"1f4ef","🔔":"1f514","🔕":"1f515","🎼":"1f3bc","🎵":"1f3b5","🎶":"1f3b6","🎙":"1f399","🎚":"1f39a","🎛":"1f39b","🎤":"1f3a4","🎧":"1f3a7","📻":"1f4fb","🎷":"1f3b7","🪗":"1fa97","🎸":"1f3b8","🎹":"1f3b9","🎺":"1f3ba","🎻":"1f3bb","🪕":"1fa95","🥁":"1f941","🪘":"1fa98","📱":"1f4f1","📲":"1f4f2","☎":"260e","📞":"1f4de","📟":"1f4df","📠":"1f4e0","🔋":"1f50b","🪫":"1faab","🔌":"1f50c","💻":"1f4bb","🖥":"1f5a5","🖨":"1f5a8","⌨":"2328","🖱":"1f5b1","🖲":"1f5b2","💽":"1f4bd","💾":"1f4be","💿":"1f4bf","📀":"1f4c0","🧮":"1f9ee","🎥":"1f3a5","🎞":"1f39e","📽":"1f4fd","🎬":"1f3ac","📺":"1f4fa","📷":"1f4f7","📸":"1f4f8","📹":"1f4f9","📼":"1f4fc","🔍":"1f50d","🔎":"1f50e","🕯":"1f56f","💡":"1f4a1","🔦":"1f526","🏮":"1f3ee","🪔":"1fa94","📔":"1f4d4","📕":"1f4d5","📖":"1f4d6","📗":"1f4d7","📘":"1f4d8","📙":"1f4d9","📚":"1f4da","📓":"1f4d3","📒":"1f4d2","📃":"1f4c3","📜":"1f4dc","📄":"1f4c4","📰":"1f4f0","🗞":"1f5de","📑":"1f4d1","🔖":"1f516","🏷":"1f3f7","💰":"1f4b0","🪙":"1fa99","💴":"1f4b4","💵":"1f4b5","💶":"1f4b6","💷":"1f4b7","💸":"1f4b8","💳":"1f4b3","🧾":"1f9fe","💹":"1f4b9","✉":"2709","📧":"1f4e7","📨":"1f4e8","📩":"1f4e9","📤":"1f4e4","📥":"1f4e5","📦":"1f4e6","📫":"1f4eb","📪":"1f4ea","📬":"1f4ec","📭":"1f4ed","📮":"1f4ee","🗳":"1f5f3","✏":"270f","✒":"2712","🖋":"1f58b","🖊":"1f58a","🖌":"1f58c","🖍":"1f58d","📝":"1f4dd","💼":"1f4bc","📁":"1f4c1","📂":"1f4c2","🗂":"1f5c2","📅":"1f4c5","📆":"1f4c6","🗒":"1f5d2","🗓":"1f5d3","📇":"1f4c7","📈":"1f4c8","📉":"1f4c9","📊":"1f4ca","📋":"1f4cb","📌":"1f4cc","📍":"1f4cd","📎":"1f4ce","🖇":"1f587","📏":"1f4cf","📐":"1f4d0","✂":"2702","🗃":"1f5c3","🗄":"1f5c4","🗑":"1f5d1","🔒":"1f512","🔓":"1f513","🔏":"1f50f","🔐":"1f510","🔑":"1f511","🗝":"1f5dd","🔨":"1f528","🪓":"1fa93","⛏":"26cf","⚒":"2692","🛠":"1f6e0","🗡":"1f5e1","⚔":"2694","🔫":"1f52b","🪃":"1fa83","🏹":"1f3f9","🛡":"1f6e1","🪚":"1fa9a","🔧":"1f527","🪛":"1fa9b","🔩":"1f529","⚙":"2699","🗜":"1f5dc","⚖":"2696","🦯":"1f9af","🔗":"1f517","⛓":"26d3","🪝":"1fa9d","🧰":"1f9f0","🧲":"1f9f2","🪜":"1fa9c","⚗":"2697","🧪":"1f9ea","🧫":"1f9eb","🧬":"1f9ec","🔬":"1f52c","🔭":"1f52d","📡":"1f4e1","💉":"1f489","🩸":"1fa78","💊":"1f48a","🩹":"1fa79","🩼":"1fa7c","🩺":"1fa7a","🩻":"1fa7b","🚪":"1f6aa","🛗":"1f6d7","🪞":"1fa9e","🪟":"1fa9f","🛏":"1f6cf","🛋":"1f6cb","🪑":"1fa91","🚽":"1f6bd","🪠":"1faa0","🚿":"1f6bf","🛁":"1f6c1","🪤":"1faa4","🪒":"1fa92","🧴":"1f9f4","🧷":"1f9f7","🧹":"1f9f9","🧺":"1f9fa","🧻":"1f9fb","🪣":"1faa3","🧼":"1f9fc","🫧":"1fae7","🪥":"1faa5","🧽":"1f9fd","🧯":"1f9ef","🛒":"1f6d2","🚬":"1f6ac","⚰":"26b0","🪦":"1faa6","⚱":"26b1","🗿":"1f5ff","🪧":"1faa7","🪪":"1faaa","🏧":"1f3e7","🚮":"1f6ae","🚰":"1f6b0","♿":"267f","🚹":"1f6b9","🚺":"1f6ba","🚻":"1f6bb","🚼":"1f6bc","🚾":"1f6be","🛂":"1f6c2","🛃":"1f6c3","🛄":"1f6c4","🛅":"1f6c5","⚠":"26a0","🚸":"1f6b8","⛔":"26d4","🚫":"1f6ab","🚳":"1f6b3","🚭":"1f6ad","🚯":"1f6af","🚱":"1f6b1","🚷":"1f6b7","📵":"1f4f5","🔞":"1f51e","☢":"2622","☣":"2623","⬆":"2b06","↗":"2197","➡":"27a1","↘":"2198","⬇":"2b07","↙":"2199","⬅":"2b05","↖":"2196","↕":"2195","↔":"2194","↩":"21a9","↪":"21aa","⤴":"2934","⤵":"2935","🔃":"1f503","🔄":"1f504","🔙":"1f519","🔚":"1f51a","🔛":"1f51b","🔜":"1f51c","🔝":"1f51d","🛐":"1f6d0","⚛":"269b","🕉":"1f549","✡":"2721","☸":"2638","☯":"262f","✝":"271d","☦":"2626","☪":"262a","☮":"262e","🕎":"1f54e","🔯":"1f52f","♈":"2648","♉":"2649","♊":"264a","♋":"264b","♌":"264c","♍":"264d","♎":"264e","♏":"264f","♐":"2650","♑":"2651","♒":"2652","♓":"2653","⛎":"26ce","🔀":"1f500","🔁":"1f501","🔂":"1f502","▶":"25b6","⏩":"23e9","⏭":"23ed","⏯":"23ef","◀":"25c0","⏪":"23ea","⏮":"23ee","🔼":"1f53c","⏫":"23eb","🔽":"1f53d","⏬":"23ec","⏸":"23f8","⏹":"23f9","⏺":"23fa","⏏":"23cf","🎦":"1f3a6","🔅":"1f505","🔆":"1f506","📶":"1f4f6","📳":"1f4f3","📴":"1f4f4","♀":"2640","♂":"2642","⚧":"26a7","✖":"2716","➕":"2795","➖":"2796","➗":"2797","🟰":"1f7f0","♾":"267e","‼":"203c","⁉":"2049","❓":"2753","❔":"2754","❕":"2755","❗":"2757","〰":"3030","💱":"1f4b1","💲":"1f4b2","⚕":"2695","♻":"267b","⚜":"269c","🔱":"1f531","📛":"1f4db","🔰":"1f530","⭕":"2b55","✅":"2705","☑":"2611","✔":"2714","❌":"274c","❎":"274e","➰":"27b0","➿":"27bf","〽":"303d","✳":"2733","✴":"2734","❇":"2747","©":"a9","®":"ae","™":"2122","🔟":"1f51f","🔠":"1f520","🔡":"1f521","🔢":"1f522","🔣":"1f523","🔤":"1f524","🅰":"1f170","🆎":"1f18e","🅱":"1f171","🆑":"1f191","🆒":"1f192","🆓":"1f193","ℹ":"2139","🆔":"1f194","Ⓜ":"24c2","🆕":"1f195","🆖":"1f196","🅾":"1f17e","🆗":"1f197","🅿":"1f17f","🆘":"1f198","🆙":"1f199","🆚":"1f19a","🈁":"1f201","🈂":"1f202","🈷":"1f237","🈶":"1f236","🈯":"1f22f","🉐":"1f250","🈹":"1f239","🈚":"1f21a","🈲":"1f232","🉑":"1f251","🈸":"1f238","🈴":"1f234","🈳":"1f233","㊗":"3297","㊙":"3299","🈺":"1f23a","🈵":"1f235","🔴":"1f534","🟠":"1f7e0","🟡":"1f7e1","🟢":"1f7e2","🔵":"1f535","🟣":"1f7e3","🟤":"1f7e4","⚫":"26ab","⚪":"26aa","🟥":"1f7e5","🟧":"1f7e7","🟨":"1f7e8","🟩":"1f7e9","🟦":"1f7e6","🟪":"1f7ea","🟫":"1f7eb","⬛":"2b1b","⬜":"2b1c","◼":"25fc","◻":"25fb","◾":"25fe","◽":"25fd","▪":"25aa","▫":"25ab","🔶":"1f536","🔷":"1f537","🔸":"1f538","🔹":"1f539","🔺":"1f53a","🔻":"1f53b","💠":"1f4a0","🔘":"1f518","🔳":"1f533","🔲":"1f532","🏁":"1f3c1","🚩":"1f6a9","🎌":"1f38c","🏴":"1f3f4","🏳":"1f3f3","☺️":"263a","☹️":"2639","☠️":"2620","❣️":"2763","❤️":"2764","🕳️":"1f573","🗨️":"1f5e8","🗯️":"1f5ef","👋🏻":"1f44b-1f3fb","👋🏼":"1f44b-1f3fc","👋🏽":"1f44b-1f3fd","👋🏾":"1f44b-1f3fe","👋🏿":"1f44b-1f3ff","🤚🏻":"1f91a-1f3fb","🤚🏼":"1f91a-1f3fc","🤚🏽":"1f91a-1f3fd","🤚🏾":"1f91a-1f3fe","🤚🏿":"1f91a-1f3ff","🖐️":"1f590","🖐🏻":"1f590-1f3fb","🖐🏼":"1f590-1f3fc","🖐🏽":"1f590-1f3fd","🖐🏾":"1f590-1f3fe","🖐🏿":"1f590-1f3ff","✋🏻":"270b-1f3fb","✋🏼":"270b-1f3fc","✋🏽":"270b-1f3fd","✋🏾":"270b-1f3fe","✋🏿":"270b-1f3ff","🖖🏻":"1f596-1f3fb","🖖🏼":"1f596-1f3fc","🖖🏽":"1f596-1f3fd","🖖🏾":"1f596-1f3fe","🖖🏿":"1f596-1f3ff","🫱🏻":"1faf1-1f3fb","🫱🏼":"1faf1-1f3fc","🫱🏽":"1faf1-1f3fd","🫱🏾":"1faf1-1f3fe","🫱🏿":"1faf1-1f3ff","🫲🏻":"1faf2-1f3fb","🫲🏼":"1faf2-1f3fc","🫲🏽":"1faf2-1f3fd","🫲🏾":"1faf2-1f3fe","🫲🏿":"1faf2-1f3ff","🫳🏻":"1faf3-1f3fb","🫳🏼":"1faf3-1f3fc","🫳🏽":"1faf3-1f3fd","🫳🏾":"1faf3-1f3fe","🫳🏿":"1faf3-1f3ff","🫴🏻":"1faf4-1f3fb","🫴🏼":"1faf4-1f3fc","🫴🏽":"1faf4-1f3fd","🫴🏾":"1faf4-1f3fe","🫴🏿":"1faf4-1f3ff","👌🏻":"1f44c-1f3fb","👌🏼":"1f44c-1f3fc","👌🏽":"1f44c-1f3fd","👌🏾":"1f44c-1f3fe","👌🏿":"1f44c-1f3ff","🤌🏻":"1f90c-1f3fb","🤌🏼":"1f90c-1f3fc","🤌🏽":"1f90c-1f3fd","🤌🏾":"1f90c-1f3fe","🤌🏿":"1f90c-1f3ff","🤏🏻":"1f90f-1f3fb","🤏🏼":"1f90f-1f3fc","🤏🏽":"1f90f-1f3fd","🤏🏾":"1f90f-1f3fe","🤏🏿":"1f90f-1f3ff","✌️":"270c","✌🏻":"270c-1f3fb","✌🏼":"270c-1f3fc","✌🏽":"270c-1f3fd","✌🏾":"270c-1f3fe","✌🏿":"270c-1f3ff","🤞🏻":"1f91e-1f3fb","🤞🏼":"1f91e-1f3fc","🤞🏽":"1f91e-1f3fd","🤞🏾":"1f91e-1f3fe","🤞🏿":"1f91e-1f3ff","🫰🏻":"1faf0-1f3fb","🫰🏼":"1faf0-1f3fc","🫰🏽":"1faf0-1f3fd","🫰🏾":"1faf0-1f3fe","🫰🏿":"1faf0-1f3ff","🤟🏻":"1f91f-1f3fb","🤟🏼":"1f91f-1f3fc","🤟🏽":"1f91f-1f3fd","🤟🏾":"1f91f-1f3fe","🤟🏿":"1f91f-1f3ff","🤘🏻":"1f918-1f3fb","🤘🏼":"1f918-1f3fc","🤘🏽":"1f918-1f3fd","🤘🏾":"1f918-1f3fe","🤘🏿":"1f918-1f3ff","🤙🏻":"1f919-1f3fb","🤙🏼":"1f919-1f3fc","🤙🏽":"1f919-1f3fd","🤙🏾":"1f919-1f3fe","🤙🏿":"1f919-1f3ff","👈🏻":"1f448-1f3fb","👈🏼":"1f448-1f3fc","👈🏽":"1f448-1f3fd","👈🏾":"1f448-1f3fe","👈🏿":"1f448-1f3ff","👉🏻":"1f449-1f3fb","👉🏼":"1f449-1f3fc","👉🏽":"1f449-1f3fd","👉🏾":"1f449-1f3fe","👉🏿":"1f449-1f3ff","👆🏻":"1f446-1f3fb","👆🏼":"1f446-1f3fc","👆🏽":"1f446-1f3fd","👆🏾":"1f446-1f3fe","👆🏿":"1f446-1f3ff","🖕🏻":"1f595-1f3fb","🖕🏼":"1f595-1f3fc","🖕🏽":"1f595-1f3fd","🖕🏾":"1f595-1f3fe","🖕🏿":"1f595-1f3ff","👇🏻":"1f447-1f3fb","👇🏼":"1f447-1f3fc","👇🏽":"1f447-1f3fd","👇🏾":"1f447-1f3fe","👇🏿":"1f447-1f3ff","☝️":"261d","☝🏻":"261d-1f3fb","☝🏼":"261d-1f3fc","☝🏽":"261d-1f3fd","☝🏾":"261d-1f3fe","☝🏿":"261d-1f3ff","🫵🏻":"1faf5-1f3fb","🫵🏼":"1faf5-1f3fc","🫵🏽":"1faf5-1f3fd","🫵🏾":"1faf5-1f3fe","🫵🏿":"1faf5-1f3ff","👍🏻":"1f44d-1f3fb","👍🏼":"1f44d-1f3fc","👍🏽":"1f44d-1f3fd","👍🏾":"1f44d-1f3fe","👍🏿":"1f44d-1f3ff","👎🏻":"1f44e-1f3fb","👎🏼":"1f44e-1f3fc","👎🏽":"1f44e-1f3fd","👎🏾":"1f44e-1f3fe","👎🏿":"1f44e-1f3ff","✊🏻":"270a-1f3fb","✊🏼":"270a-1f3fc","✊🏽":"270a-1f3fd","✊🏾":"270a-1f3fe","✊🏿":"270a-1f3ff","👊🏻":"1f44a-1f3fb","👊🏼":"1f44a-1f3fc","👊🏽":"1f44a-1f3fd","👊🏾":"1f44a-1f3fe","👊🏿":"1f44a-1f3ff","🤛🏻":"1f91b-1f3fb","🤛🏼":"1f91b-1f3fc","🤛🏽":"1f91b-1f3fd","🤛🏾":"1f91b-1f3fe","🤛🏿":"1f91b-1f3ff","🤜🏻":"1f91c-1f3fb","🤜🏼":"1f91c-1f3fc","🤜🏽":"1f91c-1f3fd","🤜🏾":"1f91c-1f3fe","🤜🏿":"1f91c-1f3ff","👏🏻":"1f44f-1f3fb","👏🏼":"1f44f-1f3fc","👏🏽":"1f44f-1f3fd","👏🏾":"1f44f-1f3fe","👏🏿":"1f44f-1f3ff","🙌🏻":"1f64c-1f3fb","🙌🏼":"1f64c-1f3fc","🙌🏽":"1f64c-1f3fd","🙌🏾":"1f64c-1f3fe","🙌🏿":"1f64c-1f3ff","🫶🏻":"1faf6-1f3fb","🫶🏼":"1faf6-1f3fc","🫶🏽":"1faf6-1f3fd","🫶🏾":"1faf6-1f3fe","🫶🏿":"1faf6-1f3ff","👐🏻":"1f450-1f3fb","👐🏼":"1f450-1f3fc","👐🏽":"1f450-1f3fd","👐🏾":"1f450-1f3fe","👐🏿":"1f450-1f3ff","🤲🏻":"1f932-1f3fb","🤲🏼":"1f932-1f3fc","🤲🏽":"1f932-1f3fd","🤲🏾":"1f932-1f3fe","🤲🏿":"1f932-1f3ff","🤝🏻":"1f91d-1f3fb","🤝🏼":"1f91d-1f3fc","🤝🏽":"1f91d-1f3fd","🤝🏾":"1f91d-1f3fe","🤝🏿":"1f91d-1f3ff","🙏🏻":"1f64f-1f3fb","🙏🏼":"1f64f-1f3fc","🙏🏽":"1f64f-1f3fd","🙏🏾":"1f64f-1f3fe","🙏🏿":"1f64f-1f3ff","✍️":"270d","✍🏻":"270d-1f3fb","✍🏼":"270d-1f3fc","✍🏽":"270d-1f3fd","✍🏾":"270d-1f3fe","✍🏿":"270d-1f3ff","💅🏻":"1f485-1f3fb","💅🏼":"1f485-1f3fc","💅🏽":"1f485-1f3fd","💅🏾":"1f485-1f3fe","💅🏿":"1f485-1f3ff","🤳🏻":"1f933-1f3fb","🤳🏼":"1f933-1f3fc","🤳🏽":"1f933-1f3fd","🤳🏾":"1f933-1f3fe","🤳🏿":"1f933-1f3ff","💪🏻":"1f4aa-1f3fb","💪🏼":"1f4aa-1f3fc","💪🏽":"1f4aa-1f3fd","💪🏾":"1f4aa-1f3fe","💪🏿":"1f4aa-1f3ff","🦵🏻":"1f9b5-1f3fb","🦵🏼":"1f9b5-1f3fc","🦵🏽":"1f9b5-1f3fd","🦵🏾":"1f9b5-1f3fe","🦵🏿":"1f9b5-1f3ff","🦶🏻":"1f9b6-1f3fb","🦶🏼":"1f9b6-1f3fc","🦶🏽":"1f9b6-1f3fd","🦶🏾":"1f9b6-1f3fe","🦶🏿":"1f9b6-1f3ff","👂🏻":"1f442-1f3fb","👂🏼":"1f442-1f3fc","👂🏽":"1f442-1f3fd","👂🏾":"1f442-1f3fe","👂🏿":"1f442-1f3ff","🦻🏻":"1f9bb-1f3fb","🦻🏼":"1f9bb-1f3fc","🦻🏽":"1f9bb-1f3fd","🦻🏾":"1f9bb-1f3fe","🦻🏿":"1f9bb-1f3ff","👃🏻":"1f443-1f3fb","👃🏼":"1f443-1f3fc","👃🏽":"1f443-1f3fd","👃🏾":"1f443-1f3fe","👃🏿":"1f443-1f3ff","👁️":"1f441","👶🏻":"1f476-1f3fb","👶🏼":"1f476-1f3fc","👶🏽":"1f476-1f3fd","👶🏾":"1f476-1f3fe","👶🏿":"1f476-1f3ff","🧒🏻":"1f9d2-1f3fb","🧒🏼":"1f9d2-1f3fc","🧒🏽":"1f9d2-1f3fd","🧒🏾":"1f9d2-1f3fe","🧒🏿":"1f9d2-1f3ff","👦🏻":"1f466-1f3fb","👦🏼":"1f466-1f3fc","👦🏽":"1f466-1f3fd","👦🏾":"1f466-1f3fe","👦🏿":"1f466-1f3ff","👧🏻":"1f467-1f3fb","👧🏼":"1f467-1f3fc","👧🏽":"1f467-1f3fd","👧🏾":"1f467-1f3fe","👧🏿":"1f467-1f3ff","🧑🏻":"1f9d1-1f3fb","🧑🏼":"1f9d1-1f3fc","🧑🏽":"1f9d1-1f3fd","🧑🏾":"1f9d1-1f3fe","🧑🏿":"1f9d1-1f3ff","👱🏻":"1f471-1f3fb","👱🏼":"1f471-1f3fc","👱🏽":"1f471-1f3fd","👱🏾":"1f471-1f3fe","👱🏿":"1f471-1f3ff","👨🏻":"1f468-1f3fb","👨🏼":"1f468-1f3fc","👨🏽":"1f468-1f3fd","👨🏾":"1f468-1f3fe","👨🏿":"1f468-1f3ff","🧔🏻":"1f9d4-1f3fb","🧔🏼":"1f9d4-1f3fc","🧔🏽":"1f9d4-1f3fd","🧔🏾":"1f9d4-1f3fe","🧔🏿":"1f9d4-1f3ff","👩🏻":"1f469-1f3fb","👩🏼":"1f469-1f3fc","👩🏽":"1f469-1f3fd","👩🏾":"1f469-1f3fe","👩🏿":"1f469-1f3ff","🧓🏻":"1f9d3-1f3fb","🧓🏼":"1f9d3-1f3fc","🧓🏽":"1f9d3-1f3fd","🧓🏾":"1f9d3-1f3fe","🧓🏿":"1f9d3-1f3ff","👴🏻":"1f474-1f3fb","👴🏼":"1f474-1f3fc","👴🏽":"1f474-1f3fd","👴🏾":"1f474-1f3fe","👴🏿":"1f474-1f3ff","👵🏻":"1f475-1f3fb","👵🏼":"1f475-1f3fc","👵🏽":"1f475-1f3fd","👵🏾":"1f475-1f3fe","👵🏿":"1f475-1f3ff","🙍🏻":"1f64d-1f3fb","🙍🏼":"1f64d-1f3fc","🙍🏽":"1f64d-1f3fd","🙍🏾":"1f64d-1f3fe","🙍🏿":"1f64d-1f3ff","🙎🏻":"1f64e-1f3fb","🙎🏼":"1f64e-1f3fc","🙎🏽":"1f64e-1f3fd","🙎🏾":"1f64e-1f3fe","🙎🏿":"1f64e-1f3ff","🙅🏻":"1f645-1f3fb","🙅🏼":"1f645-1f3fc","🙅🏽":"1f645-1f3fd","🙅🏾":"1f645-1f3fe","🙅🏿":"1f645-1f3ff","🙆🏻":"1f646-1f3fb","🙆🏼":"1f646-1f3fc","🙆🏽":"1f646-1f3fd","🙆🏾":"1f646-1f3fe","🙆🏿":"1f646-1f3ff","💁🏻":"1f481-1f3fb","💁🏼":"1f481-1f3fc","💁🏽":"1f481-1f3fd","💁🏾":"1f481-1f3fe","💁🏿":"1f481-1f3ff","🙋🏻":"1f64b-1f3fb","🙋🏼":"1f64b-1f3fc","🙋🏽":"1f64b-1f3fd","🙋🏾":"1f64b-1f3fe","🙋🏿":"1f64b-1f3ff","🧏🏻":"1f9cf-1f3fb","🧏🏼":"1f9cf-1f3fc","🧏🏽":"1f9cf-1f3fd","🧏🏾":"1f9cf-1f3fe","🧏🏿":"1f9cf-1f3ff","🙇🏻":"1f647-1f3fb","🙇🏼":"1f647-1f3fc","🙇🏽":"1f647-1f3fd","🙇🏾":"1f647-1f3fe","🙇🏿":"1f647-1f3ff","🤦🏻":"1f926-1f3fb","🤦🏼":"1f926-1f3fc","🤦🏽":"1f926-1f3fd","🤦🏾":"1f926-1f3fe","🤦🏿":"1f926-1f3ff","🤷🏻":"1f937-1f3fb","🤷🏼":"1f937-1f3fc","🤷🏽":"1f937-1f3fd","🤷🏾":"1f937-1f3fe","🤷🏿":"1f937-1f3ff","👮🏻":"1f46e-1f3fb","👮🏼":"1f46e-1f3fc","👮🏽":"1f46e-1f3fd","👮🏾":"1f46e-1f3fe","👮🏿":"1f46e-1f3ff","🕵️":"1f575","🕵🏻":"1f575-1f3fb","🕵🏼":"1f575-1f3fc","🕵🏽":"1f575-1f3fd","🕵🏾":"1f575-1f3fe","🕵🏿":"1f575-1f3ff","💂🏻":"1f482-1f3fb","💂🏼":"1f482-1f3fc","💂🏽":"1f482-1f3fd","💂🏾":"1f482-1f3fe","💂🏿":"1f482-1f3ff","🥷🏻":"1f977-1f3fb","🥷🏼":"1f977-1f3fc","🥷🏽":"1f977-1f3fd","🥷🏾":"1f977-1f3fe","🥷🏿":"1f977-1f3ff","👷🏻":"1f477-1f3fb","👷🏼":"1f477-1f3fc","👷🏽":"1f477-1f3fd","👷🏾":"1f477-1f3fe","👷🏿":"1f477-1f3ff","🫅🏻":"1fac5-1f3fb","🫅🏼":"1fac5-1f3fc","🫅🏽":"1fac5-1f3fd","🫅🏾":"1fac5-1f3fe","🫅🏿":"1fac5-1f3ff","🤴🏻":"1f934-1f3fb","🤴🏼":"1f934-1f3fc","🤴🏽":"1f934-1f3fd","🤴🏾":"1f934-1f3fe","🤴🏿":"1f934-1f3ff","👸🏻":"1f478-1f3fb","👸🏼":"1f478-1f3fc","👸🏽":"1f478-1f3fd","👸🏾":"1f478-1f3fe","👸🏿":"1f478-1f3ff","👳🏻":"1f473-1f3fb","👳🏼":"1f473-1f3fc","👳🏽":"1f473-1f3fd","👳🏾":"1f473-1f3fe","👳🏿":"1f473-1f3ff","👲🏻":"1f472-1f3fb","👲🏼":"1f472-1f3fc","👲🏽":"1f472-1f3fd","👲🏾":"1f472-1f3fe","👲🏿":"1f472-1f3ff","🧕🏻":"1f9d5-1f3fb","🧕🏼":"1f9d5-1f3fc","🧕🏽":"1f9d5-1f3fd","🧕🏾":"1f9d5-1f3fe","🧕🏿":"1f9d5-1f3ff","🤵🏻":"1f935-1f3fb","🤵🏼":"1f935-1f3fc","🤵🏽":"1f935-1f3fd","🤵🏾":"1f935-1f3fe","🤵🏿":"1f935-1f3ff","👰🏻":"1f470-1f3fb","👰🏼":"1f470-1f3fc","👰🏽":"1f470-1f3fd","👰🏾":"1f470-1f3fe","👰🏿":"1f470-1f3ff","🤰🏻":"1f930-1f3fb","🤰🏼":"1f930-1f3fc","🤰🏽":"1f930-1f3fd","🤰🏾":"1f930-1f3fe","🤰🏿":"1f930-1f3ff","🫃🏻":"1fac3-1f3fb","🫃🏼":"1fac3-1f3fc","🫃🏽":"1fac3-1f3fd","🫃🏾":"1fac3-1f3fe","🫃🏿":"1fac3-1f3ff","🫄🏻":"1fac4-1f3fb","🫄🏼":"1fac4-1f3fc","🫄🏽":"1fac4-1f3fd","🫄🏾":"1fac4-1f3fe","🫄🏿":"1fac4-1f3ff","🤱🏻":"1f931-1f3fb","🤱🏼":"1f931-1f3fc","🤱🏽":"1f931-1f3fd","🤱🏾":"1f931-1f3fe","🤱🏿":"1f931-1f3ff","👼🏻":"1f47c-1f3fb","👼🏼":"1f47c-1f3fc","👼🏽":"1f47c-1f3fd","👼🏾":"1f47c-1f3fe","👼🏿":"1f47c-1f3ff","🎅🏻":"1f385-1f3fb","🎅🏼":"1f385-1f3fc","🎅🏽":"1f385-1f3fd","🎅🏾":"1f385-1f3fe","🎅🏿":"1f385-1f3ff","🤶🏻":"1f936-1f3fb","🤶🏼":"1f936-1f3fc","🤶🏽":"1f936-1f3fd","🤶🏾":"1f936-1f3fe","🤶🏿":"1f936-1f3ff","🦸🏻":"1f9b8-1f3fb","🦸🏼":"1f9b8-1f3fc","🦸🏽":"1f9b8-1f3fd","🦸🏾":"1f9b8-1f3fe","🦸🏿":"1f9b8-1f3ff","🦹🏻":"1f9b9-1f3fb","🦹🏼":"1f9b9-1f3fc","🦹🏽":"1f9b9-1f3fd","🦹🏾":"1f9b9-1f3fe","🦹🏿":"1f9b9-1f3ff","🧙🏻":"1f9d9-1f3fb","🧙🏼":"1f9d9-1f3fc","🧙🏽":"1f9d9-1f3fd","🧙🏾":"1f9d9-1f3fe","🧙🏿":"1f9d9-1f3ff","🧚🏻":"1f9da-1f3fb","🧚🏼":"1f9da-1f3fc","🧚🏽":"1f9da-1f3fd","🧚🏾":"1f9da-1f3fe","🧚🏿":"1f9da-1f3ff","🧛🏻":"1f9db-1f3fb","🧛🏼":"1f9db-1f3fc","🧛🏽":"1f9db-1f3fd","🧛🏾":"1f9db-1f3fe","🧛🏿":"1f9db-1f3ff","🧜🏻":"1f9dc-1f3fb","🧜🏼":"1f9dc-1f3fc","🧜🏽":"1f9dc-1f3fd","🧜🏾":"1f9dc-1f3fe","🧜🏿":"1f9dc-1f3ff","🧝🏻":"1f9dd-1f3fb","🧝🏼":"1f9dd-1f3fc","🧝🏽":"1f9dd-1f3fd","🧝🏾":"1f9dd-1f3fe","🧝🏿":"1f9dd-1f3ff","💆🏻":"1f486-1f3fb","💆🏼":"1f486-1f3fc","💆🏽":"1f486-1f3fd","💆🏾":"1f486-1f3fe","💆🏿":"1f486-1f3ff","💇🏻":"1f487-1f3fb","💇🏼":"1f487-1f3fc","💇🏽":"1f487-1f3fd","💇🏾":"1f487-1f3fe","💇🏿":"1f487-1f3ff","🚶🏻":"1f6b6-1f3fb","🚶🏼":"1f6b6-1f3fc","🚶🏽":"1f6b6-1f3fd","🚶🏾":"1f6b6-1f3fe","🚶🏿":"1f6b6-1f3ff","🧍🏻":"1f9cd-1f3fb","🧍🏼":"1f9cd-1f3fc","🧍🏽":"1f9cd-1f3fd","🧍🏾":"1f9cd-1f3fe","🧍🏿":"1f9cd-1f3ff","🧎🏻":"1f9ce-1f3fb","🧎🏼":"1f9ce-1f3fc","🧎🏽":"1f9ce-1f3fd","🧎🏾":"1f9ce-1f3fe","🧎🏿":"1f9ce-1f3ff","🏃🏻":"1f3c3-1f3fb","🏃🏼":"1f3c3-1f3fc","🏃🏽":"1f3c3-1f3fd","🏃🏾":"1f3c3-1f3fe","🏃🏿":"1f3c3-1f3ff","💃🏻":"1f483-1f3fb","💃🏼":"1f483-1f3fc","💃🏽":"1f483-1f3fd","💃🏾":"1f483-1f3fe","💃🏿":"1f483-1f3ff","🕺🏻":"1f57a-1f3fb","🕺🏼":"1f57a-1f3fc","🕺🏽":"1f57a-1f3fd","🕺🏾":"1f57a-1f3fe","🕺🏿":"1f57a-1f3ff","🕴️":"1f574","🕴🏻":"1f574-1f3fb","🕴🏼":"1f574-1f3fc","🕴🏽":"1f574-1f3fd","🕴🏾":"1f574-1f3fe","🕴🏿":"1f574-1f3ff","🧖🏻":"1f9d6-1f3fb","🧖🏼":"1f9d6-1f3fc","🧖🏽":"1f9d6-1f3fd","🧖🏾":"1f9d6-1f3fe","🧖🏿":"1f9d6-1f3ff","🧗🏻":"1f9d7-1f3fb","🧗🏼":"1f9d7-1f3fc","🧗🏽":"1f9d7-1f3fd","🧗🏾":"1f9d7-1f3fe","🧗🏿":"1f9d7-1f3ff","🏇🏻":"1f3c7-1f3fb","🏇🏼":"1f3c7-1f3fc","🏇🏽":"1f3c7-1f3fd","🏇🏾":"1f3c7-1f3fe","🏇🏿":"1f3c7-1f3ff","⛷️":"26f7","🏂🏻":"1f3c2-1f3fb","🏂🏼":"1f3c2-1f3fc","🏂🏽":"1f3c2-1f3fd","🏂🏾":"1f3c2-1f3fe","🏂🏿":"1f3c2-1f3ff","🏌️":"1f3cc","🏌🏻":"1f3cc-1f3fb","🏌🏼":"1f3cc-1f3fc","🏌🏽":"1f3cc-1f3fd","🏌🏾":"1f3cc-1f3fe","🏌🏿":"1f3cc-1f3ff","🏄🏻":"1f3c4-1f3fb","🏄🏼":"1f3c4-1f3fc","🏄🏽":"1f3c4-1f3fd","🏄🏾":"1f3c4-1f3fe","🏄🏿":"1f3c4-1f3ff","🚣🏻":"1f6a3-1f3fb","🚣🏼":"1f6a3-1f3fc","🚣🏽":"1f6a3-1f3fd","🚣🏾":"1f6a3-1f3fe","🚣🏿":"1f6a3-1f3ff","🏊🏻":"1f3ca-1f3fb","🏊🏼":"1f3ca-1f3fc","🏊🏽":"1f3ca-1f3fd","🏊🏾":"1f3ca-1f3fe","🏊🏿":"1f3ca-1f3ff","⛹️":"26f9","⛹🏻":"26f9-1f3fb","⛹🏼":"26f9-1f3fc","⛹🏽":"26f9-1f3fd","⛹🏾":"26f9-1f3fe","⛹🏿":"26f9-1f3ff","🏋️":"1f3cb","🏋🏻":"1f3cb-1f3fb","🏋🏼":"1f3cb-1f3fc","🏋🏽":"1f3cb-1f3fd","🏋🏾":"1f3cb-1f3fe","🏋🏿":"1f3cb-1f3ff","🚴🏻":"1f6b4-1f3fb","🚴🏼":"1f6b4-1f3fc","🚴🏽":"1f6b4-1f3fd","🚴🏾":"1f6b4-1f3fe","🚴🏿":"1f6b4-1f3ff","🚵🏻":"1f6b5-1f3fb","🚵🏼":"1f6b5-1f3fc","🚵🏽":"1f6b5-1f3fd","🚵🏾":"1f6b5-1f3fe","🚵🏿":"1f6b5-1f3ff","🤸🏻":"1f938-1f3fb","🤸🏼":"1f938-1f3fc","🤸🏽":"1f938-1f3fd","🤸🏾":"1f938-1f3fe","🤸🏿":"1f938-1f3ff","🤽🏻":"1f93d-1f3fb","🤽🏼":"1f93d-1f3fc","🤽🏽":"1f93d-1f3fd","🤽🏾":"1f93d-1f3fe","🤽🏿":"1f93d-1f3ff","🤾🏻":"1f93e-1f3fb","🤾🏼":"1f93e-1f3fc","🤾🏽":"1f93e-1f3fd","🤾🏾":"1f93e-1f3fe","🤾🏿":"1f93e-1f3ff","🤹🏻":"1f939-1f3fb","🤹🏼":"1f939-1f3fc","🤹🏽":"1f939-1f3fd","🤹🏾":"1f939-1f3fe","🤹🏿":"1f939-1f3ff","🧘🏻":"1f9d8-1f3fb","🧘🏼":"1f9d8-1f3fc","🧘🏽":"1f9d8-1f3fd","🧘🏾":"1f9d8-1f3fe","🧘🏿":"1f9d8-1f3ff","🛀🏻":"1f6c0-1f3fb","🛀🏼":"1f6c0-1f3fc","🛀🏽":"1f6c0-1f3fd","🛀🏾":"1f6c0-1f3fe","🛀🏿":"1f6c0-1f3ff","🛌🏻":"1f6cc-1f3fb","🛌🏼":"1f6cc-1f3fc","🛌🏽":"1f6cc-1f3fd","🛌🏾":"1f6cc-1f3fe","🛌🏿":"1f6cc-1f3ff","👭🏻":"1f46d-1f3fb","👭🏼":"1f46d-1f3fc","👭🏽":"1f46d-1f3fd","👭🏾":"1f46d-1f3fe","👭🏿":"1f46d-1f3ff","👫🏻":"1f46b-1f3fb","👫🏼":"1f46b-1f3fc","👫🏽":"1f46b-1f3fd","👫🏾":"1f46b-1f3fe","👫🏿":"1f46b-1f3ff","👬🏻":"1f46c-1f3fb","👬🏼":"1f46c-1f3fc","👬🏽":"1f46c-1f3fd","👬🏾":"1f46c-1f3fe","👬🏿":"1f46c-1f3ff","💏🏻":"1f48f-1f3fb","💏🏼":"1f48f-1f3fc","💏🏽":"1f48f-1f3fd","💏🏾":"1f48f-1f3fe","💏🏿":"1f48f-1f3ff","💑🏻":"1f491-1f3fb","💑🏼":"1f491-1f3fc","💑🏽":"1f491-1f3fd","💑🏾":"1f491-1f3fe","💑🏿":"1f491-1f3ff","🗣️":"1f5e3","🐿️":"1f43f","🕊️":"1f54a","🕷️":"1f577","🕸️":"1f578","🏵️":"1f3f5","☘️":"2618","🌶️":"1f336","🍽️":"1f37d","🗺️":"1f5fa","🏔️":"1f3d4","⛰️":"26f0","🏕️":"1f3d5","🏖️":"1f3d6","🏜️":"1f3dc","🏝️":"1f3dd","🏞️":"1f3de","🏟️":"1f3df","🏛️":"1f3db","🏗️":"1f3d7","🏘️":"1f3d8","🏚️":"1f3da","⛩️":"26e9","🏙️":"1f3d9","♨️":"2668","🏎️":"1f3ce","🏍️":"1f3cd","🛣️":"1f6e3","🛤️":"1f6e4","🛢️":"1f6e2","🛳️":"1f6f3","⛴️":"26f4","🛥️":"1f6e5","✈️":"2708","🛩️":"1f6e9","🛰️":"1f6f0","🛎️":"1f6ce","⏱️":"23f1","⏲️":"23f2","🕰️":"1f570","🌡️":"1f321","☀️":"2600","☁️":"2601","⛈️":"26c8","🌤️":"1f324","🌥️":"1f325","🌦️":"1f326","🌧️":"1f327","🌨️":"1f328","🌩️":"1f329","🌪️":"1f32a","🌫️":"1f32b","🌬️":"1f32c","☂️":"2602","⛱️":"26f1","❄️":"2744","☃️":"2603","☄️":"2604","🎗️":"1f397","🎟️":"1f39f","🎖️":"1f396","⛸️":"26f8","🕹️":"1f579","♠️":"2660","♥️":"2665","♦️":"2666","♣️":"2663","♟️":"265f","🖼️":"1f5bc","🕶️":"1f576","🛍️":"1f6cd","⛑️":"26d1","🎙️":"1f399","🎚️":"1f39a","🎛️":"1f39b","☎️":"260e","🖥️":"1f5a5","🖨️":"1f5a8","⌨️":"2328","🖱️":"1f5b1","🖲️":"1f5b2","🎞️":"1f39e","📽️":"1f4fd","🕯️":"1f56f","🗞️":"1f5de","🏷️":"1f3f7","✉️":"2709","🗳️":"1f5f3","✏️":"270f","✒️":"2712","🖋️":"1f58b","🖊️":"1f58a","🖌️":"1f58c","🖍️":"1f58d","🗂️":"1f5c2","🗒️":"1f5d2","🗓️":"1f5d3","🖇️":"1f587","✂️":"2702","🗃️":"1f5c3","🗄️":"1f5c4","🗑️":"1f5d1","🗝️":"1f5dd","⛏️":"26cf","⚒️":"2692","🛠️":"1f6e0","🗡️":"1f5e1","⚔️":"2694","🛡️":"1f6e1","⚙️":"2699","🗜️":"1f5dc","⚖️":"2696","⛓️":"26d3","⚗️":"2697","🛏️":"1f6cf","🛋️":"1f6cb","⚰️":"26b0","⚱️":"26b1","⚠️":"26a0","☢️":"2622","☣️":"2623","⬆️":"2b06","↗️":"2197","➡️":"27a1","↘️":"2198","⬇️":"2b07","↙️":"2199","⬅️":"2b05","↖️":"2196","↕️":"2195","↔️":"2194","↩️":"21a9","↪️":"21aa","⤴️":"2934","⤵️":"2935","⚛️":"269b","🕉️":"1f549","✡️":"2721","☸️":"2638","☯️":"262f","✝️":"271d","☦️":"2626","☪️":"262a","☮️":"262e","▶️":"25b6","⏭️":"23ed","⏯️":"23ef","◀️":"25c0","⏮️":"23ee","⏸️":"23f8","⏹️":"23f9","⏺️":"23fa","⏏️":"23cf","♀️":"2640","♂️":"2642","⚧️":"26a7","✖️":"2716","♾️":"267e","‼️":"203c","⁉️":"2049","〰️":"3030","⚕️":"2695","♻️":"267b","⚜️":"269c","☑️":"2611","✔️":"2714","〽️":"303d","✳️":"2733","✴️":"2734","❇️":"2747","©️":"a9","®️":"ae","™️":"2122","#⃣":"23-20e3","*⃣":"2a-20e3","0⃣":"30-20e3","1⃣":"31-20e3","2⃣":"32-20e3","3⃣":"33-20e3","4⃣":"34-20e3","5⃣":"35-20e3","6⃣":"36-20e3","7⃣":"37-20e3","8⃣":"38-20e3","9⃣":"39-20e3","🅰️":"1f170","🅱️":"1f171","ℹ️":"2139","Ⓜ️":"24c2","🅾️":"1f17e","🅿️":"1f17f","🈂️":"1f202","🈷️":"1f237","㊗️":"3297","㊙️":"3299","◼️":"25fc","◻️":"25fb","▪️":"25aa","▫️":"25ab","🏳️":"1f3f3","🇦🇨":"1f1e6-1f1e8","🇦🇩":"1f1e6-1f1e9","🇦🇪":"1f1e6-1f1ea","🇦🇫":"1f1e6-1f1eb","🇦🇬":"1f1e6-1f1ec","🇦🇮":"1f1e6-1f1ee","🇦🇱":"1f1e6-1f1f1","🇦🇲":"1f1e6-1f1f2","🇦🇴":"1f1e6-1f1f4","🇦🇶":"1f1e6-1f1f6","🇦🇷":"1f1e6-1f1f7","🇦🇸":"1f1e6-1f1f8","🇦🇹":"1f1e6-1f1f9","🇦🇺":"1f1e6-1f1fa","🇦🇼":"1f1e6-1f1fc","🇦🇽":"1f1e6-1f1fd","🇦🇿":"1f1e6-1f1ff","🇧🇦":"1f1e7-1f1e6","🇧🇧":"1f1e7-1f1e7","🇧🇩":"1f1e7-1f1e9","🇧🇪":"1f1e7-1f1ea","🇧🇫":"1f1e7-1f1eb","🇧🇬":"1f1e7-1f1ec","🇧🇭":"1f1e7-1f1ed","🇧🇮":"1f1e7-1f1ee","🇧🇯":"1f1e7-1f1ef","🇧🇱":"1f1e7-1f1f1","🇧🇲":"1f1e7-1f1f2","🇧🇳":"1f1e7-1f1f3","🇧🇴":"1f1e7-1f1f4","🇧🇶":"1f1e7-1f1f6","🇧🇷":"1f1e7-1f1f7","🇧🇸":"1f1e7-1f1f8","🇧🇹":"1f1e7-1f1f9","🇧🇻":"1f1e7-1f1fb","🇧🇼":"1f1e7-1f1fc","🇧🇾":"1f1e7-1f1fe","🇧🇿":"1f1e7-1f1ff","🇨🇦":"1f1e8-1f1e6","🇨🇨":"1f1e8-1f1e8","🇨🇩":"1f1e8-1f1e9","🇨🇫":"1f1e8-1f1eb","🇨🇬":"1f1e8-1f1ec","🇨🇭":"1f1e8-1f1ed","🇨🇮":"1f1e8-1f1ee","🇨🇰":"1f1e8-1f1f0","🇨🇱":"1f1e8-1f1f1","🇨🇲":"1f1e8-1f1f2","🇨🇳":"1f1e8-1f1f3","🇨🇴":"1f1e8-1f1f4","🇨🇵":"1f1e8-1f1f5","🇨🇷":"1f1e8-1f1f7","🇨🇺":"1f1e8-1f1fa","🇨🇻":"1f1e8-1f1fb","🇨🇼":"1f1e8-1f1fc","🇨🇽":"1f1e8-1f1fd","🇨🇾":"1f1e8-1f1fe","🇨🇿":"1f1e8-1f1ff","🇩🇪":"1f1e9-1f1ea","🇩🇬":"1f1e9-1f1ec","🇩🇯":"1f1e9-1f1ef","🇩🇰":"1f1e9-1f1f0","🇩🇲":"1f1e9-1f1f2","🇩🇴":"1f1e9-1f1f4","🇩🇿":"1f1e9-1f1ff","🇪🇦":"1f1ea-1f1e6","🇪🇨":"1f1ea-1f1e8","🇪🇪":"1f1ea-1f1ea","🇪🇬":"1f1ea-1f1ec","🇪🇭":"1f1ea-1f1ed","🇪🇷":"1f1ea-1f1f7","🇪🇸":"1f1ea-1f1f8","🇪🇹":"1f1ea-1f1f9","🇪🇺":"1f1ea-1f1fa","🇫🇮":"1f1eb-1f1ee","🇫🇯":"1f1eb-1f1ef","🇫🇰":"1f1eb-1f1f0","🇫🇲":"1f1eb-1f1f2","🇫🇴":"1f1eb-1f1f4","🇫🇷":"1f1eb-1f1f7","🇬🇦":"1f1ec-1f1e6","🇬🇧":"1f1ec-1f1e7","🇬🇩":"1f1ec-1f1e9","🇬🇪":"1f1ec-1f1ea","🇬🇫":"1f1ec-1f1eb","🇬🇬":"1f1ec-1f1ec","🇬🇭":"1f1ec-1f1ed","🇬🇮":"1f1ec-1f1ee","🇬🇱":"1f1ec-1f1f1","🇬🇲":"1f1ec-1f1f2","🇬🇳":"1f1ec-1f1f3","🇬🇵":"1f1ec-1f1f5","🇬🇶":"1f1ec-1f1f6","🇬🇷":"1f1ec-1f1f7","🇬🇸":"1f1ec-1f1f8","🇬🇹":"1f1ec-1f1f9","🇬🇺":"1f1ec-1f1fa","🇬🇼":"1f1ec-1f1fc","🇬🇾":"1f1ec-1f1fe","🇭🇰":"1f1ed-1f1f0","🇭🇲":"1f1ed-1f1f2","🇭🇳":"1f1ed-1f1f3","🇭🇷":"1f1ed-1f1f7","🇭🇹":"1f1ed-1f1f9","🇭🇺":"1f1ed-1f1fa","🇮🇨":"1f1ee-1f1e8","🇮🇩":"1f1ee-1f1e9","🇮🇪":"1f1ee-1f1ea","🇮🇱":"1f1ee-1f1f1","🇮🇲":"1f1ee-1f1f2","🇮🇳":"1f1ee-1f1f3","🇮🇴":"1f1ee-1f1f4","🇮🇶":"1f1ee-1f1f6","🇮🇷":"1f1ee-1f1f7","🇮🇸":"1f1ee-1f1f8","🇮🇹":"1f1ee-1f1f9","🇯🇪":"1f1ef-1f1ea","🇯🇲":"1f1ef-1f1f2","🇯🇴":"1f1ef-1f1f4","🇯🇵":"1f1ef-1f1f5","🇰🇪":"1f1f0-1f1ea","🇰🇬":"1f1f0-1f1ec","🇰🇭":"1f1f0-1f1ed","🇰🇮":"1f1f0-1f1ee","🇰🇲":"1f1f0-1f1f2","🇰🇳":"1f1f0-1f1f3","🇰🇵":"1f1f0-1f1f5","🇰🇷":"1f1f0-1f1f7","🇰🇼":"1f1f0-1f1fc","🇰🇾":"1f1f0-1f1fe","🇰🇿":"1f1f0-1f1ff","🇱🇦":"1f1f1-1f1e6","🇱🇧":"1f1f1-1f1e7","🇱🇨":"1f1f1-1f1e8","🇱🇮":"1f1f1-1f1ee","🇱🇰":"1f1f1-1f1f0","🇱🇷":"1f1f1-1f1f7","🇱🇸":"1f1f1-1f1f8","🇱🇹":"1f1f1-1f1f9","🇱🇺":"1f1f1-1f1fa","🇱🇻":"1f1f1-1f1fb","🇱🇾":"1f1f1-1f1fe","🇲🇦":"1f1f2-1f1e6","🇲🇨":"1f1f2-1f1e8","🇲🇩":"1f1f2-1f1e9","🇲🇪":"1f1f2-1f1ea","🇲🇫":"1f1f2-1f1eb","🇲🇬":"1f1f2-1f1ec","🇲🇭":"1f1f2-1f1ed","🇲🇰":"1f1f2-1f1f0","🇲🇱":"1f1f2-1f1f1","🇲🇲":"1f1f2-1f1f2","🇲🇳":"1f1f2-1f1f3","🇲🇴":"1f1f2-1f1f4","🇲🇵":"1f1f2-1f1f5","🇲🇶":"1f1f2-1f1f6","🇲🇷":"1f1f2-1f1f7","🇲🇸":"1f1f2-1f1f8","🇲🇹":"1f1f2-1f1f9","🇲🇺":"1f1f2-1f1fa","🇲🇻":"1f1f2-1f1fb","🇲🇼":"1f1f2-1f1fc","🇲🇽":"1f1f2-1f1fd","🇲🇾":"1f1f2-1f1fe","🇲🇿":"1f1f2-1f1ff","🇳🇦":"1f1f3-1f1e6","🇳🇨":"1f1f3-1f1e8","🇳🇪":"1f1f3-1f1ea","🇳🇫":"1f1f3-1f1eb","🇳🇬":"1f1f3-1f1ec","🇳🇮":"1f1f3-1f1ee","🇳🇱":"1f1f3-1f1f1","🇳🇴":"1f1f3-1f1f4","🇳🇵":"1f1f3-1f1f5","🇳🇷":"1f1f3-1f1f7","🇳🇺":"1f1f3-1f1fa","🇳🇿":"1f1f3-1f1ff","🇴🇲":"1f1f4-1f1f2","🇵🇦":"1f1f5-1f1e6","🇵🇪":"1f1f5-1f1ea","🇵🇫":"1f1f5-1f1eb","🇵🇬":"1f1f5-1f1ec","🇵🇭":"1f1f5-1f1ed","🇵🇰":"1f1f5-1f1f0","🇵🇱":"1f1f5-1f1f1","🇵🇲":"1f1f5-1f1f2","🇵🇳":"1f1f5-1f1f3","🇵🇷":"1f1f5-1f1f7","🇵🇸":"1f1f5-1f1f8","🇵🇹":"1f1f5-1f1f9","🇵🇼":"1f1f5-1f1fc","🇵🇾":"1f1f5-1f1fe","🇶🇦":"1f1f6-1f1e6","🇷🇪":"1f1f7-1f1ea","🇷🇴":"1f1f7-1f1f4","🇷🇸":"1f1f7-1f1f8","🇷🇺":"1f1f7-1f1fa","🇷🇼":"1f1f7-1f1fc","🇸🇦":"1f1f8-1f1e6","🇸🇧":"1f1f8-1f1e7","🇸🇨":"1f1f8-1f1e8","🇸🇩":"1f1f8-1f1e9","🇸🇪":"1f1f8-1f1ea","🇸🇬":"1f1f8-1f1ec","🇸🇭":"1f1f8-1f1ed","🇸🇮":"1f1f8-1f1ee","🇸🇯":"1f1f8-1f1ef","🇸🇰":"1f1f8-1f1f0","🇸🇱":"1f1f8-1f1f1","🇸🇲":"1f1f8-1f1f2","🇸🇳":"1f1f8-1f1f3","🇸🇴":"1f1f8-1f1f4","🇸🇷":"1f1f8-1f1f7","🇸🇸":"1f1f8-1f1f8","🇸🇹":"1f1f8-1f1f9","🇸🇻":"1f1f8-1f1fb","🇸🇽":"1f1f8-1f1fd","🇸🇾":"1f1f8-1f1fe","🇸🇿":"1f1f8-1f1ff","🇹🇦":"1f1f9-1f1e6","🇹🇨":"1f1f9-1f1e8","🇹🇩":"1f1f9-1f1e9","🇹🇫":"1f1f9-1f1eb","🇹🇬":"1f1f9-1f1ec","🇹🇭":"1f1f9-1f1ed","🇹🇯":"1f1f9-1f1ef","🇹🇰":"1f1f9-1f1f0","🇹🇱":"1f1f9-1f1f1","🇹🇲":"1f1f9-1f1f2","🇹🇳":"1f1f9-1f1f3","🇹🇴":"1f1f9-1f1f4","🇹🇷":"1f1f9-1f1f7","🇹🇹":"1f1f9-1f1f9","🇹🇻":"1f1f9-1f1fb","🇹🇼":"1f1f9-1f1fc","🇹🇿":"1f1f9-1f1ff","🇺🇦":"1f1fa-1f1e6","🇺🇬":"1f1fa-1f1ec","🇺🇲":"1f1fa-1f1f2","🇺🇳":"1f1fa-1f1f3","🇺🇸":"1f1fa-1f1f8","🇺🇾":"1f1fa-1f1fe","🇺🇿":"1f1fa-1f1ff","🇻🇦":"1f1fb-1f1e6","🇻🇨":"1f1fb-1f1e8","🇻🇪":"1f1fb-1f1ea","🇻🇬":"1f1fb-1f1ec","🇻🇮":"1f1fb-1f1ee","🇻🇳":"1f1fb-1f1f3","🇻🇺":"1f1fb-1f1fa","🇼🇫":"1f1fc-1f1eb","🇼🇸":"1f1fc-1f1f8","🇽🇰":"1f1fd-1f1f0","🇾🇪":"1f1fe-1f1ea","🇾🇹":"1f1fe-1f1f9","🇿🇦":"1f1ff-1f1e6","🇿🇲":"1f1ff-1f1f2","🇿🇼":"1f1ff-1f1fc","😶‍🌫":"1f636-200d-1f32b-fe0f","😮‍💨":"1f62e-200d-1f4a8","😵‍💫":"1f635-200d-1f4ab","❤‍🔥":"2764-fe0f-200d-1f525","❤‍🩹":"2764-fe0f-200d-1fa79","👁‍🗨":"1f441-200d-1f5e8","🧔‍♂":"1f9d4-200d-2642-fe0f","🧔‍♀":"1f9d4-200d-2640-fe0f","👨‍🦰":"1f468-200d-1f9b0","👨‍🦱":"1f468-200d-1f9b1","👨‍🦳":"1f468-200d-1f9b3","👨‍🦲":"1f468-200d-1f9b2","👩‍🦰":"1f469-200d-1f9b0","🧑‍🦰":"1f9d1-200d-1f9b0","👩‍🦱":"1f469-200d-1f9b1","🧑‍🦱":"1f9d1-200d-1f9b1","👩‍🦳":"1f469-200d-1f9b3","🧑‍🦳":"1f9d1-200d-1f9b3","👩‍🦲":"1f469-200d-1f9b2","🧑‍🦲":"1f9d1-200d-1f9b2","👱‍♀":"1f471-200d-2640-fe0f","👱‍♂":"1f471-200d-2642-fe0f","🙍‍♂":"1f64d-200d-2642-fe0f","🙍‍♀":"1f64d-200d-2640-fe0f","🙎‍♂":"1f64e-200d-2642-fe0f","🙎‍♀":"1f64e-200d-2640-fe0f","🙅‍♂":"1f645-200d-2642-fe0f","🙅‍♀":"1f645-200d-2640-fe0f","🙆‍♂":"1f646-200d-2642-fe0f","🙆‍♀":"1f646-200d-2640-fe0f","💁‍♂":"1f481-200d-2642-fe0f","💁‍♀":"1f481-200d-2640-fe0f","🙋‍♂":"1f64b-200d-2642-fe0f","🙋‍♀":"1f64b-200d-2640-fe0f","🧏‍♂":"1f9cf-200d-2642-fe0f","🧏‍♀":"1f9cf-200d-2640-fe0f","🙇‍♂":"1f647-200d-2642-fe0f","🙇‍♀":"1f647-200d-2640-fe0f","🤦‍♂":"1f926-200d-2642-fe0f","🤦‍♀":"1f926-200d-2640-fe0f","🤷‍♂":"1f937-200d-2642-fe0f","🤷‍♀":"1f937-200d-2640-fe0f","🧑‍⚕":"1f9d1-200d-2695-fe0f","👨‍⚕":"1f468-200d-2695-fe0f","👩‍⚕":"1f469-200d-2695-fe0f","🧑‍🎓":"1f9d1-200d-1f393","👨‍🎓":"1f468-200d-1f393","👩‍🎓":"1f469-200d-1f393","🧑‍🏫":"1f9d1-200d-1f3eb","👨‍🏫":"1f468-200d-1f3eb","👩‍🏫":"1f469-200d-1f3eb","🧑‍⚖":"1f9d1-200d-2696-fe0f","👨‍⚖":"1f468-200d-2696-fe0f","👩‍⚖":"1f469-200d-2696-fe0f","🧑‍🌾":"1f9d1-200d-1f33e","👨‍🌾":"1f468-200d-1f33e","👩‍🌾":"1f469-200d-1f33e","🧑‍🍳":"1f9d1-200d-1f373","👨‍🍳":"1f468-200d-1f373","👩‍🍳":"1f469-200d-1f373","🧑‍🔧":"1f9d1-200d-1f527","👨‍🔧":"1f468-200d-1f527","👩‍🔧":"1f469-200d-1f527","🧑‍🏭":"1f9d1-200d-1f3ed","👨‍🏭":"1f468-200d-1f3ed","👩‍🏭":"1f469-200d-1f3ed","🧑‍💼":"1f9d1-200d-1f4bc","👨‍💼":"1f468-200d-1f4bc","👩‍💼":"1f469-200d-1f4bc","🧑‍🔬":"1f9d1-200d-1f52c","👨‍🔬":"1f468-200d-1f52c","👩‍🔬":"1f469-200d-1f52c","🧑‍💻":"1f9d1-200d-1f4bb","👨‍💻":"1f468-200d-1f4bb","👩‍💻":"1f469-200d-1f4bb","🧑‍🎤":"1f9d1-200d-1f3a4","👨‍🎤":"1f468-200d-1f3a4","👩‍🎤":"1f469-200d-1f3a4","🧑‍🎨":"1f9d1-200d-1f3a8","👨‍🎨":"1f468-200d-1f3a8","👩‍🎨":"1f469-200d-1f3a8","🧑‍✈":"1f9d1-200d-2708-fe0f","👨‍✈":"1f468-200d-2708-fe0f","👩‍✈":"1f469-200d-2708-fe0f","🧑‍🚀":"1f9d1-200d-1f680","👨‍🚀":"1f468-200d-1f680","👩‍🚀":"1f469-200d-1f680","🧑‍🚒":"1f9d1-200d-1f692","👨‍🚒":"1f468-200d-1f692","👩‍🚒":"1f469-200d-1f692","👮‍♂":"1f46e-200d-2642-fe0f","👮‍♀":"1f46e-200d-2640-fe0f","🕵‍♂":"1f575-fe0f-200d-2642-fe0f","🕵‍♀":"1f575-fe0f-200d-2640-fe0f","💂‍♂":"1f482-200d-2642-fe0f","💂‍♀":"1f482-200d-2640-fe0f","👷‍♂":"1f477-200d-2642-fe0f","👷‍♀":"1f477-200d-2640-fe0f","👳‍♂":"1f473-200d-2642-fe0f","👳‍♀":"1f473-200d-2640-fe0f","🤵‍♂":"1f935-200d-2642-fe0f","🤵‍♀":"1f935-200d-2640-fe0f","👰‍♂":"1f470-200d-2642-fe0f","👰‍♀":"1f470-200d-2640-fe0f","👩‍🍼":"1f469-200d-1f37c","👨‍🍼":"1f468-200d-1f37c","🧑‍🍼":"1f9d1-200d-1f37c","🧑‍🎄":"1f9d1-200d-1f384","🦸‍♂":"1f9b8-200d-2642-fe0f","🦸‍♀":"1f9b8-200d-2640-fe0f","🦹‍♂":"1f9b9-200d-2642-fe0f","🦹‍♀":"1f9b9-200d-2640-fe0f","🧙‍♂":"1f9d9-200d-2642-fe0f","🧙‍♀":"1f9d9-200d-2640-fe0f","🧚‍♂":"1f9da-200d-2642-fe0f","🧚‍♀":"1f9da-200d-2640-fe0f","🧛‍♂":"1f9db-200d-2642-fe0f","🧛‍♀":"1f9db-200d-2640-fe0f","🧜‍♂":"1f9dc-200d-2642-fe0f","🧜‍♀":"1f9dc-200d-2640-fe0f","🧝‍♂":"1f9dd-200d-2642-fe0f","🧝‍♀":"1f9dd-200d-2640-fe0f","🧞‍♂":"1f9de-200d-2642-fe0f","🧞‍♀":"1f9de-200d-2640-fe0f","🧟‍♂":"1f9df-200d-2642-fe0f","🧟‍♀":"1f9df-200d-2640-fe0f","💆‍♂":"1f486-200d-2642-fe0f","💆‍♀":"1f486-200d-2640-fe0f","💇‍♂":"1f487-200d-2642-fe0f","💇‍♀":"1f487-200d-2640-fe0f","🚶‍♂":"1f6b6-200d-2642-fe0f","🚶‍♀":"1f6b6-200d-2640-fe0f","🧍‍♂":"1f9cd-200d-2642-fe0f","🧍‍♀":"1f9cd-200d-2640-fe0f","🧎‍♂":"1f9ce-200d-2642-fe0f","🧎‍♀":"1f9ce-200d-2640-fe0f","🧑‍🦯":"1f9d1-200d-1f9af","👨‍🦯":"1f468-200d-1f9af","👩‍🦯":"1f469-200d-1f9af","🧑‍🦼":"1f9d1-200d-1f9bc","👨‍🦼":"1f468-200d-1f9bc","👩‍🦼":"1f469-200d-1f9bc","🧑‍🦽":"1f9d1-200d-1f9bd","👨‍🦽":"1f468-200d-1f9bd","👩‍🦽":"1f469-200d-1f9bd","🏃‍♂":"1f3c3-200d-2642-fe0f","🏃‍♀":"1f3c3-200d-2640-fe0f","👯‍♂":"1f46f-200d-2642-fe0f","👯‍♀":"1f46f-200d-2640-fe0f","🧖‍♂":"1f9d6-200d-2642-fe0f","🧖‍♀":"1f9d6-200d-2640-fe0f","🧗‍♂":"1f9d7-200d-2642-fe0f","🧗‍♀":"1f9d7-200d-2640-fe0f","🏌‍♂":"1f3cc-fe0f-200d-2642-fe0f","🏌‍♀":"1f3cc-fe0f-200d-2640-fe0f","🏄‍♂":"1f3c4-200d-2642-fe0f","🏄‍♀":"1f3c4-200d-2640-fe0f","🚣‍♂":"1f6a3-200d-2642-fe0f","🚣‍♀":"1f6a3-200d-2640-fe0f","🏊‍♂":"1f3ca-200d-2642-fe0f","🏊‍♀":"1f3ca-200d-2640-fe0f","⛹‍♂":"26f9-fe0f-200d-2642-fe0f","⛹‍♀":"26f9-fe0f-200d-2640-fe0f","🏋‍♂":"1f3cb-fe0f-200d-2642-fe0f","🏋‍♀":"1f3cb-fe0f-200d-2640-fe0f","🚴‍♂":"1f6b4-200d-2642-fe0f","🚴‍♀":"1f6b4-200d-2640-fe0f","🚵‍♂":"1f6b5-200d-2642-fe0f","🚵‍♀":"1f6b5-200d-2640-fe0f","🤸‍♂":"1f938-200d-2642-fe0f","🤸‍♀":"1f938-200d-2640-fe0f","🤼‍♂":"1f93c-200d-2642-fe0f","🤼‍♀":"1f93c-200d-2640-fe0f","🤽‍♂":"1f93d-200d-2642-fe0f","🤽‍♀":"1f93d-200d-2640-fe0f","🤾‍♂":"1f93e-200d-2642-fe0f","🤾‍♀":"1f93e-200d-2640-fe0f","🤹‍♂":"1f939-200d-2642-fe0f","🤹‍♀":"1f939-200d-2640-fe0f","🧘‍♂":"1f9d8-200d-2642-fe0f","🧘‍♀":"1f9d8-200d-2640-fe0f","👨‍👦":"1f468-200d-1f466","👨‍👧":"1f468-200d-1f467","👩‍👦":"1f469-200d-1f466","👩‍👧":"1f469-200d-1f467","🐕‍🦺":"1f415-200d-1f9ba","🐈‍⬛":"1f408-200d-2b1b","🐻‍❄":"1f43b-200d-2744-fe0f","#️⃣":"23-20e3","*️⃣":"2a-20e3","0️⃣":"30-20e3","1️⃣":"31-20e3","2️⃣":"32-20e3","3️⃣":"33-20e3","4️⃣":"34-20e3","5️⃣":"35-20e3","6️⃣":"36-20e3","7️⃣":"37-20e3","8️⃣":"38-20e3","9️⃣":"39-20e3","🏳‍🌈":"1f3f3-fe0f-200d-1f308","🏳‍⚧":"1f3f3-fe0f-200d-26a7-fe0f","🏴‍☠":"1f3f4-200d-2620-fe0f","😶‍🌫️":"1f636-200d-1f32b-fe0f","❤️‍🔥":"2764-fe0f-200d-1f525","❤️‍🩹":"2764-fe0f-200d-1fa79","👁‍🗨️":"1f441-200d-1f5e8","👁️‍🗨":"1f441-200d-1f5e8","🧔‍♂️":"1f9d4-200d-2642-fe0f","🧔🏻‍♂":"1f9d4-1f3fb-200d-2642-fe0f","🧔🏼‍♂":"1f9d4-1f3fc-200d-2642-fe0f","🧔🏽‍♂":"1f9d4-1f3fd-200d-2642-fe0f","🧔🏾‍♂":"1f9d4-1f3fe-200d-2642-fe0f","🧔🏿‍♂":"1f9d4-1f3ff-200d-2642-fe0f","🧔‍♀️":"1f9d4-200d-2640-fe0f","🧔🏻‍♀":"1f9d4-1f3fb-200d-2640-fe0f","🧔🏼‍♀":"1f9d4-1f3fc-200d-2640-fe0f","🧔🏽‍♀":"1f9d4-1f3fd-200d-2640-fe0f","🧔🏾‍♀":"1f9d4-1f3fe-200d-2640-fe0f","🧔🏿‍♀":"1f9d4-1f3ff-200d-2640-fe0f","👨🏻‍🦰":"1f468-1f3fb-200d-1f9b0","👨🏼‍🦰":"1f468-1f3fc-200d-1f9b0","👨🏽‍🦰":"1f468-1f3fd-200d-1f9b0","👨🏾‍🦰":"1f468-1f3fe-200d-1f9b0","👨🏿‍🦰":"1f468-1f3ff-200d-1f9b0","👨🏻‍🦱":"1f468-1f3fb-200d-1f9b1","👨🏼‍🦱":"1f468-1f3fc-200d-1f9b1","👨🏽‍🦱":"1f468-1f3fd-200d-1f9b1","👨🏾‍🦱":"1f468-1f3fe-200d-1f9b1","👨🏿‍🦱":"1f468-1f3ff-200d-1f9b1","👨🏻‍🦳":"1f468-1f3fb-200d-1f9b3","👨🏼‍🦳":"1f468-1f3fc-200d-1f9b3","👨🏽‍🦳":"1f468-1f3fd-200d-1f9b3","👨🏾‍🦳":"1f468-1f3fe-200d-1f9b3","👨🏿‍🦳":"1f468-1f3ff-200d-1f9b3","👨🏻‍🦲":"1f468-1f3fb-200d-1f9b2","👨🏼‍🦲":"1f468-1f3fc-200d-1f9b2","👨🏽‍🦲":"1f468-1f3fd-200d-1f9b2","👨🏾‍🦲":"1f468-1f3fe-200d-1f9b2","👨🏿‍🦲":"1f468-1f3ff-200d-1f9b2","👩🏻‍🦰":"1f469-1f3fb-200d-1f9b0","👩🏼‍🦰":"1f469-1f3fc-200d-1f9b0","👩🏽‍🦰":"1f469-1f3fd-200d-1f9b0","👩🏾‍🦰":"1f469-1f3fe-200d-1f9b0","👩🏿‍🦰":"1f469-1f3ff-200d-1f9b0","🧑🏻‍🦰":"1f9d1-1f3fb-200d-1f9b0","🧑🏼‍🦰":"1f9d1-1f3fc-200d-1f9b0","🧑🏽‍🦰":"1f9d1-1f3fd-200d-1f9b0","🧑🏾‍🦰":"1f9d1-1f3fe-200d-1f9b0","🧑🏿‍🦰":"1f9d1-1f3ff-200d-1f9b0","👩🏻‍🦱":"1f469-1f3fb-200d-1f9b1","👩🏼‍🦱":"1f469-1f3fc-200d-1f9b1","👩🏽‍🦱":"1f469-1f3fd-200d-1f9b1","👩🏾‍🦱":"1f469-1f3fe-200d-1f9b1","👩🏿‍🦱":"1f469-1f3ff-200d-1f9b1","🧑🏻‍🦱":"1f9d1-1f3fb-200d-1f9b1","🧑🏼‍🦱":"1f9d1-1f3fc-200d-1f9b1","🧑🏽‍🦱":"1f9d1-1f3fd-200d-1f9b1","🧑🏾‍🦱":"1f9d1-1f3fe-200d-1f9b1","🧑🏿‍🦱":"1f9d1-1f3ff-200d-1f9b1","👩🏻‍🦳":"1f469-1f3fb-200d-1f9b3","👩🏼‍🦳":"1f469-1f3fc-200d-1f9b3","👩🏽‍🦳":"1f469-1f3fd-200d-1f9b3","👩🏾‍🦳":"1f469-1f3fe-200d-1f9b3","👩🏿‍🦳":"1f469-1f3ff-200d-1f9b3","🧑🏻‍🦳":"1f9d1-1f3fb-200d-1f9b3","🧑🏼‍🦳":"1f9d1-1f3fc-200d-1f9b3","🧑🏽‍🦳":"1f9d1-1f3fd-200d-1f9b3","🧑🏾‍🦳":"1f9d1-1f3fe-200d-1f9b3","🧑🏿‍🦳":"1f9d1-1f3ff-200d-1f9b3","👩🏻‍🦲":"1f469-1f3fb-200d-1f9b2","👩🏼‍🦲":"1f469-1f3fc-200d-1f9b2","👩🏽‍🦲":"1f469-1f3fd-200d-1f9b2","👩🏾‍🦲":"1f469-1f3fe-200d-1f9b2","👩🏿‍🦲":"1f469-1f3ff-200d-1f9b2","🧑🏻‍🦲":"1f9d1-1f3fb-200d-1f9b2","🧑🏼‍🦲":"1f9d1-1f3fc-200d-1f9b2","🧑🏽‍🦲":"1f9d1-1f3fd-200d-1f9b2","🧑🏾‍🦲":"1f9d1-1f3fe-200d-1f9b2","🧑🏿‍🦲":"1f9d1-1f3ff-200d-1f9b2","👱‍♀️":"1f471-200d-2640-fe0f","👱🏻‍♀":"1f471-1f3fb-200d-2640-fe0f","👱🏼‍♀":"1f471-1f3fc-200d-2640-fe0f","👱🏽‍♀":"1f471-1f3fd-200d-2640-fe0f","👱🏾‍♀":"1f471-1f3fe-200d-2640-fe0f","👱🏿‍♀":"1f471-1f3ff-200d-2640-fe0f","👱‍♂️":"1f471-200d-2642-fe0f","👱🏻‍♂":"1f471-1f3fb-200d-2642-fe0f","👱🏼‍♂":"1f471-1f3fc-200d-2642-fe0f","👱🏽‍♂":"1f471-1f3fd-200d-2642-fe0f","👱🏾‍♂":"1f471-1f3fe-200d-2642-fe0f","👱🏿‍♂":"1f471-1f3ff-200d-2642-fe0f","🙍‍♂️":"1f64d-200d-2642-fe0f","🙍🏻‍♂":"1f64d-1f3fb-200d-2642-fe0f","🙍🏼‍♂":"1f64d-1f3fc-200d-2642-fe0f","🙍🏽‍♂":"1f64d-1f3fd-200d-2642-fe0f","🙍🏾‍♂":"1f64d-1f3fe-200d-2642-fe0f","🙍🏿‍♂":"1f64d-1f3ff-200d-2642-fe0f","🙍‍♀️":"1f64d-200d-2640-fe0f","🙍🏻‍♀":"1f64d-1f3fb-200d-2640-fe0f","🙍🏼‍♀":"1f64d-1f3fc-200d-2640-fe0f","🙍🏽‍♀":"1f64d-1f3fd-200d-2640-fe0f","🙍🏾‍♀":"1f64d-1f3fe-200d-2640-fe0f","🙍🏿‍♀":"1f64d-1f3ff-200d-2640-fe0f","🙎‍♂️":"1f64e-200d-2642-fe0f","🙎🏻‍♂":"1f64e-1f3fb-200d-2642-fe0f","🙎🏼‍♂":"1f64e-1f3fc-200d-2642-fe0f","🙎🏽‍♂":"1f64e-1f3fd-200d-2642-fe0f","🙎🏾‍♂":"1f64e-1f3fe-200d-2642-fe0f","🙎🏿‍♂":"1f64e-1f3ff-200d-2642-fe0f","🙎‍♀️":"1f64e-200d-2640-fe0f","🙎🏻‍♀":"1f64e-1f3fb-200d-2640-fe0f","🙎🏼‍♀":"1f64e-1f3fc-200d-2640-fe0f","🙎🏽‍♀":"1f64e-1f3fd-200d-2640-fe0f","🙎🏾‍♀":"1f64e-1f3fe-200d-2640-fe0f","🙎🏿‍♀":"1f64e-1f3ff-200d-2640-fe0f","🙅‍♂️":"1f645-200d-2642-fe0f","🙅🏻‍♂":"1f645-1f3fb-200d-2642-fe0f","🙅🏼‍♂":"1f645-1f3fc-200d-2642-fe0f","🙅🏽‍♂":"1f645-1f3fd-200d-2642-fe0f","🙅🏾‍♂":"1f645-1f3fe-200d-2642-fe0f","🙅🏿‍♂":"1f645-1f3ff-200d-2642-fe0f","🙅‍♀️":"1f645-200d-2640-fe0f","🙅🏻‍♀":"1f645-1f3fb-200d-2640-fe0f","🙅🏼‍♀":"1f645-1f3fc-200d-2640-fe0f","🙅🏽‍♀":"1f645-1f3fd-200d-2640-fe0f","🙅🏾‍♀":"1f645-1f3fe-200d-2640-fe0f","🙅🏿‍♀":"1f645-1f3ff-200d-2640-fe0f","🙆‍♂️":"1f646-200d-2642-fe0f","🙆🏻‍♂":"1f646-1f3fb-200d-2642-fe0f","🙆🏼‍♂":"1f646-1f3fc-200d-2642-fe0f","🙆🏽‍♂":"1f646-1f3fd-200d-2642-fe0f","🙆🏾‍♂":"1f646-1f3fe-200d-2642-fe0f","🙆🏿‍♂":"1f646-1f3ff-200d-2642-fe0f","🙆‍♀️":"1f646-200d-2640-fe0f","🙆🏻‍♀":"1f646-1f3fb-200d-2640-fe0f","🙆🏼‍♀":"1f646-1f3fc-200d-2640-fe0f","🙆🏽‍♀":"1f646-1f3fd-200d-2640-fe0f","🙆🏾‍♀":"1f646-1f3fe-200d-2640-fe0f","🙆🏿‍♀":"1f646-1f3ff-200d-2640-fe0f","💁‍♂️":"1f481-200d-2642-fe0f","💁🏻‍♂":"1f481-1f3fb-200d-2642-fe0f","💁🏼‍♂":"1f481-1f3fc-200d-2642-fe0f","💁🏽‍♂":"1f481-1f3fd-200d-2642-fe0f","💁🏾‍♂":"1f481-1f3fe-200d-2642-fe0f","💁🏿‍♂":"1f481-1f3ff-200d-2642-fe0f","💁‍♀️":"1f481-200d-2640-fe0f","💁🏻‍♀":"1f481-1f3fb-200d-2640-fe0f","💁🏼‍♀":"1f481-1f3fc-200d-2640-fe0f","💁🏽‍♀":"1f481-1f3fd-200d-2640-fe0f","💁🏾‍♀":"1f481-1f3fe-200d-2640-fe0f","💁🏿‍♀":"1f481-1f3ff-200d-2640-fe0f","🙋‍♂️":"1f64b-200d-2642-fe0f","🙋🏻‍♂":"1f64b-1f3fb-200d-2642-fe0f","🙋🏼‍♂":"1f64b-1f3fc-200d-2642-fe0f","🙋🏽‍♂":"1f64b-1f3fd-200d-2642-fe0f","🙋🏾‍♂":"1f64b-1f3fe-200d-2642-fe0f","🙋🏿‍♂":"1f64b-1f3ff-200d-2642-fe0f","🙋‍♀️":"1f64b-200d-2640-fe0f","🙋🏻‍♀":"1f64b-1f3fb-200d-2640-fe0f","🙋🏼‍♀":"1f64b-1f3fc-200d-2640-fe0f","🙋🏽‍♀":"1f64b-1f3fd-200d-2640-fe0f","🙋🏾‍♀":"1f64b-1f3fe-200d-2640-fe0f","🙋🏿‍♀":"1f64b-1f3ff-200d-2640-fe0f","🧏‍♂️":"1f9cf-200d-2642-fe0f","🧏🏻‍♂":"1f9cf-1f3fb-200d-2642-fe0f","🧏🏼‍♂":"1f9cf-1f3fc-200d-2642-fe0f","🧏🏽‍♂":"1f9cf-1f3fd-200d-2642-fe0f","🧏🏾‍♂":"1f9cf-1f3fe-200d-2642-fe0f","🧏🏿‍♂":"1f9cf-1f3ff-200d-2642-fe0f","🧏‍♀️":"1f9cf-200d-2640-fe0f","🧏🏻‍♀":"1f9cf-1f3fb-200d-2640-fe0f","🧏🏼‍♀":"1f9cf-1f3fc-200d-2640-fe0f","🧏🏽‍♀":"1f9cf-1f3fd-200d-2640-fe0f","🧏🏾‍♀":"1f9cf-1f3fe-200d-2640-fe0f","🧏🏿‍♀":"1f9cf-1f3ff-200d-2640-fe0f","🙇‍♂️":"1f647-200d-2642-fe0f","🙇🏻‍♂":"1f647-1f3fb-200d-2642-fe0f","🙇🏼‍♂":"1f647-1f3fc-200d-2642-fe0f","🙇🏽‍♂":"1f647-1f3fd-200d-2642-fe0f","🙇🏾‍♂":"1f647-1f3fe-200d-2642-fe0f","🙇🏿‍♂":"1f647-1f3ff-200d-2642-fe0f","🙇‍♀️":"1f647-200d-2640-fe0f","🙇🏻‍♀":"1f647-1f3fb-200d-2640-fe0f","🙇🏼‍♀":"1f647-1f3fc-200d-2640-fe0f","🙇🏽‍♀":"1f647-1f3fd-200d-2640-fe0f","🙇🏾‍♀":"1f647-1f3fe-200d-2640-fe0f","🙇🏿‍♀":"1f647-1f3ff-200d-2640-fe0f","🤦‍♂️":"1f926-200d-2642-fe0f","🤦🏻‍♂":"1f926-1f3fb-200d-2642-fe0f","🤦🏼‍♂":"1f926-1f3fc-200d-2642-fe0f","🤦🏽‍♂":"1f926-1f3fd-200d-2642-fe0f","🤦🏾‍♂":"1f926-1f3fe-200d-2642-fe0f","🤦🏿‍♂":"1f926-1f3ff-200d-2642-fe0f","🤦‍♀️":"1f926-200d-2640-fe0f","🤦🏻‍♀":"1f926-1f3fb-200d-2640-fe0f","🤦🏼‍♀":"1f926-1f3fc-200d-2640-fe0f","🤦🏽‍♀":"1f926-1f3fd-200d-2640-fe0f","🤦🏾‍♀":"1f926-1f3fe-200d-2640-fe0f","🤦🏿‍♀":"1f926-1f3ff-200d-2640-fe0f","🤷‍♂️":"1f937-200d-2642-fe0f","🤷🏻‍♂":"1f937-1f3fb-200d-2642-fe0f","🤷🏼‍♂":"1f937-1f3fc-200d-2642-fe0f","🤷🏽‍♂":"1f937-1f3fd-200d-2642-fe0f","🤷🏾‍♂":"1f937-1f3fe-200d-2642-fe0f","🤷🏿‍♂":"1f937-1f3ff-200d-2642-fe0f","🤷‍♀️":"1f937-200d-2640-fe0f","🤷🏻‍♀":"1f937-1f3fb-200d-2640-fe0f","🤷🏼‍♀":"1f937-1f3fc-200d-2640-fe0f","🤷🏽‍♀":"1f937-1f3fd-200d-2640-fe0f","🤷🏾‍♀":"1f937-1f3fe-200d-2640-fe0f","🤷🏿‍♀":"1f937-1f3ff-200d-2640-fe0f","🧑‍⚕️":"1f9d1-200d-2695-fe0f","🧑🏻‍⚕":"1f9d1-1f3fb-200d-2695-fe0f","🧑🏼‍⚕":"1f9d1-1f3fc-200d-2695-fe0f","🧑🏽‍⚕":"1f9d1-1f3fd-200d-2695-fe0f","🧑🏾‍⚕":"1f9d1-1f3fe-200d-2695-fe0f","🧑🏿‍⚕":"1f9d1-1f3ff-200d-2695-fe0f","👨‍⚕️":"1f468-200d-2695-fe0f","👨🏻‍⚕":"1f468-1f3fb-200d-2695-fe0f","👨🏼‍⚕":"1f468-1f3fc-200d-2695-fe0f","👨🏽‍⚕":"1f468-1f3fd-200d-2695-fe0f","👨🏾‍⚕":"1f468-1f3fe-200d-2695-fe0f","👨🏿‍⚕":"1f468-1f3ff-200d-2695-fe0f","👩‍⚕️":"1f469-200d-2695-fe0f","👩🏻‍⚕":"1f469-1f3fb-200d-2695-fe0f","👩🏼‍⚕":"1f469-1f3fc-200d-2695-fe0f","👩🏽‍⚕":"1f469-1f3fd-200d-2695-fe0f","👩🏾‍⚕":"1f469-1f3fe-200d-2695-fe0f","👩🏿‍⚕":"1f469-1f3ff-200d-2695-fe0f","🧑🏻‍🎓":"1f9d1-1f3fb-200d-1f393","🧑🏼‍🎓":"1f9d1-1f3fc-200d-1f393","🧑🏽‍🎓":"1f9d1-1f3fd-200d-1f393","🧑🏾‍🎓":"1f9d1-1f3fe-200d-1f393","🧑🏿‍🎓":"1f9d1-1f3ff-200d-1f393","👨🏻‍🎓":"1f468-1f3fb-200d-1f393","👨🏼‍🎓":"1f468-1f3fc-200d-1f393","👨🏽‍🎓":"1f468-1f3fd-200d-1f393","👨🏾‍🎓":"1f468-1f3fe-200d-1f393","👨🏿‍🎓":"1f468-1f3ff-200d-1f393","👩🏻‍🎓":"1f469-1f3fb-200d-1f393","👩🏼‍🎓":"1f469-1f3fc-200d-1f393","👩🏽‍🎓":"1f469-1f3fd-200d-1f393","👩🏾‍🎓":"1f469-1f3fe-200d-1f393","👩🏿‍🎓":"1f469-1f3ff-200d-1f393","🧑🏻‍🏫":"1f9d1-1f3fb-200d-1f3eb","🧑🏼‍🏫":"1f9d1-1f3fc-200d-1f3eb","🧑🏽‍🏫":"1f9d1-1f3fd-200d-1f3eb","🧑🏾‍🏫":"1f9d1-1f3fe-200d-1f3eb","🧑🏿‍🏫":"1f9d1-1f3ff-200d-1f3eb","👨🏻‍🏫":"1f468-1f3fb-200d-1f3eb","👨🏼‍🏫":"1f468-1f3fc-200d-1f3eb","👨🏽‍🏫":"1f468-1f3fd-200d-1f3eb","👨🏾‍🏫":"1f468-1f3fe-200d-1f3eb","👨🏿‍🏫":"1f468-1f3ff-200d-1f3eb","👩🏻‍🏫":"1f469-1f3fb-200d-1f3eb","👩🏼‍🏫":"1f469-1f3fc-200d-1f3eb","👩🏽‍🏫":"1f469-1f3fd-200d-1f3eb","👩🏾‍🏫":"1f469-1f3fe-200d-1f3eb","👩🏿‍🏫":"1f469-1f3ff-200d-1f3eb","🧑‍⚖️":"1f9d1-200d-2696-fe0f","🧑🏻‍⚖":"1f9d1-1f3fb-200d-2696-fe0f","🧑🏼‍⚖":"1f9d1-1f3fc-200d-2696-fe0f","🧑🏽‍⚖":"1f9d1-1f3fd-200d-2696-fe0f","🧑🏾‍⚖":"1f9d1-1f3fe-200d-2696-fe0f","🧑🏿‍⚖":"1f9d1-1f3ff-200d-2696-fe0f","👨‍⚖️":"1f468-200d-2696-fe0f","👨🏻‍⚖":"1f468-1f3fb-200d-2696-fe0f","👨🏼‍⚖":"1f468-1f3fc-200d-2696-fe0f","👨🏽‍⚖":"1f468-1f3fd-200d-2696-fe0f","👨🏾‍⚖":"1f468-1f3fe-200d-2696-fe0f","👨🏿‍⚖":"1f468-1f3ff-200d-2696-fe0f","👩‍⚖️":"1f469-200d-2696-fe0f","👩🏻‍⚖":"1f469-1f3fb-200d-2696-fe0f","👩🏼‍⚖":"1f469-1f3fc-200d-2696-fe0f","👩🏽‍⚖":"1f469-1f3fd-200d-2696-fe0f","👩🏾‍⚖":"1f469-1f3fe-200d-2696-fe0f","👩🏿‍⚖":"1f469-1f3ff-200d-2696-fe0f","🧑🏻‍🌾":"1f9d1-1f3fb-200d-1f33e","🧑🏼‍🌾":"1f9d1-1f3fc-200d-1f33e","🧑🏽‍🌾":"1f9d1-1f3fd-200d-1f33e","🧑🏾‍🌾":"1f9d1-1f3fe-200d-1f33e","🧑🏿‍🌾":"1f9d1-1f3ff-200d-1f33e","👨🏻‍🌾":"1f468-1f3fb-200d-1f33e","👨🏼‍🌾":"1f468-1f3fc-200d-1f33e","👨🏽‍🌾":"1f468-1f3fd-200d-1f33e","👨🏾‍🌾":"1f468-1f3fe-200d-1f33e","👨🏿‍🌾":"1f468-1f3ff-200d-1f33e","👩🏻‍🌾":"1f469-1f3fb-200d-1f33e","👩🏼‍🌾":"1f469-1f3fc-200d-1f33e","👩🏽‍🌾":"1f469-1f3fd-200d-1f33e","👩🏾‍🌾":"1f469-1f3fe-200d-1f33e","👩🏿‍🌾":"1f469-1f3ff-200d-1f33e","🧑🏻‍🍳":"1f9d1-1f3fb-200d-1f373","🧑🏼‍🍳":"1f9d1-1f3fc-200d-1f373","🧑🏽‍🍳":"1f9d1-1f3fd-200d-1f373","🧑🏾‍🍳":"1f9d1-1f3fe-200d-1f373","🧑🏿‍🍳":"1f9d1-1f3ff-200d-1f373","👨🏻‍🍳":"1f468-1f3fb-200d-1f373","👨🏼‍🍳":"1f468-1f3fc-200d-1f373","👨🏽‍🍳":"1f468-1f3fd-200d-1f373","👨🏾‍🍳":"1f468-1f3fe-200d-1f373","👨🏿‍🍳":"1f468-1f3ff-200d-1f373","👩🏻‍🍳":"1f469-1f3fb-200d-1f373","👩🏼‍🍳":"1f469-1f3fc-200d-1f373","👩🏽‍🍳":"1f469-1f3fd-200d-1f373","👩🏾‍🍳":"1f469-1f3fe-200d-1f373","👩🏿‍🍳":"1f469-1f3ff-200d-1f373","🧑🏻‍🔧":"1f9d1-1f3fb-200d-1f527","🧑🏼‍🔧":"1f9d1-1f3fc-200d-1f527","🧑🏽‍🔧":"1f9d1-1f3fd-200d-1f527","🧑🏾‍🔧":"1f9d1-1f3fe-200d-1f527","🧑🏿‍🔧":"1f9d1-1f3ff-200d-1f527","👨🏻‍🔧":"1f468-1f3fb-200d-1f527","👨🏼‍🔧":"1f468-1f3fc-200d-1f527","👨🏽‍🔧":"1f468-1f3fd-200d-1f527","👨🏾‍🔧":"1f468-1f3fe-200d-1f527","👨🏿‍🔧":"1f468-1f3ff-200d-1f527","👩🏻‍🔧":"1f469-1f3fb-200d-1f527","👩🏼‍🔧":"1f469-1f3fc-200d-1f527","👩🏽‍🔧":"1f469-1f3fd-200d-1f527","👩🏾‍🔧":"1f469-1f3fe-200d-1f527","👩🏿‍🔧":"1f469-1f3ff-200d-1f527","🧑🏻‍🏭":"1f9d1-1f3fb-200d-1f3ed","🧑🏼‍🏭":"1f9d1-1f3fc-200d-1f3ed","🧑🏽‍🏭":"1f9d1-1f3fd-200d-1f3ed","🧑🏾‍🏭":"1f9d1-1f3fe-200d-1f3ed","🧑🏿‍🏭":"1f9d1-1f3ff-200d-1f3ed","👨🏻‍🏭":"1f468-1f3fb-200d-1f3ed","👨🏼‍🏭":"1f468-1f3fc-200d-1f3ed","👨🏽‍🏭":"1f468-1f3fd-200d-1f3ed","👨🏾‍🏭":"1f468-1f3fe-200d-1f3ed","👨🏿‍🏭":"1f468-1f3ff-200d-1f3ed","👩🏻‍🏭":"1f469-1f3fb-200d-1f3ed","👩🏼‍🏭":"1f469-1f3fc-200d-1f3ed","👩🏽‍🏭":"1f469-1f3fd-200d-1f3ed","👩🏾‍🏭":"1f469-1f3fe-200d-1f3ed","👩🏿‍🏭":"1f469-1f3ff-200d-1f3ed","🧑🏻‍💼":"1f9d1-1f3fb-200d-1f4bc","🧑🏼‍💼":"1f9d1-1f3fc-200d-1f4bc","🧑🏽‍💼":"1f9d1-1f3fd-200d-1f4bc","🧑🏾‍💼":"1f9d1-1f3fe-200d-1f4bc","🧑🏿‍💼":"1f9d1-1f3ff-200d-1f4bc","👨🏻‍💼":"1f468-1f3fb-200d-1f4bc","👨🏼‍💼":"1f468-1f3fc-200d-1f4bc","👨🏽‍💼":"1f468-1f3fd-200d-1f4bc","👨🏾‍💼":"1f468-1f3fe-200d-1f4bc","👨🏿‍💼":"1f468-1f3ff-200d-1f4bc","👩🏻‍💼":"1f469-1f3fb-200d-1f4bc","👩🏼‍💼":"1f469-1f3fc-200d-1f4bc","👩🏽‍💼":"1f469-1f3fd-200d-1f4bc","👩🏾‍💼":"1f469-1f3fe-200d-1f4bc","👩🏿‍💼":"1f469-1f3ff-200d-1f4bc","🧑🏻‍🔬":"1f9d1-1f3fb-200d-1f52c","🧑🏼‍🔬":"1f9d1-1f3fc-200d-1f52c","🧑🏽‍🔬":"1f9d1-1f3fd-200d-1f52c","🧑🏾‍🔬":"1f9d1-1f3fe-200d-1f52c","🧑🏿‍🔬":"1f9d1-1f3ff-200d-1f52c","👨🏻‍🔬":"1f468-1f3fb-200d-1f52c","👨🏼‍🔬":"1f468-1f3fc-200d-1f52c","👨🏽‍🔬":"1f468-1f3fd-200d-1f52c","👨🏾‍🔬":"1f468-1f3fe-200d-1f52c","👨🏿‍🔬":"1f468-1f3ff-200d-1f52c","👩🏻‍🔬":"1f469-1f3fb-200d-1f52c","👩🏼‍🔬":"1f469-1f3fc-200d-1f52c","👩🏽‍🔬":"1f469-1f3fd-200d-1f52c","👩🏾‍🔬":"1f469-1f3fe-200d-1f52c","👩🏿‍🔬":"1f469-1f3ff-200d-1f52c","🧑🏻‍💻":"1f9d1-1f3fb-200d-1f4bb","🧑🏼‍💻":"1f9d1-1f3fc-200d-1f4bb","🧑🏽‍💻":"1f9d1-1f3fd-200d-1f4bb","🧑🏾‍💻":"1f9d1-1f3fe-200d-1f4bb","🧑🏿‍💻":"1f9d1-1f3ff-200d-1f4bb","👨🏻‍💻":"1f468-1f3fb-200d-1f4bb","👨🏼‍💻":"1f468-1f3fc-200d-1f4bb","👨🏽‍💻":"1f468-1f3fd-200d-1f4bb","👨🏾‍💻":"1f468-1f3fe-200d-1f4bb","👨🏿‍💻":"1f468-1f3ff-200d-1f4bb","👩🏻‍💻":"1f469-1f3fb-200d-1f4bb","👩🏼‍💻":"1f469-1f3fc-200d-1f4bb","👩🏽‍💻":"1f469-1f3fd-200d-1f4bb","👩🏾‍💻":"1f469-1f3fe-200d-1f4bb","👩🏿‍💻":"1f469-1f3ff-200d-1f4bb","🧑🏻‍🎤":"1f9d1-1f3fb-200d-1f3a4","🧑🏼‍🎤":"1f9d1-1f3fc-200d-1f3a4","🧑🏽‍🎤":"1f9d1-1f3fd-200d-1f3a4","🧑🏾‍🎤":"1f9d1-1f3fe-200d-1f3a4","🧑🏿‍🎤":"1f9d1-1f3ff-200d-1f3a4","👨🏻‍🎤":"1f468-1f3fb-200d-1f3a4","👨🏼‍🎤":"1f468-1f3fc-200d-1f3a4","👨🏽‍🎤":"1f468-1f3fd-200d-1f3a4","👨🏾‍🎤":"1f468-1f3fe-200d-1f3a4","👨🏿‍🎤":"1f468-1f3ff-200d-1f3a4","👩🏻‍🎤":"1f469-1f3fb-200d-1f3a4","👩🏼‍🎤":"1f469-1f3fc-200d-1f3a4","👩🏽‍🎤":"1f469-1f3fd-200d-1f3a4","👩🏾‍🎤":"1f469-1f3fe-200d-1f3a4","👩🏿‍🎤":"1f469-1f3ff-200d-1f3a4","🧑🏻‍🎨":"1f9d1-1f3fb-200d-1f3a8","🧑🏼‍🎨":"1f9d1-1f3fc-200d-1f3a8","🧑🏽‍🎨":"1f9d1-1f3fd-200d-1f3a8","🧑🏾‍🎨":"1f9d1-1f3fe-200d-1f3a8","🧑🏿‍🎨":"1f9d1-1f3ff-200d-1f3a8","👨🏻‍🎨":"1f468-1f3fb-200d-1f3a8","👨🏼‍🎨":"1f468-1f3fc-200d-1f3a8","👨🏽‍🎨":"1f468-1f3fd-200d-1f3a8","👨🏾‍🎨":"1f468-1f3fe-200d-1f3a8","👨🏿‍🎨":"1f468-1f3ff-200d-1f3a8","👩🏻‍🎨":"1f469-1f3fb-200d-1f3a8","👩🏼‍🎨":"1f469-1f3fc-200d-1f3a8","👩🏽‍🎨":"1f469-1f3fd-200d-1f3a8","👩🏾‍🎨":"1f469-1f3fe-200d-1f3a8","👩🏿‍🎨":"1f469-1f3ff-200d-1f3a8","🧑‍✈️":"1f9d1-200d-2708-fe0f","🧑🏻‍✈":"1f9d1-1f3fb-200d-2708-fe0f","🧑🏼‍✈":"1f9d1-1f3fc-200d-2708-fe0f","🧑🏽‍✈":"1f9d1-1f3fd-200d-2708-fe0f","🧑🏾‍✈":"1f9d1-1f3fe-200d-2708-fe0f","🧑🏿‍✈":"1f9d1-1f3ff-200d-2708-fe0f","👨‍✈️":"1f468-200d-2708-fe0f","👨🏻‍✈":"1f468-1f3fb-200d-2708-fe0f","👨🏼‍✈":"1f468-1f3fc-200d-2708-fe0f","👨🏽‍✈":"1f468-1f3fd-200d-2708-fe0f","👨🏾‍✈":"1f468-1f3fe-200d-2708-fe0f","👨🏿‍✈":"1f468-1f3ff-200d-2708-fe0f","👩‍✈️":"1f469-200d-2708-fe0f","👩🏻‍✈":"1f469-1f3fb-200d-2708-fe0f","👩🏼‍✈":"1f469-1f3fc-200d-2708-fe0f","👩🏽‍✈":"1f469-1f3fd-200d-2708-fe0f","👩🏾‍✈":"1f469-1f3fe-200d-2708-fe0f","👩🏿‍✈":"1f469-1f3ff-200d-2708-fe0f","🧑🏻‍🚀":"1f9d1-1f3fb-200d-1f680","🧑🏼‍🚀":"1f9d1-1f3fc-200d-1f680","🧑🏽‍🚀":"1f9d1-1f3fd-200d-1f680","🧑🏾‍🚀":"1f9d1-1f3fe-200d-1f680","🧑🏿‍🚀":"1f9d1-1f3ff-200d-1f680","👨🏻‍🚀":"1f468-1f3fb-200d-1f680","👨🏼‍🚀":"1f468-1f3fc-200d-1f680","👨🏽‍🚀":"1f468-1f3fd-200d-1f680","👨🏾‍🚀":"1f468-1f3fe-200d-1f680","👨🏿‍🚀":"1f468-1f3ff-200d-1f680","👩🏻‍🚀":"1f469-1f3fb-200d-1f680","👩🏼‍🚀":"1f469-1f3fc-200d-1f680","👩🏽‍🚀":"1f469-1f3fd-200d-1f680","👩🏾‍🚀":"1f469-1f3fe-200d-1f680","👩🏿‍🚀":"1f469-1f3ff-200d-1f680","🧑🏻‍🚒":"1f9d1-1f3fb-200d-1f692","🧑🏼‍🚒":"1f9d1-1f3fc-200d-1f692","🧑🏽‍🚒":"1f9d1-1f3fd-200d-1f692","🧑🏾‍🚒":"1f9d1-1f3fe-200d-1f692","🧑🏿‍🚒":"1f9d1-1f3ff-200d-1f692","👨🏻‍🚒":"1f468-1f3fb-200d-1f692","👨🏼‍🚒":"1f468-1f3fc-200d-1f692","👨🏽‍🚒":"1f468-1f3fd-200d-1f692","👨🏾‍🚒":"1f468-1f3fe-200d-1f692","👨🏿‍🚒":"1f468-1f3ff-200d-1f692","👩🏻‍🚒":"1f469-1f3fb-200d-1f692","👩🏼‍🚒":"1f469-1f3fc-200d-1f692","👩🏽‍🚒":"1f469-1f3fd-200d-1f692","👩🏾‍🚒":"1f469-1f3fe-200d-1f692","👩🏿‍🚒":"1f469-1f3ff-200d-1f692","👮‍♂️":"1f46e-200d-2642-fe0f","👮🏻‍♂":"1f46e-1f3fb-200d-2642-fe0f","👮🏼‍♂":"1f46e-1f3fc-200d-2642-fe0f","👮🏽‍♂":"1f46e-1f3fd-200d-2642-fe0f","👮🏾‍♂":"1f46e-1f3fe-200d-2642-fe0f","👮🏿‍♂":"1f46e-1f3ff-200d-2642-fe0f","👮‍♀️":"1f46e-200d-2640-fe0f","👮🏻‍♀":"1f46e-1f3fb-200d-2640-fe0f","👮🏼‍♀":"1f46e-1f3fc-200d-2640-fe0f","👮🏽‍♀":"1f46e-1f3fd-200d-2640-fe0f","👮🏾‍♀":"1f46e-1f3fe-200d-2640-fe0f","👮🏿‍♀":"1f46e-1f3ff-200d-2640-fe0f","🕵‍♂️":"1f575-fe0f-200d-2642-fe0f","🕵️‍♂":"1f575-fe0f-200d-2642-fe0f","🕵🏻‍♂":"1f575-1f3fb-200d-2642-fe0f","🕵🏼‍♂":"1f575-1f3fc-200d-2642-fe0f","🕵🏽‍♂":"1f575-1f3fd-200d-2642-fe0f","🕵🏾‍♂":"1f575-1f3fe-200d-2642-fe0f","🕵🏿‍♂":"1f575-1f3ff-200d-2642-fe0f","🕵‍♀️":"1f575-fe0f-200d-2640-fe0f","🕵️‍♀":"1f575-fe0f-200d-2640-fe0f","🕵🏻‍♀":"1f575-1f3fb-200d-2640-fe0f","🕵🏼‍♀":"1f575-1f3fc-200d-2640-fe0f","🕵🏽‍♀":"1f575-1f3fd-200d-2640-fe0f","🕵🏾‍♀":"1f575-1f3fe-200d-2640-fe0f","🕵🏿‍♀":"1f575-1f3ff-200d-2640-fe0f","💂‍♂️":"1f482-200d-2642-fe0f","💂🏻‍♂":"1f482-1f3fb-200d-2642-fe0f","💂🏼‍♂":"1f482-1f3fc-200d-2642-fe0f","💂🏽‍♂":"1f482-1f3fd-200d-2642-fe0f","💂🏾‍♂":"1f482-1f3fe-200d-2642-fe0f","💂🏿‍♂":"1f482-1f3ff-200d-2642-fe0f","💂‍♀️":"1f482-200d-2640-fe0f","💂🏻‍♀":"1f482-1f3fb-200d-2640-fe0f","💂🏼‍♀":"1f482-1f3fc-200d-2640-fe0f","💂🏽‍♀":"1f482-1f3fd-200d-2640-fe0f","💂🏾‍♀":"1f482-1f3fe-200d-2640-fe0f","💂🏿‍♀":"1f482-1f3ff-200d-2640-fe0f","👷‍♂️":"1f477-200d-2642-fe0f","👷🏻‍♂":"1f477-1f3fb-200d-2642-fe0f","👷🏼‍♂":"1f477-1f3fc-200d-2642-fe0f","👷🏽‍♂":"1f477-1f3fd-200d-2642-fe0f","👷🏾‍♂":"1f477-1f3fe-200d-2642-fe0f","👷🏿‍♂":"1f477-1f3ff-200d-2642-fe0f","👷‍♀️":"1f477-200d-2640-fe0f","👷🏻‍♀":"1f477-1f3fb-200d-2640-fe0f","👷🏼‍♀":"1f477-1f3fc-200d-2640-fe0f","👷🏽‍♀":"1f477-1f3fd-200d-2640-fe0f","👷🏾‍♀":"1f477-1f3fe-200d-2640-fe0f","👷🏿‍♀":"1f477-1f3ff-200d-2640-fe0f","👳‍♂️":"1f473-200d-2642-fe0f","👳🏻‍♂":"1f473-1f3fb-200d-2642-fe0f","👳🏼‍♂":"1f473-1f3fc-200d-2642-fe0f","👳🏽‍♂":"1f473-1f3fd-200d-2642-fe0f","👳🏾‍♂":"1f473-1f3fe-200d-2642-fe0f","👳🏿‍♂":"1f473-1f3ff-200d-2642-fe0f","👳‍♀️":"1f473-200d-2640-fe0f","👳🏻‍♀":"1f473-1f3fb-200d-2640-fe0f","👳🏼‍♀":"1f473-1f3fc-200d-2640-fe0f","👳🏽‍♀":"1f473-1f3fd-200d-2640-fe0f","👳🏾‍♀":"1f473-1f3fe-200d-2640-fe0f","👳🏿‍♀":"1f473-1f3ff-200d-2640-fe0f","🤵‍♂️":"1f935-200d-2642-fe0f","🤵🏻‍♂":"1f935-1f3fb-200d-2642-fe0f","🤵🏼‍♂":"1f935-1f3fc-200d-2642-fe0f","🤵🏽‍♂":"1f935-1f3fd-200d-2642-fe0f","🤵🏾‍♂":"1f935-1f3fe-200d-2642-fe0f","🤵🏿‍♂":"1f935-1f3ff-200d-2642-fe0f","🤵‍♀️":"1f935-200d-2640-fe0f","🤵🏻‍♀":"1f935-1f3fb-200d-2640-fe0f","🤵🏼‍♀":"1f935-1f3fc-200d-2640-fe0f","🤵🏽‍♀":"1f935-1f3fd-200d-2640-fe0f","🤵🏾‍♀":"1f935-1f3fe-200d-2640-fe0f","🤵🏿‍♀":"1f935-1f3ff-200d-2640-fe0f","👰‍♂️":"1f470-200d-2642-fe0f","👰🏻‍♂":"1f470-1f3fb-200d-2642-fe0f","👰🏼‍♂":"1f470-1f3fc-200d-2642-fe0f","👰🏽‍♂":"1f470-1f3fd-200d-2642-fe0f","👰🏾‍♂":"1f470-1f3fe-200d-2642-fe0f","👰🏿‍♂":"1f470-1f3ff-200d-2642-fe0f","👰‍♀️":"1f470-200d-2640-fe0f","👰🏻‍♀":"1f470-1f3fb-200d-2640-fe0f","👰🏼‍♀":"1f470-1f3fc-200d-2640-fe0f","👰🏽‍♀":"1f470-1f3fd-200d-2640-fe0f","👰🏾‍♀":"1f470-1f3fe-200d-2640-fe0f","👰🏿‍♀":"1f470-1f3ff-200d-2640-fe0f","👩🏻‍🍼":"1f469-1f3fb-200d-1f37c","👩🏼‍🍼":"1f469-1f3fc-200d-1f37c","👩🏽‍🍼":"1f469-1f3fd-200d-1f37c","👩🏾‍🍼":"1f469-1f3fe-200d-1f37c","👩🏿‍🍼":"1f469-1f3ff-200d-1f37c","👨🏻‍🍼":"1f468-1f3fb-200d-1f37c","👨🏼‍🍼":"1f468-1f3fc-200d-1f37c","👨🏽‍🍼":"1f468-1f3fd-200d-1f37c","👨🏾‍🍼":"1f468-1f3fe-200d-1f37c","👨🏿‍🍼":"1f468-1f3ff-200d-1f37c","🧑🏻‍🍼":"1f9d1-1f3fb-200d-1f37c","🧑🏼‍🍼":"1f9d1-1f3fc-200d-1f37c","🧑🏽‍🍼":"1f9d1-1f3fd-200d-1f37c","🧑🏾‍🍼":"1f9d1-1f3fe-200d-1f37c","🧑🏿‍🍼":"1f9d1-1f3ff-200d-1f37c","🧑🏻‍🎄":"1f9d1-1f3fb-200d-1f384","🧑🏼‍🎄":"1f9d1-1f3fc-200d-1f384","🧑🏽‍🎄":"1f9d1-1f3fd-200d-1f384","🧑🏾‍🎄":"1f9d1-1f3fe-200d-1f384","🧑🏿‍🎄":"1f9d1-1f3ff-200d-1f384","🦸‍♂️":"1f9b8-200d-2642-fe0f","🦸🏻‍♂":"1f9b8-1f3fb-200d-2642-fe0f","🦸🏼‍♂":"1f9b8-1f3fc-200d-2642-fe0f","🦸🏽‍♂":"1f9b8-1f3fd-200d-2642-fe0f","🦸🏾‍♂":"1f9b8-1f3fe-200d-2642-fe0f","🦸🏿‍♂":"1f9b8-1f3ff-200d-2642-fe0f","🦸‍♀️":"1f9b8-200d-2640-fe0f","🦸🏻‍♀":"1f9b8-1f3fb-200d-2640-fe0f","🦸🏼‍♀":"1f9b8-1f3fc-200d-2640-fe0f","🦸🏽‍♀":"1f9b8-1f3fd-200d-2640-fe0f","🦸🏾‍♀":"1f9b8-1f3fe-200d-2640-fe0f","🦸🏿‍♀":"1f9b8-1f3ff-200d-2640-fe0f","🦹‍♂️":"1f9b9-200d-2642-fe0f","🦹🏻‍♂":"1f9b9-1f3fb-200d-2642-fe0f","🦹🏼‍♂":"1f9b9-1f3fc-200d-2642-fe0f","🦹🏽‍♂":"1f9b9-1f3fd-200d-2642-fe0f","🦹🏾‍♂":"1f9b9-1f3fe-200d-2642-fe0f","🦹🏿‍♂":"1f9b9-1f3ff-200d-2642-fe0f","🦹‍♀️":"1f9b9-200d-2640-fe0f","🦹🏻‍♀":"1f9b9-1f3fb-200d-2640-fe0f","🦹🏼‍♀":"1f9b9-1f3fc-200d-2640-fe0f","🦹🏽‍♀":"1f9b9-1f3fd-200d-2640-fe0f","🦹🏾‍♀":"1f9b9-1f3fe-200d-2640-fe0f","🦹🏿‍♀":"1f9b9-1f3ff-200d-2640-fe0f","🧙‍♂️":"1f9d9-200d-2642-fe0f","🧙🏻‍♂":"1f9d9-1f3fb-200d-2642-fe0f","🧙🏼‍♂":"1f9d9-1f3fc-200d-2642-fe0f","🧙🏽‍♂":"1f9d9-1f3fd-200d-2642-fe0f","🧙🏾‍♂":"1f9d9-1f3fe-200d-2642-fe0f","🧙🏿‍♂":"1f9d9-1f3ff-200d-2642-fe0f","🧙‍♀️":"1f9d9-200d-2640-fe0f","🧙🏻‍♀":"1f9d9-1f3fb-200d-2640-fe0f","🧙🏼‍♀":"1f9d9-1f3fc-200d-2640-fe0f","🧙🏽‍♀":"1f9d9-1f3fd-200d-2640-fe0f","🧙🏾‍♀":"1f9d9-1f3fe-200d-2640-fe0f","🧙🏿‍♀":"1f9d9-1f3ff-200d-2640-fe0f","🧚‍♂️":"1f9da-200d-2642-fe0f","🧚🏻‍♂":"1f9da-1f3fb-200d-2642-fe0f","🧚🏼‍♂":"1f9da-1f3fc-200d-2642-fe0f","🧚🏽‍♂":"1f9da-1f3fd-200d-2642-fe0f","🧚🏾‍♂":"1f9da-1f3fe-200d-2642-fe0f","🧚🏿‍♂":"1f9da-1f3ff-200d-2642-fe0f","🧚‍♀️":"1f9da-200d-2640-fe0f","🧚🏻‍♀":"1f9da-1f3fb-200d-2640-fe0f","🧚🏼‍♀":"1f9da-1f3fc-200d-2640-fe0f","🧚🏽‍♀":"1f9da-1f3fd-200d-2640-fe0f","🧚🏾‍♀":"1f9da-1f3fe-200d-2640-fe0f","🧚🏿‍♀":"1f9da-1f3ff-200d-2640-fe0f","🧛‍♂️":"1f9db-200d-2642-fe0f","🧛🏻‍♂":"1f9db-1f3fb-200d-2642-fe0f","🧛🏼‍♂":"1f9db-1f3fc-200d-2642-fe0f","🧛🏽‍♂":"1f9db-1f3fd-200d-2642-fe0f","🧛🏾‍♂":"1f9db-1f3fe-200d-2642-fe0f","🧛🏿‍♂":"1f9db-1f3ff-200d-2642-fe0f","🧛‍♀️":"1f9db-200d-2640-fe0f","🧛🏻‍♀":"1f9db-1f3fb-200d-2640-fe0f","🧛🏼‍♀":"1f9db-1f3fc-200d-2640-fe0f","🧛🏽‍♀":"1f9db-1f3fd-200d-2640-fe0f","🧛🏾‍♀":"1f9db-1f3fe-200d-2640-fe0f","🧛🏿‍♀":"1f9db-1f3ff-200d-2640-fe0f","🧜‍♂️":"1f9dc-200d-2642-fe0f","🧜🏻‍♂":"1f9dc-1f3fb-200d-2642-fe0f","🧜🏼‍♂":"1f9dc-1f3fc-200d-2642-fe0f","🧜🏽‍♂":"1f9dc-1f3fd-200d-2642-fe0f","🧜🏾‍♂":"1f9dc-1f3fe-200d-2642-fe0f","🧜🏿‍♂":"1f9dc-1f3ff-200d-2642-fe0f","🧜‍♀️":"1f9dc-200d-2640-fe0f","🧜🏻‍♀":"1f9dc-1f3fb-200d-2640-fe0f","🧜🏼‍♀":"1f9dc-1f3fc-200d-2640-fe0f","🧜🏽‍♀":"1f9dc-1f3fd-200d-2640-fe0f","🧜🏾‍♀":"1f9dc-1f3fe-200d-2640-fe0f","🧜🏿‍♀":"1f9dc-1f3ff-200d-2640-fe0f","🧝‍♂️":"1f9dd-200d-2642-fe0f","🧝🏻‍♂":"1f9dd-1f3fb-200d-2642-fe0f","🧝🏼‍♂":"1f9dd-1f3fc-200d-2642-fe0f","🧝🏽‍♂":"1f9dd-1f3fd-200d-2642-fe0f","🧝🏾‍♂":"1f9dd-1f3fe-200d-2642-fe0f","🧝🏿‍♂":"1f9dd-1f3ff-200d-2642-fe0f","🧝‍♀️":"1f9dd-200d-2640-fe0f","🧝🏻‍♀":"1f9dd-1f3fb-200d-2640-fe0f","🧝🏼‍♀":"1f9dd-1f3fc-200d-2640-fe0f","🧝🏽‍♀":"1f9dd-1f3fd-200d-2640-fe0f","🧝🏾‍♀":"1f9dd-1f3fe-200d-2640-fe0f","🧝🏿‍♀":"1f9dd-1f3ff-200d-2640-fe0f","🧞‍♂️":"1f9de-200d-2642-fe0f","🧞‍♀️":"1f9de-200d-2640-fe0f","🧟‍♂️":"1f9df-200d-2642-fe0f","🧟‍♀️":"1f9df-200d-2640-fe0f","💆‍♂️":"1f486-200d-2642-fe0f","💆🏻‍♂":"1f486-1f3fb-200d-2642-fe0f","💆🏼‍♂":"1f486-1f3fc-200d-2642-fe0f","💆🏽‍♂":"1f486-1f3fd-200d-2642-fe0f","💆🏾‍♂":"1f486-1f3fe-200d-2642-fe0f","💆🏿‍♂":"1f486-1f3ff-200d-2642-fe0f","💆‍♀️":"1f486-200d-2640-fe0f","💆🏻‍♀":"1f486-1f3fb-200d-2640-fe0f","💆🏼‍♀":"1f486-1f3fc-200d-2640-fe0f","💆🏽‍♀":"1f486-1f3fd-200d-2640-fe0f","💆🏾‍♀":"1f486-1f3fe-200d-2640-fe0f","💆🏿‍♀":"1f486-1f3ff-200d-2640-fe0f","💇‍♂️":"1f487-200d-2642-fe0f","💇🏻‍♂":"1f487-1f3fb-200d-2642-fe0f","💇🏼‍♂":"1f487-1f3fc-200d-2642-fe0f","💇🏽‍♂":"1f487-1f3fd-200d-2642-fe0f","💇🏾‍♂":"1f487-1f3fe-200d-2642-fe0f","💇🏿‍♂":"1f487-1f3ff-200d-2642-fe0f","💇‍♀️":"1f487-200d-2640-fe0f","💇🏻‍♀":"1f487-1f3fb-200d-2640-fe0f","💇🏼‍♀":"1f487-1f3fc-200d-2640-fe0f","💇🏽‍♀":"1f487-1f3fd-200d-2640-fe0f","💇🏾‍♀":"1f487-1f3fe-200d-2640-fe0f","💇🏿‍♀":"1f487-1f3ff-200d-2640-fe0f","🚶‍♂️":"1f6b6-200d-2642-fe0f","🚶🏻‍♂":"1f6b6-1f3fb-200d-2642-fe0f","🚶🏼‍♂":"1f6b6-1f3fc-200d-2642-fe0f","🚶🏽‍♂":"1f6b6-1f3fd-200d-2642-fe0f","🚶🏾‍♂":"1f6b6-1f3fe-200d-2642-fe0f","🚶🏿‍♂":"1f6b6-1f3ff-200d-2642-fe0f","🚶‍♀️":"1f6b6-200d-2640-fe0f","🚶🏻‍♀":"1f6b6-1f3fb-200d-2640-fe0f","🚶🏼‍♀":"1f6b6-1f3fc-200d-2640-fe0f","🚶🏽‍♀":"1f6b6-1f3fd-200d-2640-fe0f","🚶🏾‍♀":"1f6b6-1f3fe-200d-2640-fe0f","🚶🏿‍♀":"1f6b6-1f3ff-200d-2640-fe0f","🧍‍♂️":"1f9cd-200d-2642-fe0f","🧍🏻‍♂":"1f9cd-1f3fb-200d-2642-fe0f","🧍🏼‍♂":"1f9cd-1f3fc-200d-2642-fe0f","🧍🏽‍♂":"1f9cd-1f3fd-200d-2642-fe0f","🧍🏾‍♂":"1f9cd-1f3fe-200d-2642-fe0f","🧍🏿‍♂":"1f9cd-1f3ff-200d-2642-fe0f","🧍‍♀️":"1f9cd-200d-2640-fe0f","🧍🏻‍♀":"1f9cd-1f3fb-200d-2640-fe0f","🧍🏼‍♀":"1f9cd-1f3fc-200d-2640-fe0f","🧍🏽‍♀":"1f9cd-1f3fd-200d-2640-fe0f","🧍🏾‍♀":"1f9cd-1f3fe-200d-2640-fe0f","🧍🏿‍♀":"1f9cd-1f3ff-200d-2640-fe0f","🧎‍♂️":"1f9ce-200d-2642-fe0f","🧎🏻‍♂":"1f9ce-1f3fb-200d-2642-fe0f","🧎🏼‍♂":"1f9ce-1f3fc-200d-2642-fe0f","🧎🏽‍♂":"1f9ce-1f3fd-200d-2642-fe0f","🧎🏾‍♂":"1f9ce-1f3fe-200d-2642-fe0f","🧎🏿‍♂":"1f9ce-1f3ff-200d-2642-fe0f","🧎‍♀️":"1f9ce-200d-2640-fe0f","🧎🏻‍♀":"1f9ce-1f3fb-200d-2640-fe0f","🧎🏼‍♀":"1f9ce-1f3fc-200d-2640-fe0f","🧎🏽‍♀":"1f9ce-1f3fd-200d-2640-fe0f","🧎🏾‍♀":"1f9ce-1f3fe-200d-2640-fe0f","🧎🏿‍♀":"1f9ce-1f3ff-200d-2640-fe0f","🧑🏻‍🦯":"1f9d1-1f3fb-200d-1f9af","🧑🏼‍🦯":"1f9d1-1f3fc-200d-1f9af","🧑🏽‍🦯":"1f9d1-1f3fd-200d-1f9af","🧑🏾‍🦯":"1f9d1-1f3fe-200d-1f9af","🧑🏿‍🦯":"1f9d1-1f3ff-200d-1f9af","👨🏻‍🦯":"1f468-1f3fb-200d-1f9af","👨🏼‍🦯":"1f468-1f3fc-200d-1f9af","👨🏽‍🦯":"1f468-1f3fd-200d-1f9af","👨🏾‍🦯":"1f468-1f3fe-200d-1f9af","👨🏿‍🦯":"1f468-1f3ff-200d-1f9af","👩🏻‍🦯":"1f469-1f3fb-200d-1f9af","👩🏼‍🦯":"1f469-1f3fc-200d-1f9af","👩🏽‍🦯":"1f469-1f3fd-200d-1f9af","👩🏾‍🦯":"1f469-1f3fe-200d-1f9af","👩🏿‍🦯":"1f469-1f3ff-200d-1f9af","🧑🏻‍🦼":"1f9d1-1f3fb-200d-1f9bc","🧑🏼‍🦼":"1f9d1-1f3fc-200d-1f9bc","🧑🏽‍🦼":"1f9d1-1f3fd-200d-1f9bc","🧑🏾‍🦼":"1f9d1-1f3fe-200d-1f9bc","🧑🏿‍🦼":"1f9d1-1f3ff-200d-1f9bc","👨🏻‍🦼":"1f468-1f3fb-200d-1f9bc","👨🏼‍🦼":"1f468-1f3fc-200d-1f9bc","👨🏽‍🦼":"1f468-1f3fd-200d-1f9bc","👨🏾‍🦼":"1f468-1f3fe-200d-1f9bc","👨🏿‍🦼":"1f468-1f3ff-200d-1f9bc","👩🏻‍🦼":"1f469-1f3fb-200d-1f9bc","👩🏼‍🦼":"1f469-1f3fc-200d-1f9bc","👩🏽‍🦼":"1f469-1f3fd-200d-1f9bc","👩🏾‍🦼":"1f469-1f3fe-200d-1f9bc","👩🏿‍🦼":"1f469-1f3ff-200d-1f9bc","🧑🏻‍🦽":"1f9d1-1f3fb-200d-1f9bd","🧑🏼‍🦽":"1f9d1-1f3fc-200d-1f9bd","🧑🏽‍🦽":"1f9d1-1f3fd-200d-1f9bd","🧑🏾‍🦽":"1f9d1-1f3fe-200d-1f9bd","🧑🏿‍🦽":"1f9d1-1f3ff-200d-1f9bd","👨🏻‍🦽":"1f468-1f3fb-200d-1f9bd","👨🏼‍🦽":"1f468-1f3fc-200d-1f9bd","👨🏽‍🦽":"1f468-1f3fd-200d-1f9bd","👨🏾‍🦽":"1f468-1f3fe-200d-1f9bd","👨🏿‍🦽":"1f468-1f3ff-200d-1f9bd","👩🏻‍🦽":"1f469-1f3fb-200d-1f9bd","👩🏼‍🦽":"1f469-1f3fc-200d-1f9bd","👩🏽‍🦽":"1f469-1f3fd-200d-1f9bd","👩🏾‍🦽":"1f469-1f3fe-200d-1f9bd","👩🏿‍🦽":"1f469-1f3ff-200d-1f9bd","🏃‍♂️":"1f3c3-200d-2642-fe0f","🏃🏻‍♂":"1f3c3-1f3fb-200d-2642-fe0f","🏃🏼‍♂":"1f3c3-1f3fc-200d-2642-fe0f","🏃🏽‍♂":"1f3c3-1f3fd-200d-2642-fe0f","🏃🏾‍♂":"1f3c3-1f3fe-200d-2642-fe0f","🏃🏿‍♂":"1f3c3-1f3ff-200d-2642-fe0f","🏃‍♀️":"1f3c3-200d-2640-fe0f","🏃🏻‍♀":"1f3c3-1f3fb-200d-2640-fe0f","🏃🏼‍♀":"1f3c3-1f3fc-200d-2640-fe0f","🏃🏽‍♀":"1f3c3-1f3fd-200d-2640-fe0f","🏃🏾‍♀":"1f3c3-1f3fe-200d-2640-fe0f","🏃🏿‍♀":"1f3c3-1f3ff-200d-2640-fe0f","👯‍♂️":"1f46f-200d-2642-fe0f","👯‍♀️":"1f46f-200d-2640-fe0f","🧖‍♂️":"1f9d6-200d-2642-fe0f","🧖🏻‍♂":"1f9d6-1f3fb-200d-2642-fe0f","🧖🏼‍♂":"1f9d6-1f3fc-200d-2642-fe0f","🧖🏽‍♂":"1f9d6-1f3fd-200d-2642-fe0f","🧖🏾‍♂":"1f9d6-1f3fe-200d-2642-fe0f","🧖🏿‍♂":"1f9d6-1f3ff-200d-2642-fe0f","🧖‍♀️":"1f9d6-200d-2640-fe0f","🧖🏻‍♀":"1f9d6-1f3fb-200d-2640-fe0f","🧖🏼‍♀":"1f9d6-1f3fc-200d-2640-fe0f","🧖🏽‍♀":"1f9d6-1f3fd-200d-2640-fe0f","🧖🏾‍♀":"1f9d6-1f3fe-200d-2640-fe0f","🧖🏿‍♀":"1f9d6-1f3ff-200d-2640-fe0f","🧗‍♂️":"1f9d7-200d-2642-fe0f","🧗🏻‍♂":"1f9d7-1f3fb-200d-2642-fe0f","🧗🏼‍♂":"1f9d7-1f3fc-200d-2642-fe0f","🧗🏽‍♂":"1f9d7-1f3fd-200d-2642-fe0f","🧗🏾‍♂":"1f9d7-1f3fe-200d-2642-fe0f","🧗🏿‍♂":"1f9d7-1f3ff-200d-2642-fe0f","🧗‍♀️":"1f9d7-200d-2640-fe0f","🧗🏻‍♀":"1f9d7-1f3fb-200d-2640-fe0f","🧗🏼‍♀":"1f9d7-1f3fc-200d-2640-fe0f","🧗🏽‍♀":"1f9d7-1f3fd-200d-2640-fe0f","🧗🏾‍♀":"1f9d7-1f3fe-200d-2640-fe0f","🧗🏿‍♀":"1f9d7-1f3ff-200d-2640-fe0f","🏌‍♂️":"1f3cc-fe0f-200d-2642-fe0f","🏌️‍♂":"1f3cc-fe0f-200d-2642-fe0f","🏌🏻‍♂":"1f3cc-1f3fb-200d-2642-fe0f","🏌🏼‍♂":"1f3cc-1f3fc-200d-2642-fe0f","🏌🏽‍♂":"1f3cc-1f3fd-200d-2642-fe0f","🏌🏾‍♂":"1f3cc-1f3fe-200d-2642-fe0f","🏌🏿‍♂":"1f3cc-1f3ff-200d-2642-fe0f","🏌‍♀️":"1f3cc-fe0f-200d-2640-fe0f","🏌️‍♀":"1f3cc-fe0f-200d-2640-fe0f","🏌🏻‍♀":"1f3cc-1f3fb-200d-2640-fe0f","🏌🏼‍♀":"1f3cc-1f3fc-200d-2640-fe0f","🏌🏽‍♀":"1f3cc-1f3fd-200d-2640-fe0f","🏌🏾‍♀":"1f3cc-1f3fe-200d-2640-fe0f","🏌🏿‍♀":"1f3cc-1f3ff-200d-2640-fe0f","🏄‍♂️":"1f3c4-200d-2642-fe0f","🏄🏻‍♂":"1f3c4-1f3fb-200d-2642-fe0f","🏄🏼‍♂":"1f3c4-1f3fc-200d-2642-fe0f","🏄🏽‍♂":"1f3c4-1f3fd-200d-2642-fe0f","🏄🏾‍♂":"1f3c4-1f3fe-200d-2642-fe0f","🏄🏿‍♂":"1f3c4-1f3ff-200d-2642-fe0f","🏄‍♀️":"1f3c4-200d-2640-fe0f","🏄🏻‍♀":"1f3c4-1f3fb-200d-2640-fe0f","🏄🏼‍♀":"1f3c4-1f3fc-200d-2640-fe0f","🏄🏽‍♀":"1f3c4-1f3fd-200d-2640-fe0f","🏄🏾‍♀":"1f3c4-1f3fe-200d-2640-fe0f","🏄🏿‍♀":"1f3c4-1f3ff-200d-2640-fe0f","🚣‍♂️":"1f6a3-200d-2642-fe0f","🚣🏻‍♂":"1f6a3-1f3fb-200d-2642-fe0f","🚣🏼‍♂":"1f6a3-1f3fc-200d-2642-fe0f","🚣🏽‍♂":"1f6a3-1f3fd-200d-2642-fe0f","🚣🏾‍♂":"1f6a3-1f3fe-200d-2642-fe0f","🚣🏿‍♂":"1f6a3-1f3ff-200d-2642-fe0f","🚣‍♀️":"1f6a3-200d-2640-fe0f","🚣🏻‍♀":"1f6a3-1f3fb-200d-2640-fe0f","🚣🏼‍♀":"1f6a3-1f3fc-200d-2640-fe0f","🚣🏽‍♀":"1f6a3-1f3fd-200d-2640-fe0f","🚣🏾‍♀":"1f6a3-1f3fe-200d-2640-fe0f","🚣🏿‍♀":"1f6a3-1f3ff-200d-2640-fe0f","🏊‍♂️":"1f3ca-200d-2642-fe0f","🏊🏻‍♂":"1f3ca-1f3fb-200d-2642-fe0f","🏊🏼‍♂":"1f3ca-1f3fc-200d-2642-fe0f","🏊🏽‍♂":"1f3ca-1f3fd-200d-2642-fe0f","🏊🏾‍♂":"1f3ca-1f3fe-200d-2642-fe0f","🏊🏿‍♂":"1f3ca-1f3ff-200d-2642-fe0f","🏊‍♀️":"1f3ca-200d-2640-fe0f","🏊🏻‍♀":"1f3ca-1f3fb-200d-2640-fe0f","🏊🏼‍♀":"1f3ca-1f3fc-200d-2640-fe0f","🏊🏽‍♀":"1f3ca-1f3fd-200d-2640-fe0f","🏊🏾‍♀":"1f3ca-1f3fe-200d-2640-fe0f","🏊🏿‍♀":"1f3ca-1f3ff-200d-2640-fe0f","⛹‍♂️":"26f9-fe0f-200d-2642-fe0f","⛹️‍♂":"26f9-fe0f-200d-2642-fe0f","⛹🏻‍♂":"26f9-1f3fb-200d-2642-fe0f","⛹🏼‍♂":"26f9-1f3fc-200d-2642-fe0f","⛹🏽‍♂":"26f9-1f3fd-200d-2642-fe0f","⛹🏾‍♂":"26f9-1f3fe-200d-2642-fe0f","⛹🏿‍♂":"26f9-1f3ff-200d-2642-fe0f","⛹‍♀️":"26f9-fe0f-200d-2640-fe0f","⛹️‍♀":"26f9-fe0f-200d-2640-fe0f","⛹🏻‍♀":"26f9-1f3fb-200d-2640-fe0f","⛹🏼‍♀":"26f9-1f3fc-200d-2640-fe0f","⛹🏽‍♀":"26f9-1f3fd-200d-2640-fe0f","⛹🏾‍♀":"26f9-1f3fe-200d-2640-fe0f","⛹🏿‍♀":"26f9-1f3ff-200d-2640-fe0f","🏋‍♂️":"1f3cb-fe0f-200d-2642-fe0f","🏋️‍♂":"1f3cb-fe0f-200d-2642-fe0f","🏋🏻‍♂":"1f3cb-1f3fb-200d-2642-fe0f","🏋🏼‍♂":"1f3cb-1f3fc-200d-2642-fe0f","🏋🏽‍♂":"1f3cb-1f3fd-200d-2642-fe0f","🏋🏾‍♂":"1f3cb-1f3fe-200d-2642-fe0f","🏋🏿‍♂":"1f3cb-1f3ff-200d-2642-fe0f","🏋‍♀️":"1f3cb-fe0f-200d-2640-fe0f","🏋️‍♀":"1f3cb-fe0f-200d-2640-fe0f","🏋🏻‍♀":"1f3cb-1f3fb-200d-2640-fe0f","🏋🏼‍♀":"1f3cb-1f3fc-200d-2640-fe0f","🏋🏽‍♀":"1f3cb-1f3fd-200d-2640-fe0f","🏋🏾‍♀":"1f3cb-1f3fe-200d-2640-fe0f","🏋🏿‍♀":"1f3cb-1f3ff-200d-2640-fe0f","🚴‍♂️":"1f6b4-200d-2642-fe0f","🚴🏻‍♂":"1f6b4-1f3fb-200d-2642-fe0f","🚴🏼‍♂":"1f6b4-1f3fc-200d-2642-fe0f","🚴🏽‍♂":"1f6b4-1f3fd-200d-2642-fe0f","🚴🏾‍♂":"1f6b4-1f3fe-200d-2642-fe0f","🚴🏿‍♂":"1f6b4-1f3ff-200d-2642-fe0f","🚴‍♀️":"1f6b4-200d-2640-fe0f","🚴🏻‍♀":"1f6b4-1f3fb-200d-2640-fe0f","🚴🏼‍♀":"1f6b4-1f3fc-200d-2640-fe0f","🚴🏽‍♀":"1f6b4-1f3fd-200d-2640-fe0f","🚴🏾‍♀":"1f6b4-1f3fe-200d-2640-fe0f","🚴🏿‍♀":"1f6b4-1f3ff-200d-2640-fe0f","🚵‍♂️":"1f6b5-200d-2642-fe0f","🚵🏻‍♂":"1f6b5-1f3fb-200d-2642-fe0f","🚵🏼‍♂":"1f6b5-1f3fc-200d-2642-fe0f","🚵🏽‍♂":"1f6b5-1f3fd-200d-2642-fe0f","🚵🏾‍♂":"1f6b5-1f3fe-200d-2642-fe0f","🚵🏿‍♂":"1f6b5-1f3ff-200d-2642-fe0f","🚵‍♀️":"1f6b5-200d-2640-fe0f","🚵🏻‍♀":"1f6b5-1f3fb-200d-2640-fe0f","🚵🏼‍♀":"1f6b5-1f3fc-200d-2640-fe0f","🚵🏽‍♀":"1f6b5-1f3fd-200d-2640-fe0f","🚵🏾‍♀":"1f6b5-1f3fe-200d-2640-fe0f","🚵🏿‍♀":"1f6b5-1f3ff-200d-2640-fe0f","🤸‍♂️":"1f938-200d-2642-fe0f","🤸🏻‍♂":"1f938-1f3fb-200d-2642-fe0f","🤸🏼‍♂":"1f938-1f3fc-200d-2642-fe0f","🤸🏽‍♂":"1f938-1f3fd-200d-2642-fe0f","🤸🏾‍♂":"1f938-1f3fe-200d-2642-fe0f","🤸🏿‍♂":"1f938-1f3ff-200d-2642-fe0f","🤸‍♀️":"1f938-200d-2640-fe0f","🤸🏻‍♀":"1f938-1f3fb-200d-2640-fe0f","🤸🏼‍♀":"1f938-1f3fc-200d-2640-fe0f","🤸🏽‍♀":"1f938-1f3fd-200d-2640-fe0f","🤸🏾‍♀":"1f938-1f3fe-200d-2640-fe0f","🤸🏿‍♀":"1f938-1f3ff-200d-2640-fe0f","🤼‍♂️":"1f93c-200d-2642-fe0f","🤼‍♀️":"1f93c-200d-2640-fe0f","🤽‍♂️":"1f93d-200d-2642-fe0f","🤽🏻‍♂":"1f93d-1f3fb-200d-2642-fe0f","🤽🏼‍♂":"1f93d-1f3fc-200d-2642-fe0f","🤽🏽‍♂":"1f93d-1f3fd-200d-2642-fe0f","🤽🏾‍♂":"1f93d-1f3fe-200d-2642-fe0f","🤽🏿‍♂":"1f93d-1f3ff-200d-2642-fe0f","🤽‍♀️":"1f93d-200d-2640-fe0f","🤽🏻‍♀":"1f93d-1f3fb-200d-2640-fe0f","🤽🏼‍♀":"1f93d-1f3fc-200d-2640-fe0f","🤽🏽‍♀":"1f93d-1f3fd-200d-2640-fe0f","🤽🏾‍♀":"1f93d-1f3fe-200d-2640-fe0f","🤽🏿‍♀":"1f93d-1f3ff-200d-2640-fe0f","🤾‍♂️":"1f93e-200d-2642-fe0f","🤾🏻‍♂":"1f93e-1f3fb-200d-2642-fe0f","🤾🏼‍♂":"1f93e-1f3fc-200d-2642-fe0f","🤾🏽‍♂":"1f93e-1f3fd-200d-2642-fe0f","🤾🏾‍♂":"1f93e-1f3fe-200d-2642-fe0f","🤾🏿‍♂":"1f93e-1f3ff-200d-2642-fe0f","🤾‍♀️":"1f93e-200d-2640-fe0f","🤾🏻‍♀":"1f93e-1f3fb-200d-2640-fe0f","🤾🏼‍♀":"1f93e-1f3fc-200d-2640-fe0f","🤾🏽‍♀":"1f93e-1f3fd-200d-2640-fe0f","🤾🏾‍♀":"1f93e-1f3fe-200d-2640-fe0f","🤾🏿‍♀":"1f93e-1f3ff-200d-2640-fe0f","🤹‍♂️":"1f939-200d-2642-fe0f","🤹🏻‍♂":"1f939-1f3fb-200d-2642-fe0f","🤹🏼‍♂":"1f939-1f3fc-200d-2642-fe0f","🤹🏽‍♂":"1f939-1f3fd-200d-2642-fe0f","🤹🏾‍♂":"1f939-1f3fe-200d-2642-fe0f","🤹🏿‍♂":"1f939-1f3ff-200d-2642-fe0f","🤹‍♀️":"1f939-200d-2640-fe0f","🤹🏻‍♀":"1f939-1f3fb-200d-2640-fe0f","🤹🏼‍♀":"1f939-1f3fc-200d-2640-fe0f","🤹🏽‍♀":"1f939-1f3fd-200d-2640-fe0f","🤹🏾‍♀":"1f939-1f3fe-200d-2640-fe0f","🤹🏿‍♀":"1f939-1f3ff-200d-2640-fe0f","🧘‍♂️":"1f9d8-200d-2642-fe0f","🧘🏻‍♂":"1f9d8-1f3fb-200d-2642-fe0f","🧘🏼‍♂":"1f9d8-1f3fc-200d-2642-fe0f","🧘🏽‍♂":"1f9d8-1f3fd-200d-2642-fe0f","🧘🏾‍♂":"1f9d8-1f3fe-200d-2642-fe0f","🧘🏿‍♂":"1f9d8-1f3ff-200d-2642-fe0f","🧘‍♀️":"1f9d8-200d-2640-fe0f","🧘🏻‍♀":"1f9d8-1f3fb-200d-2640-fe0f","🧘🏼‍♀":"1f9d8-1f3fc-200d-2640-fe0f","🧘🏽‍♀":"1f9d8-1f3fd-200d-2640-fe0f","🧘🏾‍♀":"1f9d8-1f3fe-200d-2640-fe0f","🧘🏿‍♀":"1f9d8-1f3ff-200d-2640-fe0f","🐻‍❄️":"1f43b-200d-2744-fe0f","🏳️‍🌈":"1f3f3-fe0f-200d-1f308","🏳‍⚧️":"1f3f3-fe0f-200d-26a7-fe0f","🏳️‍⚧":"1f3f3-fe0f-200d-26a7-fe0f","🏴‍☠️":"1f3f4-200d-2620-fe0f","👁️‍🗨️":"1f441-200d-1f5e8","🫱🏻‍🫲🏼":"1faf1-1f3fb-200d-1faf2-1f3fc","🫱🏻‍🫲🏽":"1faf1-1f3fb-200d-1faf2-1f3fd","🫱🏻‍🫲🏾":"1faf1-1f3fb-200d-1faf2-1f3fe","🫱🏻‍🫲🏿":"1faf1-1f3fb-200d-1faf2-1f3ff","🫱🏼‍🫲🏻":"1faf1-1f3fc-200d-1faf2-1f3fb","🫱🏼‍🫲🏽":"1faf1-1f3fc-200d-1faf2-1f3fd","🫱🏼‍🫲🏾":"1faf1-1f3fc-200d-1faf2-1f3fe","🫱🏼‍🫲🏿":"1faf1-1f3fc-200d-1faf2-1f3ff","🫱🏽‍🫲🏻":"1faf1-1f3fd-200d-1faf2-1f3fb","🫱🏽‍🫲🏼":"1faf1-1f3fd-200d-1faf2-1f3fc","🫱🏽‍🫲🏾":"1faf1-1f3fd-200d-1faf2-1f3fe","🫱🏽‍🫲🏿":"1faf1-1f3fd-200d-1faf2-1f3ff","🫱🏾‍🫲🏻":"1faf1-1f3fe-200d-1faf2-1f3fb","🫱🏾‍🫲🏼":"1faf1-1f3fe-200d-1faf2-1f3fc","🫱🏾‍🫲🏽":"1faf1-1f3fe-200d-1faf2-1f3fd","🫱🏾‍🫲🏿":"1faf1-1f3fe-200d-1faf2-1f3ff","🫱🏿‍🫲🏻":"1faf1-1f3ff-200d-1faf2-1f3fb","🫱🏿‍🫲🏼":"1faf1-1f3ff-200d-1faf2-1f3fc","🫱🏿‍🫲🏽":"1faf1-1f3ff-200d-1faf2-1f3fd","🫱🏿‍🫲🏾":"1faf1-1f3ff-200d-1faf2-1f3fe","🧔🏻‍♂️":"1f9d4-1f3fb-200d-2642-fe0f","🧔🏼‍♂️":"1f9d4-1f3fc-200d-2642-fe0f","🧔🏽‍♂️":"1f9d4-1f3fd-200d-2642-fe0f","🧔🏾‍♂️":"1f9d4-1f3fe-200d-2642-fe0f","🧔🏿‍♂️":"1f9d4-1f3ff-200d-2642-fe0f","🧔🏻‍♀️":"1f9d4-1f3fb-200d-2640-fe0f","🧔🏼‍♀️":"1f9d4-1f3fc-200d-2640-fe0f","🧔🏽‍♀️":"1f9d4-1f3fd-200d-2640-fe0f","🧔🏾‍♀️":"1f9d4-1f3fe-200d-2640-fe0f","🧔🏿‍♀️":"1f9d4-1f3ff-200d-2640-fe0f","👱🏻‍♀️":"1f471-1f3fb-200d-2640-fe0f","👱🏼‍♀️":"1f471-1f3fc-200d-2640-fe0f","👱🏽‍♀️":"1f471-1f3fd-200d-2640-fe0f","👱🏾‍♀️":"1f471-1f3fe-200d-2640-fe0f","👱🏿‍♀️":"1f471-1f3ff-200d-2640-fe0f","👱🏻‍♂️":"1f471-1f3fb-200d-2642-fe0f","👱🏼‍♂️":"1f471-1f3fc-200d-2642-fe0f","👱🏽‍♂️":"1f471-1f3fd-200d-2642-fe0f","👱🏾‍♂️":"1f471-1f3fe-200d-2642-fe0f","👱🏿‍♂️":"1f471-1f3ff-200d-2642-fe0f","🙍🏻‍♂️":"1f64d-1f3fb-200d-2642-fe0f","🙍🏼‍♂️":"1f64d-1f3fc-200d-2642-fe0f","🙍🏽‍♂️":"1f64d-1f3fd-200d-2642-fe0f","🙍🏾‍♂️":"1f64d-1f3fe-200d-2642-fe0f","🙍🏿‍♂️":"1f64d-1f3ff-200d-2642-fe0f","🙍🏻‍♀️":"1f64d-1f3fb-200d-2640-fe0f","🙍🏼‍♀️":"1f64d-1f3fc-200d-2640-fe0f","🙍🏽‍♀️":"1f64d-1f3fd-200d-2640-fe0f","🙍🏾‍♀️":"1f64d-1f3fe-200d-2640-fe0f","🙍🏿‍♀️":"1f64d-1f3ff-200d-2640-fe0f","🙎🏻‍♂️":"1f64e-1f3fb-200d-2642-fe0f","🙎🏼‍♂️":"1f64e-1f3fc-200d-2642-fe0f","🙎🏽‍♂️":"1f64e-1f3fd-200d-2642-fe0f","🙎🏾‍♂️":"1f64e-1f3fe-200d-2642-fe0f","🙎🏿‍♂️":"1f64e-1f3ff-200d-2642-fe0f","🙎🏻‍♀️":"1f64e-1f3fb-200d-2640-fe0f","🙎🏼‍♀️":"1f64e-1f3fc-200d-2640-fe0f","🙎🏽‍♀️":"1f64e-1f3fd-200d-2640-fe0f","🙎🏾‍♀️":"1f64e-1f3fe-200d-2640-fe0f","🙎🏿‍♀️":"1f64e-1f3ff-200d-2640-fe0f","🙅🏻‍♂️":"1f645-1f3fb-200d-2642-fe0f","🙅🏼‍♂️":"1f645-1f3fc-200d-2642-fe0f","🙅🏽‍♂️":"1f645-1f3fd-200d-2642-fe0f","🙅🏾‍♂️":"1f645-1f3fe-200d-2642-fe0f","🙅🏿‍♂️":"1f645-1f3ff-200d-2642-fe0f","🙅🏻‍♀️":"1f645-1f3fb-200d-2640-fe0f","🙅🏼‍♀️":"1f645-1f3fc-200d-2640-fe0f","🙅🏽‍♀️":"1f645-1f3fd-200d-2640-fe0f","🙅🏾‍♀️":"1f645-1f3fe-200d-2640-fe0f","🙅🏿‍♀️":"1f645-1f3ff-200d-2640-fe0f","🙆🏻‍♂️":"1f646-1f3fb-200d-2642-fe0f","🙆🏼‍♂️":"1f646-1f3fc-200d-2642-fe0f","🙆🏽‍♂️":"1f646-1f3fd-200d-2642-fe0f","🙆🏾‍♂️":"1f646-1f3fe-200d-2642-fe0f","🙆🏿‍♂️":"1f646-1f3ff-200d-2642-fe0f","🙆🏻‍♀️":"1f646-1f3fb-200d-2640-fe0f","🙆🏼‍♀️":"1f646-1f3fc-200d-2640-fe0f","🙆🏽‍♀️":"1f646-1f3fd-200d-2640-fe0f","🙆🏾‍♀️":"1f646-1f3fe-200d-2640-fe0f","🙆🏿‍♀️":"1f646-1f3ff-200d-2640-fe0f","💁🏻‍♂️":"1f481-1f3fb-200d-2642-fe0f","💁🏼‍♂️":"1f481-1f3fc-200d-2642-fe0f","💁🏽‍♂️":"1f481-1f3fd-200d-2642-fe0f","💁🏾‍♂️":"1f481-1f3fe-200d-2642-fe0f","💁🏿‍♂️":"1f481-1f3ff-200d-2642-fe0f","💁🏻‍♀️":"1f481-1f3fb-200d-2640-fe0f","💁🏼‍♀️":"1f481-1f3fc-200d-2640-fe0f","💁🏽‍♀️":"1f481-1f3fd-200d-2640-fe0f","💁🏾‍♀️":"1f481-1f3fe-200d-2640-fe0f","💁🏿‍♀️":"1f481-1f3ff-200d-2640-fe0f","🙋🏻‍♂️":"1f64b-1f3fb-200d-2642-fe0f","🙋🏼‍♂️":"1f64b-1f3fc-200d-2642-fe0f","🙋🏽‍♂️":"1f64b-1f3fd-200d-2642-fe0f","🙋🏾‍♂️":"1f64b-1f3fe-200d-2642-fe0f","🙋🏿‍♂️":"1f64b-1f3ff-200d-2642-fe0f","🙋🏻‍♀️":"1f64b-1f3fb-200d-2640-fe0f","🙋🏼‍♀️":"1f64b-1f3fc-200d-2640-fe0f","🙋🏽‍♀️":"1f64b-1f3fd-200d-2640-fe0f","🙋🏾‍♀️":"1f64b-1f3fe-200d-2640-fe0f","🙋🏿‍♀️":"1f64b-1f3ff-200d-2640-fe0f","🧏🏻‍♂️":"1f9cf-1f3fb-200d-2642-fe0f","🧏🏼‍♂️":"1f9cf-1f3fc-200d-2642-fe0f","🧏🏽‍♂️":"1f9cf-1f3fd-200d-2642-fe0f","🧏🏾‍♂️":"1f9cf-1f3fe-200d-2642-fe0f","🧏🏿‍♂️":"1f9cf-1f3ff-200d-2642-fe0f","🧏🏻‍♀️":"1f9cf-1f3fb-200d-2640-fe0f","🧏🏼‍♀️":"1f9cf-1f3fc-200d-2640-fe0f","🧏🏽‍♀️":"1f9cf-1f3fd-200d-2640-fe0f","🧏🏾‍♀️":"1f9cf-1f3fe-200d-2640-fe0f","🧏🏿‍♀️":"1f9cf-1f3ff-200d-2640-fe0f","🙇🏻‍♂️":"1f647-1f3fb-200d-2642-fe0f","🙇🏼‍♂️":"1f647-1f3fc-200d-2642-fe0f","🙇🏽‍♂️":"1f647-1f3fd-200d-2642-fe0f","🙇🏾‍♂️":"1f647-1f3fe-200d-2642-fe0f","🙇🏿‍♂️":"1f647-1f3ff-200d-2642-fe0f","🙇🏻‍♀️":"1f647-1f3fb-200d-2640-fe0f","🙇🏼‍♀️":"1f647-1f3fc-200d-2640-fe0f","🙇🏽‍♀️":"1f647-1f3fd-200d-2640-fe0f","🙇🏾‍♀️":"1f647-1f3fe-200d-2640-fe0f","🙇🏿‍♀️":"1f647-1f3ff-200d-2640-fe0f","🤦🏻‍♂️":"1f926-1f3fb-200d-2642-fe0f","🤦🏼‍♂️":"1f926-1f3fc-200d-2642-fe0f","🤦🏽‍♂️":"1f926-1f3fd-200d-2642-fe0f","🤦🏾‍♂️":"1f926-1f3fe-200d-2642-fe0f","🤦🏿‍♂️":"1f926-1f3ff-200d-2642-fe0f","🤦🏻‍♀️":"1f926-1f3fb-200d-2640-fe0f","🤦🏼‍♀️":"1f926-1f3fc-200d-2640-fe0f","🤦🏽‍♀️":"1f926-1f3fd-200d-2640-fe0f","🤦🏾‍♀️":"1f926-1f3fe-200d-2640-fe0f","🤦🏿‍♀️":"1f926-1f3ff-200d-2640-fe0f","🤷🏻‍♂️":"1f937-1f3fb-200d-2642-fe0f","🤷🏼‍♂️":"1f937-1f3fc-200d-2642-fe0f","🤷🏽‍♂️":"1f937-1f3fd-200d-2642-fe0f","🤷🏾‍♂️":"1f937-1f3fe-200d-2642-fe0f","🤷🏿‍♂️":"1f937-1f3ff-200d-2642-fe0f","🤷🏻‍♀️":"1f937-1f3fb-200d-2640-fe0f","🤷🏼‍♀️":"1f937-1f3fc-200d-2640-fe0f","🤷🏽‍♀️":"1f937-1f3fd-200d-2640-fe0f","🤷🏾‍♀️":"1f937-1f3fe-200d-2640-fe0f","🤷🏿‍♀️":"1f937-1f3ff-200d-2640-fe0f","🧑🏻‍⚕️":"1f9d1-1f3fb-200d-2695-fe0f","🧑🏼‍⚕️":"1f9d1-1f3fc-200d-2695-fe0f","🧑🏽‍⚕️":"1f9d1-1f3fd-200d-2695-fe0f","🧑🏾‍⚕️":"1f9d1-1f3fe-200d-2695-fe0f","🧑🏿‍⚕️":"1f9d1-1f3ff-200d-2695-fe0f","👨🏻‍⚕️":"1f468-1f3fb-200d-2695-fe0f","👨🏼‍⚕️":"1f468-1f3fc-200d-2695-fe0f","👨🏽‍⚕️":"1f468-1f3fd-200d-2695-fe0f","👨🏾‍⚕️":"1f468-1f3fe-200d-2695-fe0f","👨🏿‍⚕️":"1f468-1f3ff-200d-2695-fe0f","👩🏻‍⚕️":"1f469-1f3fb-200d-2695-fe0f","👩🏼‍⚕️":"1f469-1f3fc-200d-2695-fe0f","👩🏽‍⚕️":"1f469-1f3fd-200d-2695-fe0f","👩🏾‍⚕️":"1f469-1f3fe-200d-2695-fe0f","👩🏿‍⚕️":"1f469-1f3ff-200d-2695-fe0f","🧑🏻‍⚖️":"1f9d1-1f3fb-200d-2696-fe0f","🧑🏼‍⚖️":"1f9d1-1f3fc-200d-2696-fe0f","🧑🏽‍⚖️":"1f9d1-1f3fd-200d-2696-fe0f","🧑🏾‍⚖️":"1f9d1-1f3fe-200d-2696-fe0f","🧑🏿‍⚖️":"1f9d1-1f3ff-200d-2696-fe0f","👨🏻‍⚖️":"1f468-1f3fb-200d-2696-fe0f","👨🏼‍⚖️":"1f468-1f3fc-200d-2696-fe0f","👨🏽‍⚖️":"1f468-1f3fd-200d-2696-fe0f","👨🏾‍⚖️":"1f468-1f3fe-200d-2696-fe0f","👨🏿‍⚖️":"1f468-1f3ff-200d-2696-fe0f","👩🏻‍⚖️":"1f469-1f3fb-200d-2696-fe0f","👩🏼‍⚖️":"1f469-1f3fc-200d-2696-fe0f","👩🏽‍⚖️":"1f469-1f3fd-200d-2696-fe0f","👩🏾‍⚖️":"1f469-1f3fe-200d-2696-fe0f","👩🏿‍⚖️":"1f469-1f3ff-200d-2696-fe0f","🧑🏻‍✈️":"1f9d1-1f3fb-200d-2708-fe0f","🧑🏼‍✈️":"1f9d1-1f3fc-200d-2708-fe0f","🧑🏽‍✈️":"1f9d1-1f3fd-200d-2708-fe0f","🧑🏾‍✈️":"1f9d1-1f3fe-200d-2708-fe0f","🧑🏿‍✈️":"1f9d1-1f3ff-200d-2708-fe0f","👨🏻‍✈️":"1f468-1f3fb-200d-2708-fe0f","👨🏼‍✈️":"1f468-1f3fc-200d-2708-fe0f","👨🏽‍✈️":"1f468-1f3fd-200d-2708-fe0f","👨🏾‍✈️":"1f468-1f3fe-200d-2708-fe0f","👨🏿‍✈️":"1f468-1f3ff-200d-2708-fe0f","👩🏻‍✈️":"1f469-1f3fb-200d-2708-fe0f","👩🏼‍✈️":"1f469-1f3fc-200d-2708-fe0f","👩🏽‍✈️":"1f469-1f3fd-200d-2708-fe0f","👩🏾‍✈️":"1f469-1f3fe-200d-2708-fe0f","👩🏿‍✈️":"1f469-1f3ff-200d-2708-fe0f","👮🏻‍♂️":"1f46e-1f3fb-200d-2642-fe0f","👮🏼‍♂️":"1f46e-1f3fc-200d-2642-fe0f","👮🏽‍♂️":"1f46e-1f3fd-200d-2642-fe0f","👮🏾‍♂️":"1f46e-1f3fe-200d-2642-fe0f","👮🏿‍♂️":"1f46e-1f3ff-200d-2642-fe0f","👮🏻‍♀️":"1f46e-1f3fb-200d-2640-fe0f","👮🏼‍♀️":"1f46e-1f3fc-200d-2640-fe0f","👮🏽‍♀️":"1f46e-1f3fd-200d-2640-fe0f","👮🏾‍♀️":"1f46e-1f3fe-200d-2640-fe0f","👮🏿‍♀️":"1f46e-1f3ff-200d-2640-fe0f","🕵️‍♂️":"1f575-fe0f-200d-2642-fe0f","🕵🏻‍♂️":"1f575-1f3fb-200d-2642-fe0f","🕵🏼‍♂️":"1f575-1f3fc-200d-2642-fe0f","🕵🏽‍♂️":"1f575-1f3fd-200d-2642-fe0f","🕵🏾‍♂️":"1f575-1f3fe-200d-2642-fe0f","🕵🏿‍♂️":"1f575-1f3ff-200d-2642-fe0f","🕵️‍♀️":"1f575-fe0f-200d-2640-fe0f","🕵🏻‍♀️":"1f575-1f3fb-200d-2640-fe0f","🕵🏼‍♀️":"1f575-1f3fc-200d-2640-fe0f","🕵🏽‍♀️":"1f575-1f3fd-200d-2640-fe0f","🕵🏾‍♀️":"1f575-1f3fe-200d-2640-fe0f","🕵🏿‍♀️":"1f575-1f3ff-200d-2640-fe0f","💂🏻‍♂️":"1f482-1f3fb-200d-2642-fe0f","💂🏼‍♂️":"1f482-1f3fc-200d-2642-fe0f","💂🏽‍♂️":"1f482-1f3fd-200d-2642-fe0f","💂🏾‍♂️":"1f482-1f3fe-200d-2642-fe0f","💂🏿‍♂️":"1f482-1f3ff-200d-2642-fe0f","💂🏻‍♀️":"1f482-1f3fb-200d-2640-fe0f","💂🏼‍♀️":"1f482-1f3fc-200d-2640-fe0f","💂🏽‍♀️":"1f482-1f3fd-200d-2640-fe0f","💂🏾‍♀️":"1f482-1f3fe-200d-2640-fe0f","💂🏿‍♀️":"1f482-1f3ff-200d-2640-fe0f","👷🏻‍♂️":"1f477-1f3fb-200d-2642-fe0f","👷🏼‍♂️":"1f477-1f3fc-200d-2642-fe0f","👷🏽‍♂️":"1f477-1f3fd-200d-2642-fe0f","👷🏾‍♂️":"1f477-1f3fe-200d-2642-fe0f","👷🏿‍♂️":"1f477-1f3ff-200d-2642-fe0f","👷🏻‍♀️":"1f477-1f3fb-200d-2640-fe0f","👷🏼‍♀️":"1f477-1f3fc-200d-2640-fe0f","👷🏽‍♀️":"1f477-1f3fd-200d-2640-fe0f","👷🏾‍♀️":"1f477-1f3fe-200d-2640-fe0f","👷🏿‍♀️":"1f477-1f3ff-200d-2640-fe0f","👳🏻‍♂️":"1f473-1f3fb-200d-2642-fe0f","👳🏼‍♂️":"1f473-1f3fc-200d-2642-fe0f","👳🏽‍♂️":"1f473-1f3fd-200d-2642-fe0f","👳🏾‍♂️":"1f473-1f3fe-200d-2642-fe0f","👳🏿‍♂️":"1f473-1f3ff-200d-2642-fe0f","👳🏻‍♀️":"1f473-1f3fb-200d-2640-fe0f","👳🏼‍♀️":"1f473-1f3fc-200d-2640-fe0f","👳🏽‍♀️":"1f473-1f3fd-200d-2640-fe0f","👳🏾‍♀️":"1f473-1f3fe-200d-2640-fe0f","👳🏿‍♀️":"1f473-1f3ff-200d-2640-fe0f","🤵🏻‍♂️":"1f935-1f3fb-200d-2642-fe0f","🤵🏼‍♂️":"1f935-1f3fc-200d-2642-fe0f","🤵🏽‍♂️":"1f935-1f3fd-200d-2642-fe0f","🤵🏾‍♂️":"1f935-1f3fe-200d-2642-fe0f","🤵🏿‍♂️":"1f935-1f3ff-200d-2642-fe0f","🤵🏻‍♀️":"1f935-1f3fb-200d-2640-fe0f","🤵🏼‍♀️":"1f935-1f3fc-200d-2640-fe0f","🤵🏽‍♀️":"1f935-1f3fd-200d-2640-fe0f","🤵🏾‍♀️":"1f935-1f3fe-200d-2640-fe0f","🤵🏿‍♀️":"1f935-1f3ff-200d-2640-fe0f","👰🏻‍♂️":"1f470-1f3fb-200d-2642-fe0f","👰🏼‍♂️":"1f470-1f3fc-200d-2642-fe0f","👰🏽‍♂️":"1f470-1f3fd-200d-2642-fe0f","👰🏾‍♂️":"1f470-1f3fe-200d-2642-fe0f","👰🏿‍♂️":"1f470-1f3ff-200d-2642-fe0f","👰🏻‍♀️":"1f470-1f3fb-200d-2640-fe0f","👰🏼‍♀️":"1f470-1f3fc-200d-2640-fe0f","👰🏽‍♀️":"1f470-1f3fd-200d-2640-fe0f","👰🏾‍♀️":"1f470-1f3fe-200d-2640-fe0f","👰🏿‍♀️":"1f470-1f3ff-200d-2640-fe0f","🦸🏻‍♂️":"1f9b8-1f3fb-200d-2642-fe0f","🦸🏼‍♂️":"1f9b8-1f3fc-200d-2642-fe0f","🦸🏽‍♂️":"1f9b8-1f3fd-200d-2642-fe0f","🦸🏾‍♂️":"1f9b8-1f3fe-200d-2642-fe0f","🦸🏿‍♂️":"1f9b8-1f3ff-200d-2642-fe0f","🦸🏻‍♀️":"1f9b8-1f3fb-200d-2640-fe0f","🦸🏼‍♀️":"1f9b8-1f3fc-200d-2640-fe0f","🦸🏽‍♀️":"1f9b8-1f3fd-200d-2640-fe0f","🦸🏾‍♀️":"1f9b8-1f3fe-200d-2640-fe0f","🦸🏿‍♀️":"1f9b8-1f3ff-200d-2640-fe0f","🦹🏻‍♂️":"1f9b9-1f3fb-200d-2642-fe0f","🦹🏼‍♂️":"1f9b9-1f3fc-200d-2642-fe0f","🦹🏽‍♂️":"1f9b9-1f3fd-200d-2642-fe0f","🦹🏾‍♂️":"1f9b9-1f3fe-200d-2642-fe0f","🦹🏿‍♂️":"1f9b9-1f3ff-200d-2642-fe0f","🦹🏻‍♀️":"1f9b9-1f3fb-200d-2640-fe0f","🦹🏼‍♀️":"1f9b9-1f3fc-200d-2640-fe0f","🦹🏽‍♀️":"1f9b9-1f3fd-200d-2640-fe0f","🦹🏾‍♀️":"1f9b9-1f3fe-200d-2640-fe0f","🦹🏿‍♀️":"1f9b9-1f3ff-200d-2640-fe0f","🧙🏻‍♂️":"1f9d9-1f3fb-200d-2642-fe0f","🧙🏼‍♂️":"1f9d9-1f3fc-200d-2642-fe0f","🧙🏽‍♂️":"1f9d9-1f3fd-200d-2642-fe0f","🧙🏾‍♂️":"1f9d9-1f3fe-200d-2642-fe0f","🧙🏿‍♂️":"1f9d9-1f3ff-200d-2642-fe0f","🧙🏻‍♀️":"1f9d9-1f3fb-200d-2640-fe0f","🧙🏼‍♀️":"1f9d9-1f3fc-200d-2640-fe0f","🧙🏽‍♀️":"1f9d9-1f3fd-200d-2640-fe0f","🧙🏾‍♀️":"1f9d9-1f3fe-200d-2640-fe0f","🧙🏿‍♀️":"1f9d9-1f3ff-200d-2640-fe0f","🧚🏻‍♂️":"1f9da-1f3fb-200d-2642-fe0f","🧚🏼‍♂️":"1f9da-1f3fc-200d-2642-fe0f","🧚🏽‍♂️":"1f9da-1f3fd-200d-2642-fe0f","🧚🏾‍♂️":"1f9da-1f3fe-200d-2642-fe0f","🧚🏿‍♂️":"1f9da-1f3ff-200d-2642-fe0f","🧚🏻‍♀️":"1f9da-1f3fb-200d-2640-fe0f","🧚🏼‍♀️":"1f9da-1f3fc-200d-2640-fe0f","🧚🏽‍♀️":"1f9da-1f3fd-200d-2640-fe0f","🧚🏾‍♀️":"1f9da-1f3fe-200d-2640-fe0f","🧚🏿‍♀️":"1f9da-1f3ff-200d-2640-fe0f","🧛🏻‍♂️":"1f9db-1f3fb-200d-2642-fe0f","🧛🏼‍♂️":"1f9db-1f3fc-200d-2642-fe0f","🧛🏽‍♂️":"1f9db-1f3fd-200d-2642-fe0f","🧛🏾‍♂️":"1f9db-1f3fe-200d-2642-fe0f","🧛🏿‍♂️":"1f9db-1f3ff-200d-2642-fe0f","🧛🏻‍♀️":"1f9db-1f3fb-200d-2640-fe0f","🧛🏼‍♀️":"1f9db-1f3fc-200d-2640-fe0f","🧛🏽‍♀️":"1f9db-1f3fd-200d-2640-fe0f","🧛🏾‍♀️":"1f9db-1f3fe-200d-2640-fe0f","🧛🏿‍♀️":"1f9db-1f3ff-200d-2640-fe0f","🧜🏻‍♂️":"1f9dc-1f3fb-200d-2642-fe0f","🧜🏼‍♂️":"1f9dc-1f3fc-200d-2642-fe0f","🧜🏽‍♂️":"1f9dc-1f3fd-200d-2642-fe0f","🧜🏾‍♂️":"1f9dc-1f3fe-200d-2642-fe0f","🧜🏿‍♂️":"1f9dc-1f3ff-200d-2642-fe0f","🧜🏻‍♀️":"1f9dc-1f3fb-200d-2640-fe0f","🧜🏼‍♀️":"1f9dc-1f3fc-200d-2640-fe0f","🧜🏽‍♀️":"1f9dc-1f3fd-200d-2640-fe0f","🧜🏾‍♀️":"1f9dc-1f3fe-200d-2640-fe0f","🧜🏿‍♀️":"1f9dc-1f3ff-200d-2640-fe0f","🧝🏻‍♂️":"1f9dd-1f3fb-200d-2642-fe0f","🧝🏼‍♂️":"1f9dd-1f3fc-200d-2642-fe0f","🧝🏽‍♂️":"1f9dd-1f3fd-200d-2642-fe0f","🧝🏾‍♂️":"1f9dd-1f3fe-200d-2642-fe0f","🧝🏿‍♂️":"1f9dd-1f3ff-200d-2642-fe0f","🧝🏻‍♀️":"1f9dd-1f3fb-200d-2640-fe0f","🧝🏼‍♀️":"1f9dd-1f3fc-200d-2640-fe0f","🧝🏽‍♀️":"1f9dd-1f3fd-200d-2640-fe0f","🧝🏾‍♀️":"1f9dd-1f3fe-200d-2640-fe0f","🧝🏿‍♀️":"1f9dd-1f3ff-200d-2640-fe0f","💆🏻‍♂️":"1f486-1f3fb-200d-2642-fe0f","💆🏼‍♂️":"1f486-1f3fc-200d-2642-fe0f","💆🏽‍♂️":"1f486-1f3fd-200d-2642-fe0f","💆🏾‍♂️":"1f486-1f3fe-200d-2642-fe0f","💆🏿‍♂️":"1f486-1f3ff-200d-2642-fe0f","💆🏻‍♀️":"1f486-1f3fb-200d-2640-fe0f","💆🏼‍♀️":"1f486-1f3fc-200d-2640-fe0f","💆🏽‍♀️":"1f486-1f3fd-200d-2640-fe0f","💆🏾‍♀️":"1f486-1f3fe-200d-2640-fe0f","💆🏿‍♀️":"1f486-1f3ff-200d-2640-fe0f","💇🏻‍♂️":"1f487-1f3fb-200d-2642-fe0f","💇🏼‍♂️":"1f487-1f3fc-200d-2642-fe0f","💇🏽‍♂️":"1f487-1f3fd-200d-2642-fe0f","💇🏾‍♂️":"1f487-1f3fe-200d-2642-fe0f","💇🏿‍♂️":"1f487-1f3ff-200d-2642-fe0f","💇🏻‍♀️":"1f487-1f3fb-200d-2640-fe0f","💇🏼‍♀️":"1f487-1f3fc-200d-2640-fe0f","💇🏽‍♀️":"1f487-1f3fd-200d-2640-fe0f","💇🏾‍♀️":"1f487-1f3fe-200d-2640-fe0f","💇🏿‍♀️":"1f487-1f3ff-200d-2640-fe0f","🚶🏻‍♂️":"1f6b6-1f3fb-200d-2642-fe0f","🚶🏼‍♂️":"1f6b6-1f3fc-200d-2642-fe0f","🚶🏽‍♂️":"1f6b6-1f3fd-200d-2642-fe0f","🚶🏾‍♂️":"1f6b6-1f3fe-200d-2642-fe0f","🚶🏿‍♂️":"1f6b6-1f3ff-200d-2642-fe0f","🚶🏻‍♀️":"1f6b6-1f3fb-200d-2640-fe0f","🚶🏼‍♀️":"1f6b6-1f3fc-200d-2640-fe0f","🚶🏽‍♀️":"1f6b6-1f3fd-200d-2640-fe0f","🚶🏾‍♀️":"1f6b6-1f3fe-200d-2640-fe0f","🚶🏿‍♀️":"1f6b6-1f3ff-200d-2640-fe0f","🧍🏻‍♂️":"1f9cd-1f3fb-200d-2642-fe0f","🧍🏼‍♂️":"1f9cd-1f3fc-200d-2642-fe0f","🧍🏽‍♂️":"1f9cd-1f3fd-200d-2642-fe0f","🧍🏾‍♂️":"1f9cd-1f3fe-200d-2642-fe0f","🧍🏿‍♂️":"1f9cd-1f3ff-200d-2642-fe0f","🧍🏻‍♀️":"1f9cd-1f3fb-200d-2640-fe0f","🧍🏼‍♀️":"1f9cd-1f3fc-200d-2640-fe0f","🧍🏽‍♀️":"1f9cd-1f3fd-200d-2640-fe0f","🧍🏾‍♀️":"1f9cd-1f3fe-200d-2640-fe0f","🧍🏿‍♀️":"1f9cd-1f3ff-200d-2640-fe0f","🧎🏻‍♂️":"1f9ce-1f3fb-200d-2642-fe0f","🧎🏼‍♂️":"1f9ce-1f3fc-200d-2642-fe0f","🧎🏽‍♂️":"1f9ce-1f3fd-200d-2642-fe0f","🧎🏾‍♂️":"1f9ce-1f3fe-200d-2642-fe0f","🧎🏿‍♂️":"1f9ce-1f3ff-200d-2642-fe0f","🧎🏻‍♀️":"1f9ce-1f3fb-200d-2640-fe0f","🧎🏼‍♀️":"1f9ce-1f3fc-200d-2640-fe0f","🧎🏽‍♀️":"1f9ce-1f3fd-200d-2640-fe0f","🧎🏾‍♀️":"1f9ce-1f3fe-200d-2640-fe0f","🧎🏿‍♀️":"1f9ce-1f3ff-200d-2640-fe0f","🏃🏻‍♂️":"1f3c3-1f3fb-200d-2642-fe0f","🏃🏼‍♂️":"1f3c3-1f3fc-200d-2642-fe0f","🏃🏽‍♂️":"1f3c3-1f3fd-200d-2642-fe0f","🏃🏾‍♂️":"1f3c3-1f3fe-200d-2642-fe0f","🏃🏿‍♂️":"1f3c3-1f3ff-200d-2642-fe0f","🏃🏻‍♀️":"1f3c3-1f3fb-200d-2640-fe0f","🏃🏼‍♀️":"1f3c3-1f3fc-200d-2640-fe0f","🏃🏽‍♀️":"1f3c3-1f3fd-200d-2640-fe0f","🏃🏾‍♀️":"1f3c3-1f3fe-200d-2640-fe0f","🏃🏿‍♀️":"1f3c3-1f3ff-200d-2640-fe0f","🧖🏻‍♂️":"1f9d6-1f3fb-200d-2642-fe0f","🧖🏼‍♂️":"1f9d6-1f3fc-200d-2642-fe0f","🧖🏽‍♂️":"1f9d6-1f3fd-200d-2642-fe0f","🧖🏾‍♂️":"1f9d6-1f3fe-200d-2642-fe0f","🧖🏿‍♂️":"1f9d6-1f3ff-200d-2642-fe0f","🧖🏻‍♀️":"1f9d6-1f3fb-200d-2640-fe0f","🧖🏼‍♀️":"1f9d6-1f3fc-200d-2640-fe0f","🧖🏽‍♀️":"1f9d6-1f3fd-200d-2640-fe0f","🧖🏾‍♀️":"1f9d6-1f3fe-200d-2640-fe0f","🧖🏿‍♀️":"1f9d6-1f3ff-200d-2640-fe0f","🧗🏻‍♂️":"1f9d7-1f3fb-200d-2642-fe0f","🧗🏼‍♂️":"1f9d7-1f3fc-200d-2642-fe0f","🧗🏽‍♂️":"1f9d7-1f3fd-200d-2642-fe0f","🧗🏾‍♂️":"1f9d7-1f3fe-200d-2642-fe0f","🧗🏿‍♂️":"1f9d7-1f3ff-200d-2642-fe0f","🧗🏻‍♀️":"1f9d7-1f3fb-200d-2640-fe0f","🧗🏼‍♀️":"1f9d7-1f3fc-200d-2640-fe0f","🧗🏽‍♀️":"1f9d7-1f3fd-200d-2640-fe0f","🧗🏾‍♀️":"1f9d7-1f3fe-200d-2640-fe0f","🧗🏿‍♀️":"1f9d7-1f3ff-200d-2640-fe0f","🏌️‍♂️":"1f3cc-fe0f-200d-2642-fe0f","🏌🏻‍♂️":"1f3cc-1f3fb-200d-2642-fe0f","🏌🏼‍♂️":"1f3cc-1f3fc-200d-2642-fe0f","🏌🏽‍♂️":"1f3cc-1f3fd-200d-2642-fe0f","🏌🏾‍♂️":"1f3cc-1f3fe-200d-2642-fe0f","🏌🏿‍♂️":"1f3cc-1f3ff-200d-2642-fe0f","🏌️‍♀️":"1f3cc-fe0f-200d-2640-fe0f","🏌🏻‍♀️":"1f3cc-1f3fb-200d-2640-fe0f","🏌🏼‍♀️":"1f3cc-1f3fc-200d-2640-fe0f","🏌🏽‍♀️":"1f3cc-1f3fd-200d-2640-fe0f","🏌🏾‍♀️":"1f3cc-1f3fe-200d-2640-fe0f","🏌🏿‍♀️":"1f3cc-1f3ff-200d-2640-fe0f","🏄🏻‍♂️":"1f3c4-1f3fb-200d-2642-fe0f","🏄🏼‍♂️":"1f3c4-1f3fc-200d-2642-fe0f","🏄🏽‍♂️":"1f3c4-1f3fd-200d-2642-fe0f","🏄🏾‍♂️":"1f3c4-1f3fe-200d-2642-fe0f","🏄🏿‍♂️":"1f3c4-1f3ff-200d-2642-fe0f","🏄🏻‍♀️":"1f3c4-1f3fb-200d-2640-fe0f","🏄🏼‍♀️":"1f3c4-1f3fc-200d-2640-fe0f","🏄🏽‍♀️":"1f3c4-1f3fd-200d-2640-fe0f","🏄🏾‍♀️":"1f3c4-1f3fe-200d-2640-fe0f","🏄🏿‍♀️":"1f3c4-1f3ff-200d-2640-fe0f","🚣🏻‍♂️":"1f6a3-1f3fb-200d-2642-fe0f","🚣🏼‍♂️":"1f6a3-1f3fc-200d-2642-fe0f","🚣🏽‍♂️":"1f6a3-1f3fd-200d-2642-fe0f","🚣🏾‍♂️":"1f6a3-1f3fe-200d-2642-fe0f","🚣🏿‍♂️":"1f6a3-1f3ff-200d-2642-fe0f","🚣🏻‍♀️":"1f6a3-1f3fb-200d-2640-fe0f","🚣🏼‍♀️":"1f6a3-1f3fc-200d-2640-fe0f","🚣🏽‍♀️":"1f6a3-1f3fd-200d-2640-fe0f","🚣🏾‍♀️":"1f6a3-1f3fe-200d-2640-fe0f","🚣🏿‍♀️":"1f6a3-1f3ff-200d-2640-fe0f","🏊🏻‍♂️":"1f3ca-1f3fb-200d-2642-fe0f","🏊🏼‍♂️":"1f3ca-1f3fc-200d-2642-fe0f","🏊🏽‍♂️":"1f3ca-1f3fd-200d-2642-fe0f","🏊🏾‍♂️":"1f3ca-1f3fe-200d-2642-fe0f","🏊🏿‍♂️":"1f3ca-1f3ff-200d-2642-fe0f","🏊🏻‍♀️":"1f3ca-1f3fb-200d-2640-fe0f","🏊🏼‍♀️":"1f3ca-1f3fc-200d-2640-fe0f","🏊🏽‍♀️":"1f3ca-1f3fd-200d-2640-fe0f","🏊🏾‍♀️":"1f3ca-1f3fe-200d-2640-fe0f","🏊🏿‍♀️":"1f3ca-1f3ff-200d-2640-fe0f","⛹️‍♂️":"26f9-fe0f-200d-2642-fe0f","⛹🏻‍♂️":"26f9-1f3fb-200d-2642-fe0f","⛹🏼‍♂️":"26f9-1f3fc-200d-2642-fe0f","⛹🏽‍♂️":"26f9-1f3fd-200d-2642-fe0f","⛹🏾‍♂️":"26f9-1f3fe-200d-2642-fe0f","⛹🏿‍♂️":"26f9-1f3ff-200d-2642-fe0f","⛹️‍♀️":"26f9-fe0f-200d-2640-fe0f","⛹🏻‍♀️":"26f9-1f3fb-200d-2640-fe0f","⛹🏼‍♀️":"26f9-1f3fc-200d-2640-fe0f","⛹🏽‍♀️":"26f9-1f3fd-200d-2640-fe0f","⛹🏾‍♀️":"26f9-1f3fe-200d-2640-fe0f","⛹🏿‍♀️":"26f9-1f3ff-200d-2640-fe0f","🏋️‍♂️":"1f3cb-fe0f-200d-2642-fe0f","🏋🏻‍♂️":"1f3cb-1f3fb-200d-2642-fe0f","🏋🏼‍♂️":"1f3cb-1f3fc-200d-2642-fe0f","🏋🏽‍♂️":"1f3cb-1f3fd-200d-2642-fe0f","🏋🏾‍♂️":"1f3cb-1f3fe-200d-2642-fe0f","🏋🏿‍♂️":"1f3cb-1f3ff-200d-2642-fe0f","🏋️‍♀️":"1f3cb-fe0f-200d-2640-fe0f","🏋🏻‍♀️":"1f3cb-1f3fb-200d-2640-fe0f","🏋🏼‍♀️":"1f3cb-1f3fc-200d-2640-fe0f","🏋🏽‍♀️":"1f3cb-1f3fd-200d-2640-fe0f","🏋🏾‍♀️":"1f3cb-1f3fe-200d-2640-fe0f","🏋🏿‍♀️":"1f3cb-1f3ff-200d-2640-fe0f","🚴🏻‍♂️":"1f6b4-1f3fb-200d-2642-fe0f","🚴🏼‍♂️":"1f6b4-1f3fc-200d-2642-fe0f","🚴🏽‍♂️":"1f6b4-1f3fd-200d-2642-fe0f","🚴🏾‍♂️":"1f6b4-1f3fe-200d-2642-fe0f","🚴🏿‍♂️":"1f6b4-1f3ff-200d-2642-fe0f","🚴🏻‍♀️":"1f6b4-1f3fb-200d-2640-fe0f","🚴🏼‍♀️":"1f6b4-1f3fc-200d-2640-fe0f","🚴🏽‍♀️":"1f6b4-1f3fd-200d-2640-fe0f","🚴🏾‍♀️":"1f6b4-1f3fe-200d-2640-fe0f","🚴🏿‍♀️":"1f6b4-1f3ff-200d-2640-fe0f","🚵🏻‍♂️":"1f6b5-1f3fb-200d-2642-fe0f","🚵🏼‍♂️":"1f6b5-1f3fc-200d-2642-fe0f","🚵🏽‍♂️":"1f6b5-1f3fd-200d-2642-fe0f","🚵🏾‍♂️":"1f6b5-1f3fe-200d-2642-fe0f","🚵🏿‍♂️":"1f6b5-1f3ff-200d-2642-fe0f","🚵🏻‍♀️":"1f6b5-1f3fb-200d-2640-fe0f","🚵🏼‍♀️":"1f6b5-1f3fc-200d-2640-fe0f","🚵🏽‍♀️":"1f6b5-1f3fd-200d-2640-fe0f","🚵🏾‍♀️":"1f6b5-1f3fe-200d-2640-fe0f","🚵🏿‍♀️":"1f6b5-1f3ff-200d-2640-fe0f","🤸🏻‍♂️":"1f938-1f3fb-200d-2642-fe0f","🤸🏼‍♂️":"1f938-1f3fc-200d-2642-fe0f","🤸🏽‍♂️":"1f938-1f3fd-200d-2642-fe0f","🤸🏾‍♂️":"1f938-1f3fe-200d-2642-fe0f","🤸🏿‍♂️":"1f938-1f3ff-200d-2642-fe0f","🤸🏻‍♀️":"1f938-1f3fb-200d-2640-fe0f","🤸🏼‍♀️":"1f938-1f3fc-200d-2640-fe0f","🤸🏽‍♀️":"1f938-1f3fd-200d-2640-fe0f","🤸🏾‍♀️":"1f938-1f3fe-200d-2640-fe0f","🤸🏿‍♀️":"1f938-1f3ff-200d-2640-fe0f","🤽🏻‍♂️":"1f93d-1f3fb-200d-2642-fe0f","🤽🏼‍♂️":"1f93d-1f3fc-200d-2642-fe0f","🤽🏽‍♂️":"1f93d-1f3fd-200d-2642-fe0f","🤽🏾‍♂️":"1f93d-1f3fe-200d-2642-fe0f","🤽🏿‍♂️":"1f93d-1f3ff-200d-2642-fe0f","🤽🏻‍♀️":"1f93d-1f3fb-200d-2640-fe0f","🤽🏼‍♀️":"1f93d-1f3fc-200d-2640-fe0f","🤽🏽‍♀️":"1f93d-1f3fd-200d-2640-fe0f","🤽🏾‍♀️":"1f93d-1f3fe-200d-2640-fe0f","🤽🏿‍♀️":"1f93d-1f3ff-200d-2640-fe0f","🤾🏻‍♂️":"1f93e-1f3fb-200d-2642-fe0f","🤾🏼‍♂️":"1f93e-1f3fc-200d-2642-fe0f","🤾🏽‍♂️":"1f93e-1f3fd-200d-2642-fe0f","🤾🏾‍♂️":"1f93e-1f3fe-200d-2642-fe0f","🤾🏿‍♂️":"1f93e-1f3ff-200d-2642-fe0f","🤾🏻‍♀️":"1f93e-1f3fb-200d-2640-fe0f","🤾🏼‍♀️":"1f93e-1f3fc-200d-2640-fe0f","🤾🏽‍♀️":"1f93e-1f3fd-200d-2640-fe0f","🤾🏾‍♀️":"1f93e-1f3fe-200d-2640-fe0f","🤾🏿‍♀️":"1f93e-1f3ff-200d-2640-fe0f","🤹🏻‍♂️":"1f939-1f3fb-200d-2642-fe0f","🤹🏼‍♂️":"1f939-1f3fc-200d-2642-fe0f","🤹🏽‍♂️":"1f939-1f3fd-200d-2642-fe0f","🤹🏾‍♂️":"1f939-1f3fe-200d-2642-fe0f","🤹🏿‍♂️":"1f939-1f3ff-200d-2642-fe0f","🤹🏻‍♀️":"1f939-1f3fb-200d-2640-fe0f","🤹🏼‍♀️":"1f939-1f3fc-200d-2640-fe0f","🤹🏽‍♀️":"1f939-1f3fd-200d-2640-fe0f","🤹🏾‍♀️":"1f939-1f3fe-200d-2640-fe0f","🤹🏿‍♀️":"1f939-1f3ff-200d-2640-fe0f","🧘🏻‍♂️":"1f9d8-1f3fb-200d-2642-fe0f","🧘🏼‍♂️":"1f9d8-1f3fc-200d-2642-fe0f","🧘🏽‍♂️":"1f9d8-1f3fd-200d-2642-fe0f","🧘🏾‍♂️":"1f9d8-1f3fe-200d-2642-fe0f","🧘🏿‍♂️":"1f9d8-1f3ff-200d-2642-fe0f","🧘🏻‍♀️":"1f9d8-1f3fb-200d-2640-fe0f","🧘🏼‍♀️":"1f9d8-1f3fc-200d-2640-fe0f","🧘🏽‍♀️":"1f9d8-1f3fd-200d-2640-fe0f","🧘🏾‍♀️":"1f9d8-1f3fe-200d-2640-fe0f","🧘🏿‍♀️":"1f9d8-1f3ff-200d-2640-fe0f","🧑‍🤝‍🧑":"1f9d1-200d-1f91d-200d-1f9d1","👩‍❤‍👨":"1f469-200d-2764-fe0f-200d-1f468","👨‍❤‍👨":"1f468-200d-2764-fe0f-200d-1f468","👩‍❤‍👩":"1f469-200d-2764-fe0f-200d-1f469","👨‍👩‍👦":"1f468-200d-1f469-200d-1f466","👨‍👩‍👧":"1f468-200d-1f469-200d-1f467","👨‍👨‍👦":"1f468-200d-1f468-200d-1f466","👨‍👨‍👧":"1f468-200d-1f468-200d-1f467","👩‍👩‍👦":"1f469-200d-1f469-200d-1f466","👩‍👩‍👧":"1f469-200d-1f469-200d-1f467","👨‍👦‍👦":"1f468-200d-1f466-200d-1f466","👨‍👧‍👦":"1f468-200d-1f467-200d-1f466","👨‍👧‍👧":"1f468-200d-1f467-200d-1f467","👩‍👦‍👦":"1f469-200d-1f466-200d-1f466","👩‍👧‍👦":"1f469-200d-1f467-200d-1f466","👩‍👧‍👧":"1f469-200d-1f467-200d-1f467","🏳️‍⚧️":"1f3f3-fe0f-200d-26a7-fe0f","👩‍❤️‍👨":"1f469-200d-2764-fe0f-200d-1f468","👨‍❤️‍👨":"1f468-200d-2764-fe0f-200d-1f468","👩‍❤️‍👩":"1f469-200d-2764-fe0f-200d-1f469","🧑🏻‍🤝‍🧑🏻":"1f9d1-1f3fb-200d-1f91d-200d-1f9d1-1f3fb","🧑🏻‍🤝‍🧑🏼":"1f9d1-1f3fb-200d-1f91d-200d-1f9d1-1f3fc","🧑🏻‍🤝‍🧑🏽":"1f9d1-1f3fb-200d-1f91d-200d-1f9d1-1f3fd","🧑🏻‍🤝‍🧑🏾":"1f9d1-1f3fb-200d-1f91d-200d-1f9d1-1f3fe","🧑🏻‍🤝‍🧑🏿":"1f9d1-1f3fb-200d-1f91d-200d-1f9d1-1f3ff","🧑🏼‍🤝‍🧑🏻":"1f9d1-1f3fc-200d-1f91d-200d-1f9d1-1f3fb","🧑🏼‍🤝‍🧑🏼":"1f9d1-1f3fc-200d-1f91d-200d-1f9d1-1f3fc","🧑🏼‍🤝‍🧑🏽":"1f9d1-1f3fc-200d-1f91d-200d-1f9d1-1f3fd","🧑🏼‍🤝‍🧑🏾":"1f9d1-1f3fc-200d-1f91d-200d-1f9d1-1f3fe","🧑🏼‍🤝‍🧑🏿":"1f9d1-1f3fc-200d-1f91d-200d-1f9d1-1f3ff","🧑🏽‍🤝‍🧑🏻":"1f9d1-1f3fd-200d-1f91d-200d-1f9d1-1f3fb","🧑🏽‍🤝‍🧑🏼":"1f9d1-1f3fd-200d-1f91d-200d-1f9d1-1f3fc","🧑🏽‍🤝‍🧑🏽":"1f9d1-1f3fd-200d-1f91d-200d-1f9d1-1f3fd","🧑🏽‍🤝‍🧑🏾":"1f9d1-1f3fd-200d-1f91d-200d-1f9d1-1f3fe","🧑🏽‍🤝‍🧑🏿":"1f9d1-1f3fd-200d-1f91d-200d-1f9d1-1f3ff","🧑🏾‍🤝‍🧑🏻":"1f9d1-1f3fe-200d-1f91d-200d-1f9d1-1f3fb","🧑🏾‍🤝‍🧑🏼":"1f9d1-1f3fe-200d-1f91d-200d-1f9d1-1f3fc","🧑🏾‍🤝‍🧑🏽":"1f9d1-1f3fe-200d-1f91d-200d-1f9d1-1f3fd","🧑🏾‍🤝‍🧑🏾":"1f9d1-1f3fe-200d-1f91d-200d-1f9d1-1f3fe","🧑🏾‍🤝‍🧑🏿":"1f9d1-1f3fe-200d-1f91d-200d-1f9d1-1f3ff","🧑🏿‍🤝‍🧑🏻":"1f9d1-1f3ff-200d-1f91d-200d-1f9d1-1f3fb","🧑🏿‍🤝‍🧑🏼":"1f9d1-1f3ff-200d-1f91d-200d-1f9d1-1f3fc","🧑🏿‍🤝‍🧑🏽":"1f9d1-1f3ff-200d-1f91d-200d-1f9d1-1f3fd","🧑🏿‍🤝‍🧑🏾":"1f9d1-1f3ff-200d-1f91d-200d-1f9d1-1f3fe","🧑🏿‍🤝‍🧑🏿":"1f9d1-1f3ff-200d-1f91d-200d-1f9d1-1f3ff","👩🏻‍🤝‍👩🏼":"1f469-1f3fb-200d-1f91d-200d-1f469-1f3fc","👩🏻‍🤝‍👩🏽":"1f469-1f3fb-200d-1f91d-200d-1f469-1f3fd","👩🏻‍🤝‍👩🏾":"1f469-1f3fb-200d-1f91d-200d-1f469-1f3fe","👩🏻‍🤝‍👩🏿":"1f469-1f3fb-200d-1f91d-200d-1f469-1f3ff","👩🏼‍🤝‍👩🏻":"1f469-1f3fc-200d-1f91d-200d-1f469-1f3fb","👩🏼‍🤝‍👩🏽":"1f469-1f3fc-200d-1f91d-200d-1f469-1f3fd","👩🏼‍🤝‍👩🏾":"1f469-1f3fc-200d-1f91d-200d-1f469-1f3fe","👩🏼‍🤝‍👩🏿":"1f469-1f3fc-200d-1f91d-200d-1f469-1f3ff","👩🏽‍🤝‍👩🏻":"1f469-1f3fd-200d-1f91d-200d-1f469-1f3fb","👩🏽‍🤝‍👩🏼":"1f469-1f3fd-200d-1f91d-200d-1f469-1f3fc","👩🏽‍🤝‍👩🏾":"1f469-1f3fd-200d-1f91d-200d-1f469-1f3fe","👩🏽‍🤝‍👩🏿":"1f469-1f3fd-200d-1f91d-200d-1f469-1f3ff","👩🏾‍🤝‍👩🏻":"1f469-1f3fe-200d-1f91d-200d-1f469-1f3fb","👩🏾‍🤝‍👩🏼":"1f469-1f3fe-200d-1f91d-200d-1f469-1f3fc","👩🏾‍🤝‍👩🏽":"1f469-1f3fe-200d-1f91d-200d-1f469-1f3fd","👩🏾‍🤝‍👩🏿":"1f469-1f3fe-200d-1f91d-200d-1f469-1f3ff","👩🏿‍🤝‍👩🏻":"1f469-1f3ff-200d-1f91d-200d-1f469-1f3fb","👩🏿‍🤝‍👩🏼":"1f469-1f3ff-200d-1f91d-200d-1f469-1f3fc","👩🏿‍🤝‍👩🏽":"1f469-1f3ff-200d-1f91d-200d-1f469-1f3fd","👩🏿‍🤝‍👩🏾":"1f469-1f3ff-200d-1f91d-200d-1f469-1f3fe","👩🏻‍🤝‍👨🏼":"1f469-1f3fb-200d-1f91d-200d-1f468-1f3fc","👩🏻‍🤝‍👨🏽":"1f469-1f3fb-200d-1f91d-200d-1f468-1f3fd","👩🏻‍🤝‍👨🏾":"1f469-1f3fb-200d-1f91d-200d-1f468-1f3fe","👩🏻‍🤝‍👨🏿":"1f469-1f3fb-200d-1f91d-200d-1f468-1f3ff","👩🏼‍🤝‍👨🏻":"1f469-1f3fc-200d-1f91d-200d-1f468-1f3fb","👩🏼‍🤝‍👨🏽":"1f469-1f3fc-200d-1f91d-200d-1f468-1f3fd","👩🏼‍🤝‍👨🏾":"1f469-1f3fc-200d-1f91d-200d-1f468-1f3fe","👩🏼‍🤝‍👨🏿":"1f469-1f3fc-200d-1f91d-200d-1f468-1f3ff","👩🏽‍🤝‍👨🏻":"1f469-1f3fd-200d-1f91d-200d-1f468-1f3fb","👩🏽‍🤝‍👨🏼":"1f469-1f3fd-200d-1f91d-200d-1f468-1f3fc","👩🏽‍🤝‍👨🏾":"1f469-1f3fd-200d-1f91d-200d-1f468-1f3fe","👩🏽‍🤝‍👨🏿":"1f469-1f3fd-200d-1f91d-200d-1f468-1f3ff","👩🏾‍🤝‍👨🏻":"1f469-1f3fe-200d-1f91d-200d-1f468-1f3fb","👩🏾‍🤝‍👨🏼":"1f469-1f3fe-200d-1f91d-200d-1f468-1f3fc","👩🏾‍🤝‍👨🏽":"1f469-1f3fe-200d-1f91d-200d-1f468-1f3fd","👩🏾‍🤝‍👨🏿":"1f469-1f3fe-200d-1f91d-200d-1f468-1f3ff","👩🏿‍🤝‍👨🏻":"1f469-1f3ff-200d-1f91d-200d-1f468-1f3fb","👩🏿‍🤝‍👨🏼":"1f469-1f3ff-200d-1f91d-200d-1f468-1f3fc","👩🏿‍🤝‍👨🏽":"1f469-1f3ff-200d-1f91d-200d-1f468-1f3fd","👩🏿‍🤝‍👨🏾":"1f469-1f3ff-200d-1f91d-200d-1f468-1f3fe","👨🏻‍🤝‍👨🏼":"1f468-1f3fb-200d-1f91d-200d-1f468-1f3fc","👨🏻‍🤝‍👨🏽":"1f468-1f3fb-200d-1f91d-200d-1f468-1f3fd","👨🏻‍🤝‍👨🏾":"1f468-1f3fb-200d-1f91d-200d-1f468-1f3fe","👨🏻‍🤝‍👨🏿":"1f468-1f3fb-200d-1f91d-200d-1f468-1f3ff","👨🏼‍🤝‍👨🏻":"1f468-1f3fc-200d-1f91d-200d-1f468-1f3fb","👨🏼‍🤝‍👨🏽":"1f468-1f3fc-200d-1f91d-200d-1f468-1f3fd","👨🏼‍🤝‍👨🏾":"1f468-1f3fc-200d-1f91d-200d-1f468-1f3fe","👨🏼‍🤝‍👨🏿":"1f468-1f3fc-200d-1f91d-200d-1f468-1f3ff","👨🏽‍🤝‍👨🏻":"1f468-1f3fd-200d-1f91d-200d-1f468-1f3fb","👨🏽‍🤝‍👨🏼":"1f468-1f3fd-200d-1f91d-200d-1f468-1f3fc","👨🏽‍🤝‍👨🏾":"1f468-1f3fd-200d-1f91d-200d-1f468-1f3fe","👨🏽‍🤝‍👨🏿":"1f468-1f3fd-200d-1f91d-200d-1f468-1f3ff","👨🏾‍🤝‍👨🏻":"1f468-1f3fe-200d-1f91d-200d-1f468-1f3fb","👨🏾‍🤝‍👨🏼":"1f468-1f3fe-200d-1f91d-200d-1f468-1f3fc","👨🏾‍🤝‍👨🏽":"1f468-1f3fe-200d-1f91d-200d-1f468-1f3fd","👨🏾‍🤝‍👨🏿":"1f468-1f3fe-200d-1f91d-200d-1f468-1f3ff","👨🏿‍🤝‍👨🏻":"1f468-1f3ff-200d-1f91d-200d-1f468-1f3fb","👨🏿‍🤝‍👨🏼":"1f468-1f3ff-200d-1f91d-200d-1f468-1f3fc","👨🏿‍🤝‍👨🏽":"1f468-1f3ff-200d-1f91d-200d-1f468-1f3fd","👨🏿‍🤝‍👨🏾":"1f468-1f3ff-200d-1f91d-200d-1f468-1f3fe","👩‍❤‍💋‍👨":"1f469-200d-2764-fe0f-200d-1f48b-200d-1f468","👨‍❤‍💋‍👨":"1f468-200d-2764-fe0f-200d-1f48b-200d-1f468","👩‍❤‍💋‍👩":"1f469-200d-2764-fe0f-200d-1f48b-200d-1f469","🧑🏻‍❤‍🧑🏼":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏻‍❤‍🧑🏽":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏻‍❤‍🧑🏾":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3fe","🧑🏻‍❤‍🧑🏿":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏼‍❤‍🧑🏻":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏼‍❤‍🧑🏽":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏼‍❤‍🧑🏾":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3fe","🧑🏼‍❤‍🧑🏿":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏽‍❤‍🧑🏻":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏽‍❤‍🧑🏼":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏽‍❤‍🧑🏾":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3fe","🧑🏽‍❤‍🧑🏿":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏾‍❤‍🧑🏻":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏾‍❤‍🧑🏼":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏾‍❤‍🧑🏽":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏾‍❤‍🧑🏿":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏿‍❤‍🧑🏻":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏿‍❤‍🧑🏼":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏿‍❤‍🧑🏽":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏿‍❤‍🧑🏾":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fe","👩🏻‍❤‍👨🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fb","👩🏻‍❤‍👨🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fc","👩🏻‍❤‍👨🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fd","👩🏻‍❤‍👨🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fe","👩🏻‍❤‍👨🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3ff","👩🏼‍❤‍👨🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fb","👩🏼‍❤‍👨🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fc","👩🏼‍❤‍👨🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fd","👩🏼‍❤‍👨🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fe","👩🏼‍❤‍👨🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3ff","👩🏽‍❤‍👨🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fb","👩🏽‍❤‍👨🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fc","👩🏽‍❤‍👨🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fd","👩🏽‍❤‍👨🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fe","👩🏽‍❤‍👨🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3ff","👩🏾‍❤‍👨🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fb","👩🏾‍❤‍👨🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fc","👩🏾‍❤‍👨🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fd","👩🏾‍❤‍👨🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fe","👩🏾‍❤‍👨🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3ff","👩🏿‍❤‍👨🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fb","👩🏿‍❤‍👨🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fc","👩🏿‍❤‍👨🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fd","👩🏿‍❤‍👨🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fe","👩🏿‍❤‍👨🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3ff","👨🏻‍❤‍👨🏻":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fb","👨🏻‍❤‍👨🏼":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fc","👨🏻‍❤‍👨🏽":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fd","👨🏻‍❤‍👨🏾":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fe","👨🏻‍❤‍👨🏿":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3ff","👨🏼‍❤‍👨🏻":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fb","👨🏼‍❤‍👨🏼":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fc","👨🏼‍❤‍👨🏽":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fd","👨🏼‍❤‍👨🏾":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fe","👨🏼‍❤‍👨🏿":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3ff","👨🏽‍❤‍👨🏻":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fb","👨🏽‍❤‍👨🏼":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fc","👨🏽‍❤‍👨🏽":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fd","👨🏽‍❤‍👨🏾":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fe","👨🏽‍❤‍👨🏿":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3ff","👨🏾‍❤‍👨🏻":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fb","👨🏾‍❤‍👨🏼":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fc","👨🏾‍❤‍👨🏽":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fd","👨🏾‍❤‍👨🏾":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fe","👨🏾‍❤‍👨🏿":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3ff","👨🏿‍❤‍👨🏻":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fb","👨🏿‍❤‍👨🏼":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fc","👨🏿‍❤‍👨🏽":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fd","👨🏿‍❤‍👨🏾":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fe","👨🏿‍❤‍👨🏿":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3ff","👩🏻‍❤‍👩🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fb","👩🏻‍❤‍👩🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fc","👩🏻‍❤‍👩🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fd","👩🏻‍❤‍👩🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fe","👩🏻‍❤‍👩🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3ff","👩🏼‍❤‍👩🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fb","👩🏼‍❤‍👩🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fc","👩🏼‍❤‍👩🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fd","👩🏼‍❤‍👩🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fe","👩🏼‍❤‍👩🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3ff","👩🏽‍❤‍👩🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fb","👩🏽‍❤‍👩🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fc","👩🏽‍❤‍👩🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fd","👩🏽‍❤‍👩🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fe","👩🏽‍❤‍👩🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3ff","👩🏾‍❤‍👩🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fb","👩🏾‍❤‍👩🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fc","👩🏾‍❤‍👩🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fd","👩🏾‍❤‍👩🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fe","👩🏾‍❤‍👩🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3ff","👩🏿‍❤‍👩🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fb","👩🏿‍❤‍👩🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fc","👩🏿‍❤‍👩🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fd","👩🏿‍❤‍👩🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fe","👩🏿‍❤‍👩🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3ff","👨‍👩‍👧‍👦":"1f468-200d-1f469-200d-1f467-200d-1f466","👨‍👩‍👦‍👦":"1f468-200d-1f469-200d-1f466-200d-1f466","👨‍👩‍👧‍👧":"1f468-200d-1f469-200d-1f467-200d-1f467","👨‍👨‍👧‍👦":"1f468-200d-1f468-200d-1f467-200d-1f466","👨‍👨‍👦‍👦":"1f468-200d-1f468-200d-1f466-200d-1f466","👨‍👨‍👧‍👧":"1f468-200d-1f468-200d-1f467-200d-1f467","👩‍👩‍👧‍👦":"1f469-200d-1f469-200d-1f467-200d-1f466","👩‍👩‍👦‍👦":"1f469-200d-1f469-200d-1f466-200d-1f466","👩‍👩‍👧‍👧":"1f469-200d-1f469-200d-1f467-200d-1f467","🏴󠁧󠁢󠁥󠁮󠁧󠁿":"1f3f4-e0067-e0062-e0065-e006e-e0067-e007f","🏴󠁧󠁢󠁳󠁣󠁴󠁿":"1f3f4-e0067-e0062-e0073-e0063-e0074-e007f","🏴󠁧󠁢󠁷󠁬󠁳󠁿":"1f3f4-e0067-e0062-e0077-e006c-e0073-e007f","👩‍❤️‍💋‍👨":"1f469-200d-2764-fe0f-200d-1f48b-200d-1f468","👨‍❤️‍💋‍👨":"1f468-200d-2764-fe0f-200d-1f48b-200d-1f468","👩‍❤️‍💋‍👩":"1f469-200d-2764-fe0f-200d-1f48b-200d-1f469","🧑🏻‍❤️‍🧑🏼":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏻‍❤️‍🧑🏽":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏻‍❤️‍🧑🏾":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3fe","🧑🏻‍❤️‍🧑🏿":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏼‍❤️‍🧑🏻":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏼‍❤️‍🧑🏽":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏼‍❤️‍🧑🏾":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3fe","🧑🏼‍❤️‍🧑🏿":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏽‍❤️‍🧑🏻":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏽‍❤️‍🧑🏼":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏽‍❤️‍🧑🏾":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3fe","🧑🏽‍❤️‍🧑🏿":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏾‍❤️‍🧑🏻":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏾‍❤️‍🧑🏼":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏾‍❤️‍🧑🏽":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏾‍❤️‍🧑🏿":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f9d1-1f3ff","🧑🏿‍❤️‍🧑🏻":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fb","🧑🏿‍❤️‍🧑🏼":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fc","🧑🏿‍❤️‍🧑🏽":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fd","🧑🏿‍❤️‍🧑🏾":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f9d1-1f3fe","👩🏻‍❤️‍👨🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fb","👩🏻‍❤️‍👨🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fc","👩🏻‍❤️‍👨🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fd","👩🏻‍❤️‍👨🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3fe","👩🏻‍❤️‍👨🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f468-1f3ff","👩🏼‍❤️‍👨🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fb","👩🏼‍❤️‍👨🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fc","👩🏼‍❤️‍👨🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fd","👩🏼‍❤️‍👨🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3fe","👩🏼‍❤️‍👨🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f468-1f3ff","👩🏽‍❤️‍👨🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fb","👩🏽‍❤️‍👨🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fc","👩🏽‍❤️‍👨🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fd","👩🏽‍❤️‍👨🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3fe","👩🏽‍❤️‍👨🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f468-1f3ff","👩🏾‍❤️‍👨🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fb","👩🏾‍❤️‍👨🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fc","👩🏾‍❤️‍👨🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fd","👩🏾‍❤️‍👨🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3fe","👩🏾‍❤️‍👨🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f468-1f3ff","👩🏿‍❤️‍👨🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fb","👩🏿‍❤️‍👨🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fc","👩🏿‍❤️‍👨🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fd","👩🏿‍❤️‍👨🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3fe","👩🏿‍❤️‍👨🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f468-1f3ff","👨🏻‍❤️‍👨🏻":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fb","👨🏻‍❤️‍👨🏼":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fc","👨🏻‍❤️‍👨🏽":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fd","👨🏻‍❤️‍👨🏾":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3fe","👨🏻‍❤️‍👨🏿":"1f468-1f3fb-200d-2764-fe0f-200d-1f468-1f3ff","👨🏼‍❤️‍👨🏻":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fb","👨🏼‍❤️‍👨🏼":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fc","👨🏼‍❤️‍👨🏽":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fd","👨🏼‍❤️‍👨🏾":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3fe","👨🏼‍❤️‍👨🏿":"1f468-1f3fc-200d-2764-fe0f-200d-1f468-1f3ff","👨🏽‍❤️‍👨🏻":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fb","👨🏽‍❤️‍👨🏼":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fc","👨🏽‍❤️‍👨🏽":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fd","👨🏽‍❤️‍👨🏾":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3fe","👨🏽‍❤️‍👨🏿":"1f468-1f3fd-200d-2764-fe0f-200d-1f468-1f3ff","👨🏾‍❤️‍👨🏻":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fb","👨🏾‍❤️‍👨🏼":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fc","👨🏾‍❤️‍👨🏽":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fd","👨🏾‍❤️‍👨🏾":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3fe","👨🏾‍❤️‍👨🏿":"1f468-1f3fe-200d-2764-fe0f-200d-1f468-1f3ff","👨🏿‍❤️‍👨🏻":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fb","👨🏿‍❤️‍👨🏼":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fc","👨🏿‍❤️‍👨🏽":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fd","👨🏿‍❤️‍👨🏾":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3fe","👨🏿‍❤️‍👨🏿":"1f468-1f3ff-200d-2764-fe0f-200d-1f468-1f3ff","👩🏻‍❤️‍👩🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fb","👩🏻‍❤️‍👩🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fc","👩🏻‍❤️‍👩🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fd","👩🏻‍❤️‍👩🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3fe","👩🏻‍❤️‍👩🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f469-1f3ff","👩🏼‍❤️‍👩🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fb","👩🏼‍❤️‍👩🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fc","👩🏼‍❤️‍👩🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fd","👩🏼‍❤️‍👩🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3fe","👩🏼‍❤️‍👩🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f469-1f3ff","👩🏽‍❤️‍👩🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fb","👩🏽‍❤️‍👩🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fc","👩🏽‍❤️‍👩🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fd","👩🏽‍❤️‍👩🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3fe","👩🏽‍❤️‍👩🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f469-1f3ff","👩🏾‍❤️‍👩🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fb","👩🏾‍❤️‍👩🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fc","👩🏾‍❤️‍👩🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fd","👩🏾‍❤️‍👩🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3fe","👩🏾‍❤️‍👩🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f469-1f3ff","👩🏿‍❤️‍👩🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fb","👩🏿‍❤️‍👩🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fc","👩🏿‍❤️‍👩🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fd","👩🏿‍❤️‍👩🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3fe","👩🏿‍❤️‍👩🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f469-1f3ff","🧑🏻‍❤‍💋‍🧑🏼":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏻‍❤‍💋‍🧑🏽":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏻‍❤‍💋‍🧑🏾":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","🧑🏻‍❤‍💋‍🧑🏿":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏼‍❤‍💋‍🧑🏻":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏼‍❤‍💋‍🧑🏽":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏼‍❤‍💋‍🧑🏾":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","🧑🏼‍❤‍💋‍🧑🏿":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏽‍❤‍💋‍🧑🏻":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏽‍❤‍💋‍🧑🏼":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏽‍❤‍💋‍🧑🏾":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","🧑🏽‍❤‍💋‍🧑🏿":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏾‍❤‍💋‍🧑🏻":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏾‍❤‍💋‍🧑🏼":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏾‍❤‍💋‍🧑🏽":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏾‍❤‍💋‍🧑🏿":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏿‍❤‍💋‍🧑🏻":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏿‍❤‍💋‍🧑🏼":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏿‍❤‍💋‍🧑🏽":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏿‍❤‍💋‍🧑🏾":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","👩🏻‍❤‍💋‍👨🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏻‍❤‍💋‍👨🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏻‍❤‍💋‍👨🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏻‍❤‍💋‍👨🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏻‍❤‍💋‍👨🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏼‍❤‍💋‍👨🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏼‍❤‍💋‍👨🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏼‍❤‍💋‍👨🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏼‍❤‍💋‍👨🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏼‍❤‍💋‍👨🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏽‍❤‍💋‍👨🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏽‍❤‍💋‍👨🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏽‍❤‍💋‍👨🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏽‍❤‍💋‍👨🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏽‍❤‍💋‍👨🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏾‍❤‍💋‍👨🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏾‍❤‍💋‍👨🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏾‍❤‍💋‍👨🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏾‍❤‍💋‍👨🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏾‍❤‍💋‍👨🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏿‍❤‍💋‍👨🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏿‍❤‍💋‍👨🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏿‍❤‍💋‍👨🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏿‍❤‍💋‍👨🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏿‍❤‍💋‍👨🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏻‍❤‍💋‍👨🏻":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏻‍❤‍💋‍👨🏼":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏻‍❤‍💋‍👨🏽":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏻‍❤‍💋‍👨🏾":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏻‍❤‍💋‍👨🏿":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏼‍❤‍💋‍👨🏻":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏼‍❤‍💋‍👨🏼":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏼‍❤‍💋‍👨🏽":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏼‍❤‍💋‍👨🏾":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏼‍❤‍💋‍👨🏿":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏽‍❤‍💋‍👨🏻":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏽‍❤‍💋‍👨🏼":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏽‍❤‍💋‍👨🏽":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏽‍❤‍💋‍👨🏾":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏽‍❤‍💋‍👨🏿":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏾‍❤‍💋‍👨🏻":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏾‍❤‍💋‍👨🏼":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏾‍❤‍💋‍👨🏽":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏾‍❤‍💋‍👨🏾":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏾‍❤‍💋‍👨🏿":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏿‍❤‍💋‍👨🏻":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏿‍❤‍💋‍👨🏼":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏿‍❤‍💋‍👨🏽":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏿‍❤‍💋‍👨🏾":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏿‍❤‍💋‍👨🏿":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏻‍❤‍💋‍👩🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏻‍❤‍💋‍👩🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏻‍❤‍💋‍👩🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏻‍❤‍💋‍👩🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏻‍❤‍💋‍👩🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏼‍❤‍💋‍👩🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏼‍❤‍💋‍👩🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏼‍❤‍💋‍👩🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏼‍❤‍💋‍👩🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏼‍❤‍💋‍👩🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏽‍❤‍💋‍👩🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏽‍❤‍💋‍👩🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏽‍❤‍💋‍👩🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏽‍❤‍💋‍👩🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏽‍❤‍💋‍👩🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏾‍❤‍💋‍👩🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏾‍❤‍💋‍👩🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏾‍❤‍💋‍👩🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏾‍❤‍💋‍👩🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏾‍❤‍💋‍👩🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏿‍❤‍💋‍👩🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏿‍❤‍💋‍👩🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏿‍❤‍💋‍👩🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏿‍❤‍💋‍👩🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏿‍❤‍💋‍👩🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","🧑🏻‍❤️‍💋‍🧑🏼":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏻‍❤️‍💋‍🧑🏽":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏻‍❤️‍💋‍🧑🏾":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","🧑🏻‍❤️‍💋‍🧑🏿":"1f9d1-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏼‍❤️‍💋‍🧑🏻":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏼‍❤️‍💋‍🧑🏽":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏼‍❤️‍💋‍🧑🏾":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","🧑🏼‍❤️‍💋‍🧑🏿":"1f9d1-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏽‍❤️‍💋‍🧑🏻":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏽‍❤️‍💋‍🧑🏼":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏽‍❤️‍💋‍🧑🏾":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","🧑🏽‍❤️‍💋‍🧑🏿":"1f9d1-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏾‍❤️‍💋‍🧑🏻":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏾‍❤️‍💋‍🧑🏼":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏾‍❤️‍💋‍🧑🏽":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏾‍❤️‍💋‍🧑🏿":"1f9d1-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3ff","🧑🏿‍❤️‍💋‍🧑🏻":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fb","🧑🏿‍❤️‍💋‍🧑🏼":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fc","🧑🏿‍❤️‍💋‍🧑🏽":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fd","🧑🏿‍❤️‍💋‍🧑🏾":"1f9d1-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f9d1-1f3fe","👩🏻‍❤️‍💋‍👨🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏻‍❤️‍💋‍👨🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏻‍❤️‍💋‍👨🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏻‍❤️‍💋‍👨🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏻‍❤️‍💋‍👨🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏼‍❤️‍💋‍👨🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏼‍❤️‍💋‍👨🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏼‍❤️‍💋‍👨🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏼‍❤️‍💋‍👨🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏼‍❤️‍💋‍👨🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏽‍❤️‍💋‍👨🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏽‍❤️‍💋‍👨🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏽‍❤️‍💋‍👨🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏽‍❤️‍💋‍👨🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏽‍❤️‍💋‍👨🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏾‍❤️‍💋‍👨🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏾‍❤️‍💋‍👨🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏾‍❤️‍💋‍👨🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏾‍❤️‍💋‍👨🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏾‍❤️‍💋‍👨🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏿‍❤️‍💋‍👨🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👩🏿‍❤️‍💋‍👨🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👩🏿‍❤️‍💋‍👨🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👩🏿‍❤️‍💋‍👨🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👩🏿‍❤️‍💋‍👨🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏻‍❤️‍💋‍👨🏻":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏻‍❤️‍💋‍👨🏼":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏻‍❤️‍💋‍👨🏽":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏻‍❤️‍💋‍👨🏾":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏻‍❤️‍💋‍👨🏿":"1f468-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏼‍❤️‍💋‍👨🏻":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏼‍❤️‍💋‍👨🏼":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏼‍❤️‍💋‍👨🏽":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏼‍❤️‍💋‍👨🏾":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏼‍❤️‍💋‍👨🏿":"1f468-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏽‍❤️‍💋‍👨🏻":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏽‍❤️‍💋‍👨🏼":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏽‍❤️‍💋‍👨🏽":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏽‍❤️‍💋‍👨🏾":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏽‍❤️‍💋‍👨🏿":"1f468-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏾‍❤️‍💋‍👨🏻":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏾‍❤️‍💋‍👨🏼":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏾‍❤️‍💋‍👨🏽":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏾‍❤️‍💋‍👨🏾":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏾‍❤️‍💋‍👨🏿":"1f468-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👨🏿‍❤️‍💋‍👨🏻":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fb","👨🏿‍❤️‍💋‍👨🏼":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fc","👨🏿‍❤️‍💋‍👨🏽":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fd","👨🏿‍❤️‍💋‍👨🏾":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3fe","👨🏿‍❤️‍💋‍👨🏿":"1f468-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f468-1f3ff","👩🏻‍❤️‍💋‍👩🏻":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏻‍❤️‍💋‍👩🏼":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏻‍❤️‍💋‍👩🏽":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏻‍❤️‍💋‍👩🏾":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏻‍❤️‍💋‍👩🏿":"1f469-1f3fb-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏼‍❤️‍💋‍👩🏻":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏼‍❤️‍💋‍👩🏼":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏼‍❤️‍💋‍👩🏽":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏼‍❤️‍💋‍👩🏾":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏼‍❤️‍💋‍👩🏿":"1f469-1f3fc-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏽‍❤️‍💋‍👩🏻":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏽‍❤️‍💋‍👩🏼":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏽‍❤️‍💋‍👩🏽":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏽‍❤️‍💋‍👩🏾":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏽‍❤️‍💋‍👩🏿":"1f469-1f3fd-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏾‍❤️‍💋‍👩🏻":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏾‍❤️‍💋‍👩🏼":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏾‍❤️‍💋‍👩🏽":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏾‍❤️‍💋‍👩🏾":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏾‍❤️‍💋‍👩🏿":"1f469-1f3fe-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff","👩🏿‍❤️‍💋‍👩🏻":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fb","👩🏿‍❤️‍💋‍👩🏼":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fc","👩🏿‍❤️‍💋‍👩🏽":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fd","👩🏿‍❤️‍💋‍👩🏾":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3fe","👩🏿‍❤️‍💋‍👩🏿":"1f469-1f3ff-200d-2764-fe0f-200d-1f48b-200d-1f469-1f3ff"}
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_mart_data_light.ts b/app/javascript/flavours/blobfox/features/emoji/emoji_mart_data_light.ts
new file mode 100644
index 00000000000000..ffca1f8b0630a3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_mart_data_light.ts
@@ -0,0 +1,43 @@
+// The output of this module is designed to mimic emoji-mart's
+// "data" object, such that we can use it for a light version of emoji-mart's
+// emojiIndex.search functionality.
+import type { BaseEmoji } from 'emoji-mart';
+import type { Emoji } from 'emoji-mart/dist-es/utils/data';
+
+import type { Search, ShortCodesToEmojiData } from './emoji_compressed';
+import emojiCompressed from './emoji_compressed';
+import { unicodeToUnifiedName } from './unicode_to_unified_name';
+
+type Emojis = {
+  [key in NonNullable<keyof ShortCodesToEmojiData>]: {
+    native: BaseEmoji['native'];
+    search: Search;
+    short_names: Emoji['short_names'];
+    unified: Emoji['unified'];
+  };
+};
+
+const [
+  shortCodesToEmojiData,
+  skins,
+  categories,
+  short_names,
+  _emojisWithoutShortCodes,
+] = emojiCompressed;
+
+const emojis: Emojis = {};
+
+// decompress
+Object.keys(shortCodesToEmojiData).forEach((shortCode) => {
+  const [_filenameData, searchData] = shortCodesToEmojiData[shortCode];
+  const [native, short_names, search, unified] = searchData;
+
+  emojis[shortCode] = {
+    native,
+    search,
+    short_names: short_names ? [shortCode].concat(short_names) : undefined,
+    unified: unified ?? unicodeToUnifiedName(native),
+  };
+});
+
+export { emojis, skins, categories, short_names };
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_mart_search_light.js b/app/javascript/flavours/blobfox/features/emoji/emoji_mart_search_light.js
new file mode 100644
index 00000000000000..83e154b0b287e5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_mart_search_light.js
@@ -0,0 +1,185 @@
+// This code is largely borrowed from:
+// https://github.com/missive/emoji-mart/blob/5f2ffcc/src/utils/emoji-index.js
+
+import * as data from './emoji_mart_data_light';
+import { getData, getSanitizedData, uniq, intersect } from './emoji_utils';
+
+let originalPool = {};
+let index = {};
+let emojisList = {};
+let emoticonsList = {};
+let customEmojisList = [];
+
+for (let emoji in data.emojis) {
+  let emojiData = data.emojis[emoji];
+  let { short_names, emoticons } = emojiData;
+  let id = short_names[0];
+
+  if (emoticons) {
+    emoticons.forEach(emoticon => {
+      if (emoticonsList[emoticon]) {
+        return;
+      }
+
+      emoticonsList[emoticon] = id;
+    });
+  }
+
+  emojisList[id] = getSanitizedData(id);
+  originalPool[id] = emojiData;
+}
+
+function clearCustomEmojis(pool) {
+  customEmojisList.forEach((emoji) => {
+    let emojiId = emoji.id || emoji.short_names[0];
+
+    delete pool[emojiId];
+    delete emojisList[emojiId];
+  });
+}
+
+function addCustomToPool(custom, pool) {
+  if (customEmojisList.length) clearCustomEmojis(pool);
+
+  custom.forEach((emoji) => {
+    let emojiId = emoji.id || emoji.short_names[0];
+
+    if (emojiId && !pool[emojiId]) {
+      pool[emojiId] = getData(emoji);
+      emojisList[emojiId] = getSanitizedData(emoji);
+    }
+  });
+
+  customEmojisList = custom;
+  index = {};
+}
+
+function search(value, { emojisToShowFilter, maxResults, include, exclude, custom } = {}) {
+  if (custom !== undefined) {
+    if (customEmojisList !== custom)
+      addCustomToPool(custom, originalPool);
+  } else {
+    custom = [];
+  }
+
+  maxResults = maxResults || 75;
+  include = include || [];
+  exclude = exclude || [];
+
+  let results = null,
+    pool = originalPool;
+
+  if (value.length) {
+    if (value === '-' || value === '-1') {
+      return [emojisList['-1']];
+    }
+
+    let values = value.toLowerCase().split(/[\s|,\-_]+/),
+      allResults = [];
+
+    if (values.length > 2) {
+      values = [values[0], values[1]];
+    }
+
+    if (include.length || exclude.length) {
+      pool = {};
+
+      data.categories.forEach(category => {
+        let isIncluded = include && include.length ? include.indexOf(category.name.toLowerCase()) > -1 : true;
+        let isExcluded = exclude && exclude.length ? exclude.indexOf(category.name.toLowerCase()) > -1 : false;
+        if (!isIncluded || isExcluded) {
+          return;
+        }
+
+        category.emojis.forEach(emojiId => pool[emojiId] = data.emojis[emojiId]);
+      });
+
+      if (custom.length) {
+        let customIsIncluded = include && include.length ? include.indexOf('custom') > -1 : true;
+        let customIsExcluded = exclude && exclude.length ? exclude.indexOf('custom') > -1 : false;
+        if (customIsIncluded && !customIsExcluded) {
+          addCustomToPool(custom, pool);
+        }
+      }
+    }
+
+    const searchValue = (value) => {
+      let aPool = pool,
+        aIndex = index,
+        length = 0;
+
+      for (let charIndex = 0; charIndex < value.length; charIndex++) {
+        const char = value[charIndex];
+        length++;
+
+        aIndex[char] = aIndex[char] || {};
+        aIndex = aIndex[char];
+
+        if (!aIndex.results) {
+          let scores = {};
+
+          aIndex.results = [];
+          aIndex.pool = {};
+
+          for (let id in aPool) {
+            let emoji = aPool[id],
+              { search } = emoji,
+              sub = value.slice(0, length),
+              subIndex = search.indexOf(sub);
+
+            if (subIndex !== -1) {
+              let score = subIndex + 1;
+              if (sub === id) score = 0;
+
+              aIndex.results.push(emojisList[id]);
+              aIndex.pool[id] = emoji;
+
+              scores[id] = score;
+            }
+          }
+
+          aIndex.results.sort((a, b) => {
+            let aScore = scores[a.id],
+              bScore = scores[b.id];
+
+            return aScore - bScore;
+          });
+        }
+
+        aPool = aIndex.pool;
+      }
+
+      return aIndex.results;
+    };
+
+    if (values.length > 1) {
+      results = searchValue(value);
+    } else {
+      results = [];
+    }
+
+    allResults = values.map(searchValue).filter(a => a);
+
+    if (allResults.length > 1) {
+      allResults = intersect.apply(null, allResults);
+    } else if (allResults.length) {
+      allResults = allResults[0];
+    }
+
+    results = uniq(results.concat(allResults));
+  }
+
+  if (results) {
+    if (emojisToShowFilter) {
+      results = results.filter((result) => emojisToShowFilter(data.emojis[result.id]));
+    }
+
+    if (results && results.length > maxResults) {
+      results = results.slice(0, maxResults);
+    }
+  }
+
+  return results;
+}
+
+export { search };
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_picker.js b/app/javascript/flavours/blobfox/features/emoji/emoji_picker.js
new file mode 100644
index 00000000000000..8725d39ecd78a3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_picker.js
@@ -0,0 +1,7 @@
+import Emoji from 'emoji-mart/dist-es/components/emoji/emoji';
+import Picker from 'emoji-mart/dist-es/components/picker/picker';
+
+export {
+  Picker,
+  Emoji,
+};
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_unicode_mapping_light.ts b/app/javascript/flavours/blobfox/features/emoji/emoji_unicode_mapping_light.ts
new file mode 100644
index 00000000000000..191419496fb2a9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_unicode_mapping_light.ts
@@ -0,0 +1,60 @@
+// A mapping of unicode strings to an object containing the filename
+// (i.e. the svg filename) and a shortCode intended to be shown
+// as a "title" attribute in an HTML element (aka tooltip).
+
+import type {
+  FilenameData,
+  ShortCodesToEmojiDataKey,
+} from './emoji_compressed';
+import emojiCompressed from './emoji_compressed';
+import { unicodeToFilename } from './unicode_to_filename';
+
+type UnicodeMapping = {
+  [key in FilenameData[number][0]]: {
+    shortCode: ShortCodesToEmojiDataKey;
+    filename: FilenameData[number][number];
+  };
+};
+
+const [
+  shortCodesToEmojiData,
+  _skins,
+  _categories,
+  _short_names,
+  emojisWithoutShortCodes,
+] = emojiCompressed;
+
+// decompress
+const unicodeMapping: UnicodeMapping = {};
+
+function processEmojiMapData(
+  emojiMapData: FilenameData[number],
+  shortCode?: ShortCodesToEmojiDataKey,
+) {
+  const [native, _filename] = emojiMapData;
+  let filename = emojiMapData[1];
+  if (!filename) {
+    // filename name can be derived from unicodeToFilename
+    filename = unicodeToFilename(native);
+  }
+  unicodeMapping[native] = {
+    shortCode,
+    filename,
+  };
+}
+
+Object.keys(shortCodesToEmojiData).forEach(
+  (shortCode: ShortCodesToEmojiDataKey) => {
+    if (shortCode === undefined) return;
+    const [filenameData, _searchData] = shortCodesToEmojiData[shortCode];
+    filenameData.forEach((emojiMapData) => {
+      processEmojiMapData(emojiMapData, shortCode);
+    });
+  },
+);
+
+emojisWithoutShortCodes.forEach((emojiMapData) => {
+  processEmojiMapData(emojiMapData);
+});
+
+export { unicodeMapping };
diff --git a/app/javascript/flavours/blobfox/features/emoji/emoji_utils.js b/app/javascript/flavours/blobfox/features/emoji/emoji_utils.js
new file mode 100644
index 00000000000000..83bcc9d82f9f14
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/emoji_utils.js
@@ -0,0 +1,258 @@
+// This code is largely borrowed from:
+// https://github.com/missive/emoji-mart/blob/5f2ffcc/src/utils/index.js
+
+import * as data from './emoji_mart_data_light';
+
+const buildSearch = (data) => {
+  const search = [];
+
+  let addToSearch = (strings, split) => {
+    if (!strings) {
+      return;
+    }
+
+    (Array.isArray(strings) ? strings : [strings]).forEach((string) => {
+      (split ? string.split(/[-|_|\s]+/) : [string]).forEach((s) => {
+        s = s.toLowerCase();
+
+        if (search.indexOf(s) === -1) {
+          search.push(s);
+        }
+      });
+    });
+  };
+
+  addToSearch(data.short_names, true);
+  addToSearch(data.name, true);
+  addToSearch(data.keywords, false);
+  addToSearch(data.emoticons, false);
+
+  return search.join(',');
+};
+
+const _String = String;
+
+const stringFromCodePoint = _String.fromCodePoint || function () {
+  let MAX_SIZE = 0x4000;
+  let codeUnits = [];
+  let highSurrogate;
+  let lowSurrogate;
+  let index = -1;
+  let length = arguments.length;
+  if (!length) {
+    return '';
+  }
+  let result = '';
+  while (++index < length) {
+    let codePoint = Number(arguments[index]);
+    if (
+      !isFinite(codePoint) ||       // `NaN`, `+Infinity`, or `-Infinity`
+      codePoint < 0 ||              // not a valid Unicode code point
+      codePoint > 0x10FFFF ||       // not a valid Unicode code point
+      Math.floor(codePoint) !== codePoint // not an integer
+    ) {
+      throw RangeError('Invalid code point: ' + codePoint);
+    }
+    if (codePoint <= 0xFFFF) { // BMP code point
+      codeUnits.push(codePoint);
+    } else { // Astral code point; split in surrogate halves
+      // http://mathiasbynens.be/notes/javascript-encoding#surrogate-formulae
+      codePoint -= 0x10000;
+      highSurrogate = (codePoint >> 10) + 0xD800;
+      lowSurrogate = (codePoint % 0x400) + 0xDC00;
+      codeUnits.push(highSurrogate, lowSurrogate);
+    }
+    if (index + 1 === length || codeUnits.length > MAX_SIZE) {
+      result += String.fromCharCode.apply(null, codeUnits);
+      codeUnits.length = 0;
+    }
+  }
+  return result;
+};
+
+
+const _JSON = JSON;
+
+const COLONS_REGEX = /^(?::([^:]+):)(?::skin-tone-(\d):)?$/;
+const SKINS = [
+  '1F3FA', '1F3FB', '1F3FC',
+  '1F3FD', '1F3FE', '1F3FF',
+];
+
+function unifiedToNative(unified) {
+  let unicodes = unified.split('-'),
+    codePoints = unicodes.map((u) => `0x${u}`);
+
+  return stringFromCodePoint.apply(null, codePoints);
+}
+
+function sanitize(emoji) {
+  let { name, short_names, skin_tone, skin_variations, emoticons, unified, custom, imageUrl } = emoji,
+    id = emoji.id || short_names[0],
+    colons = `:${id}:`;
+
+  if (custom) {
+    return {
+      id,
+      name,
+      colons,
+      emoticons,
+      custom,
+      imageUrl,
+    };
+  }
+
+  if (skin_tone) {
+    colons += `:skin-tone-${skin_tone}:`;
+  }
+
+  return {
+    id,
+    name,
+    colons,
+    emoticons,
+    unified: unified.toLowerCase(),
+    skin: skin_tone || (skin_variations ? 1 : null),
+    native: unifiedToNative(unified),
+  };
+}
+
+function getSanitizedData() {
+  return sanitize(getData(...arguments));
+}
+
+function getData(emoji, skin, set) {
+  let emojiData = {};
+
+  if (typeof emoji === 'string') {
+    let matches = emoji.match(COLONS_REGEX);
+
+    if (matches) {
+      emoji = matches[1];
+
+      if (matches[2]) {
+        skin = parseInt(matches[2]);
+      }
+    }
+
+    if (Object.prototype.hasOwnProperty.call(data.short_names, emoji)) {
+      emoji = data.short_names[emoji];
+    }
+
+    if (Object.prototype.hasOwnProperty.call(data.emojis, emoji)) {
+      emojiData = data.emojis[emoji];
+    }
+  } else if (emoji.id) {
+    if (Object.prototype.hasOwnProperty.call(data.short_names, emoji.id)) {
+      emoji.id = data.short_names[emoji.id];
+    }
+
+    if (Object.prototype.hasOwnProperty.call(data.emojis, emoji.id)) {
+      emojiData = data.emojis[emoji.id];
+      skin = skin || emoji.skin;
+    }
+  }
+
+  if (!Object.keys(emojiData).length) {
+    emojiData = emoji;
+    emojiData.custom = true;
+
+    if (!emojiData.search) {
+      emojiData.search = buildSearch(emoji);
+    }
+  }
+
+  emojiData.emoticons = emojiData.emoticons || [];
+  emojiData.variations = emojiData.variations || [];
+
+  if (emojiData.skin_variations && skin > 1 && set) {
+    emojiData = JSON.parse(_JSON.stringify(emojiData));
+
+    let skinKey = SKINS[skin - 1],
+      variationData = emojiData.skin_variations[skinKey];
+
+    if (!variationData.variations && emojiData.variations) {
+      delete emojiData.variations;
+    }
+
+    if (variationData[`has_img_${set}`]) {
+      emojiData.skin_tone = skin;
+
+      for (let k in variationData) {
+        let v = variationData[k];
+        emojiData[k] = v;
+      }
+    }
+  }
+
+  if (emojiData.variations && emojiData.variations.length) {
+    emojiData = JSON.parse(_JSON.stringify(emojiData));
+    emojiData.unified = emojiData.variations.shift();
+  }
+
+  return emojiData;
+}
+
+function uniq(arr) {
+  return arr.reduce((acc, item) => {
+    if (acc.indexOf(item) === -1) {
+      acc.push(item);
+    }
+    return acc;
+  }, []);
+}
+
+function intersect(a, b) {
+  const uniqA = uniq(a);
+  const uniqB = uniq(b);
+
+  return uniqA.filter(item => uniqB.indexOf(item) >= 0);
+}
+
+function deepMerge(a, b) {
+  let o = {};
+
+  for (let key in a) {
+    let originalValue = a[key],
+      value = originalValue;
+
+    if (Object.prototype.hasOwnProperty.call(b, key)) {
+      value = b[key];
+    }
+
+    if (typeof value === 'object') {
+      value = deepMerge(originalValue, value);
+    }
+
+    o[key] = value;
+  }
+
+  return o;
+}
+
+// https://github.com/sonicdoe/measure-scrollbar
+function measureScrollbar() {
+  const div = document.createElement('div');
+
+  div.style.width = '100px';
+  div.style.height = '100px';
+  div.style.overflow = 'scroll';
+  div.style.position = 'absolute';
+  div.style.top = '-9999px';
+
+  document.body.appendChild(div);
+  const scrollbarWidth = div.offsetWidth - div.clientWidth;
+  document.body.removeChild(div);
+
+  return scrollbarWidth;
+}
+
+export {
+  getData,
+  getSanitizedData,
+  uniq,
+  intersect,
+  deepMerge,
+  unifiedToNative,
+  measureScrollbar,
+};
diff --git a/app/javascript/flavours/blobfox/features/emoji/unicode_to_filename.js b/app/javascript/flavours/blobfox/features/emoji/unicode_to_filename.js
new file mode 100644
index 00000000000000..3395c77174cc9d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/unicode_to_filename.js
@@ -0,0 +1,29 @@
+/* eslint-disable import/no-commonjs --
+   We need to use CommonJS here as its imported into a preval file (`emoji_compressed.js`) */
+
+// taken from:
+// https://github.com/twitter/twemoji/blob/47732c7/twemoji-generator.js#L848-L866
+exports.unicodeToFilename = (str) => {
+  let result = '';
+  let charCode = 0;
+  let p = 0;
+  let i = 0;
+  while (i < str.length) {
+    charCode = str.charCodeAt(i++);
+    if (p) {
+      if (result.length > 0) {
+        result += '-';
+      }
+      result += (0x10000 + ((p - 0xD800) << 10) + (charCode - 0xDC00)).toString(16);
+      p = 0;
+    } else if (0xD800 <= charCode && charCode <= 0xDBFF) {
+      p = charCode;
+    } else {
+      if (result.length > 0) {
+        result += '-';
+      }
+      result += charCode.toString(16);
+    }
+  }
+  return result;
+};
diff --git a/app/javascript/flavours/blobfox/features/emoji/unicode_to_unified_name.js b/app/javascript/flavours/blobfox/features/emoji/unicode_to_unified_name.js
new file mode 100644
index 00000000000000..108b911222adc0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/emoji/unicode_to_unified_name.js
@@ -0,0 +1,24 @@
+/* eslint-disable import/no-commonjs --
+   We need to use CommonJS here as its imported into a preval file (`emoji_compressed.js`) */
+
+function padLeft(str, num) {
+  while (str.length < num) {
+    str = '0' + str;
+  }
+
+  return str;
+}
+
+exports.unicodeToUnifiedName = (str) => {
+  let output = '';
+
+  for (let i = 0; i < str.length; i += 2) {
+    if (i > 0) {
+      output += '-';
+    }
+
+    output += padLeft(str.codePointAt(i).toString(16).toUpperCase(), 4);
+  }
+
+  return output;
+};
diff --git a/app/javascript/flavours/blobfox/features/explore/components/search_section.jsx b/app/javascript/flavours/blobfox/features/explore/components/search_section.jsx
new file mode 100644
index 00000000000000..c84e3f7cef6f84
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/components/search_section.jsx
@@ -0,0 +1,20 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+export const SearchSection = ({ title, onClickMore, children }) => (
+  <div className='search-results__section'>
+    <div className='search-results__section__header'>
+      <h3>{title}</h3>
+      {onClickMore && <button onClick={onClickMore}><FormattedMessage id='search_results.see_all' defaultMessage='See all' /></button>}
+    </div>
+
+    {children}
+  </div>
+);
+
+SearchSection.propTypes = {
+  title: PropTypes.node.isRequired,
+  onClickMore: PropTypes.func,
+  children: PropTypes.children,
+};
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/features/explore/components/story.jsx b/app/javascript/flavours/blobfox/features/explore/components/story.jsx
new file mode 100644
index 00000000000000..da1a170580d667
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/components/story.jsx
@@ -0,0 +1,61 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { Blurhash } from 'flavours/blobfox/components/blurhash';
+import { accountsCountRenderer } from 'flavours/blobfox/components/hashtag';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+
+export default class Story extends PureComponent {
+
+  static propTypes = {
+    url: PropTypes.string,
+    title: PropTypes.string,
+    lang: PropTypes.string,
+    publisher: PropTypes.string,
+    publishedAt: PropTypes.string,
+    author: PropTypes.string,
+    sharedTimes: PropTypes.number,
+    thumbnail: PropTypes.string,
+    thumbnailDescription: PropTypes.string,
+    blurhash: PropTypes.string,
+    expanded: PropTypes.bool,
+  };
+
+  state = {
+    thumbnailLoaded: false,
+  };
+
+  handleImageLoad = () => this.setState({ thumbnailLoaded: true });
+
+  render () {
+    const { expanded, url, title, lang, publisher, author, publishedAt, sharedTimes, thumbnail, thumbnailDescription, blurhash } = this.props;
+
+    const { thumbnailLoaded } = this.state;
+
+    return (
+      <a className={classNames('story', { expanded })} href={url} target='blank' rel='noopener'>
+        <div className='story__details'>
+          <div className='story__details__publisher'>{publisher ? <span lang={lang}>{publisher}</span> : <Skeleton width={50} />}{publishedAt && <> · <RelativeTimestamp timestamp={publishedAt} /></>}</div>
+          <div className='story__details__title' lang={lang}>{title ? title : <Skeleton />}</div>
+          <div className='story__details__shared'>{author && <><FormattedMessage id='link_preview.author' defaultMessage='By {name}' values={{ name: <strong>{author}</strong> }} /> · </>}{typeof sharedTimes === 'number' ? <ShortNumber value={sharedTimes} renderer={accountsCountRenderer} /> : <Skeleton width={100} />}</div>
+        </div>
+
+        <div className='story__thumbnail'>
+          {thumbnail ? (
+            <>
+              <div className={classNames('story__thumbnail__preview', { 'story__thumbnail__preview--hidden': thumbnailLoaded })}><Blurhash hash={blurhash} /></div>
+              <img src={thumbnail} onLoad={this.handleImageLoad} alt={thumbnailDescription} title={thumbnailDescription} lang={lang} />
+            </>
+          ) : <Skeleton />}
+        </div>
+      </a>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/explore/index.jsx b/app/javascript/flavours/blobfox/features/explore/index.jsx
new file mode 100644
index 00000000000000..a63368f0f5eee2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/index.jsx
@@ -0,0 +1,114 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+import { NavLink, Switch, Route } from 'react-router-dom';
+
+import { connect } from 'react-redux';
+
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import Search from 'flavours/blobfox/features/compose/containers/search_container';
+import { trendsEnabled } from 'flavours/blobfox/initial_state';
+
+import Links from './links';
+import SearchResults from './results';
+import Statuses from './statuses';
+import Suggestions from './suggestions';
+import Tags from './tags';
+
+const messages = defineMessages({
+  title: { id: 'explore.title', defaultMessage: 'Explore' },
+  searchResults: { id: 'explore.search_results', defaultMessage: 'Search results' },
+});
+
+const mapStateToProps = state => ({
+  layout: state.getIn(['meta', 'layout']),
+  isSearching: state.getIn(['search', 'submitted']) || !trendsEnabled,
+});
+
+class Explore extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+    isSearching: PropTypes.bool,
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  render() {
+    const { intl, multiColumn, isSearching } = this.props;
+    const { signedIn } = this.context.identity;
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.title)}>
+        <ColumnHeader
+          icon={isSearching ? 'search' : 'hashtag'}
+          title={intl.formatMessage(isSearching ? messages.searchResults : messages.title)}
+          onClick={this.handleHeaderClick}
+          multiColumn={multiColumn}
+        />
+
+        <div className='explore__search-header'>
+          <Search />
+        </div>
+
+        {isSearching ? (
+          <SearchResults />
+        ) : (
+          <>
+            <div className='account__section-headline'>
+              <NavLink exact to='/explore'>
+                <FormattedMessage tagName='div' id='explore.trending_statuses' defaultMessage='Posts' />
+              </NavLink>
+
+              <NavLink exact to='/explore/tags'>
+                <FormattedMessage tagName='div' id='explore.trending_tags' defaultMessage='Hashtags' />
+              </NavLink>
+
+              {signedIn && (
+                <NavLink exact to='/explore/suggestions'>
+                  <FormattedMessage tagName='div' id='explore.suggested_follows' defaultMessage='People' />
+                </NavLink>
+              )}
+
+              <NavLink exact to='/explore/links'>
+                <FormattedMessage tagName='div' id='explore.trending_links' defaultMessage='News' />
+              </NavLink>
+            </div>
+
+            <Switch>
+              <Route path='/explore/tags' component={Tags} />
+              <Route path='/explore/links' component={Links} />
+              <Route path='/explore/suggestions' component={Suggestions} />
+              <Route exact path={['/explore', '/explore/posts', '/search']}>
+                <Statuses multiColumn={multiColumn} />
+              </Route>
+            </Switch>
+
+            <Helmet>
+              <title>{intl.formatMessage(messages.title)}</title>
+              <meta name='robots' content={isSearching ? 'noindex' : 'all'} />
+            </Helmet>
+          </>
+        )}
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Explore));
diff --git a/app/javascript/flavours/blobfox/features/explore/links.jsx b/app/javascript/flavours/blobfox/features/explore/links.jsx
new file mode 100644
index 00000000000000..83836304f616de
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/links.jsx
@@ -0,0 +1,90 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { fetchTrendingLinks } from 'flavours/blobfox/actions/trends';
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import Story from './components/story';
+
+const mapStateToProps = state => ({
+  links: state.getIn(['trends', 'links', 'items']),
+  isLoading: state.getIn(['trends', 'links', 'isLoading']),
+});
+
+class Links extends PureComponent {
+
+  static propTypes = {
+    links: ImmutablePropTypes.list,
+    isLoading: PropTypes.bool,
+    dispatch: PropTypes.func.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  componentDidMount () {
+    const { dispatch, links, history } = this.props;
+
+    // If we're navigating back to the screen, do not trigger a reload
+    if (history.action === 'POP' && links.size > 0) {
+      return;
+    }
+
+    dispatch(fetchTrendingLinks());
+  }
+
+  render () {
+    const { isLoading, links } = this.props;
+
+    const banner = (
+      <DismissableBanner id='explore/links'>
+        <FormattedMessage id='dismissable_banner.explore_links' defaultMessage='These are news stories being shared the most on the social web today. Newer news stories posted by more different people are ranked higher.' />
+      </DismissableBanner>
+    );
+
+    if (!isLoading && links.isEmpty()) {
+      return (
+        <div className='explore__links scrollable scrollable--flex'>
+          {banner}
+
+          <div className='empty-column-indicator'>
+            <FormattedMessage id='empty_column.explore_statuses' defaultMessage='Nothing is trending right now. Check back later!' />
+          </div>
+        </div>
+      );
+    }
+
+    return (
+      <div className='explore__links scrollable' data-nosnippet>
+        {banner}
+
+        {isLoading ? (<LoadingIndicator />) : links.map((link, i) => (
+          <Story
+            key={link.get('id')}
+            expanded={i === 0}
+            lang={link.get('language')}
+            url={link.get('url')}
+            title={link.get('title')}
+            publisher={link.get('provider_name')}
+            publishedAt={link.get('published_at')}
+            author={link.get('author_name')}
+            sharedTimes={link.getIn(['history', 0, 'accounts']) * 1 + link.getIn(['history', 1, 'accounts']) * 1}
+            thumbnail={link.get('image')}
+            thumbnailDescription={link.get('image_description')}
+            blurhash={link.get('blurhash')}
+          />
+        ))}
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(withRouter(Links));
diff --git a/app/javascript/flavours/blobfox/features/explore/results.jsx b/app/javascript/flavours/blobfox/features/explore/results.jsx
new file mode 100644
index 00000000000000..63d2a41af67bba
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/results.jsx
@@ -0,0 +1,229 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { submitSearch, expandSearch } from 'flavours/blobfox/actions/search';
+import { ImmutableHashtag as Hashtag } from 'flavours/blobfox/components/hashtag';
+import { Icon } from 'flavours/blobfox/components/icon';
+import ScrollableList from 'flavours/blobfox/components/scrollable_list';
+import Account from 'flavours/blobfox/containers/account_container';
+import Status from 'flavours/blobfox/containers/status_container';
+
+import { SearchSection } from './components/search_section';
+
+const messages = defineMessages({
+  title: { id: 'search_results.title', defaultMessage: 'Search for {q}' },
+});
+
+const mapStateToProps = state => ({
+  isLoading: state.getIn(['search', 'isLoading']),
+  results: state.getIn(['search', 'results']),
+  q: state.getIn(['search', 'searchTerm']),
+  submittedType: state.getIn(['search', 'type']),
+});
+
+const INITIAL_PAGE_LIMIT = 10;
+const INITIAL_DISPLAY = 4;
+
+const hidePeek = list => {
+  if (list.size > INITIAL_PAGE_LIMIT && list.size % INITIAL_PAGE_LIMIT === 1) {
+    return list.skipLast(1);
+  } else {
+    return list;
+  }
+};
+
+const renderAccounts = accounts => hidePeek(accounts).map(id => (
+  <Account key={id} id={id} />
+));
+
+const renderHashtags = hashtags => hidePeek(hashtags).map(hashtag => (
+  <Hashtag key={hashtag.get('name')} hashtag={hashtag} />
+));
+
+const renderStatuses = statuses => hidePeek(statuses).map(id => (
+  <Status key={id} id={id} />
+));
+
+class Results extends PureComponent {
+
+  static propTypes = {
+    results: ImmutablePropTypes.contains({
+      accounts: ImmutablePropTypes.orderedSet,
+      statuses: ImmutablePropTypes.orderedSet,
+      hashtags: ImmutablePropTypes.orderedSet,
+    }),
+    isLoading: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    dispatch: PropTypes.func.isRequired,
+    q: PropTypes.string,
+    intl: PropTypes.object,
+    submittedType: PropTypes.oneOf(['accounts', 'statuses', 'hashtags']),
+  };
+
+  state = {
+    type: this.props.submittedType || 'all',
+  };
+
+  static getDerivedStateFromProps(props, state) {
+    if (props.submittedType !== state.type) {
+      return {
+        type: props.submittedType || 'all',
+      };
+    }
+
+    return null;
+  }
+
+  handleSelectAll = () => {
+    const { submittedType, dispatch } = this.props;
+
+    // If we originally searched for a specific type, we need to resubmit
+    // the query to get all types of results
+    if (submittedType) {
+      dispatch(submitSearch());
+    }
+
+    this.setState({ type: 'all' });
+  };
+
+  handleSelectAccounts = () => {
+    const { submittedType, dispatch } = this.props;
+
+    // If we originally searched for something else (but not everything),
+    // we need to resubmit the query for this specific type
+    if (submittedType !== 'accounts') {
+      dispatch(submitSearch('accounts'));
+    }
+
+    this.setState({ type: 'accounts' });
+  };
+
+  handleSelectHashtags = () => {
+    const { submittedType, dispatch } = this.props;
+
+    // If we originally searched for something else (but not everything),
+    // we need to resubmit the query for this specific type
+    if (submittedType !== 'hashtags') {
+      dispatch(submitSearch('hashtags'));
+    }
+
+    this.setState({ type: 'hashtags' });
+  };
+
+  handleSelectStatuses = () => {
+    const { submittedType, dispatch } = this.props;
+
+    // If we originally searched for something else (but not everything),
+    // we need to resubmit the query for this specific type
+    if (submittedType !== 'statuses') {
+      dispatch(submitSearch('statuses'));
+    }
+
+    this.setState({ type: 'statuses' });
+  };
+
+  handleLoadMoreAccounts = () => this._loadMore('accounts');
+  handleLoadMoreStatuses = () => this._loadMore('statuses');
+  handleLoadMoreHashtags = () => this._loadMore('hashtags');
+
+  _loadMore (type) {
+    const { dispatch } = this.props;
+    dispatch(expandSearch(type));
+  }
+
+  handleLoadMore = () => {
+    const { type } = this.state;
+
+    if (type !== 'all') {
+      this._loadMore(type);
+    }
+  };
+
+  render () {
+    const { intl, isLoading, q, results } = this.props;
+    const { type } = this.state;
+
+    // We request 1 more result than we display so we can tell if there'd be a next page
+    const hasMore = type !== 'all' ? results.get(type, ImmutableList()).size > INITIAL_PAGE_LIMIT && results.get(type).size % INITIAL_PAGE_LIMIT === 1 : false;
+
+    let filteredResults;
+
+    const accounts = results.get('accounts', ImmutableList());
+    const hashtags = results.get('hashtags', ImmutableList());
+    const statuses = results.get('statuses', ImmutableList());
+
+    switch(type) {
+    case 'all':
+      filteredResults = (accounts.size + hashtags.size + statuses.size) > 0 ? (
+        <>
+          {accounts.size > 0 && (
+            <SearchSection key='accounts' title={<><Icon id='users' fixedWidth /><FormattedMessage id='search_results.accounts' defaultMessage='Profiles' /></>} onClickMore={this.handleLoadMoreAccounts}>
+              {accounts.take(INITIAL_DISPLAY).map(id => <Account key={id} id={id} />)}
+            </SearchSection>
+          )}
+
+          {hashtags.size > 0 && (
+            <SearchSection key='hashtags' title={<><Icon id='hashtag' fixedWidth /><FormattedMessage id='search_results.hashtags' defaultMessage='Hashtags' /></>} onClickMore={this.handleLoadMoreHashtags}>
+              {hashtags.take(INITIAL_DISPLAY).map(hashtag => <Hashtag key={hashtag.get('name')} hashtag={hashtag} />)}
+            </SearchSection>
+          )}
+
+          {statuses.size > 0 && (
+            <SearchSection key='statuses' title={<><Icon id='quote-right' fixedWidth /><FormattedMessage id='search_results.statuses' defaultMessage='Posts' /></>} onClickMore={this.handleLoadMoreStatuses}>
+              {statuses.take(INITIAL_DISPLAY).map(id => <Status key={id} id={id} />)}
+            </SearchSection>
+          )}
+        </>
+      ) : [];
+      break;
+    case 'accounts':
+      filteredResults = renderAccounts(accounts);
+      break;
+    case 'hashtags':
+      filteredResults = renderHashtags(hashtags);
+      break;
+    case 'statuses':
+      filteredResults = renderStatuses(statuses);
+      break;
+    }
+
+    return (
+      <>
+        <div className='account__section-headline'>
+          <button onClick={this.handleSelectAll} className={type === 'all' ? 'active' : undefined}><FormattedMessage id='search_results.all' defaultMessage='All' /></button>
+          <button onClick={this.handleSelectAccounts} className={type === 'accounts' ? 'active' : undefined}><FormattedMessage id='search_results.accounts' defaultMessage='Profiles' /></button>
+          <button onClick={this.handleSelectHashtags} className={type === 'hashtags' ? 'active' : undefined}><FormattedMessage id='search_results.hashtags' defaultMessage='Hashtags' /></button>
+          <button onClick={this.handleSelectStatuses} className={type === 'statuses' ? 'active' : undefined}><FormattedMessage id='search_results.statuses' defaultMessage='Posts' /></button>
+        </div>
+
+        <div className='explore__search-results' data-nosnippet>
+          <ScrollableList
+            scrollKey='search-results'
+            isLoading={isLoading}
+            onLoadMore={this.handleLoadMore}
+            hasMore={hasMore}
+            emptyMessage={<FormattedMessage id='search_results.nothing_found' defaultMessage='Could not find anything for these search terms' />}
+            bindToDocument
+          >
+            {filteredResults}
+          </ScrollableList>
+        </div>
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title, { q })}</title>
+        </Helmet>
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Results));
diff --git a/app/javascript/flavours/blobfox/features/explore/statuses.jsx b/app/javascript/flavours/blobfox/features/explore/statuses.jsx
new file mode 100644
index 00000000000000..3039709fe78fd2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/statuses.jsx
@@ -0,0 +1,78 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+
+import { fetchTrendingStatuses, expandTrendingStatuses } from 'flavours/blobfox/actions/trends';
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import StatusList from 'flavours/blobfox/components/status_list';
+import { getStatusList } from 'flavours/blobfox/selectors';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const mapStateToProps = state => ({
+  statusIds: getStatusList(state, 'trending'),
+  isLoading: state.getIn(['status_lists', 'trending', 'isLoading'], true),
+  hasMore: !!state.getIn(['status_lists', 'trending', 'next']),
+});
+
+class Statuses extends PureComponent {
+
+  static propTypes = {
+    statusIds: ImmutablePropTypes.list,
+    isLoading: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    dispatch: PropTypes.func.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  componentDidMount () {
+    const { dispatch, statusIds, history } = this.props;
+
+    // If we're navigating back to the screen, do not trigger a reload
+    if (history.action === 'POP' && statusIds.size > 0) {
+      return;
+    }
+
+    dispatch(fetchTrendingStatuses());
+  }
+
+  handleLoadMore = debounce(() => {
+    const { dispatch } = this.props;
+    dispatch(expandTrendingStatuses());
+  }, 300, { leading: true });
+
+  render () {
+    const { isLoading, hasMore, statusIds, multiColumn } = this.props;
+
+    const emptyMessage = <FormattedMessage id='empty_column.explore_statuses' defaultMessage='Nothing is trending right now. Check back later!' />;
+
+    return (
+      <StatusList
+        trackScroll
+        prepend={<DismissableBanner id='explore/statuses'><FormattedMessage id='dismissable_banner.explore_statuses' defaultMessage='These are posts from across the social web that are gaining traction today. Newer posts with more boosts and favorites are ranked higher.' /></DismissableBanner>}
+        alwaysPrepend
+        timelineId='explore'
+        statusIds={statusIds}
+        scrollKey='explore-statuses'
+        hasMore={hasMore}
+        isLoading={isLoading}
+        onLoadMore={this.handleLoadMore}
+        emptyMessage={emptyMessage}
+        bindToDocument={!multiColumn}
+        withCounters
+      />
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(withRouter(Statuses));
diff --git a/app/javascript/flavours/blobfox/features/explore/suggestions.jsx b/app/javascript/flavours/blobfox/features/explore/suggestions.jsx
new file mode 100644
index 00000000000000..405b05c2f5eb06
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/suggestions.jsx
@@ -0,0 +1,70 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { fetchSuggestions, dismissSuggestion } from 'flavours/blobfox/actions/suggestions';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import AccountCard from 'flavours/blobfox/features/directory/components/account_card';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const mapStateToProps = state => ({
+  suggestions: state.getIn(['suggestions', 'items']),
+  isLoading: state.getIn(['suggestions', 'isLoading']),
+});
+
+class Suggestions extends PureComponent {
+
+  static propTypes = {
+    isLoading: PropTypes.bool,
+    suggestions: ImmutablePropTypes.list,
+    dispatch: PropTypes.func.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  componentDidMount () {
+    const { dispatch, suggestions, history } = this.props;
+
+    // If we're navigating back to the screen, do not trigger a reload
+    if (history.action === 'POP' && suggestions.size > 0) {
+      return;
+    }
+
+    dispatch(fetchSuggestions(true));
+  }
+
+  handleDismiss = (accountId) => {
+    const { dispatch } = this.props;
+    dispatch(dismissSuggestion(accountId));
+  };
+
+  render () {
+    const { isLoading, suggestions } = this.props;
+
+    if (!isLoading && suggestions.isEmpty()) {
+      return (
+        <div className='explore__suggestions scrollable scrollable--flex'>
+          <div className='empty-column-indicator'>
+            <FormattedMessage id='empty_column.explore_statuses' defaultMessage='Nothing is trending right now. Check back later!' />
+          </div>
+        </div>
+      );
+    }
+
+    return (
+      <div className='explore__suggestions scrollable' data-nosnippet>
+        {isLoading ? <LoadingIndicator /> : suggestions.map(suggestion => (
+          <AccountCard key={suggestion.get('account')} id={suggestion.get('account')} onDismiss={suggestion.get('source') === 'past_interactions' ? this.handleDismiss : null} />
+        ))}
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(withRouter(Suggestions));
diff --git a/app/javascript/flavours/blobfox/features/explore/tags.jsx b/app/javascript/flavours/blobfox/features/explore/tags.jsx
new file mode 100644
index 00000000000000..c08acac515ccce
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/explore/tags.jsx
@@ -0,0 +1,76 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { fetchTrendingHashtags } from 'flavours/blobfox/actions/trends';
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import { ImmutableHashtag as Hashtag } from 'flavours/blobfox/components/hashtag';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const mapStateToProps = state => ({
+  hashtags: state.getIn(['trends', 'tags', 'items']),
+  isLoadingHashtags: state.getIn(['trends', 'tags', 'isLoading']),
+});
+
+class Tags extends PureComponent {
+
+  static propTypes = {
+    hashtags: ImmutablePropTypes.list,
+    isLoading: PropTypes.bool,
+    dispatch: PropTypes.func.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  componentDidMount () {
+    const { dispatch, history, hashtags } = this.props;
+
+    // If we're navigating back to the screen, do not trigger a reload
+    if (history.action === 'POP' && hashtags.size > 0) {
+      return;
+    }
+
+    dispatch(fetchTrendingHashtags());
+  }
+
+  render () {
+    const { isLoading, hashtags } = this.props;
+
+    const banner = (
+      <DismissableBanner id='explore/tags'>
+        <FormattedMessage id='dismissable_banner.explore_tags' defaultMessage='These are hashtags that are gaining traction on the social web today. Hashtags that are used by more different people are ranked higher.' />
+      </DismissableBanner>
+    );
+
+    if (!isLoading && hashtags.isEmpty()) {
+      return (
+        <div className='explore__links scrollable scrollable--flex'>
+          {banner}
+
+          <div className='empty-column-indicator'>
+            <FormattedMessage id='empty_column.explore_statuses' defaultMessage='Nothing is trending right now. Check back later!' />
+          </div>
+        </div>
+      );
+    }
+
+    return (
+      <div className='scrollable explore__links' data-nosnippet>
+        {banner}
+
+        {isLoading ? (<LoadingIndicator />) : hashtags.map(hashtag => (
+          <Hashtag key={hashtag.get('name')} hashtag={hashtag} />
+        ))}
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(withRouter(Tags));
diff --git a/app/javascript/flavours/blobfox/features/favourited_statuses/index.jsx b/app/javascript/flavours/blobfox/features/favourited_statuses/index.jsx
new file mode 100644
index 00000000000000..2be1489dc325c7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/favourited_statuses/index.jsx
@@ -0,0 +1,113 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { addColumn, removeColumn, moveColumn } from 'flavours/blobfox/actions/columns';
+import { fetchFavouritedStatuses, expandFavouritedStatuses } from 'flavours/blobfox/actions/favourites';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import StatusList from 'flavours/blobfox/components/status_list';
+import Column from 'flavours/blobfox/features/ui/components/column';
+import { getStatusList } from 'flavours/blobfox/selectors';
+
+const messages = defineMessages({
+  heading: { id: 'column.favourites', defaultMessage: 'Favorites' },
+});
+
+const mapStateToProps = state => ({
+  statusIds: getStatusList(state, 'favourites'),
+  isLoading: state.getIn(['status_lists', 'favourites', 'isLoading'], true),
+  hasMore: !!state.getIn(['status_lists', 'favourites', 'next']),
+});
+
+class Favourites extends ImmutablePureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    statusIds: ImmutablePropTypes.list.isRequired,
+    intl: PropTypes.object.isRequired,
+    columnId: PropTypes.string,
+    multiColumn: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchFavouritedStatuses());
+  }
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('FAVOURITES', {}));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandFavouritedStatuses());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, statusIds, columnId, multiColumn, hasMore, isLoading } = this.props;
+    const pinned = !!columnId;
+
+    const emptyMessage = <FormattedMessage id='empty_column.favourited_statuses' defaultMessage="You don't have any favorite posts yet. When you favorite one, it will show up here." />;
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.heading)}>
+        <ColumnHeader
+          icon='star'
+          title={intl.formatMessage(messages.heading)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+          showBackButton
+        />
+
+        <StatusList
+          trackScroll={!pinned}
+          statusIds={statusIds}
+          scrollKey={`favourited_statuses-${columnId}`}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          onLoadMore={this.handleLoadMore}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        />
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.heading)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Favourites));
diff --git a/app/javascript/flavours/blobfox/features/favourites/index.jsx b/app/javascript/flavours/blobfox/features/favourites/index.jsx
new file mode 100644
index 00000000000000..4451d047c2a433
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/favourites/index.jsx
@@ -0,0 +1,114 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchFavourites, expandFavourites } from 'flavours/blobfox/actions/interactions';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import ScrollableList from 'flavours/blobfox/components/scrollable_list';
+import AccountContainer from 'flavours/blobfox/containers/account_container';
+import Column from 'flavours/blobfox/features/ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'column.favourited_by', defaultMessage: 'Favourited by' },
+  refresh: { id: 'refresh', defaultMessage: 'Refresh' },
+});
+
+const mapStateToProps = (state, props) => ({
+  accountIds: state.getIn(['user_lists', 'favourited_by', props.params.statusId, 'items']),
+  hasMore: !!state.getIn(['user_lists', 'favourited_by', props.params.statusId, 'next']),
+  isLoading: state.getIn(['user_lists', 'favourited_by', props.params.statusId, 'isLoading'], true),
+});
+
+class Favourites extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    accountIds: ImmutablePropTypes.list,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  UNSAFE_componentWillMount () {
+    if (!this.props.accountIds) {
+      this.props.dispatch(fetchFavourites(this.props.params.statusId));
+    }
+  }
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleRefresh = () => {
+    this.props.dispatch(fetchFavourites(this.props.params.statusId));
+  };
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandFavourites(this.props.params.statusId));
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, accountIds, hasMore, isLoading, multiColumn } = this.props;
+
+    if (!accountIds) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    const emptyMessage = <FormattedMessage id='empty_column.favourites' defaultMessage='No one has favorited this post yet. When someone does, they will show up here.' />;
+
+    return (
+      <Column ref={this.setRef}>
+        <ColumnHeader
+          icon='star'
+          title={intl.formatMessage(messages.heading)}
+          onClick={this.handleHeaderClick}
+          showBackButton
+          multiColumn={multiColumn}
+          extraButton={(
+            <button type='button' className='column-header__button' title={intl.formatMessage(messages.refresh)} aria-label={intl.formatMessage(messages.refresh)} onClick={this.handleRefresh}><Icon id='refresh' /></button>
+          )}
+        />
+
+        <ScrollableList
+          scrollKey='favourites'
+          onLoadMore={this.handleLoadMore}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {accountIds.map(id =>
+            <AccountContainer key={id} id={id} withNote={false} />,
+          )}
+        </ScrollableList>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Favourites));
diff --git a/app/javascript/flavours/blobfox/features/filters/added_to_filter.jsx b/app/javascript/flavours/blobfox/features/filters/added_to_filter.jsx
new file mode 100644
index 00000000000000..e4264b90ceaccb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/filters/added_to_filter.jsx
@@ -0,0 +1,106 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { Button } from 'flavours/blobfox/components/button';
+import { toServerSideType } from 'flavours/blobfox/utils/filters';
+
+const mapStateToProps = (state, { filterId }) => ({
+  filter: state.getIn(['filters', filterId]),
+});
+
+class AddedToFilter extends PureComponent {
+
+  static propTypes = {
+    onClose: PropTypes.func.isRequired,
+    contextType: PropTypes.string,
+    filter: ImmutablePropTypes.map.isRequired,
+    dispatch: PropTypes.func.isRequired,
+  };
+
+  handleCloseClick = () => {
+    const { onClose } = this.props;
+    onClose();
+  };
+
+  render () {
+    const { filter, contextType } = this.props;
+
+    let expiredMessage = null;
+    if (filter.get('expires_at') && filter.get('expires_at') < new Date()) {
+      expiredMessage = (
+        <>
+          <h4 className='report-dialog-modal__subtitle'><FormattedMessage id='filter_modal.added.expired_title' defaultMessage='Expired filter!' /></h4>
+          <p className='report-dialog-modal__lead'>
+            <FormattedMessage
+              id='filter_modal.added.expired_explanation'
+              defaultMessage='This filter category has expired, you will need to change the expiration date for it to apply.'
+            />
+          </p>
+        </>
+      );
+    }
+
+    let contextMismatchMessage = null;
+    if (contextType && !filter.get('context').includes(toServerSideType(contextType))) {
+      contextMismatchMessage = (
+        <>
+          <h4 className='report-dialog-modal__subtitle'><FormattedMessage id='filter_modal.added.context_mismatch_title' defaultMessage='Context mismatch!' /></h4>
+          <p className='report-dialog-modal__lead'>
+            <FormattedMessage
+              id='filter_modal.added.context_mismatch_explanation'
+              defaultMessage='This filter category does not apply to the context in which you have accessed this post. If you want the post to be filtered in this context too, you will have to edit the filter.'
+            />
+          </p>
+        </>
+      );
+    }
+
+    const settings_link = (
+      <a href={`/filters/${filter.get('id')}/edit`}>
+        <FormattedMessage
+          id='filter_modal.added.settings_link'
+          defaultMessage='settings page'
+        />
+      </a>
+    );
+
+    return (
+      <>
+        <h3 className='report-dialog-modal__title'><FormattedMessage id='filter_modal.added.title' defaultMessage='Filter added!' /></h3>
+        <p className='report-dialog-modal__lead'>
+          <FormattedMessage
+            id='filter_modal.added.short_explanation'
+            defaultMessage='This post has been added to the following filter category: {title}.'
+            values={{ title: filter.get('title') }}
+          />
+        </p>
+
+        {expiredMessage}
+        {contextMismatchMessage}
+
+        <h4 className='report-dialog-modal__subtitle'><FormattedMessage id='filter_modal.added.review_and_configure_title' defaultMessage='Filter settings' /></h4>
+        <p className='report-dialog-modal__lead'>
+          <FormattedMessage
+            id='filter_modal.added.review_and_configure'
+            defaultMessage='To review and further configure this filter category, go to the {settings_link}.'
+            values={{ settings_link }}
+          />
+        </p>
+
+        <div className='flex-spacer' />
+
+        <div className='report-dialog-modal__actions'>
+          <Button onClick={this.handleCloseClick}><FormattedMessage id='report.close' defaultMessage='Done' /></Button>
+        </div>
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(AddedToFilter);
diff --git a/app/javascript/flavours/blobfox/features/filters/select_filter.jsx b/app/javascript/flavours/blobfox/features/filters/select_filter.jsx
new file mode 100644
index 00000000000000..23ce2cc73b8a39
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/filters/select_filter.jsx
@@ -0,0 +1,196 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import fuzzysort from 'fuzzysort';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { toServerSideType } from 'flavours/blobfox/utils/filters';
+import { loupeIcon, deleteIcon } from 'flavours/blobfox/utils/icons';
+
+const messages = defineMessages({
+  search: { id: 'filter_modal.select_filter.search', defaultMessage: 'Search or create' },
+  clear: { id: 'emoji_button.clear', defaultMessage: 'Clear' },
+});
+
+const mapStateToProps = (state, { contextType }) => ({
+  filters: Array.from(state.get('filters').values()).map((filter) => [
+    filter.get('id'),
+    filter.get('title'),
+    filter.get('keywords')?.map((keyword) => keyword.get('keyword')).join('\n'),
+    filter.get('expires_at') && filter.get('expires_at') < new Date(),
+    contextType && !filter.get('context').includes(toServerSideType(contextType)),
+  ]),
+});
+
+class SelectFilter extends PureComponent {
+
+  static propTypes = {
+    onSelectFilter: PropTypes.func.isRequired,
+    onNewFilter: PropTypes.func.isRequired,
+    filters: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.object)),
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    searchValue: '',
+  };
+
+  search () {
+    const { filters } = this.props;
+    const { searchValue } = this.state;
+
+    if (searchValue === '') {
+      return filters;
+    }
+
+    return fuzzysort.go(searchValue, filters, {
+      keys: ['1', '2'],
+      limit: 5,
+      threshold: -10000,
+    }).map(result => result.obj);
+  }
+
+  renderItem = filter => {
+    let warning = null;
+    if (filter[3] || filter[4]) {
+      warning = (
+        <span className='language-dropdown__dropdown__results__item__common-name'>
+          (
+          {filter[3] && <FormattedMessage id='filter_modal.select_filter.expired' defaultMessage='expired' />}
+          {filter[3] && filter[4] && ', '}
+          {filter[4] && <FormattedMessage id='filter_modal.select_filter.context_mismatch' defaultMessage='does not apply to this context' />}
+          )
+        </span>
+      );
+    }
+
+    return (
+      <div key={filter[0]} role='button' tabIndex={0} data-index={filter[0]} className='language-dropdown__dropdown__results__item' onClick={this.handleItemClick} onKeyDown={this.handleKeyDown}>
+        <span className='language-dropdown__dropdown__results__item__native-name'>{filter[1]}</span> {warning}
+      </div>
+    );
+  };
+
+  renderCreateNew (name) {
+    return (
+      <div key='add-new-filter' role='button' tabIndex={0} className='language-dropdown__dropdown__results__item' onClick={this.handleNewFilterClick} onKeyDown={this.handleKeyDown}>
+        <Icon id='plus' fixedWidth /> <FormattedMessage id='filter_modal.select_filter.prompt_new' defaultMessage='New category: {name}' values={{ name }} />
+      </div>
+    );
+  }
+
+  handleSearchChange = ({ target }) => {
+    this.setState({ searchValue: target.value });
+  };
+
+  setListRef = c => {
+    this.listNode = c;
+  };
+
+  handleKeyDown = e => {
+    const index = Array.from(this.listNode.childNodes).findIndex(node => node === e.currentTarget);
+
+    let element = null;
+
+    switch(e.key) {
+    case ' ':
+    case 'Enter':
+      e.currentTarget.click();
+      break;
+    case 'ArrowDown':
+      element = this.listNode.childNodes[index + 1] || this.listNode.firstChild;
+      break;
+    case 'ArrowUp':
+      element = this.listNode.childNodes[index - 1] || this.listNode.lastChild;
+      break;
+    case 'Tab':
+      if (e.shiftKey) {
+        element = this.listNode.childNodes[index - 1] || this.listNode.lastChild;
+      } else {
+        element = this.listNode.childNodes[index + 1] || this.listNode.firstChild;
+      }
+      break;
+    case 'Home':
+      element = this.listNode.firstChild;
+      break;
+    case 'End':
+      element = this.listNode.lastChild;
+      break;
+    }
+
+    if (element) {
+      element.focus();
+      e.preventDefault();
+      e.stopPropagation();
+    }
+  };
+
+  handleSearchKeyDown = e => {
+    let element = null;
+
+    switch(e.key) {
+    case 'Tab':
+    case 'ArrowDown':
+      element = this.listNode.firstChild;
+
+      if (element) {
+        element.focus();
+        e.preventDefault();
+        e.stopPropagation();
+      }
+
+      break;
+    }
+  };
+
+  handleClear = () => {
+    this.setState({ searchValue: '' });
+  };
+
+  handleItemClick = e => {
+    const value = e.currentTarget.getAttribute('data-index');
+
+    e.preventDefault();
+
+    this.props.onSelectFilter(value);
+  };
+
+  handleNewFilterClick = e => {
+    e.preventDefault();
+
+    this.props.onNewFilter(this.state.searchValue);
+  };
+
+  render () {
+    const { intl } = this.props;
+
+    const { searchValue } = this.state;
+    const isSearching = searchValue !== '';
+    const results = this.search();
+
+    return (
+      <>
+        <h3 className='report-dialog-modal__title'><FormattedMessage id='filter_modal.select_filter.title' defaultMessage='Filter this post' /></h3>
+        <p className='report-dialog-modal__lead'><FormattedMessage id='filter_modal.select_filter.subtitle' defaultMessage='Use an existing category or create a new one' /></p>
+
+        <div className='emoji-mart-search'>
+          <input type='search' value={searchValue} onChange={this.handleSearchChange} onKeyDown={this.handleSearchKeyDown} placeholder={intl.formatMessage(messages.search)} autoFocus />
+          <button className='emoji-mart-search-icon' disabled={!isSearching} aria-label={intl.formatMessage(messages.clear)} onClick={this.handleClear}>{!isSearching ? loupeIcon : deleteIcon}</button>
+        </div>
+
+        <div className='language-dropdown__dropdown__results emoji-mart-scroll' role='listbox' ref={this.setListRef}>
+          {results.map(this.renderItem)}
+          {isSearching && this.renderCreateNew(searchValue) }
+        </div>
+
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(SelectFilter));
diff --git a/app/javascript/flavours/blobfox/features/firehose/index.jsx b/app/javascript/flavours/blobfox/features/firehose/index.jsx
new file mode 100644
index 00000000000000..86a3c894f1a593
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/firehose/index.jsx
@@ -0,0 +1,229 @@
+import PropTypes from 'prop-types';
+import { useRef, useCallback, useEffect } from 'react';
+
+import { useIntl, defineMessages, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+import { NavLink } from 'react-router-dom';
+
+import { addColumn } from 'flavours/blobfox/actions/columns';
+import { changeSetting } from 'flavours/blobfox/actions/settings';
+import { connectPublicStream, connectCommunityStream } from 'flavours/blobfox/actions/streaming';
+import { expandPublicTimeline, expandCommunityTimeline } from 'flavours/blobfox/actions/timelines';
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import SettingText from 'flavours/blobfox/components/setting_text';
+import initialState, { domain } from 'flavours/blobfox/initial_state';
+import { useAppDispatch, useAppSelector } from 'flavours/blobfox/store';
+
+import Column from '../../components/column';
+import ColumnHeader from '../../components/column_header';
+import SettingToggle from '../notifications/components/setting_toggle';
+import StatusListContainer from '../ui/containers/status_list_container';
+
+const messages = defineMessages({
+  title: { id: 'column.firehose', defaultMessage: 'Live feeds' },
+  filter_regex: { id: 'home.column_settings.filter_regex', defaultMessage: 'Filter out by regular expressions' },
+});
+
+// TODO: use a proper React context later on
+const useIdentity = () => ({
+  signedIn: !!initialState.meta.me,
+  accountId: initialState.meta.me,
+  disabledAccountId: initialState.meta.disabled_account_id,
+  accessToken: initialState.meta.access_token,
+  permissions: initialState.role ? initialState.role.permissions : 0,
+});
+
+const ColumnSettings = () => {
+  const intl = useIntl();
+  const dispatch = useAppDispatch();
+  const settings = useAppSelector((state) => state.getIn(['settings', 'firehose']));
+  const onChange = useCallback(
+    (key, checked) => dispatch(changeSetting(['firehose', ...key], checked)),
+    [dispatch],
+  );
+
+  return (
+    <div>
+      <div className='column-settings__row'>
+        <SettingToggle
+          settings={settings}
+          settingPath={['onlyMedia']}
+          onChange={onChange}
+          label={<FormattedMessage id='community.column_settings.media_only' defaultMessage='Media only' />}
+        />
+        <SettingToggle
+          settings={settings}
+          settingPath={['allowLocalOnly']}
+          onChange={onChange}
+          label={<FormattedMessage id='firehose.column_settings.allow_local_only' defaultMessage='Show local-only posts in "All"' />}
+        />
+        <span className='column-settings__section'><FormattedMessage id='home.column_settings.advanced' defaultMessage='Advanced' /></span>
+        <SettingText
+          settings={settings}
+          settingPath={['regex', 'body']}
+          onChange={onChange}
+          label={intl.formatMessage(messages.filter_regex)}
+        />
+      </div>
+    </div>
+  );
+};
+
+const Firehose = ({ feedType, multiColumn }) => {
+  const dispatch = useAppDispatch();
+  const intl = useIntl();
+  const { signedIn } = useIdentity();
+  const columnRef = useRef(null);
+
+  const allowLocalOnly = useAppSelector((state) => state.getIn(['settings', 'firehose', 'allowLocalOnly']));
+  const regex = useAppSelector((state) => state.getIn(['settings', 'firehose', 'regex', 'body']));
+
+  const onlyMedia = useAppSelector((state) => state.getIn(['settings', 'firehose', 'onlyMedia'], false));
+  const hasUnread = useAppSelector((state) => state.getIn(['timelines', `${feedType}${feedType === 'public' && allowLocalOnly ? ':allow_local_only' : ''}${onlyMedia ? ':media' : ''}`, 'unread'], 0) > 0);
+
+  const handlePin = useCallback(
+    () => {
+      switch(feedType) {
+      case 'community':
+        dispatch(addColumn('COMMUNITY', { other: { onlyMedia }, regex: { body: regex } }));
+        break;
+      case 'public':
+        dispatch(addColumn('PUBLIC', { other: { onlyMedia, allowLocalOnly }, regex: { body: regex }  }));
+        break;
+      case 'public:remote':
+        dispatch(addColumn('REMOTE', { other: { onlyMedia, onlyRemote: true }, regex: { body: regex }  }));
+        break;
+      }
+    },
+    [dispatch, onlyMedia, feedType, allowLocalOnly, regex],
+  );
+
+  const handleLoadMore = useCallback(
+    (maxId) => {
+      switch(feedType) {
+      case 'community':
+        dispatch(expandCommunityTimeline({ maxId, onlyMedia }));
+        break;
+      case 'public':
+        dispatch(expandPublicTimeline({ maxId, onlyMedia, allowLocalOnly }));
+        break;
+      case 'public:remote':
+        dispatch(expandPublicTimeline({ maxId, onlyMedia, onlyRemote: true }));
+        break;
+      }
+    },
+    [dispatch, onlyMedia, allowLocalOnly, feedType],
+  );
+
+  const handleHeaderClick = useCallback(() => columnRef.current?.scrollTop(), []);
+
+  useEffect(() => {
+    let disconnect;
+
+    switch(feedType) {
+    case 'community':
+      dispatch(expandCommunityTimeline({ onlyMedia }));
+      if (signedIn) {
+        disconnect = dispatch(connectCommunityStream({ onlyMedia }));
+      }
+      break;
+    case 'public':
+      dispatch(expandPublicTimeline({ onlyMedia, allowLocalOnly }));
+      if (signedIn) {
+        disconnect = dispatch(connectPublicStream({ onlyMedia, allowLocalOnly }));
+      }
+      break;
+    case 'public:remote':
+      dispatch(expandPublicTimeline({ onlyMedia, onlyRemote: true }));
+      if (signedIn) {
+        disconnect = dispatch(connectPublicStream({ onlyMedia, onlyRemote: true }));
+      }
+      break;
+    }
+
+    return () => disconnect?.();
+  }, [dispatch, signedIn, feedType, onlyMedia, allowLocalOnly]);
+
+  const prependBanner = feedType === 'community' ? (
+    <DismissableBanner id='community_timeline'>
+      <FormattedMessage
+        id='dismissable_banner.community_timeline'
+        defaultMessage='These are the most recent public posts from people whose accounts are hosted by {domain}.'
+        values={{ domain }}
+      />
+    </DismissableBanner>
+  ) : (
+    <DismissableBanner id='public_timeline'>
+      <FormattedMessage
+        id='dismissable_banner.public_timeline'
+        defaultMessage='These are the most recent public posts from people on the social web that people on {domain} follow.'
+        values={{ domain }}
+      />
+    </DismissableBanner>
+  );
+
+  const emptyMessage = feedType === 'community' ? (
+    <FormattedMessage
+      id='empty_column.community'
+      defaultMessage='The local timeline is empty. Write something publicly to get the ball rolling!'
+    />
+  ) : (
+    <FormattedMessage
+      id='empty_column.public'
+      defaultMessage='There is nothing here! Write something publicly, or manually follow users from other servers to fill it up'
+    />
+  );
+
+  return (
+    <Column bindToDocument={!multiColumn} ref={columnRef} label={intl.formatMessage(messages.title)}>
+      <ColumnHeader
+        icon='globe'
+        active={hasUnread}
+        title={intl.formatMessage(messages.title)}
+        onPin={handlePin}
+        onClick={handleHeaderClick}
+        multiColumn={multiColumn}
+      >
+        <ColumnSettings />
+      </ColumnHeader>
+
+      <div className='account__section-headline'>
+        <NavLink exact to='/public/local'>
+          <FormattedMessage tagName='div' id='firehose.local' defaultMessage='This server' />
+        </NavLink>
+
+        <NavLink exact to='/public/remote'>
+          <FormattedMessage tagName='div' id='firehose.remote' defaultMessage='Other servers' />
+        </NavLink>
+
+        <NavLink exact to='/public'>
+          <FormattedMessage tagName='div' id='firehose.all' defaultMessage='All' />
+        </NavLink>
+      </div>
+
+      <StatusListContainer
+        prepend={prependBanner}
+        timelineId={`${feedType}${feedType === 'public' && allowLocalOnly ? ':allow_local_only' : ''}${onlyMedia ? ':media' : ''}`}
+        onLoadMore={handleLoadMore}
+        trackScroll
+        scrollKey='firehose'
+        emptyMessage={emptyMessage}
+        bindToDocument={!multiColumn}
+        regex={regex}
+      />
+
+      <Helmet>
+        <title>{intl.formatMessage(messages.title)}</title>
+        <meta name='robots' content='noindex' />
+      </Helmet>
+    </Column>
+  );
+};
+
+Firehose.propTypes = {
+  multiColumn: PropTypes.bool,
+  feedType: PropTypes.string,
+};
+
+export default Firehose;
diff --git a/app/javascript/flavours/blobfox/features/follow_requests/components/account_authorize.jsx b/app/javascript/flavours/blobfox/features/follow_requests/components/account_authorize.jsx
new file mode 100644
index 00000000000000..5746af1b392a5f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/follow_requests/components/account_authorize.jsx
@@ -0,0 +1,52 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Avatar } from '../../../components/avatar';
+import { DisplayName } from '../../../components/display_name';
+import { IconButton } from '../../../components/icon_button';
+import Permalink from '../../../components/permalink';
+
+const messages = defineMessages({
+  authorize: { id: 'follow_request.authorize', defaultMessage: 'Authorize' },
+  reject: { id: 'follow_request.reject', defaultMessage: 'Reject' },
+});
+
+class AccountAuthorize extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    onAuthorize: PropTypes.func.isRequired,
+    onReject: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { intl, account, onAuthorize, onReject } = this.props;
+    const content = { __html: account.get('note_emojified') };
+
+    return (
+      <div className='account-authorize__wrapper'>
+        <div className='account-authorize'>
+          <Permalink href={account.get('url')} to={`/@${account.get('acct')}`} className='detailed-status__display-name'>
+            <div className='account-authorize__avatar'><Avatar account={account} size={48} /></div>
+            <DisplayName account={account} />
+          </Permalink>
+
+          <div className='account__header__content translate' dangerouslySetInnerHTML={content} />
+        </div>
+
+        <div className='account--panel'>
+          <div className='account--panel__button'><IconButton title={intl.formatMessage(messages.authorize)} icon='check' onClick={onAuthorize} /></div>
+          <div className='account--panel__button'><IconButton title={intl.formatMessage(messages.reject)} icon='times' onClick={onReject} /></div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(AccountAuthorize);
diff --git a/app/javascript/flavours/blobfox/features/follow_requests/containers/account_authorize_container.js b/app/javascript/flavours/blobfox/features/follow_requests/containers/account_authorize_container.js
new file mode 100644
index 00000000000000..c9c8dd7d874d13
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/follow_requests/containers/account_authorize_container.js
@@ -0,0 +1,27 @@
+import { connect } from 'react-redux';
+
+import { authorizeFollowRequest, rejectFollowRequest } from '../../../actions/accounts';
+import { makeGetAccount } from '../../../selectors';
+import AccountAuthorize from '../components/account_authorize';
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, props) => ({
+    account: getAccount(state, props.id),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { id }) => ({
+  onAuthorize () {
+    dispatch(authorizeFollowRequest(id));
+  },
+
+  onReject () {
+    dispatch(rejectFollowRequest(id));
+  },
+});
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(AccountAuthorize);
diff --git a/app/javascript/flavours/blobfox/features/follow_requests/index.jsx b/app/javascript/flavours/blobfox/features/follow_requests/index.jsx
new file mode 100644
index 00000000000000..796254e0ec9928
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/follow_requests/index.jsx
@@ -0,0 +1,96 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchFollowRequests, expandFollowRequests } from '../../actions/accounts';
+import ColumnBackButtonSlim from '../../components/column_back_button_slim';
+import ScrollableList from '../../components/scrollable_list';
+import { me } from '../../initial_state';
+import Column from '../ui/components/column';
+
+import AccountAuthorizeContainer from './containers/account_authorize_container';
+
+const messages = defineMessages({
+  heading: { id: 'column.follow_requests', defaultMessage: 'Follow requests' },
+});
+
+const mapStateToProps = state => ({
+  accountIds: state.getIn(['user_lists', 'follow_requests', 'items']),
+  isLoading: state.getIn(['user_lists', 'follow_requests', 'isLoading'], true),
+  hasMore: !!state.getIn(['user_lists', 'follow_requests', 'next']),
+  locked: !!state.getIn(['accounts', me, 'locked']),
+  domain: state.getIn(['meta', 'domain']),
+});
+
+class FollowRequests extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    accountIds: ImmutablePropTypes.list,
+    locked: PropTypes.bool,
+    domain: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchFollowRequests());
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandFollowRequests());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, accountIds, hasMore, multiColumn, locked, domain, isLoading } = this.props;
+
+    const emptyMessage = <FormattedMessage id='empty_column.follow_requests' defaultMessage="You don't have any follow requests yet. When you receive one, it will show up here." />;
+    const unlockedPrependMessage = !locked && accountIds.size > 0 && (
+      <div className='follow_requests-unlocked_explanation'>
+        <FormattedMessage
+          id='follow_requests.unlocked_explanation'
+          defaultMessage='Even though your account is not locked, the {domain} staff thought you might want to review follow requests from these accounts manually.'
+          values={{ domain: domain }}
+        />
+      </div>
+    );
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='user-plus' heading={intl.formatMessage(messages.heading)}>
+        <ColumnBackButtonSlim />
+        <ScrollableList
+          scrollKey='follow_requests'
+          onLoadMore={this.handleLoadMore}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          showLoading={isLoading && accountIds.size === 0}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+          prepend={unlockedPrependMessage}
+        >
+          {accountIds.map(id =>
+            <AccountAuthorizeContainer key={id} id={id} />,
+          )}
+        </ScrollableList>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(FollowRequests));
diff --git a/app/javascript/flavours/blobfox/features/followed_tags/index.jsx b/app/javascript/flavours/blobfox/features/followed_tags/index.jsx
new file mode 100644
index 00000000000000..7d877402b668dd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/followed_tags/index.jsx
@@ -0,0 +1,93 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { expandFollowedHashtags, fetchFollowedHashtags } from 'flavours/blobfox/actions/tags';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import Hashtag from 'flavours/blobfox/components/hashtag';
+import ScrollableList from 'flavours/blobfox/components/scrollable_list';
+import Column from 'flavours/blobfox/features/ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'followed_tags', defaultMessage: 'Followed hashtags' },
+});
+
+const mapStateToProps = state => ({
+  hashtags: state.getIn(['followed_tags', 'items']),
+  isLoading: state.getIn(['followed_tags', 'isLoading'], true),
+  hasMore: !!state.getIn(['followed_tags', 'next']),
+});
+
+class FollowedTags extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    hashtags: ImmutablePropTypes.list,
+    isLoading: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+  };
+
+  componentDidMount() {
+    this.props.dispatch(fetchFollowedHashtags());
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandFollowedHashtags());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, hashtags, isLoading, hasMore, multiColumn } = this.props;
+
+    const emptyMessage = <FormattedMessage id='empty_column.followed_tags' defaultMessage='You have not followed any hashtags yet. When you do, they will show up here.' />;
+
+    return (
+      <Column bindToDocument={!multiColumn}>
+        <ColumnHeader
+          icon='hashtag'
+          title={intl.formatMessage(messages.heading)}
+          showBackButton
+          multiColumn={multiColumn}
+        />
+
+        <ScrollableList
+          scrollKey='followed_tags'
+          emptyMessage={emptyMessage}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          onLoadMore={this.handleLoadMore}
+          bindToDocument={!multiColumn}
+        >
+          {hashtags.map((hashtag) => (
+            <Hashtag
+              key={hashtag.get('name')}
+              name={hashtag.get('name')}
+              to={`/tags/${hashtag.get('name')}`}
+              withGraph={false}
+              // Taken from ImmutableHashtag. Should maybe refactor ImmutableHashtag to accept more options?
+              people={hashtag.getIn(['history', 0, 'accounts']) * 1 + hashtag.getIn(['history', 1, 'accounts']) * 1}
+              history={hashtag.get('history').reverse().map((day) => day.get('uses')).toArray()}
+            />
+          ))}
+        </ScrollableList>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(FollowedTags));
diff --git a/app/javascript/flavours/blobfox/features/followers/index.jsx b/app/javascript/flavours/blobfox/features/followers/index.jsx
new file mode 100644
index 00000000000000..85cc79b362122d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/followers/index.jsx
@@ -0,0 +1,177 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { TimelineHint } from 'flavours/blobfox/components/timeline_hint';
+import BundleColumnError from 'flavours/blobfox/features/ui/components/bundle_column_error';
+import { normalizeForLookup } from 'flavours/blobfox/reducers/accounts_map';
+import { getAccountHidden } from 'flavours/blobfox/selectors';
+
+import {
+  lookupAccount,
+  fetchAccount,
+  fetchFollowers,
+  expandFollowers,
+} from '../../actions/accounts';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import ScrollableList from '../../components/scrollable_list';
+import AccountContainer from '../../containers/account_container';
+import ProfileColumnHeader from '../account/components/profile_column_header';
+import { LimitedAccountHint } from '../account_timeline/components/limited_account_hint';
+import HeaderContainer from '../account_timeline/containers/header_container';
+import Column from '../ui/components/column';
+
+const mapStateToProps = (state, { params: { acct, id } }) => {
+  const accountId = id || state.getIn(['accounts_map', normalizeForLookup(acct)]);
+
+  if (!accountId) {
+    return {
+      isLoading: true,
+    };
+  }
+
+  return {
+    accountId,
+    remote: !!(state.getIn(['accounts', accountId, 'acct']) !== state.getIn(['accounts', accountId, 'username'])),
+    remoteUrl: state.getIn(['accounts', accountId, 'url']),
+    isAccount: !!state.getIn(['accounts', accountId]),
+    accountIds: state.getIn(['user_lists', 'followers', accountId, 'items']),
+    hasMore: !!state.getIn(['user_lists', 'followers', accountId, 'next']),
+    isLoading: state.getIn(['user_lists', 'followers', accountId, 'isLoading'], true),
+    suspended: state.getIn(['accounts', accountId, 'suspended'], false),
+    hidden: getAccountHidden(state, accountId),
+  };
+};
+
+const RemoteHint = ({ url }) => (
+  <TimelineHint url={url} resource={<FormattedMessage id='timeline_hint.resources.followers' defaultMessage='Followers' />} />
+);
+
+RemoteHint.propTypes = {
+  url: PropTypes.string.isRequired,
+};
+
+class Followers extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.shape({
+      acct: PropTypes.string,
+      id: PropTypes.string,
+    }).isRequired,
+    accountId: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    accountIds: ImmutablePropTypes.list,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    isAccount: PropTypes.bool,
+    suspended: PropTypes.bool,
+    hidden: PropTypes.bool,
+    remote: PropTypes.bool,
+    remoteUrl: PropTypes.string,
+    multiColumn: PropTypes.bool,
+  };
+
+  _load () {
+    const { accountId, isAccount, dispatch } = this.props;
+
+    if (!isAccount) dispatch(fetchAccount(accountId));
+    dispatch(fetchFollowers(accountId));
+  }
+
+  componentDidMount () {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (accountId) {
+      this._load();
+    } else {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (prevProps.accountId !== accountId && accountId) {
+      this._load();
+    } else if (prevProps.params.acct !== acct) {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandFollowers(this.props.accountId));
+  }, 300, { leading: true });
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  render () {
+    const { accountId, accountIds, hasMore, isAccount, multiColumn, isLoading, suspended, hidden, remote, remoteUrl } = this.props;
+
+    if (!isAccount) {
+      return (
+        <BundleColumnError multiColumn={multiColumn} errorType='routing' />
+      );
+    }
+
+    if (!accountIds) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    let emptyMessage;
+
+    const forceEmptyState = suspended || hidden;
+
+    if (suspended) {
+      emptyMessage = <FormattedMessage id='empty_column.account_suspended' defaultMessage='Account suspended' />;
+    } else if (hidden) {
+      emptyMessage = <LimitedAccountHint accountId={accountId} />;
+    } else if (remote && accountIds.isEmpty()) {
+      emptyMessage = <RemoteHint url={remoteUrl} />;
+    } else {
+      emptyMessage = <FormattedMessage id='account.followers.empty' defaultMessage='No one follows this user yet.' />;
+    }
+
+    const remoteMessage = remote ? <RemoteHint url={remoteUrl} /> : null;
+
+    return (
+      <Column ref={this.setRef}>
+        <ProfileColumnHeader onClick={this.handleHeaderClick} multiColumn={multiColumn} />
+
+        <ScrollableList
+          scrollKey='followers'
+          hasMore={!forceEmptyState && hasMore}
+          isLoading={isLoading}
+          onLoadMore={this.handleLoadMore}
+          prepend={<HeaderContainer accountId={this.props.accountId} hideTabs />}
+          alwaysPrepend
+          append={remoteMessage}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {accountIds.map(id =>
+            <AccountContainer key={id} id={id} withNote={false} />,
+          )}
+        </ScrollableList>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(Followers);
diff --git a/app/javascript/flavours/blobfox/features/following/index.jsx b/app/javascript/flavours/blobfox/features/following/index.jsx
new file mode 100644
index 00000000000000..42336864d7551b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/following/index.jsx
@@ -0,0 +1,177 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { TimelineHint } from 'flavours/blobfox/components/timeline_hint';
+import BundleColumnError from 'flavours/blobfox/features/ui/components/bundle_column_error';
+import { normalizeForLookup } from 'flavours/blobfox/reducers/accounts_map';
+import { getAccountHidden } from 'flavours/blobfox/selectors';
+
+import {
+  lookupAccount,
+  fetchAccount,
+  fetchFollowing,
+  expandFollowing,
+} from '../../actions/accounts';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import ScrollableList from '../../components/scrollable_list';
+import AccountContainer from '../../containers/account_container';
+import ProfileColumnHeader from '../account/components/profile_column_header';
+import { LimitedAccountHint } from '../account_timeline/components/limited_account_hint';
+import HeaderContainer from '../account_timeline/containers/header_container';
+import Column from '../ui/components/column';
+
+const mapStateToProps = (state, { params: { acct, id } }) => {
+  const accountId = id || state.getIn(['accounts_map', normalizeForLookup(acct)]);
+
+  if (!accountId) {
+    return {
+      isLoading: true,
+    };
+  }
+
+  return {
+    accountId,
+    remote: !!(state.getIn(['accounts', accountId, 'acct']) !== state.getIn(['accounts', accountId, 'username'])),
+    remoteUrl: state.getIn(['accounts', accountId, 'url']),
+    isAccount: !!state.getIn(['accounts', accountId]),
+    accountIds: state.getIn(['user_lists', 'following', accountId, 'items']),
+    hasMore: !!state.getIn(['user_lists', 'following', accountId, 'next']),
+    isLoading: state.getIn(['user_lists', 'following', accountId, 'isLoading'], true),
+    suspended: state.getIn(['accounts', accountId, 'suspended'], false),
+    hidden: getAccountHidden(state, accountId),
+  };
+};
+
+const RemoteHint = ({ url }) => (
+  <TimelineHint url={url} resource={<FormattedMessage id='timeline_hint.resources.follows' defaultMessage='Follows' />} />
+);
+
+RemoteHint.propTypes = {
+  url: PropTypes.string.isRequired,
+};
+
+class Following extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.shape({
+      acct: PropTypes.string,
+      id: PropTypes.string,
+    }).isRequired,
+    accountId: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    accountIds: ImmutablePropTypes.list,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    isAccount: PropTypes.bool,
+    suspended: PropTypes.bool,
+    hidden: PropTypes.bool,
+    remote: PropTypes.bool,
+    remoteUrl: PropTypes.string,
+    multiColumn: PropTypes.bool,
+  };
+
+  _load () {
+    const { accountId, isAccount, dispatch } = this.props;
+
+    if (!isAccount) dispatch(fetchAccount(accountId));
+    dispatch(fetchFollowing(accountId));
+  }
+
+  componentDidMount () {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (accountId) {
+      this._load();
+    } else {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { params: { acct }, accountId, dispatch } = this.props;
+
+    if (prevProps.accountId !== accountId && accountId) {
+      this._load();
+    } else if (prevProps.params.acct !== acct) {
+      dispatch(lookupAccount(acct));
+    }
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandFollowing(this.props.accountId));
+  }, 300, { leading: true });
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  render () {
+    const { accountId, accountIds, hasMore, isAccount, multiColumn, isLoading, suspended, hidden, remote, remoteUrl } = this.props;
+
+    if (!isAccount) {
+      return (
+        <BundleColumnError multiColumn={multiColumn} errorType='routing' />
+      );
+    }
+
+    if (!accountIds) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    let emptyMessage;
+
+    const forceEmptyState = suspended || hidden;
+
+    if (suspended) {
+      emptyMessage = <FormattedMessage id='empty_column.account_suspended' defaultMessage='Account suspended' />;
+    } else if (hidden) {
+      emptyMessage = <LimitedAccountHint accountId={accountId} />;
+    } else if (remote && accountIds.isEmpty()) {
+      emptyMessage = <RemoteHint url={remoteUrl} />;
+    } else {
+      emptyMessage = <FormattedMessage id='account.follows.empty' defaultMessage="This user doesn't follow anyone yet." />;
+    }
+
+    const remoteMessage = remote ? <RemoteHint url={remoteUrl} /> : null;
+
+    return (
+      <Column ref={this.setRef}>
+        <ProfileColumnHeader onClick={this.handleHeaderClick} multiColumn={multiColumn} />
+
+        <ScrollableList
+          scrollKey='following'
+          hasMore={!forceEmptyState && hasMore}
+          isLoading={isLoading}
+          onLoadMore={this.handleLoadMore}
+          prepend={<HeaderContainer accountId={this.props.accountId} hideTabs />}
+          alwaysPrepend
+          append={remoteMessage}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {accountIds.map(id =>
+            <AccountContainer key={id} id={id} withNote={false} />,
+          )}
+        </ScrollableList>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(Following);
diff --git a/app/javascript/flavours/blobfox/features/getting_started/components/announcements.jsx b/app/javascript/flavours/blobfox/features/getting_started/components/announcements.jsx
new file mode 100644
index 00000000000000..982053a3953b51
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/getting_started/components/announcements.jsx
@@ -0,0 +1,455 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage, FormattedDate } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import TransitionMotion from 'react-motion/lib/TransitionMotion';
+import spring from 'react-motion/lib/spring';
+import ReactSwipeableViews from 'react-swipeable-views';
+
+import { AnimatedNumber } from 'flavours/blobfox/components/animated_number';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import EmojiPickerDropdown from 'flavours/blobfox/features/compose/containers/emoji_picker_dropdown_container';
+import { unicodeMapping } from 'flavours/blobfox/features/emoji/emoji_unicode_mapping_light';
+import { autoPlayGif, reduceMotion, disableSwiping, mascot } from 'flavours/blobfox/initial_state';
+import { assetHost } from 'flavours/blobfox/utils/config';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+import elephantUIPlane from 'mastodon/../images/elephant_ui_plane.svg';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+  previous: { id: 'lightbox.previous', defaultMessage: 'Previous' },
+  next: { id: 'lightbox.next', defaultMessage: 'Next' },
+});
+
+class ContentWithRouter extends ImmutablePureComponent {
+  static propTypes = {
+    announcement: ImmutablePropTypes.map.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  componentDidMount () {
+    this._updateLinks();
+  }
+
+  componentDidUpdate () {
+    this._updateLinks();
+  }
+
+  _updateLinks () {
+    const node = this.node;
+
+    if (!node) {
+      return;
+    }
+
+    const links = node.querySelectorAll('a');
+
+    for (var i = 0; i < links.length; ++i) {
+      let link = links[i];
+
+      if (link.classList.contains('status-link')) {
+        continue;
+      }
+
+      link.classList.add('status-link');
+
+      let mention = this.props.announcement.get('mentions').find(item => link.href === item.get('url'));
+
+      if (mention) {
+        link.addEventListener('click', this.onMentionClick.bind(this, mention), false);
+        link.setAttribute('title', mention.get('acct'));
+      } else if (link.textContent[0] === '#' || (link.previousSibling && link.previousSibling.textContent && link.previousSibling.textContent[link.previousSibling.textContent.length - 1] === '#')) {
+        link.addEventListener('click', this.onHashtagClick.bind(this, link.text), false);
+      } else {
+        let status = this.props.announcement.get('statuses').find(item => link.href === item.get('url'));
+        if (status) {
+          link.addEventListener('click', this.onStatusClick.bind(this, status), false);
+        }
+        link.setAttribute('title', link.href);
+        link.classList.add('unhandled-link');
+      }
+
+      link.setAttribute('target', '_blank');
+      link.setAttribute('rel', 'noopener noreferrer');
+    }
+  }
+
+  onMentionClick = (mention, e) => {
+    if (this.props.history && e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+      this.props.history.push(`/@${mention.get('acct')}`);
+    }
+  };
+
+  onHashtagClick = (hashtag, e) => {
+    hashtag = hashtag.replace(/^#/, '');
+
+    if (this.props.history&& e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+      this.props.history.push(`/tags/${hashtag}`);
+    }
+  };
+
+  onStatusClick = (status, e) => {
+    if (this.props.history && e.button === 0 && !(e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+      this.props.history.push(`/@${status.getIn(['account', 'acct'])}/${status.get('id')}`);
+    }
+  };
+
+  handleMouseEnter = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-original');
+    }
+  };
+
+  handleMouseLeave = ({ currentTarget }) => {
+    if (autoPlayGif) {
+      return;
+    }
+
+    const emojis = currentTarget.querySelectorAll('.custom-emoji');
+
+    for (var i = 0; i < emojis.length; i++) {
+      let emoji = emojis[i];
+      emoji.src = emoji.getAttribute('data-static');
+    }
+  };
+
+  render () {
+    const { announcement } = this.props;
+
+    return (
+      <div
+        className='announcements__item__content translate'
+        ref={this.setRef}
+        dangerouslySetInnerHTML={{ __html: announcement.get('contentHtml') }}
+        onMouseEnter={this.handleMouseEnter}
+        onMouseLeave={this.handleMouseLeave}
+      />
+    );
+  }
+
+}
+
+const Content = withRouter(ContentWithRouter);
+
+class Emoji extends PureComponent {
+
+  static propTypes = {
+    emoji: PropTypes.string.isRequired,
+    emojiMap: ImmutablePropTypes.map.isRequired,
+    hovered: PropTypes.bool.isRequired,
+  };
+
+  render () {
+    const { emoji, emojiMap, hovered } = this.props;
+
+    if (unicodeMapping[emoji]) {
+      const { filename, shortCode } = unicodeMapping[this.props.emoji];
+      const title = shortCode ? `:${shortCode}:` : '';
+
+      return (
+        <img
+          draggable='false'
+          className='emojione'
+          alt={emoji}
+          title={title}
+          src={`${assetHost}/emoji/${filename}.svg`}
+        />
+      );
+    } else if (emojiMap.get(emoji)) {
+      const filename  = (autoPlayGif || hovered) ? emojiMap.getIn([emoji, 'url']) : emojiMap.getIn([emoji, 'static_url']);
+      const shortCode = `:${emoji}:`;
+
+      return (
+        <img
+          draggable='false'
+          className='emojione custom-emoji'
+          alt={shortCode}
+          title={shortCode}
+          src={filename}
+        />
+      );
+    } else {
+      return null;
+    }
+  }
+
+}
+
+class Reaction extends ImmutablePureComponent {
+
+  static propTypes = {
+    announcementId: PropTypes.string.isRequired,
+    reaction: ImmutablePropTypes.map.isRequired,
+    addReaction: PropTypes.func.isRequired,
+    removeReaction: PropTypes.func.isRequired,
+    emojiMap: ImmutablePropTypes.map.isRequired,
+    style: PropTypes.object,
+  };
+
+  state = {
+    hovered: false,
+  };
+
+  handleClick = () => {
+    const { reaction, announcementId, addReaction, removeReaction } = this.props;
+
+    if (reaction.get('me')) {
+      removeReaction(announcementId, reaction.get('name'));
+    } else {
+      addReaction(announcementId, reaction.get('name'));
+    }
+  };
+
+  handleMouseEnter = () => this.setState({ hovered: true });
+
+  handleMouseLeave = () => this.setState({ hovered: false });
+
+  render () {
+    const { reaction } = this.props;
+
+    let shortCode = reaction.get('name');
+
+    if (unicodeMapping[shortCode]) {
+      shortCode = unicodeMapping[shortCode].shortCode;
+    }
+
+    return (
+      <button className={classNames('reactions-bar__item', { active: reaction.get('me') })} onClick={this.handleClick} onMouseEnter={this.handleMouseEnter} onMouseLeave={this.handleMouseLeave} title={`:${shortCode}:`} style={this.props.style}>
+        <span className='reactions-bar__item__emoji'><Emoji hovered={this.state.hovered} emoji={reaction.get('name')} emojiMap={this.props.emojiMap} /></span>
+        <span className='reactions-bar__item__count'><AnimatedNumber value={reaction.get('count')} /></span>
+      </button>
+    );
+  }
+
+}
+
+class ReactionsBar extends ImmutablePureComponent {
+
+  static propTypes = {
+    announcementId: PropTypes.string.isRequired,
+    reactions: ImmutablePropTypes.list.isRequired,
+    addReaction: PropTypes.func.isRequired,
+    removeReaction: PropTypes.func.isRequired,
+    emojiMap: ImmutablePropTypes.map.isRequired,
+  };
+
+  handleEmojiPick = data => {
+    const { addReaction, announcementId } = this.props;
+    addReaction(announcementId, data.native.replace(/:/g, ''));
+  };
+
+  willEnter () {
+    return { scale: reduceMotion ? 1 : 0 };
+  }
+
+  willLeave () {
+    return { scale: reduceMotion ? 0 : spring(0, { stiffness: 170, damping: 26 }) };
+  }
+
+  render () {
+    const { reactions } = this.props;
+    const visibleReactions = reactions.filter(x => x.get('count') > 0);
+
+    const styles = visibleReactions.map(reaction => ({
+      key: reaction.get('name'),
+      data: reaction,
+      style: { scale: reduceMotion ? 1 : spring(1, { stiffness: 150, damping: 13 }) },
+    })).toArray();
+
+    return (
+      <TransitionMotion styles={styles} willEnter={this.willEnter} willLeave={this.willLeave}>
+        {items => (
+          <div className={classNames('reactions-bar', { 'reactions-bar--empty': visibleReactions.isEmpty() })}>
+            {items.map(({ key, data, style }) => (
+              <Reaction
+                key={key}
+                reaction={data}
+                style={{ transform: `scale(${style.scale})`, position: style.scale < 0.5 ? 'absolute' : 'static' }}
+                announcementId={this.props.announcementId}
+                addReaction={this.props.addReaction}
+                removeReaction={this.props.removeReaction}
+                emojiMap={this.props.emojiMap}
+              />
+            ))}
+
+            {visibleReactions.size < 8 && <EmojiPickerDropdown onPickEmoji={this.handleEmojiPick} button={<Icon id='plus' />} />}
+          </div>
+        )}
+      </TransitionMotion>
+    );
+  }
+
+}
+
+class Announcement extends ImmutablePureComponent {
+
+  static propTypes = {
+    announcement: ImmutablePropTypes.map.isRequired,
+    emojiMap: ImmutablePropTypes.map.isRequired,
+    addReaction: PropTypes.func.isRequired,
+    removeReaction: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    selected: PropTypes.bool,
+  };
+
+  state = {
+    unread: !this.props.announcement.get('read'),
+  };
+
+  componentDidUpdate () {
+    const { selected, announcement } = this.props;
+    if (!selected && this.state.unread !== !announcement.get('read')) {
+      this.setState({ unread: !announcement.get('read') });
+    }
+  }
+
+  render () {
+    const { announcement } = this.props;
+    const { unread } = this.state;
+    const startsAt = announcement.get('starts_at') && new Date(announcement.get('starts_at'));
+    const endsAt = announcement.get('ends_at') && new Date(announcement.get('ends_at'));
+    const now = new Date();
+    const hasTimeRange = startsAt && endsAt;
+    const skipYear = hasTimeRange && startsAt.getFullYear() === endsAt.getFullYear() && endsAt.getFullYear() === now.getFullYear();
+    const skipEndDate = hasTimeRange && startsAt.getDate() === endsAt.getDate() && startsAt.getMonth() === endsAt.getMonth() && startsAt.getFullYear() === endsAt.getFullYear();
+    const skipTime = announcement.get('all_day');
+
+    return (
+      <div className='announcements__item'>
+        <strong className='announcements__item__range'>
+          <FormattedMessage id='announcement.announcement' defaultMessage='Announcement' />
+          {hasTimeRange && <span> · <FormattedDate value={startsAt} hour12={false} year={(skipYear || startsAt.getFullYear() === now.getFullYear()) ? undefined : 'numeric'} month='short' day='2-digit' hour={skipTime ? undefined : '2-digit'} minute={skipTime ? undefined : '2-digit'} /> - <FormattedDate value={endsAt} hour12={false} year={(skipYear || endsAt.getFullYear() === now.getFullYear()) ? undefined : 'numeric'} month={skipEndDate ? undefined : 'short'} day={skipEndDate ? undefined : '2-digit'} hour={skipTime ? undefined : '2-digit'} minute={skipTime ? undefined : '2-digit'} /></span>}
+        </strong>
+
+        <Content announcement={announcement} />
+
+        <ReactionsBar
+          reactions={announcement.get('reactions')}
+          announcementId={announcement.get('id')}
+          addReaction={this.props.addReaction}
+          removeReaction={this.props.removeReaction}
+          emojiMap={this.props.emojiMap}
+        />
+
+        {unread && <span className='announcements__item__unread' />}
+      </div>
+    );
+  }
+
+}
+
+class Announcements extends ImmutablePureComponent {
+
+  static propTypes = {
+    announcements: ImmutablePropTypes.list,
+    emojiMap: ImmutablePropTypes.map.isRequired,
+    dismissAnnouncement: PropTypes.func.isRequired,
+    addReaction: PropTypes.func.isRequired,
+    removeReaction: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    index: 0,
+  };
+
+  static getDerivedStateFromProps(props, state) {
+    if (props.announcements.size > 0 && state.index >= props.announcements.size) {
+      return { index: props.announcements.size - 1 };
+    } else {
+      return null;
+    }
+  }
+
+  componentDidMount () {
+    this._markAnnouncementAsRead();
+  }
+
+  componentDidUpdate () {
+    this._markAnnouncementAsRead();
+  }
+
+  _markAnnouncementAsRead () {
+    const { dismissAnnouncement, announcements } = this.props;
+    const { index } = this.state;
+    const announcement = announcements.get(announcements.size - 1 - index);
+    if (!announcement.get('read')) dismissAnnouncement(announcement.get('id'));
+  }
+
+  handleChangeIndex = index => {
+    this.setState({ index: index % this.props.announcements.size });
+  };
+
+  handleNextClick = () => {
+    this.setState({ index: (this.state.index + 1) % this.props.announcements.size });
+  };
+
+  handlePrevClick = () => {
+    this.setState({ index: (this.props.announcements.size + this.state.index - 1) % this.props.announcements.size });
+  };
+
+  render () {
+    const { announcements, intl } = this.props;
+    const { index } = this.state;
+
+    if (announcements.isEmpty()) {
+      return null;
+    }
+
+    return (
+      <div className='announcements'>
+        <img className='announcements__mastodon' alt='' draggable='false' src={mascot || elephantUIPlane} />
+
+        <div className='announcements__container'>
+          <ReactSwipeableViews animateHeight animateTransitions={!reduceMotion} index={index} onChangeIndex={this.handleChangeIndex}>
+            {announcements.map((announcement, idx) => (
+              <Announcement
+                key={announcement.get('id')}
+                announcement={announcement}
+                emojiMap={this.props.emojiMap}
+                addReaction={this.props.addReaction}
+                removeReaction={this.props.removeReaction}
+                intl={intl}
+                selected={index === idx}
+                disabled={disableSwiping}
+              />
+            )).reverse()}
+          </ReactSwipeableViews>
+
+          {announcements.size > 1 && (
+            <div className='announcements__pagination'>
+              <IconButton disabled={announcements.size === 1} title={intl.formatMessage(messages.previous)} icon='chevron-left' onClick={this.handlePrevClick} size={13} />
+              <span>{index + 1} / {announcements.size}</span>
+              <IconButton disabled={announcements.size === 1} title={intl.formatMessage(messages.next)} icon='chevron-right' onClick={this.handleNextClick} size={13} />
+            </div>
+          )}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Announcements);
diff --git a/app/javascript/flavours/blobfox/features/getting_started/components/trends.jsx b/app/javascript/flavours/blobfox/features/getting_started/components/trends.jsx
new file mode 100644
index 00000000000000..0971cce1352270
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/getting_started/components/trends.jsx
@@ -0,0 +1,54 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { ImmutableHashtag as Hashtag } from 'flavours/blobfox/components/hashtag';
+
+export default class Trends extends ImmutablePureComponent {
+
+  static defaultProps = {
+    loading: false,
+  };
+
+  static propTypes = {
+    trends: ImmutablePropTypes.list,
+    fetchTrends: PropTypes.func.isRequired,
+  };
+
+  componentDidMount () {
+    this.props.fetchTrends();
+    this.refreshInterval = setInterval(() => this.props.fetchTrends(), 900 * 1000);
+  }
+
+  componentWillUnmount () {
+    if (this.refreshInterval) {
+      clearInterval(this.refreshInterval);
+    }
+  }
+
+  render () {
+    const { trends } = this.props;
+
+    if (!trends || trends.isEmpty()) {
+      return null;
+    }
+
+    return (
+      <div className='getting-started__trends'>
+        <h4>
+          <Link to={'/explore/tags'}>
+            <FormattedMessage id='trends.trending_now' defaultMessage='Trending now' />
+          </Link>
+        </h4>
+
+        {trends.take(3).map(hashtag => <Hashtag key={hashtag.get('name')} hashtag={hashtag} />)}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/getting_started/containers/announcements_container.js b/app/javascript/flavours/blobfox/features/getting_started/containers/announcements_container.js
new file mode 100644
index 00000000000000..3af2628a403965
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/getting_started/containers/announcements_container.js
@@ -0,0 +1,22 @@
+import { Map as ImmutableMap } from 'immutable';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { addReaction, removeReaction, dismissAnnouncement } from 'flavours/blobfox/actions/announcements';
+
+import Announcements from '../components/announcements';
+
+const customEmojiMap = createSelector([state => state.get('custom_emojis')], items => items.reduce((map, emoji) => map.set(emoji.get('shortcode'), emoji), ImmutableMap()));
+
+const mapStateToProps = state => ({
+  announcements: state.getIn(['announcements', 'items']),
+  emojiMap: customEmojiMap(state),
+});
+
+const mapDispatchToProps = dispatch => ({
+  dismissAnnouncement: id => dispatch(dismissAnnouncement(id)),
+  addReaction: (id, name) => dispatch(addReaction(id, name)),
+  removeReaction: (id, name) => dispatch(removeReaction(id, name)),
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Announcements);
diff --git a/app/javascript/flavours/blobfox/features/getting_started/containers/trends_container.js b/app/javascript/flavours/blobfox/features/getting_started/containers/trends_container.js
new file mode 100644
index 00000000000000..e35c0c46f54336
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/getting_started/containers/trends_container.js
@@ -0,0 +1,15 @@
+import { connect } from 'react-redux';
+
+import { fetchTrendingHashtags } from 'flavours/blobfox/actions/trends';
+
+import Trends from '../components/trends';
+
+const mapStateToProps = state => ({
+  trends: state.getIn(['trends', 'tags', 'items']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  fetchTrends: () => dispatch(fetchTrendingHashtags()),
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(Trends);
diff --git a/app/javascript/flavours/blobfox/features/getting_started/index.jsx b/app/javascript/flavours/blobfox/features/getting_started/index.jsx
new file mode 100644
index 00000000000000..e1bc9cb9fbfe2b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/getting_started/index.jsx
@@ -0,0 +1,208 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { fetchFollowRequests } from 'flavours/blobfox/actions/accounts';
+import { fetchLists } from 'flavours/blobfox/actions/lists';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import Column from 'flavours/blobfox/features/ui/components/column';
+import LinkFooter from 'flavours/blobfox/features/ui/components/link_footer';
+import { preferencesLink } from 'flavours/blobfox/utils/backend_links';
+
+import { me, showTrends } from '../../initial_state';
+import NavigationBar from '../compose/components/navigation_bar';
+import ColumnLink from '../ui/components/column_link';
+import ColumnSubheading from '../ui/components/column_subheading';
+
+import TrendsContainer from './containers/trends_container';
+
+const messages = defineMessages({
+  heading: { id: 'getting_started.heading', defaultMessage: 'Getting started' },
+  home_timeline: { id: 'tabs_bar.home', defaultMessage: 'Home' },
+  notifications: { id: 'tabs_bar.notifications', defaultMessage: 'Notifications' },
+  public_timeline: { id: 'navigation_bar.public_timeline', defaultMessage: 'Federated timeline' },
+  navigation_subheading: { id: 'column_subheading.navigation', defaultMessage: 'Navigation' },
+  settings_subheading: { id: 'column_subheading.settings', defaultMessage: 'Settings' },
+  community_timeline: { id: 'navigation_bar.community_timeline', defaultMessage: 'Local timeline' },
+  explore: { id: 'navigation_bar.explore', defaultMessage: 'Explore' },
+  direct: { id: 'navigation_bar.direct', defaultMessage: 'Private mentions' },
+  bookmarks: { id: 'navigation_bar.bookmarks', defaultMessage: 'Bookmarks' },
+  preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
+  settings: { id: 'navigation_bar.app_settings', defaultMessage: 'App settings' },
+  follow_requests: { id: 'navigation_bar.follow_requests', defaultMessage: 'Follow requests' },
+  lists: { id: 'navigation_bar.lists', defaultMessage: 'Lists' },
+  keyboard_shortcuts: { id: 'navigation_bar.keyboard_shortcuts', defaultMessage: 'Keyboard shortcuts' },
+  lists_subheading: { id: 'column_subheading.lists', defaultMessage: 'Lists' },
+  misc: { id: 'navigation_bar.misc', defaultMessage: 'Misc' },
+  menu: { id: 'getting_started.heading', defaultMessage: 'Getting started' },
+});
+
+const makeMapStateToProps = () => {
+  const getOrderedLists = createSelector([state => state.get('lists')], lists => {
+    if (!lists) {
+      return lists;
+    }
+
+    return lists.toList().filter(item => !!item).sort((a, b) => a.get('title').localeCompare(b.get('title')));
+  });
+
+  const mapStateToProps = state => ({
+    lists: getOrderedLists(state),
+    myAccount: state.getIn(['accounts', me]),
+    columns: state.getIn(['settings', 'columns']),
+    unreadFollowRequests: state.getIn(['user_lists', 'follow_requests', 'items'], ImmutableList()).size,
+    unreadNotifications: state.getIn(['notifications', 'unread']),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = dispatch => ({
+  fetchFollowRequests: () => dispatch(fetchFollowRequests()),
+  fetchLists: () => dispatch(fetchLists()),
+  openSettings: () => dispatch(openModal({
+    modalType: 'SETTINGS',
+    modalProps: {},
+  })),
+});
+
+const badgeDisplay = (number, limit) => {
+  if (number === 0) {
+    return undefined;
+  } else if (limit && number >= limit) {
+    return `${limit}+`;
+  } else {
+    return number;
+  }
+};
+
+class GettingStarted extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    myAccount: ImmutablePropTypes.map,
+    columns: ImmutablePropTypes.list,
+    multiColumn: PropTypes.bool,
+    fetchFollowRequests: PropTypes.func.isRequired,
+    unreadFollowRequests: PropTypes.number,
+    unreadNotifications: PropTypes.number,
+    lists: ImmutablePropTypes.list,
+    fetchLists: PropTypes.func.isRequired,
+    openSettings: PropTypes.func.isRequired,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.fetchLists();
+  }
+
+  componentDidMount () {
+    const { fetchFollowRequests } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (!signedIn) {
+      return;
+    }
+
+    fetchFollowRequests();
+  }
+
+  render () {
+    const { intl, myAccount, columns, multiColumn, unreadFollowRequests, unreadNotifications, lists, openSettings } = this.props;
+    const { signedIn } = this.context.identity;
+
+    const navItems = [];
+    let listItems = [];
+
+    if (multiColumn) {
+      if (signedIn && !columns.find(item => item.get('id') === 'HOME')) {
+        navItems.push(<ColumnLink key='home' icon='home' text={intl.formatMessage(messages.home_timeline)} to='/home' />);
+      }
+
+      if (!columns.find(item => item.get('id') === 'NOTIFICATIONS')) {
+        navItems.push(<ColumnLink key='notifications' icon='bell' text={intl.formatMessage(messages.notifications)} badge={badgeDisplay(unreadNotifications)} to='/notifications' />);
+      }
+
+      if (!columns.find(item => item.get('id') === 'COMMUNITY')) {
+        navItems.push(<ColumnLink key='community_timeline' icon='users' text={intl.formatMessage(messages.community_timeline)} to='/public/local' />);
+      }
+
+      if (!columns.find(item => item.get('id') === 'PUBLIC')) {
+        navItems.push(<ColumnLink key='public_timeline' icon='globe' text={intl.formatMessage(messages.public_timeline)} to='/public' />);
+      }
+    }
+
+    if (showTrends) {
+      navItems.push(<ColumnLink key='explore' icon='hashtag' text={intl.formatMessage(messages.explore)} to='/explore' />);
+    }
+
+    if (signedIn) {
+      if (!multiColumn || !columns.find(item => item.get('id') === 'DIRECT')) {
+        navItems.push(<ColumnLink key='conversations' icon='envelope' text={intl.formatMessage(messages.direct)} to='/conversations' />);
+      }
+
+      if (!multiColumn || !columns.find(item => item.get('id') === 'BOOKMARKS')) {
+        navItems.push(<ColumnLink key='bookmarks' icon='bookmark' text={intl.formatMessage(messages.bookmarks)} to='/bookmarks' />);
+      }
+
+      if (myAccount.get('locked') || unreadFollowRequests > 0) {
+        navItems.push(<ColumnLink key='follow_requests' icon='user-plus' text={intl.formatMessage(messages.follow_requests)} badge={badgeDisplay(unreadFollowRequests, 40)} to='/follow_requests' />);
+      }
+
+      navItems.push(<ColumnLink key='getting_started' icon='ellipsis-h' text={intl.formatMessage(messages.misc)} to='/getting-started-misc' />);
+
+      listItems = listItems.concat([
+        <div key='9'>
+          <ColumnLink key='lists' icon='bars' text={intl.formatMessage(messages.lists)} to='/lists' />
+          {lists.filter(list => !columns.find(item => item.get('id') === 'LIST' && item.getIn(['params', 'id']) === list.get('id'))).map(list =>
+            <ColumnLink key={`list-${list.get('id')}`} to={`/lists/${list.get('id')}`} icon='list-ul' text={list.get('title')} />,
+          )}
+        </div>,
+      ]);
+    }
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='asterisk' heading={intl.formatMessage(messages.heading)} label={intl.formatMessage(messages.menu)} hideHeadingOnMobile>
+        <div className='scrollable optionally-scrollable'>
+          <div className='getting-started__wrapper'>
+            {!multiColumn && signedIn && <NavigationBar account={myAccount} />}
+            {multiColumn && <ColumnSubheading text={intl.formatMessage(messages.navigation_subheading)} />}
+            {navItems}
+            {signedIn && (
+              <>
+                <ColumnSubheading text={intl.formatMessage(messages.lists_subheading)} />
+                {listItems}
+                <ColumnSubheading text={intl.formatMessage(messages.settings_subheading)} />
+                { preferencesLink !== undefined && <ColumnLink icon='cog' text={intl.formatMessage(messages.preferences)} href={preferencesLink} /> }
+                <ColumnLink icon='cogs' text={intl.formatMessage(messages.settings)} onClick={openSettings} />
+              </>
+            )}
+          </div>
+
+          <LinkFooter multiColumn />
+        </div>
+
+        {(multiColumn && showTrends) && <TrendsContainer />}
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.menu)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(injectIntl(GettingStarted));
diff --git a/app/javascript/flavours/blobfox/features/getting_started_misc/index.jsx b/app/javascript/flavours/blobfox/features/getting_started_misc/index.jsx
new file mode 100644
index 00000000000000..b1d94e608b597b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/getting_started_misc/index.jsx
@@ -0,0 +1,68 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import ColumnBackButtonSlim from 'flavours/blobfox/components/column_back_button_slim';
+import Column from 'flavours/blobfox/features/ui/components/column';
+import ColumnLink from 'flavours/blobfox/features/ui/components/column_link';
+import ColumnSubheading from 'flavours/blobfox/features/ui/components/column_subheading';
+
+
+const messages = defineMessages({
+  heading: { id: 'column.heading', defaultMessage: 'Misc' },
+  subheading: { id: 'column.subheading', defaultMessage: 'Miscellaneous options' },
+  favourites: { id: 'navigation_bar.favourites', defaultMessage: 'Favorites' },
+  blocks: { id: 'navigation_bar.blocks', defaultMessage: 'Blocked users' },
+  domain_blocks: { id: 'navigation_bar.domain_blocks', defaultMessage: 'Blocked domains' },
+  mutes: { id: 'navigation_bar.mutes', defaultMessage: 'Muted users' },
+  pins: { id: 'navigation_bar.pins', defaultMessage: 'Pinned posts' },
+  keyboard_shortcuts: { id: 'navigation_bar.keyboard_shortcuts', defaultMessage: 'Keyboard shortcuts' },
+  featured_users: { id: 'navigation_bar.featured_users', defaultMessage: 'Featured users' },
+});
+
+class GettingStartedMisc extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+  };
+
+  openFeaturedAccountsModal = () => {
+    this.props.dispatch(openModal({
+      modalType: 'PINNED_ACCOUNTS_EDITOR',
+    }));
+  };
+
+  render () {
+    const { intl } = this.props;
+    const { signedIn } = this.context.identity;
+
+    return (
+      <Column icon='ellipsis-h' heading={intl.formatMessage(messages.heading)}>
+        <ColumnBackButtonSlim />
+
+        <div className='scrollable'>
+          <ColumnSubheading text={intl.formatMessage(messages.subheading)} />
+          {signedIn && (<ColumnLink key='favourites' icon='star' text={intl.formatMessage(messages.favourites)} to='/favourites' />)}
+          {signedIn && (<ColumnLink key='pinned' icon='thumb-tack' text={intl.formatMessage(messages.pins)} to='/pinned' />)}
+          {signedIn && (<ColumnLink key='featured_users' icon='users' text={intl.formatMessage(messages.featured_users)} onClick={this.openFeaturedAccountsModal} />)}
+          {signedIn && (<ColumnLink key='mutes' icon='volume-off' text={intl.formatMessage(messages.mutes)} to='/mutes' />)}
+          {signedIn && (<ColumnLink key='blocks' icon='ban' text={intl.formatMessage(messages.blocks)} to='/blocks' />)}
+          {signedIn && (<ColumnLink key='domain_blocks' icon='minus-circle' text={intl.formatMessage(messages.domain_blocks)} to='/domain_blocks' />)}
+          <ColumnLink key='shortcuts' icon='question' text={intl.formatMessage(messages.keyboard_shortcuts)} to='/keyboard-shortcuts' />
+        </div>
+      </Column>
+    );
+  }
+
+}
+
+export default connect()(injectIntl(GettingStartedMisc));
diff --git a/app/javascript/flavours/blobfox/features/hashtag_timeline/components/column_settings.jsx b/app/javascript/flavours/blobfox/features/hashtag_timeline/components/column_settings.jsx
new file mode 100644
index 00000000000000..c60de4c5189469
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/hashtag_timeline/components/column_settings.jsx
@@ -0,0 +1,138 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { NonceProvider } from 'react-select';
+import AsyncSelect from 'react-select/async';
+import Toggle from 'react-toggle';
+
+import SettingToggle from '../../notifications/components/setting_toggle';
+
+const messages = defineMessages({
+  placeholder: { id: 'hashtag.column_settings.select.placeholder', defaultMessage: 'Enter hashtags…' },
+  noOptions: { id: 'hashtag.column_settings.select.no_options_message', defaultMessage: 'No suggestions found' },
+});
+
+class ColumnSettings extends PureComponent {
+
+  static propTypes = {
+    settings: ImmutablePropTypes.map.isRequired,
+    onChange: PropTypes.func.isRequired,
+    onLoad: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    open: this.hasTags(),
+  };
+
+  hasTags () {
+    return ['all', 'any', 'none'].map(mode => this.tags(mode).length > 0).includes(true);
+  }
+
+  tags (mode) {
+    let tags = this.props.settings.getIn(['tags', mode]) || [];
+
+    if (tags.toJS) {
+      return tags.toJS();
+    } else {
+      return tags;
+    }
+  }
+
+  onSelect = mode => value => {
+    const oldValue = this.tags(mode);
+
+    // Prevent changes that add more than 4 tags, but allow removing
+    // tags that were already added before
+    if ((value.length > 4) && !(value < oldValue)) {
+      return;
+    }
+
+    this.props.onChange(['tags', mode], value);
+  };
+
+  onToggle = () => {
+    if (this.state.open && this.hasTags()) {
+      this.props.onChange('tags', {});
+    }
+
+    this.setState({ open: !this.state.open });
+  };
+
+  noOptionsMessage = () => this.props.intl.formatMessage(messages.noOptions);
+
+  modeSelect (mode) {
+    return (
+      <div className='column-settings__row'>
+        <span className='column-settings__section'>
+          {this.modeLabel(mode)}
+        </span>
+
+        <NonceProvider nonce={document.querySelector('meta[name=style-nonce]').content} cacheKey='tags'>
+          <AsyncSelect
+            isMulti
+            autoFocus
+            value={this.tags(mode)}
+            onChange={this.onSelect(mode)}
+            loadOptions={this.props.onLoad}
+            className='column-select__container'
+            classNamePrefix='column-select'
+            name='tags'
+            placeholder={this.props.intl.formatMessage(messages.placeholder)}
+            noOptionsMessage={this.noOptionsMessage}
+          />
+        </NonceProvider>
+      </div>
+    );
+  }
+
+  modeLabel (mode) {
+    switch(mode) {
+    case 'any':
+      return <FormattedMessage id='hashtag.column_settings.tag_mode.any' defaultMessage='Any of these' />;
+    case 'all':
+      return <FormattedMessage id='hashtag.column_settings.tag_mode.all' defaultMessage='All of these' />;
+    case 'none':
+      return <FormattedMessage id='hashtag.column_settings.tag_mode.none' defaultMessage='None of these' />;
+    default:
+      return '';
+    }
+  }
+
+  render () {
+    const { settings, onChange } = this.props;
+
+    return (
+      <div>
+        <div className='column-settings__row'>
+          <div className='setting-toggle'>
+            <Toggle id='hashtag.column_settings.tag_toggle' onChange={this.onToggle} checked={this.state.open} />
+
+            <span className='setting-toggle__label'>
+              <FormattedMessage id='hashtag.column_settings.tag_toggle' defaultMessage='Include additional tags in this column' />
+            </span>
+          </div>
+        </div>
+
+        {this.state.open && (
+          <div className='column-settings__hashtags'>
+            {this.modeSelect('any')}
+            {this.modeSelect('all')}
+            {this.modeSelect('none')}
+          </div>
+        )}
+
+        <div className='column-settings__row'>
+          <SettingToggle settings={settings} settingPath={['local']} onChange={onChange} label={<FormattedMessage id='community.column_settings.local_only' defaultMessage='Local only' />} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/hashtag_timeline/components/hashtag_header.jsx b/app/javascript/flavours/blobfox/features/hashtag_timeline/components/hashtag_header.jsx
new file mode 100644
index 00000000000000..cf1eb76b8a7dbe
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/hashtag_timeline/components/hashtag_header.jsx
@@ -0,0 +1,79 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Button } from 'flavours/blobfox/components/button';
+import { ShortNumber } from 'flavours/blobfox/components/short_number';
+
+const messages = defineMessages({
+  followHashtag: { id: 'hashtag.follow', defaultMessage: 'Follow hashtag' },
+  unfollowHashtag: { id: 'hashtag.unfollow', defaultMessage: 'Unfollow hashtag' },
+});
+
+const usesRenderer = (displayNumber, pluralReady) => (
+  <FormattedMessage
+    id='hashtag.counter_by_uses'
+    defaultMessage='{count, plural, one {{counter} post} other {{counter} posts}}'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+    }}
+  />
+);
+
+const peopleRenderer = (displayNumber, pluralReady) => (
+  <FormattedMessage
+    id='hashtag.counter_by_accounts'
+    defaultMessage='{count, plural, one {{counter} participant} other {{counter} participants}}'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+    }}
+  />
+);
+
+const usesTodayRenderer = (displayNumber, pluralReady) => (
+  <FormattedMessage
+    id='hashtag.counter_by_uses_today'
+    defaultMessage='{count, plural, one {{counter} post} other {{counter} posts}} today'
+    values={{
+      count: pluralReady,
+      counter: <strong>{displayNumber}</strong>,
+    }}
+  />
+);
+
+export const HashtagHeader = injectIntl(({ tag, intl, disabled, onClick }) => {
+  if (!tag) {
+    return null;
+  }
+
+  const [uses, people] = tag.get('history').reduce((arr, day) => [arr[0] + day.get('uses') * 1, arr[1] + day.get('accounts') * 1], [0, 0]);
+  const dividingCircle = <span aria-hidden>{' · '}</span>;
+
+  return (
+    <div className='hashtag-header'>
+      <div className='hashtag-header__header'>
+        <h1>#{tag.get('name')}</h1>
+        <Button onClick={onClick} text={intl.formatMessage(tag.get('following') ? messages.unfollowHashtag : messages.followHashtag)} disabled={disabled} />
+      </div>
+
+      <div>
+        <ShortNumber value={uses} renderer={usesRenderer} />
+        {dividingCircle}
+        <ShortNumber value={people} renderer={peopleRenderer} />
+        {dividingCircle}
+        <ShortNumber value={tag.getIn(['history', 0, 'uses']) * 1} renderer={usesTodayRenderer} />
+      </div>
+    </div>
+  );
+});
+
+HashtagHeader.propTypes = {
+  tag: ImmutablePropTypes.map,
+  disabled: PropTypes.bool,
+  onClick: PropTypes.func,
+  intl: PropTypes.object,
+};
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/features/hashtag_timeline/containers/column_settings_container.js b/app/javascript/flavours/blobfox/features/hashtag_timeline/containers/column_settings_container.js
new file mode 100644
index 00000000000000..be95004cc7bfaf
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/hashtag_timeline/containers/column_settings_container.js
@@ -0,0 +1,33 @@
+import { connect } from 'react-redux';
+
+import { changeColumnParams } from '../../../actions/columns';
+import api from '../../../api';
+import ColumnSettings from '../components/column_settings';
+
+const mapStateToProps = (state, { columnId }) => {
+  const columns = state.getIn(['settings', 'columns']);
+  const index   = columns.findIndex(c => c.get('uuid') === columnId);
+
+  if (!(columnId && index >= 0)) {
+    return {};
+  }
+
+  return {
+    settings: columns.get(index).get('params'),
+    onLoad (value) {
+      return api(() => state).get('/api/v2/search', { params: { q: value, type: 'hashtags' } }).then(response => {
+        return (response.data.hashtags || []).map((tag) => {
+          return { value: tag.name, label: `#${tag.name}` };
+        });
+      });
+    },
+  };
+};
+
+const mapDispatchToProps = (dispatch, { columnId }) => ({
+  onChange (key, value) {
+    dispatch(changeColumnParams(columnId, key, value));
+  },
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/hashtag_timeline/index.jsx b/app/javascript/flavours/blobfox/features/hashtag_timeline/index.jsx
new file mode 100644
index 00000000000000..bcc2e99aee450f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/hashtag_timeline/index.jsx
@@ -0,0 +1,226 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { isEqual } from 'lodash';
+
+import { addColumn, removeColumn, moveColumn } from 'flavours/blobfox/actions/columns';
+import { connectHashtagStream } from 'flavours/blobfox/actions/streaming';
+import { fetchHashtag, followHashtag, unfollowHashtag } from 'flavours/blobfox/actions/tags';
+import { expandHashtagTimeline, clearTimeline } from 'flavours/blobfox/actions/timelines';
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+
+import StatusListContainer from '../ui/containers/status_list_container';
+
+import { HashtagHeader } from './components/hashtag_header';
+import ColumnSettingsContainer from './containers/column_settings_container';
+
+const mapStateToProps = (state, props) => ({
+  hasUnread: state.getIn(['timelines', `hashtag:${props.params.id}${props.params.local ? ':local' : ''}`, 'unread']) > 0,
+  tag: state.getIn(['tags', props.params.id]),
+});
+
+class HashtagTimeline extends PureComponent {
+
+  disconnects = [];
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    columnId: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    hasUnread: PropTypes.bool,
+    tag: ImmutablePropTypes.map,
+    multiColumn: PropTypes.bool,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('HASHTAG', { id: this.props.params.id }));
+    }
+  };
+
+  title = () => {
+    const { id } = this.props.params;
+    const title  = [id];
+
+    if (this.additionalFor('any')) {
+      title.push(' ', <FormattedMessage key='any' id='hashtag.column_header.tag_mode.any'  values={{ additional: this.additionalFor('any') }} defaultMessage='or {additional}' />);
+    }
+
+    if (this.additionalFor('all')) {
+      title.push(' ', <FormattedMessage key='all' id='hashtag.column_header.tag_mode.all'  values={{ additional: this.additionalFor('all') }} defaultMessage='and {additional}' />);
+    }
+
+    if (this.additionalFor('none')) {
+      title.push(' ', <FormattedMessage key='none' id='hashtag.column_header.tag_mode.none' values={{ additional: this.additionalFor('none') }} defaultMessage='without {additional}' />);
+    }
+
+    return title;
+  };
+
+  additionalFor = (mode) => {
+    const { tags } = this.props.params;
+
+    if (tags && (tags[mode] || []).length > 0) {
+      return tags[mode].map(tag => tag.value).join('/');
+    } else {
+      return '';
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  _subscribe (dispatch, id, tags = {}, local) {
+    const { signedIn } = this.context.identity;
+
+    if (!signedIn) {
+      return;
+    }
+
+    let any  = (tags.any || []).map(tag => tag.value);
+    let all  = (tags.all || []).map(tag => tag.value);
+    let none = (tags.none || []).map(tag => tag.value);
+
+    [id, ...any].map(tag => {
+      this.disconnects.push(dispatch(connectHashtagStream(id, tag, local, status => {
+        let tags = status.tags.map(tag => tag.name);
+
+        return all.filter(tag => tags.includes(tag)).length === all.length &&
+               none.filter(tag => tags.includes(tag)).length === 0;
+      })));
+    });
+  }
+
+  _unsubscribe () {
+    this.disconnects.map(disconnect => disconnect());
+    this.disconnects = [];
+  }
+
+  _unload () {
+    const { dispatch } = this.props;
+    const { id, local } = this.props.params;
+
+    this._unsubscribe();
+    dispatch(clearTimeline(`hashtag:${id}${local ? ':local' : ''}`));
+  }
+
+  _load() {
+    const { dispatch } = this.props;
+    const { id, tags, local } = this.props.params;
+
+    this._subscribe(dispatch, id, tags, local);
+    dispatch(expandHashtagTimeline(id, { tags, local }));
+    dispatch(fetchHashtag(id));
+  }
+
+  componentDidMount () {
+    this._load();
+  }
+
+  componentDidUpdate (prevProps) {
+    const { params } = this.props;
+    const { id, tags, local } = prevProps.params;
+
+    if (id !== params.id || !isEqual(tags, params.tags) || !isEqual(local, params.local)) {
+      this._unload();
+      this._load();
+    }
+  }
+
+  componentWillUnmount () {
+    this._unsubscribe();
+  }
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = maxId => {
+    const { dispatch, params } = this.props;
+    const { id, tags, local }  = params;
+
+    dispatch(expandHashtagTimeline(id, { maxId, tags, local }));
+  };
+
+  handleFollow = () => {
+    const { dispatch, params, tag } = this.props;
+    const { id } = params;
+    const { signedIn } = this.context.identity;
+
+    if (!signedIn) {
+      return;
+    }
+
+    if (tag.get('following')) {
+      dispatch(unfollowHashtag(id));
+    } else {
+      dispatch(followHashtag(id));
+    }
+  };
+
+  render () {
+    const { hasUnread, columnId, multiColumn, tag } = this.props;
+    const { id, local } = this.props.params;
+    const pinned = !!columnId;
+    const { signedIn } = this.context.identity;
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={`#${id}`}>
+        <ColumnHeader
+          icon='hashtag'
+          active={hasUnread}
+          title={this.title()}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+          showBackButton
+        >
+          {columnId && <ColumnSettingsContainer columnId={columnId} />}
+        </ColumnHeader>
+
+        <StatusListContainer
+          prepend={pinned ? null : <HashtagHeader tag={tag} disabled={!signedIn} onClick={this.handleFollow} />}
+          alwaysPrepend
+          trackScroll={!pinned}
+          scrollKey={`hashtag_timeline-${columnId}`}
+          timelineId={`hashtag:${id}${local ? ':local' : ''}`}
+          onLoadMore={this.handleLoadMore}
+          emptyMessage={<FormattedMessage id='empty_column.hashtag' defaultMessage='There is nothing in this hashtag yet.' />}
+          bindToDocument={!multiColumn}
+        />
+
+        <Helmet>
+          <title>#{id}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(HashtagTimeline);
diff --git a/app/javascript/flavours/blobfox/features/home_timeline/components/column_settings.tsx b/app/javascript/flavours/blobfox/features/home_timeline/components/column_settings.tsx
new file mode 100644
index 00000000000000..70fc35cbb18d1a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/home_timeline/components/column_settings.tsx
@@ -0,0 +1,109 @@
+/* eslint-disable @typescript-eslint/no-unsafe-call,
+                  @typescript-eslint/no-unsafe-return,
+                  @typescript-eslint/no-unsafe-assignment,
+                  @typescript-eslint/no-unsafe-member-access
+                  -- the settings store is not yet typed */
+import { useCallback } from 'react';
+
+import { FormattedMessage, defineMessages, useIntl } from 'react-intl';
+
+import SettingText from 'flavours/blobfox/components/setting_text';
+import { useAppSelector, useAppDispatch } from 'flavours/blobfox/store';
+
+import { changeSetting } from '../../../actions/settings';
+import SettingToggle from '../../notifications/components/setting_toggle';
+
+const messages = defineMessages({
+  filter_regex: {
+    id: 'home.column_settings.filter_regex',
+    defaultMessage: 'Filter out by regular expressions',
+  },
+  settings: { id: 'home.settings', defaultMessage: 'Column settings' },
+});
+
+export const ColumnSettings: React.FC = () => {
+  const settings = useAppSelector((state) => state.settings.get('home'));
+
+  const intl = useIntl();
+
+  const dispatch = useAppDispatch();
+  const onChange = useCallback(
+    (key: string, checked: boolean) => {
+      dispatch(changeSetting(['home', ...key], checked));
+    },
+    [dispatch],
+  );
+
+  return (
+    <div>
+      <span className='column-settings__section'>
+        <FormattedMessage
+          id='home.column_settings.basic'
+          defaultMessage='Basic'
+        />
+      </span>
+
+      <div className='column-settings__row'>
+        <SettingToggle
+          prefix='home_timeline'
+          settings={settings}
+          settingPath={['shows', 'reblog']}
+          onChange={onChange}
+          label={
+            <FormattedMessage
+              id='home.column_settings.show_reblogs'
+              defaultMessage='Show boosts'
+            />
+          }
+        />
+      </div>
+
+      <div className='column-settings__row'>
+        <SettingToggle
+          prefix='home_timeline'
+          settings={settings}
+          settingPath={['shows', 'reply']}
+          onChange={onChange}
+          label={
+            <FormattedMessage
+              id='home.column_settings.show_replies'
+              defaultMessage='Show replies'
+            />
+          }
+        />
+      </div>
+
+      <div className='column-settings__row'>
+        <SettingToggle
+          prefix='home_timeline'
+          settings={settings}
+          settingPath={['shows', 'direct']}
+          onChange={onChange}
+          label={
+            <FormattedMessage
+              id='home.column_settings.show_direct'
+              defaultMessage='Show private mentions'
+            />
+          }
+        />
+      </div>
+
+      <span className='column-settings__section'>
+        <FormattedMessage
+          id='home.column_settings.advanced'
+          defaultMessage='Advanced'
+        />
+      </span>
+
+      <div className='column-settings__row'>
+        <SettingText
+          prefix='home_timeline'
+          settings={settings}
+          settingPath={['regex', 'body']}
+          onChange={onChange}
+          label={intl.formatMessage(messages.filter_regex)}
+        />
+      </div>
+    </div>
+  );
+};
diff --git a/app/javascript/flavours/blobfox/features/home_timeline/components/critical_update_banner.tsx b/app/javascript/flavours/blobfox/features/home_timeline/components/critical_update_banner.tsx
new file mode 100644
index 00000000000000..d0dd2b6acda66f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/home_timeline/components/critical_update_banner.tsx
@@ -0,0 +1,26 @@
+import { FormattedMessage } from 'react-intl';
+
+export const CriticalUpdateBanner = () => (
+  <div className='warning-banner'>
+    <div className='warning-banner__message'>
+      <h1>
+        <FormattedMessage
+          id='home.pending_critical_update.title'
+          defaultMessage='Critical security update available!'
+        />
+      </h1>
+      <p>
+        <FormattedMessage
+          id='home.pending_critical_update.body'
+          defaultMessage='Please update your Mastodon server as soon as possible!'
+        />{' '}
+        <a href='/admin/software_updates'>
+          <FormattedMessage
+            id='home.pending_critical_update.link'
+            defaultMessage='See updates'
+          />
+        </a>
+      </p>
+    </div>
+  </div>
+);
diff --git a/app/javascript/flavours/blobfox/features/home_timeline/components/explore_prompt.tsx b/app/javascript/flavours/blobfox/features/home_timeline/components/explore_prompt.tsx
new file mode 100644
index 00000000000000..8d39d6372d793f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/home_timeline/components/explore_prompt.tsx
@@ -0,0 +1,46 @@
+import { FormattedMessage } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import background from 'mastodon/../images/friends-cropped.png';
+
+export const ExplorePrompt = () => (
+  <DismissableBanner id='home.explore_prompt'>
+    <img
+      src={background}
+      alt=''
+      className='dismissable-banner__background-image'
+    />
+
+    <h1>
+      <FormattedMessage
+        id='home.explore_prompt.title'
+        defaultMessage='This is your home base within Mastodon.'
+      />
+    </h1>
+    <p>
+      <FormattedMessage
+        id='home.explore_prompt.body'
+        defaultMessage="Your home feed will have a mix of posts from the hashtags you've chosen to follow, the people you've chosen to follow, and the posts they boost. If that feels too quiet, you may want to:"
+      />
+    </p>
+
+    <div className='dismissable-banner__message__wrapper'>
+      <div className='dismissable-banner__message__actions'>
+        <Link to='/explore' className='button'>
+          <FormattedMessage
+            id='home.actions.go_to_explore'
+            defaultMessage="See what's trending"
+          />
+        </Link>
+        <Link to='/explore/suggestions' className='button button-tertiary'>
+          <FormattedMessage
+            id='home.actions.go_to_suggestions'
+            defaultMessage='Find people to follow'
+          />
+        </Link>
+      </div>
+    </div>
+  </DismissableBanner>
+);
diff --git a/app/javascript/flavours/blobfox/features/home_timeline/index.jsx b/app/javascript/flavours/blobfox/features/home_timeline/index.jsx
new file mode 100644
index 00000000000000..0164ed006a050b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/home_timeline/index.jsx
@@ -0,0 +1,239 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+
+import { List as ImmutableList } from 'immutable';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { fetchAnnouncements, toggleShowAnnouncements } from 'flavours/blobfox/actions/announcements';
+import { IconWithBadge } from 'flavours/blobfox/components/icon_with_badge';
+import { NotSignedInIndicator } from 'flavours/blobfox/components/not_signed_in_indicator';
+import AnnouncementsContainer from 'flavours/blobfox/features/getting_started/containers/announcements_container';
+import { me, criticalUpdatesPending } from 'flavours/blobfox/initial_state';
+
+import { addColumn, removeColumn, moveColumn } from '../../actions/columns';
+import { expandHomeTimeline } from '../../actions/timelines';
+import Column from '../../components/column';
+import ColumnHeader from '../../components/column_header';
+import StatusListContainer from '../ui/containers/status_list_container';
+
+import { ColumnSettings } from './components/column_settings';
+import { CriticalUpdateBanner } from './components/critical_update_banner';
+import { ExplorePrompt } from './components/explore_prompt';
+
+const messages = defineMessages({
+  title: { id: 'column.home', defaultMessage: 'Home' },
+  show_announcements: { id: 'home.show_announcements', defaultMessage: 'Show announcements' },
+  hide_announcements: { id: 'home.hide_announcements', defaultMessage: 'Hide announcements' },
+});
+
+const getHomeFeedSpeed = createSelector([
+  state => state.getIn(['timelines', 'home', 'items'], ImmutableList()),
+  state => state.getIn(['timelines', 'home', 'pendingItems'], ImmutableList()),
+  state => state.get('statuses'),
+], (statusIds, pendingStatusIds, statusMap) => {
+  const recentStatusIds = pendingStatusIds.concat(statusIds);
+  const statuses = recentStatusIds.filter(id => id !== null).map(id => statusMap.get(id)).filter(status => status?.get('account') !== me).take(20);
+
+  if (statuses.isEmpty()) {
+    return {
+      gap: 0,
+      newest: new Date(0),
+    };
+  }
+
+  const datetimes = statuses.map(status => status.get('created_at', 0));
+  const oldest = new Date(datetimes.min());
+  const newest = new Date(datetimes.max());
+  const averageGap = (newest - oldest) / (1000 * (statuses.size + 1)); // Average gap between posts on first page in seconds
+
+  return {
+    gap: averageGap,
+    newest,
+  };
+});
+
+const homeTooSlow = createSelector([
+  state => state.getIn(['timelines', 'home', 'isLoading']),
+  state => state.getIn(['timelines', 'home', 'isPartial']),
+  getHomeFeedSpeed,
+], (isLoading, isPartial, speed) =>
+  !isLoading && !isPartial // Only if the home feed has finished loading
+  && (
+    (speed.gap > (30 * 60) // If the average gap between posts is more than 30 minutes
+    || (Date.now() - speed.newest) > (1000 * 3600)) // If the most recent post is from over an hour ago
+  )
+);
+
+const mapStateToProps = state => ({
+  hasUnread: state.getIn(['timelines', 'home', 'unread']) > 0,
+  isPartial: state.getIn(['timelines', 'home', 'isPartial']),
+  hasAnnouncements: !state.getIn(['announcements', 'items']).isEmpty(),
+  unreadAnnouncements: state.getIn(['announcements', 'items']).count(item => !item.get('read')),
+  showAnnouncements: state.getIn(['announcements', 'show']),
+  tooSlow: homeTooSlow(state),
+  regex: state.getIn(['settings', 'home', 'regex', 'body']),
+});
+
+class HomeTimeline extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    hasUnread: PropTypes.bool,
+    isPartial: PropTypes.bool,
+    columnId: PropTypes.string,
+    multiColumn: PropTypes.bool,
+    hasAnnouncements: PropTypes.bool,
+    unreadAnnouncements: PropTypes.number,
+    showAnnouncements: PropTypes.bool,
+    tooSlow: PropTypes.bool,
+    regex: PropTypes.string,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('HOME', {}));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = maxId => {
+    this.props.dispatch(expandHomeTimeline({ maxId }));
+  };
+
+  componentDidMount () {
+    setTimeout(() => this.props.dispatch(fetchAnnouncements()), 700);
+    this._checkIfReloadNeeded(false, this.props.isPartial);
+  }
+
+  componentDidUpdate (prevProps) {
+    this._checkIfReloadNeeded(prevProps.isPartial, this.props.isPartial);
+  }
+
+  componentWillUnmount () {
+    this._stopPolling();
+  }
+
+  _checkIfReloadNeeded (wasPartial, isPartial) {
+    const { dispatch } = this.props;
+
+    if (wasPartial === isPartial) {
+      return;
+    } else if (!wasPartial && isPartial) {
+      this.polling = setInterval(() => {
+        dispatch(expandHomeTimeline());
+      }, 3000);
+    } else if (wasPartial && !isPartial) {
+      this._stopPolling();
+    }
+  }
+
+  _stopPolling () {
+    if (this.polling) {
+      clearInterval(this.polling);
+      this.polling = null;
+    }
+  }
+
+  handleToggleAnnouncementsClick = (e) => {
+    e.stopPropagation();
+    this.props.dispatch(toggleShowAnnouncements());
+  };
+
+  render () {
+    const { intl, hasUnread, columnId, multiColumn, tooSlow, hasAnnouncements, unreadAnnouncements, showAnnouncements } = this.props;
+    const pinned = !!columnId;
+    const { signedIn } = this.context.identity;
+    const banners = [];
+
+    let announcementsButton;
+
+    if (hasAnnouncements) {
+      announcementsButton = (
+        <button
+          className={classNames('column-header__button', { 'active': showAnnouncements })}
+          title={intl.formatMessage(showAnnouncements ? messages.hide_announcements : messages.show_announcements)}
+          aria-label={intl.formatMessage(showAnnouncements ? messages.hide_announcements : messages.show_announcements)}
+          onClick={this.handleToggleAnnouncementsClick}
+        >
+          <IconWithBadge id='bullhorn' count={unreadAnnouncements} />
+        </button>
+      );
+    }
+
+    if (criticalUpdatesPending) {
+      banners.push(<CriticalUpdateBanner key='critical-update-banner' />);
+    }
+
+    if (tooSlow) {
+      banners.push(<ExplorePrompt key='explore-prompt' />);
+    }
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.title)}>
+        <ColumnHeader
+          icon='home'
+          active={hasUnread}
+          title={intl.formatMessage(messages.title)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+          extraButton={announcementsButton}
+          appendContent={hasAnnouncements && showAnnouncements && <AnnouncementsContainer />}
+        >
+          <ColumnSettings />
+        </ColumnHeader>
+
+        {signedIn ? (
+          <StatusListContainer
+            prepend={banners}
+            alwaysPrepend
+            trackScroll={!pinned}
+            scrollKey={`home_timeline-${columnId}`}
+            onLoadMore={this.handleLoadMore}
+            timelineId='home'
+            emptyMessage={<FormattedMessage id='empty_column.home' defaultMessage='Your home timeline is empty! Follow more people to fill it up.' />}
+            bindToDocument={!multiColumn}
+            regex={this.props.regex}
+          />
+        ) : <NotSignedInIndicator />}
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(HomeTimeline));
diff --git a/app/javascript/flavours/blobfox/features/interaction_modal/index.jsx b/app/javascript/flavours/blobfox/features/interaction_modal/index.jsx
new file mode 100644
index 00000000000000..f6ac45d07dbc42
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/interaction_modal/index.jsx
@@ -0,0 +1,417 @@
+import PropTypes from 'prop-types';
+import React from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { connect } from 'react-redux';
+
+import { throttle, escapeRegExp } from 'lodash';
+
+import { openModal, closeModal } from 'flavours/blobfox/actions/modal';
+import api from 'flavours/blobfox/api';
+import { Button } from 'flavours/blobfox/components/button';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { registrationsOpen, sso_redirect } from 'flavours/blobfox/initial_state';
+
+const messages = defineMessages({
+  loginPrompt: { id: 'interaction_modal.login.prompt', defaultMessage: 'Domain of your home server, e.g. mastodon.social' },
+});
+
+const mapStateToProps = (state, { accountId }) => ({
+  displayNameHtml: state.getIn(['accounts', accountId, 'display_name_html']),
+  signupUrl: state.getIn(['server', 'server', 'registrations', 'url'], null) || '/auth/sign_up',
+});
+
+const mapDispatchToProps = (dispatch) => ({
+  onSignupClick() {
+    dispatch(closeModal({
+      modalType: undefined,
+      ignoreFocus: false,
+    }));
+    dispatch(openModal({ modalType: 'CLOSED_REGISTRATIONS' }));
+  },
+});
+
+const PERSISTENCE_KEY = 'mastodon_home';
+
+const isValidDomain = value => {
+  const url = new URL('https:///path');
+  url.hostname = value;
+  return url.hostname === value;
+};
+
+const valueToDomain = value => {
+  // If the user starts typing an URL
+  if (/^https?:\/\//.test(value)) {
+    try {
+      const url = new URL(value);
+
+      // Consider that if there is a path, the URL is more meaningful than a bare domain
+      if (url.pathname.length > 1) {
+        return '';
+      }
+
+      return url.host;
+    } catch {
+      return undefined;
+    }
+  // If the user writes their full handle including username
+  } else if (value.includes('@')) {
+    if (value.replace(/^@/, '').split('@').length > 2) {
+      return undefined;
+    }
+    return '';
+  }
+
+  return value;
+};
+
+const addInputToOptions = (value, options) => {
+  value = value.trim();
+
+  if (value.includes('.') && isValidDomain(value)) {
+    return [value].concat(options.filter((x) => x !== value));
+  }
+
+  return options;
+};
+
+class LoginForm extends React.PureComponent {
+
+  static propTypes = {
+    resourceUrl: PropTypes.string,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    value: localStorage ? (localStorage.getItem(PERSISTENCE_KEY) || '') : '',
+    expanded: false,
+    selectedOption: -1,
+    isLoading: false,
+    isSubmitting: false,
+    error: false,
+    options: [],
+    networkOptions: [],
+  };
+
+  setRef = c => {
+    this.input = c;
+  };
+
+  isValueValid = (value) => {
+    let likelyAcct = false;
+    let url = null;
+
+    if (value.startsWith('/')) {
+      return false;
+    }
+
+    if (value.startsWith('@')) {
+      value = value.slice(1);
+      likelyAcct = true;
+    }
+
+    // The user is in the middle of typing something, do not error out
+    if (value === '') {
+      return true;
+    }
+
+    if (/^https?:\/\//.test(value) && !likelyAcct) {
+      url = value;
+    } else {
+      url = `https://${value}`;
+    }
+
+    try {
+      new URL(url);
+      return true;
+    } catch(_) {
+      return false;
+    }
+  };
+
+  handleChange = ({ target }) => {
+    const error = !this.isValueValid(target.value);
+    this.setState(state => ({ error, value: target.value, isLoading: true, options: addInputToOptions(target.value, state.networkOptions) }), () => this._loadOptions());
+  };
+
+  handleMessage = (event) => {
+    const { resourceUrl } = this.props;
+
+    if (event.origin !== window.origin || event.source !== this.iframeRef.contentWindow) {
+      return;
+    }
+
+    if (event.data?.type === 'fetchInteractionURL-failure') {
+      this.setState({ isSubmitting: false, error: true });
+    } else if (event.data?.type === 'fetchInteractionURL-success') {
+      if (/^https?:\/\//.test(event.data.template)) {
+        try {
+          const url = new URL(event.data.template.replace('{uri}', encodeURIComponent(resourceUrl)));
+
+          if (localStorage) {
+            localStorage.setItem(PERSISTENCE_KEY, event.data.uri_or_domain);
+          }
+
+          window.location.href = url;
+        } catch (e) {
+          console.error(e);
+          this.setState({ isSubmitting: false, error: true });
+        }
+      } else {
+        this.setState({ isSubmitting: false, error: true });
+      }
+    }
+  };
+
+  componentDidMount () {
+    window.addEventListener('message', this.handleMessage);
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('message', this.handleMessage);
+  }
+
+  handleSubmit = () => {
+    const { value } = this.state;
+
+    this.setState({ isSubmitting: true });
+
+    this.iframeRef.contentWindow.postMessage({
+      type: 'fetchInteractionURL',
+      uri_or_domain: value.trim(),
+    }, window.origin);
+  };
+
+  setIFrameRef = (iframe) => {
+    this.iframeRef = iframe;
+  };
+
+  handleFocus = () => {
+    this.setState({ expanded: true });
+  };
+
+  handleBlur = () => {
+    this.setState({ expanded: false });
+  };
+
+  handleKeyDown = (e) => {
+    const { options, selectedOption } = this.state;
+
+    switch(e.key) {
+    case 'ArrowDown':
+      e.preventDefault();
+
+      if (options.length > 0) {
+        this.setState({ selectedOption: Math.min(selectedOption + 1, options.length - 1) });
+      }
+
+      break;
+    case 'ArrowUp':
+      e.preventDefault();
+
+      if (options.length > 0) {
+        this.setState({ selectedOption: Math.max(selectedOption - 1, -1) });
+      }
+
+      break;
+    case 'Enter':
+      e.preventDefault();
+
+      if (selectedOption === -1) {
+        this.handleSubmit();
+      } else if (options.length > 0) {
+        this.setState({ value: options[selectedOption], error: false }, () => this.handleSubmit());
+      }
+
+      break;
+    }
+  };
+
+  handleOptionClick = e => {
+    const index  = Number(e.currentTarget.getAttribute('data-index'));
+    const option = this.state.options[index];
+
+    e.preventDefault();
+    this.setState({ selectedOption: index, value: option, error: false }, () => this.handleSubmit());
+  };
+
+  _loadOptions = throttle(() => {
+    const { value } = this.state;
+
+    const domain = valueToDomain(value.trim());
+
+    if (typeof domain === 'undefined') {
+      this.setState({ options: [], networkOptions: [], isLoading: false, error: true });
+      return;
+    }
+
+    if (domain.length === 0) {
+      this.setState({ options: [], networkOptions: [], isLoading: false });
+      return;
+    }
+
+    api().get('/api/v1/peers/search', { params: { q: domain } }).then(({ data }) => {
+      if (!data) {
+        data = [];
+      }
+
+      this.setState((state) => ({ networkOptions: data, options: addInputToOptions(state.value, data), isLoading: false }));
+    }).catch(() => {
+      this.setState({ isLoading: false });
+    });
+  }, 200, { leading: true, trailing: true });
+
+  render () {
+    const { intl } = this.props;
+    const { value, expanded, options, selectedOption, error, isSubmitting } = this.state;
+    const domain = (valueToDomain(value) || '').trim();
+    const domainRegExp = new RegExp(`(${escapeRegExp(domain)})`, 'gi');
+    const hasPopOut = domain.length > 0 && options.length > 0;
+
+    return (
+      <div className={classNames('interaction-modal__login', { focused: expanded, expanded: hasPopOut, invalid: error })}>
+
+        <iframe
+          ref={this.setIFrameRef}
+          style={{display: 'none'}}
+          src='/remote_interaction_helper'
+          sandbox='allow-scripts allow-same-origin'
+          title='remote interaction helper'
+        />
+
+        <div className='interaction-modal__login__input'>
+          <input
+            ref={this.setRef}
+            type='text'
+            value={value}
+            placeholder={intl.formatMessage(messages.loginPrompt)}
+            aria-label={intl.formatMessage(messages.loginPrompt)}
+            autoFocus
+            onChange={this.handleChange}
+            onFocus={this.handleFocus}
+            onBlur={this.handleBlur}
+            onKeyDown={this.handleKeyDown}
+            autoComplete='off'
+            autoCapitalize='off'
+            spellCheck='false'
+          />
+
+          <Button onClick={this.handleSubmit} disabled={isSubmitting || error}><FormattedMessage id='interaction_modal.login.action' defaultMessage='Take me home' /></Button>
+        </div>
+
+        {hasPopOut && (
+          <div className='search__popout'>
+            <div className='search__popout__menu'>
+              {options.map((option, i) => (
+                <button key={option} onMouseDown={this.handleOptionClick} data-index={i} className={classNames('search__popout__menu__item', { selected: selectedOption === i })}>
+                  {option.split(domainRegExp).map((part, i) => (
+                    part.toLowerCase() === domain.toLowerCase() ? (
+                      <mark key={i}>
+                        {part}
+                      </mark>
+                    ) : (
+                      <span key={i}>
+                        {part}
+                      </span>
+                    )
+                  ))}
+                </button>
+              ))}
+            </div>
+          </div>
+        )}
+      </div>
+    );
+  }
+
+}
+
+const IntlLoginForm = injectIntl(LoginForm);
+
+class InteractionModal extends React.PureComponent {
+
+  static propTypes = {
+    displayNameHtml: PropTypes.string,
+    url: PropTypes.string,
+    type: PropTypes.oneOf(['reply', 'reblog', 'favourite', 'follow']),
+    onSignupClick: PropTypes.func.isRequired,
+    signupUrl: PropTypes.string.isRequired,
+  };
+
+  handleSignupClick = () => {
+    this.props.onSignupClick();
+  };
+
+  render () {
+    const { url, type, displayNameHtml, signupUrl } = this.props;
+
+    const name = <bdi dangerouslySetInnerHTML={{ __html: displayNameHtml }} />;
+
+    let title, actionDescription, icon;
+
+    switch(type) {
+    case 'reply':
+      icon = <Icon id='reply' />;
+      title = <FormattedMessage id='interaction_modal.title.reply' defaultMessage="Reply to {name}'s post" values={{ name }} />;
+      actionDescription = <FormattedMessage id='interaction_modal.description.reply' defaultMessage='With an account on Mastodon, you can respond to this post.' />;
+      break;
+    case 'reblog':
+      icon = <Icon id='retweet' />;
+      title = <FormattedMessage id='interaction_modal.title.reblog' defaultMessage="Boost {name}'s post" values={{ name }} />;
+      actionDescription = <FormattedMessage id='interaction_modal.description.reblog' defaultMessage='With an account on Mastodon, you can boost this post to share it with your own followers.' />;
+      break;
+    case 'favourite':
+      icon = <Icon id='star' />;
+      title = <FormattedMessage id='interaction_modal.title.favourite' defaultMessage="Favorite {name}'s post" values={{ name }} />;
+      actionDescription = <FormattedMessage id='interaction_modal.description.favourite' defaultMessage='With an account on Mastodon, you can favorite this post to let the author know you appreciate it and save it for later.' />;
+      break;
+    case 'follow':
+      icon = <Icon id='user-plus' />;
+      title = <FormattedMessage id='interaction_modal.title.follow' defaultMessage='Follow {name}' values={{ name }} />;
+      actionDescription = <FormattedMessage id='interaction_modal.description.follow' defaultMessage='With an account on Mastodon, you can follow {name} to receive their posts in your home feed.' values={{ name }} />;
+      break;
+    }
+
+    let signupButton;
+
+    if (sso_redirect) {
+      signupButton = (
+        <a href={sso_redirect} data-method='post' className='link-button'>
+          <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+        </a>
+      );
+    } else if (registrationsOpen) {
+      signupButton = (
+        <a href={signupUrl} className='link-button'>
+          <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+        </a>
+      );
+    } else {
+      signupButton = (
+        <button className='link-button' onClick={this.handleSignupClick}>
+          <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+        </button>
+      );
+    }
+
+    return (
+      <div className='modal-root__modal interaction-modal'>
+        <div className='interaction-modal__lead'>
+          <h3><span className='interaction-modal__icon'>{icon}</span> {title}</h3>
+          <p>{actionDescription} <strong><FormattedMessage id='interaction_modal.sign_in' defaultMessage='You are not logged in to this server. Where is your account hosted?' /></strong></p>
+        </div>
+
+        <IntlLoginForm resourceUrl={url} />
+
+        <p className='hint'><FormattedMessage id='interaction_modal.sign_in_hint' defaultMessage="Tip: That's the website where you signed up. If you don't remember, look for the welcome e-mail in your inbox. You can also enter your full username! (e.g. @Mastodon@mastodon.social)" /></p>
+        <p><FormattedMessage id='interaction_modal.no_account_yet' defaultMessage='Not on Mastodon?' /> {signupButton}</p>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(InteractionModal);
diff --git a/app/javascript/flavours/blobfox/features/keyboard_shortcuts/index.jsx b/app/javascript/flavours/blobfox/features/keyboard_shortcuts/index.jsx
new file mode 100644
index 00000000000000..220cc37ccc6b75
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/keyboard_shortcuts/index.jsx
@@ -0,0 +1,152 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+
+const messages = defineMessages({
+  heading: { id: 'keyboard_shortcuts.heading', defaultMessage: 'Keyboard Shortcuts' },
+});
+
+const mapStateToProps = state => ({
+  collapseEnabled: state.getIn(['local_settings', 'collapsed', 'enabled']),
+});
+
+class KeyboardShortcuts extends ImmutablePureComponent {
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+    collapseEnabled: PropTypes.bool,
+  };
+
+  render () {
+    const { intl, collapseEnabled, multiColumn } = this.props;
+
+    return (
+      <Column>
+        <ColumnHeader
+          title={intl.formatMessage(messages.heading)}
+          icon='question'
+          multiColumn={multiColumn}
+        />
+
+        <div className='keyboard-shortcuts scrollable optionally-scrollable'>
+          <table>
+            <thead>
+              <tr>
+                <th><FormattedMessage id='keyboard_shortcuts.hotkey' defaultMessage='Hotkey' /></th>
+                <th><FormattedMessage id='keyboard_shortcuts.description' defaultMessage='Description' /></th>
+              </tr>
+            </thead>
+            <tbody>
+              <tr>
+                <td><kbd>r</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.reply' defaultMessage='to reply' /></td>
+              </tr>
+              <tr>
+                <td><kbd>m</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.mention' defaultMessage='to mention author' /></td>
+              </tr>
+              <tr>
+                <td><kbd>p</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.profile' defaultMessage="to open author's profile" /></td>
+              </tr>
+              <tr>
+                <td><kbd>f</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.favourite' defaultMessage='to favorite' /></td>
+              </tr>
+              <tr>
+                <td><kbd>b</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.boost' defaultMessage='to boost' /></td>
+              </tr>
+              <tr>
+                <td><kbd>d</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.bookmark' defaultMessage='to bookmark' /></td>
+              </tr>
+              <tr>
+                <td><kbd>enter</kbd>, <kbd>o</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.enter' defaultMessage='to open status' /></td>
+              </tr>
+              <tr>
+                <td><kbd>e</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.open_media' defaultMessage='to open media' /></td>
+              </tr>
+              <tr>
+                <td><kbd>x</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.toggle_hidden' defaultMessage='to show/hide text behind CW' /></td>
+              </tr>
+              <tr>
+                <td><kbd>h</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.toggle_sensitivity' defaultMessage='to show/hide media' /></td>
+              </tr>
+              {collapseEnabled && (
+                <tr>
+                  <td><kbd>shift</kbd>+<kbd>x</kbd></td>
+                  <td><FormattedMessage id='keyboard_shortcuts.toggle_collapse' defaultMessage='to collapse/uncollapse toots' /></td>
+                </tr>
+              )}
+              <tr>
+                <td><kbd>up</kbd>, <kbd>k</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.up' defaultMessage='to move up in the list' /></td>
+              </tr>
+              <tr>
+                <td><kbd>down</kbd>, <kbd>j</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.down' defaultMessage='to move down in the list' /></td>
+              </tr>
+              <tr>
+                <td><kbd>1</kbd>-<kbd>9</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.column' defaultMessage='to focus a status in one of the columns' /></td>
+              </tr>
+              <tr>
+                <td><kbd>n</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.compose' defaultMessage='to focus the compose textarea' /></td>
+              </tr>
+              <tr>
+                <td><kbd>alt</kbd>+<kbd>n</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.toot' defaultMessage='to start a brand new post' /></td>
+              </tr>
+              <tr>
+                <td><kbd>alt</kbd>+<kbd>x</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.spoilers' defaultMessage='to show/hide CW field' /></td>
+              </tr>
+              <tr>
+                <td><kbd>backspace</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.back' defaultMessage='to navigate back' /></td>
+              </tr>
+              <tr>
+                <td><kbd>s</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.search' defaultMessage='to focus search' /></td>
+              </tr>
+              <tr>
+                <td><kbd>alt</kbd>+<kbd>enter</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.secondary_toot' defaultMessage='to send toot using secondary privacy setting' /></td>
+              </tr>
+              <tr>
+                <td><kbd>esc</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.unfocus' defaultMessage='to un-focus compose textarea/search' /></td>
+              </tr>
+              <tr>
+                <td><kbd>?</kbd></td>
+                <td><FormattedMessage id='keyboard_shortcuts.legend' defaultMessage='to display this legend' /></td>
+              </tr>
+            </tbody>
+          </table>
+        </div>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(KeyboardShortcuts));
diff --git a/app/javascript/flavours/blobfox/features/list_adder/components/account.jsx b/app/javascript/flavours/blobfox/features/list_adder/components/account.jsx
new file mode 100644
index 00000000000000..94a90726e33342
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_adder/components/account.jsx
@@ -0,0 +1,43 @@
+import { injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { Avatar } from '../../../components/avatar';
+import { DisplayName } from '../../../components/display_name';
+import { makeGetAccount } from '../../../selectors';
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { accountId }) => ({
+    account: getAccount(state, accountId),
+  });
+
+  return mapStateToProps;
+};
+
+class Account extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+  };
+
+  render () {
+    const { account } = this.props;
+    return (
+      <div className='account'>
+        <div className='account__wrapper'>
+          <div className='account__display-name'>
+            <div className='account__avatar-wrapper'><Avatar account={account} size={36} /></div>
+            <DisplayName account={account} />
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(makeMapStateToProps)(injectIntl(Account));
diff --git a/app/javascript/flavours/blobfox/features/list_adder/components/list.jsx b/app/javascript/flavours/blobfox/features/list_adder/components/list.jsx
new file mode 100644
index 00000000000000..385bd9f404a44f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_adder/components/list.jsx
@@ -0,0 +1,72 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+import { removeFromListAdder, addToListAdder } from '../../../actions/lists';
+import { IconButton }  from '../../../components/icon_button';
+
+const messages = defineMessages({
+  remove: { id: 'lists.account.remove', defaultMessage: 'Remove from list' },
+  add: { id: 'lists.account.add', defaultMessage: 'Add to list' },
+});
+
+const MapStateToProps = (state, { listId, added }) => ({
+  list: state.get('lists').get(listId),
+  added: typeof added === 'undefined' ? state.getIn(['listAdder', 'lists', 'items']).includes(listId) : added,
+});
+
+const mapDispatchToProps = (dispatch, { listId }) => ({
+  onRemove: () => dispatch(removeFromListAdder(listId)),
+  onAdd: () => dispatch(addToListAdder(listId)),
+});
+
+class List extends ImmutablePureComponent {
+
+  static propTypes = {
+    list: ImmutablePropTypes.map.isRequired,
+    intl: PropTypes.object.isRequired,
+    onRemove: PropTypes.func.isRequired,
+    onAdd: PropTypes.func.isRequired,
+    added: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    added: false,
+  };
+
+  render () {
+    const { list, intl, onRemove, onAdd, added } = this.props;
+
+    let button;
+
+    if (added) {
+      button = <IconButton icon='times' title={intl.formatMessage(messages.remove)} onClick={onRemove} />;
+    } else {
+      button = <IconButton icon='plus' title={intl.formatMessage(messages.add)} onClick={onAdd} />;
+    }
+
+    return (
+      <div className='list'>
+        <div className='list__wrapper'>
+          <div className='list__display-name'>
+            <Icon id='list-ul' className='column-link__icon' fixedWidth />
+            {list.get('title')}
+          </div>
+
+          <div className='account__relationship'>
+            {button}
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(MapStateToProps, mapDispatchToProps)(injectIntl(List));
diff --git a/app/javascript/flavours/blobfox/features/list_adder/index.jsx b/app/javascript/flavours/blobfox/features/list_adder/index.jsx
new file mode 100644
index 00000000000000..1ba9972e00d346
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_adder/index.jsx
@@ -0,0 +1,76 @@
+import PropTypes from 'prop-types';
+
+import { injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { setupListAdder, resetListAdder } from '../../actions/lists';
+import NewListForm from '../lists/components/new_list_form';
+
+import Account from './components/account';
+import List from './components/list';
+// hack
+
+const getOrderedLists = createSelector([state => state.get('lists')], lists => {
+  if (!lists) {
+    return lists;
+  }
+
+  return lists.toList().filter(item => !!item).sort((a, b) => a.get('title').localeCompare(b.get('title')));
+});
+
+const mapStateToProps = state => ({
+  listIds: getOrderedLists(state).map(list=>list.get('id')),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onInitialize: accountId => dispatch(setupListAdder(accountId)),
+  onReset: () => dispatch(resetListAdder()),
+});
+
+class ListAdder extends ImmutablePureComponent {
+
+  static propTypes = {
+    accountId: PropTypes.string.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    onInitialize: PropTypes.func.isRequired,
+    onReset: PropTypes.func.isRequired,
+    listIds: ImmutablePropTypes.list.isRequired,
+  };
+
+  componentDidMount () {
+    const { onInitialize, accountId } = this.props;
+    onInitialize(accountId);
+  }
+
+  componentWillUnmount () {
+    const { onReset } = this.props;
+    onReset();
+  }
+
+  render () {
+    const { accountId, listIds } = this.props;
+
+    return (
+      <div className='modal-root__modal list-adder'>
+        <div className='list-adder__account'>
+          <Account accountId={accountId} />
+        </div>
+
+        <NewListForm />
+
+
+        <div className='list-adder__lists'>
+          {listIds.map(ListId => <List key={ListId} listId={ListId} />)}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(ListAdder));
diff --git a/app/javascript/flavours/blobfox/features/list_editor/components/account.jsx b/app/javascript/flavours/blobfox/features/list_editor/components/account.jsx
new file mode 100644
index 00000000000000..96b5e96df87e55
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_editor/components/account.jsx
@@ -0,0 +1,79 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { removeFromListEditor, addToListEditor } from '../../../actions/lists';
+import { Avatar } from '../../../components/avatar';
+import { DisplayName } from '../../../components/display_name';
+import { IconButton } from '../../../components/icon_button';
+import { makeGetAccount } from '../../../selectors';
+
+const messages = defineMessages({
+  remove: { id: 'lists.account.remove', defaultMessage: 'Remove from list' },
+  add: { id: 'lists.account.add', defaultMessage: 'Add to list' },
+});
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { accountId, added }) => ({
+    account: getAccount(state, accountId),
+    added: typeof added === 'undefined' ? state.getIn(['listEditor', 'accounts', 'items']).includes(accountId) : added,
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { accountId }) => ({
+  onRemove: () => dispatch(removeFromListEditor(accountId)),
+  onAdd: () => dispatch(addToListEditor(accountId)),
+});
+
+class Account extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    intl: PropTypes.object.isRequired,
+    onRemove: PropTypes.func.isRequired,
+    onAdd: PropTypes.func.isRequired,
+    added: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    added: false,
+  };
+
+  render () {
+    const { account, intl, onRemove, onAdd, added } = this.props;
+
+    let button;
+
+    if (added) {
+      button = <IconButton icon='times' title={intl.formatMessage(messages.remove)} onClick={onRemove} />;
+    } else {
+      button = <IconButton icon='plus' title={intl.formatMessage(messages.add)} onClick={onAdd} />;
+    }
+
+    return (
+      <div className='account'>
+        <div className='account__wrapper'>
+          <div className='account__display-name'>
+            <div className='account__avatar-wrapper'><Avatar account={account} size={36} /></div>
+            <DisplayName account={account} />
+          </div>
+
+          <div className='account__relationship'>
+            {button}
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(injectIntl(Account));
diff --git a/app/javascript/flavours/blobfox/features/list_editor/components/edit_list_form.jsx b/app/javascript/flavours/blobfox/features/list_editor/components/edit_list_form.jsx
new file mode 100644
index 00000000000000..9e087a97d714d9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_editor/components/edit_list_form.jsx
@@ -0,0 +1,73 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { changeListEditorTitle, submitListEditor } from '../../../actions/lists';
+import { IconButton } from '../../../components/icon_button';
+
+const messages = defineMessages({
+  title: { id: 'lists.edit.submit', defaultMessage: 'Change title' },
+});
+
+const mapStateToProps = state => ({
+  value: state.getIn(['listEditor', 'title']),
+  disabled: !state.getIn(['listEditor', 'isChanged']) || !state.getIn(['listEditor', 'title']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onChange: value => dispatch(changeListEditorTitle(value)),
+  onSubmit: () => dispatch(submitListEditor(false)),
+});
+
+class ListForm extends PureComponent {
+
+  static propTypes = {
+    value: PropTypes.string.isRequired,
+    disabled: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    onChange: PropTypes.func.isRequired,
+    onSubmit: PropTypes.func.isRequired,
+  };
+
+  handleChange = e => {
+    this.props.onChange(e.target.value);
+  };
+
+  handleSubmit = e => {
+    e.preventDefault();
+    this.props.onSubmit();
+  };
+
+  handleClick = () => {
+    this.props.onSubmit();
+  };
+
+  render () {
+    const { value, disabled, intl } = this.props;
+
+    const title = intl.formatMessage(messages.title);
+
+    return (
+      <form className='column-inline-form' onSubmit={this.handleSubmit}>
+        <input
+          className='setting-text'
+          value={value}
+          onChange={this.handleChange}
+        />
+
+        <IconButton
+          disabled={disabled}
+          icon='check'
+          title={title}
+          onClick={this.handleClick}
+        />
+      </form>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(ListForm));
diff --git a/app/javascript/flavours/blobfox/features/list_editor/components/search.jsx b/app/javascript/flavours/blobfox/features/list_editor/components/search.jsx
new file mode 100644
index 00000000000000..04899ac8a719a1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_editor/components/search.jsx
@@ -0,0 +1,81 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { connect } from 'react-redux';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+import { fetchListSuggestions, clearListSuggestions, changeListSuggestions } from '../../../actions/lists';
+
+const messages = defineMessages({
+  search: { id: 'lists.search', defaultMessage: 'Search among people you follow' },
+});
+
+const mapStateToProps = state => ({
+  value: state.getIn(['listEditor', 'suggestions', 'value']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onSubmit: value => dispatch(fetchListSuggestions(value)),
+  onClear: () => dispatch(clearListSuggestions()),
+  onChange: value => dispatch(changeListSuggestions(value)),
+});
+
+class Search extends PureComponent {
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    value: PropTypes.string.isRequired,
+    onChange: PropTypes.func.isRequired,
+    onSubmit: PropTypes.func.isRequired,
+    onClear: PropTypes.func.isRequired,
+  };
+
+  handleChange = e => {
+    this.props.onChange(e.target.value);
+  };
+
+  handleKeyUp = e => {
+    if (e.keyCode === 13) {
+      this.props.onSubmit(this.props.value);
+    }
+  };
+
+  handleClear = () => {
+    this.props.onClear();
+  };
+
+  render () {
+    const { value, intl } = this.props;
+    const hasValue = value.length > 0;
+
+    return (
+      <div className='list-editor__search search'>
+        <label>
+          <span style={{ display: 'none' }}>{intl.formatMessage(messages.search)}</span>
+
+          <input
+            className='search__input'
+            type='text'
+            value={value}
+            onChange={this.handleChange}
+            onKeyUp={this.handleKeyUp}
+            placeholder={intl.formatMessage(messages.search)}
+          />
+        </label>
+
+        <div role='button' tabIndex={0} className='search__icon' onClick={this.handleClear}>
+          <Icon id='search' className={classNames({ active: !hasValue })} />
+          <Icon id='times-circle' aria-label={intl.formatMessage(messages.search)} className={classNames({ active: hasValue })} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(Search));
diff --git a/app/javascript/flavours/blobfox/features/list_editor/index.jsx b/app/javascript/flavours/blobfox/features/list_editor/index.jsx
new file mode 100644
index 00000000000000..85e90169e80b2c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_editor/index.jsx
@@ -0,0 +1,83 @@
+import PropTypes from 'prop-types';
+
+import { injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import spring from 'react-motion/lib/spring';
+
+import { setupListEditor, clearListSuggestions, resetListEditor } from '../../actions/lists';
+import Motion from '../ui/util/optional_motion';
+
+import Account from './components/account';
+import EditListForm from './components/edit_list_form';
+import Search from './components/search';
+
+const mapStateToProps = state => ({
+  accountIds: state.getIn(['listEditor', 'accounts', 'items']),
+  searchAccountIds: state.getIn(['listEditor', 'suggestions', 'items']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onInitialize: listId => dispatch(setupListEditor(listId)),
+  onClear: () => dispatch(clearListSuggestions()),
+  onReset: () => dispatch(resetListEditor()),
+});
+
+class ListEditor extends ImmutablePureComponent {
+
+  static propTypes = {
+    listId: PropTypes.string.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    onInitialize: PropTypes.func.isRequired,
+    onClear: PropTypes.func.isRequired,
+    onReset: PropTypes.func.isRequired,
+    accountIds: ImmutablePropTypes.list.isRequired,
+    searchAccountIds: ImmutablePropTypes.list.isRequired,
+  };
+
+  componentDidMount () {
+    const { onInitialize, listId } = this.props;
+    onInitialize(listId);
+  }
+
+  componentWillUnmount () {
+    const { onReset } = this.props;
+    onReset();
+  }
+
+  render () {
+    const { accountIds, searchAccountIds, onClear } = this.props;
+    const showSearch = searchAccountIds.size > 0;
+
+    return (
+      <div className='modal-root__modal list-editor'>
+        <EditListForm />
+
+        <Search />
+
+        <div className='drawer__pager'>
+          <div className='drawer__inner list-editor__accounts'>
+            {accountIds.map(accountId => <Account key={accountId} accountId={accountId} added />)}
+          </div>
+
+          {showSearch && <div role='button' tabIndex={-1} className='drawer__backdrop' onClick={onClear} />}
+
+          <Motion defaultStyle={{ x: -100 }} style={{ x: spring(showSearch ? 0 : -100, { stiffness: 210, damping: 20 }) }}>
+            {({ x }) => (
+              <div className='drawer__inner backdrop' style={{ transform: x === 0 ? null : `translateX(${x}%)`, visibility: x === -100 ? 'hidden' : 'visible' }}>
+                {searchAccountIds.map(accountId => <Account key={accountId} accountId={accountId} />)}
+              </div>
+            )}
+          </Motion>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(ListEditor));
diff --git a/app/javascript/flavours/blobfox/features/list_timeline/index.jsx b/app/javascript/flavours/blobfox/features/list_timeline/index.jsx
new file mode 100644
index 00000000000000..e3ddab06abcfff
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/list_timeline/index.jsx
@@ -0,0 +1,244 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import Toggle from 'react-toggle';
+
+import { addColumn, removeColumn, moveColumn } from 'flavours/blobfox/actions/columns';
+import { fetchList, deleteList, updateList } from 'flavours/blobfox/actions/lists';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { connectListStream } from 'flavours/blobfox/actions/streaming';
+import { expandListTimeline } from 'flavours/blobfox/actions/timelines';
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import { RadioButton } from 'flavours/blobfox/components/radio_button';
+import BundleColumnError from 'flavours/blobfox/features/ui/components/bundle_column_error';
+import StatusListContainer from 'flavours/blobfox/features/ui/containers/status_list_container';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const messages = defineMessages({
+  deleteMessage: { id: 'confirmations.delete_list.message', defaultMessage: 'Are you sure you want to permanently delete this list?' },
+  deleteConfirm: { id: 'confirmations.delete_list.confirm', defaultMessage: 'Delete' },
+  followed:   { id: 'lists.replies_policy.followed', defaultMessage: 'Any followed user' },
+  none:    { id: 'lists.replies_policy.none', defaultMessage: 'No one' },
+  list:  { id: 'lists.replies_policy.list', defaultMessage: 'Members of the list' },
+});
+
+const mapStateToProps = (state, props) => ({
+  list: state.getIn(['lists', props.params.id]),
+  hasUnread: state.getIn(['timelines', `list:${props.params.id}`, 'unread']) > 0,
+});
+
+class ListTimeline extends PureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    columnId: PropTypes.string,
+    hasUnread: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    list: PropTypes.oneOfType([ImmutablePropTypes.map, PropTypes.bool]),
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('LIST', { id: this.props.params.id }));
+      this.props.history.push('/');
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    const { id } = this.props.params;
+
+    dispatch(fetchList(id));
+    dispatch(expandListTimeline(id));
+
+    this.disconnect = dispatch(connectListStream(id));
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    const { dispatch } = this.props;
+    const { id } = nextProps.params;
+
+    if (id !== this.props.params.id) {
+      if (this.disconnect) {
+        this.disconnect();
+        this.disconnect = null;
+      }
+
+      dispatch(fetchList(id));
+      dispatch(expandListTimeline(id));
+
+      this.disconnect = dispatch(connectListStream(id));
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.disconnect) {
+      this.disconnect();
+      this.disconnect = null;
+    }
+  }
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = maxId => {
+    const { id } = this.props.params;
+    this.props.dispatch(expandListTimeline(id, { maxId }));
+  };
+
+  handleEditClick = () => {
+    this.props.dispatch(openModal({
+      modalType: 'LIST_EDITOR',
+      modalProps: { listId: this.props.params.id },
+    }));
+  };
+
+  handleDeleteClick = () => {
+    const { dispatch, columnId, intl } = this.props;
+    const { id } = this.props.params;
+
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.deleteMessage),
+        confirm: intl.formatMessage(messages.deleteConfirm),
+        onConfirm: () => {
+          dispatch(deleteList(id));
+
+          if (columnId) {
+            dispatch(removeColumn(columnId));
+          } else {
+            this.props.history.push('/lists');
+          }
+        },
+      },
+    }));
+  };
+
+  handleRepliesPolicyChange = ({ target }) => {
+    const { dispatch } = this.props;
+    const { id } = this.props.params;
+    dispatch(updateList(id, undefined, false, undefined, target.value));
+  };
+
+  onExclusiveToggle = ({ target }) => {
+    const { dispatch } = this.props;
+    const { id } = this.props.params;
+    dispatch(updateList(id, undefined, false, target.checked, undefined));
+  };
+
+  render () {
+    const { hasUnread, columnId, multiColumn, list, intl } = this.props;
+    const { id } = this.props.params;
+    const pinned = !!columnId;
+    const title  = list ? list.get('title') : id;
+    const replies_policy = list ? list.get('replies_policy') : undefined;
+    const isExclusive = list ? list.get('exclusive') : undefined;
+
+    if (typeof list === 'undefined') {
+      return (
+        <Column>
+          <div className='scrollable'>
+            <LoadingIndicator />
+          </div>
+        </Column>
+      );
+    } else if (list === false) {
+      return (
+        <BundleColumnError multiColumn={multiColumn} errorType='routing' />
+      );
+    }
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={title}>
+        <ColumnHeader
+          icon='list-ul'
+          active={hasUnread}
+          title={title}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+        >
+          <div className='column-settings__row column-header__links'>
+            <button type='button' className='text-btn column-header__setting-btn' tabIndex={0} onClick={this.handleEditClick}>
+              <Icon id='pencil' /> <FormattedMessage id='lists.edit' defaultMessage='Edit list' />
+            </button>
+
+            <button type='button' className='text-btn column-header__setting-btn' tabIndex={0} onClick={this.handleDeleteClick}>
+              <Icon id='trash' /> <FormattedMessage id='lists.delete' defaultMessage='Delete list' />
+            </button>
+          </div>
+
+          <div className='setting-toggle'>
+            <Toggle id={`list-${id}-exclusive`} checked={isExclusive} onChange={this.onExclusiveToggle} />
+            <label htmlFor={`list-${id}-exclusive`} className='setting-toggle__label'>
+              <FormattedMessage id='lists.exclusive' defaultMessage='Hide these posts from home' />
+            </label>
+          </div>
+
+          { replies_policy !== undefined && (
+            <div role='group' aria-labelledby={`list-${id}-replies-policy`}>
+              <span id={`list-${id}-replies-policy`} className='column-settings__section'>
+                <FormattedMessage id='lists.replies_policy.title' defaultMessage='Show replies to:' />
+              </span>
+              <div className='column-settings__row'>
+                { ['none', 'list', 'followed'].map(policy => (
+                  <RadioButton name='order' key={policy} value={policy} label={intl.formatMessage(messages[policy])} checked={replies_policy === policy} onChange={this.handleRepliesPolicyChange} />
+                ))}
+              </div>
+            </div>
+          )}
+
+          <hr />
+        </ColumnHeader>
+
+        <StatusListContainer
+          trackScroll={!pinned}
+          scrollKey={`list_timeline-${columnId}`}
+          timelineId={`list:${id}`}
+          onLoadMore={this.handleLoadMore}
+          emptyMessage={<FormattedMessage id='empty_column.list' defaultMessage='There is nothing in this list yet. When members of this list post new statuses, they will appear here.' />}
+          bindToDocument={!multiColumn}
+        />
+
+        <Helmet>
+          <title>{title}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default withRouter(connect(mapStateToProps)(injectIntl(ListTimeline)));
diff --git a/app/javascript/flavours/blobfox/features/lists/components/new_list_form.jsx b/app/javascript/flavours/blobfox/features/lists/components/new_list_form.jsx
new file mode 100644
index 00000000000000..f5581a765a7a7c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/lists/components/new_list_form.jsx
@@ -0,0 +1,81 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { changeListEditorTitle, submitListEditor } from 'flavours/blobfox/actions/lists';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+const messages = defineMessages({
+  label: { id: 'lists.new.title_placeholder', defaultMessage: 'New list title' },
+  title: { id: 'lists.new.create', defaultMessage: 'Add list' },
+});
+
+const mapStateToProps = state => ({
+  value: state.getIn(['listEditor', 'title']),
+  disabled: state.getIn(['listEditor', 'isSubmitting']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onChange: value => dispatch(changeListEditorTitle(value)),
+  onSubmit: () => dispatch(submitListEditor(true)),
+});
+
+class NewListForm extends PureComponent {
+
+  static propTypes = {
+    value: PropTypes.string.isRequired,
+    disabled: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    onChange: PropTypes.func.isRequired,
+    onSubmit: PropTypes.func.isRequired,
+  };
+
+  handleChange = e => {
+    this.props.onChange(e.target.value);
+  };
+
+  handleSubmit = e => {
+    e.preventDefault();
+    this.props.onSubmit();
+  };
+
+  handleClick = () => {
+    this.props.onSubmit();
+  };
+
+  render () {
+    const { value, disabled, intl } = this.props;
+
+    const label = intl.formatMessage(messages.label);
+    const title = intl.formatMessage(messages.title);
+
+    return (
+      <form className='column-inline-form' onSubmit={this.handleSubmit}>
+        <label>
+          <span style={{ display: 'none' }}>{label}</span>
+
+          <input
+            className='setting-text'
+            value={value}
+            disabled={disabled}
+            onChange={this.handleChange}
+            placeholder={label}
+          />
+        </label>
+
+        <IconButton
+          disabled={disabled || !value}
+          icon='plus'
+          title={title}
+          onClick={this.handleClick}
+        />
+      </form>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(NewListForm));
diff --git a/app/javascript/flavours/blobfox/features/lists/index.jsx b/app/javascript/flavours/blobfox/features/lists/index.jsx
new file mode 100644
index 00000000000000..9384df3cf210e1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/lists/index.jsx
@@ -0,0 +1,93 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { fetchLists } from 'flavours/blobfox/actions/lists';
+import ColumnBackButtonSlim from 'flavours/blobfox/components/column_back_button_slim';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import ScrollableList from 'flavours/blobfox/components/scrollable_list';
+import Column from 'flavours/blobfox/features/ui/components/column';
+import ColumnLink from 'flavours/blobfox/features/ui/components/column_link';
+import ColumnSubheading from 'flavours/blobfox/features/ui/components/column_subheading';
+
+import NewListForm from './components/new_list_form';
+
+const messages = defineMessages({
+  heading: { id: 'column.lists', defaultMessage: 'Lists' },
+  subheading: { id: 'lists.subheading', defaultMessage: 'Your lists' },
+});
+
+const getOrderedLists = createSelector([state => state.get('lists')], lists => {
+  if (!lists) {
+    return lists;
+  }
+
+  return lists.toList().filter(item => !!item).sort((a, b) => a.get('title').localeCompare(b.get('title')));
+});
+
+const mapStateToProps = state => ({
+  lists: getOrderedLists(state),
+});
+
+class Lists extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    lists: ImmutablePropTypes.list,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchLists());
+  }
+
+  render () {
+    const { intl, lists, multiColumn } = this.props;
+
+    if (!lists) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    const emptyMessage = <FormattedMessage id='empty_column.lists' defaultMessage="You don't have any lists yet. When you create one, it will show up here." />;
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='bars' heading={intl.formatMessage(messages.heading)}>
+        <ColumnBackButtonSlim />
+
+        <NewListForm />
+
+        <ColumnSubheading text={intl.formatMessage(messages.subheading)} />
+        <ScrollableList
+          scrollKey='lists'
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {lists.map(list =>
+            <ColumnLink key={list.get('id')} to={`/lists/${list.get('id')}`} icon='list-ul' text={list.get('title')} />,
+          )}
+        </ScrollableList>
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.heading)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Lists));
diff --git a/app/javascript/flavours/blobfox/features/local_settings/index.jsx b/app/javascript/flavours/blobfox/features/local_settings/index.jsx
new file mode 100644
index 00000000000000..c6062c2bd06c7c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/local_settings/index.jsx
@@ -0,0 +1,70 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+//  Our imports
+import { changeLocalSetting } from 'flavours/blobfox/actions/local_settings';
+import { closeModal } from 'flavours/blobfox/actions/modal';
+
+import LocalSettingsNavigation from './navigation';
+import LocalSettingsPage from './page';
+
+const mapStateToProps = state => ({
+  settings: state.get('local_settings'),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onChange (setting, value) {
+    dispatch(changeLocalSetting(setting, value));
+  },
+  onClose () {
+    dispatch(closeModal({
+      modalType: undefined,
+      ignoreFocus: false,
+    }));
+  },
+});
+
+class LocalSettings extends PureComponent {
+
+  static propTypes = {
+    onChange: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    settings: ImmutablePropTypes.map.isRequired,
+  };
+
+  state = {
+    currentIndex: 0,
+  };
+
+  navigateTo = (index) =>
+    this.setState({ currentIndex: +index });
+
+  render () {
+
+    const { navigateTo } = this;
+    const { onChange, onClose, settings } = this.props;
+    const { currentIndex } = this.state;
+
+    return (
+      <div className='blobfox modal-root__modal local-settings'>
+        <LocalSettingsNavigation
+          index={currentIndex}
+          onClose={onClose}
+          onNavigate={navigateTo}
+        />
+        <LocalSettingsPage
+          index={currentIndex}
+          onChange={onChange}
+          settings={settings}
+        />
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(LocalSettings);
diff --git a/app/javascript/flavours/blobfox/features/local_settings/navigation/index.jsx b/app/javascript/flavours/blobfox/features/local_settings/navigation/index.jsx
new file mode 100644
index 00000000000000..1f9f3a3edb84cc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/local_settings/navigation/index.jsx
@@ -0,0 +1,95 @@
+//  Package imports
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+//  Our imports
+import { preferencesLink } from 'flavours/blobfox/utils/backend_links';
+
+import LocalSettingsNavigationItem from './item';
+
+//  * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
+
+const messages = defineMessages({
+  general: {  id: 'settings.general', defaultMessage: 'General' },
+  compose: {  id: 'settings.compose_box_opts', defaultMessage: 'Compose box' },
+  content_warnings: { id: 'settings.content_warnings', defaultMessage: 'Content Warnings' },
+  collapsed: { id: 'settings.collapsed_statuses', defaultMessage: 'Collapsed toots' },
+  media: { id: 'settings.media', defaultMessage: 'Media' },
+  preferences: { id: 'settings.preferences', defaultMessage: 'Preferences' },
+  close: { id: 'settings.close', defaultMessage: 'Close' },
+});
+
+class LocalSettingsNavigation extends PureComponent {
+
+  static propTypes = {
+    index      : PropTypes.number,
+    intl       : PropTypes.object.isRequired,
+    onClose    : PropTypes.func.isRequired,
+    onNavigate : PropTypes.func.isRequired,
+  };
+
+  render () {
+
+    const { index, intl, onClose, onNavigate } = this.props;
+
+    return (
+      <nav className='blobfox local-settings__navigation'>
+        <LocalSettingsNavigationItem
+          active={index === 0}
+          index={0}
+          onNavigate={onNavigate}
+          icon='cogs'
+          title={intl.formatMessage(messages.general)}
+        />
+        <LocalSettingsNavigationItem
+          active={index === 1}
+          index={1}
+          onNavigate={onNavigate}
+          icon='pencil'
+          title={intl.formatMessage(messages.compose)}
+        />
+        <LocalSettingsNavigationItem
+          active={index === 2}
+          index={2}
+          onNavigate={onNavigate}
+          textIcon='CW'
+          title={intl.formatMessage(messages.content_warnings)}
+        />
+        <LocalSettingsNavigationItem
+          active={index === 3}
+          index={3}
+          onNavigate={onNavigate}
+          icon='angle-double-up'
+          title={intl.formatMessage(messages.collapsed)}
+        />
+        <LocalSettingsNavigationItem
+          active={index === 4}
+          index={4}
+          onNavigate={onNavigate}
+          icon='image'
+          title={intl.formatMessage(messages.media)}
+        />
+        <LocalSettingsNavigationItem
+          active={index === 5}
+          href={preferencesLink}
+          index={5}
+          icon='cog'
+          title={intl.formatMessage(messages.preferences)}
+        />
+        <LocalSettingsNavigationItem
+          active={index === 6}
+          className='close'
+          index={6}
+          onNavigate={onClose}
+          icon='times'
+          title={intl.formatMessage(messages.close)}
+        />
+      </nav>
+    );
+  }
+
+}
+
+export default injectIntl(LocalSettingsNavigation);
diff --git a/app/javascript/flavours/blobfox/features/local_settings/navigation/item/index.jsx b/app/javascript/flavours/blobfox/features/local_settings/navigation/item/index.jsx
new file mode 100644
index 00000000000000..30e692a92f0844
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/local_settings/navigation/item/index.jsx
@@ -0,0 +1,73 @@
+//  Package imports
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+//  * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
+
+export default class LocalSettingsPage extends PureComponent {
+
+  static propTypes = {
+    active: PropTypes.bool,
+    className: PropTypes.string,
+    href: PropTypes.string,
+    icon: PropTypes.string,
+    textIcon: PropTypes.string,
+    index: PropTypes.number.isRequired,
+    onNavigate: PropTypes.func,
+    title: PropTypes.string,
+  };
+
+  handleClick = (e) => {
+    const { index, onNavigate } = this.props;
+    if (onNavigate) {
+      onNavigate(index);
+      e.preventDefault();
+    }
+  };
+
+  render () {
+    const { handleClick } = this;
+    const {
+      active,
+      className,
+      href,
+      icon,
+      textIcon,
+      onNavigate,
+      title,
+    } = this.props;
+
+    const finalClassName = classNames('blobfox', 'local-settings__navigation__item', {
+      active,
+    }, className);
+
+    const iconElem = icon ? <Icon fixedWidth id={icon} /> : (textIcon ? <span className='text-icon-button'>{textIcon}</span> : null);
+
+    if (href) return (
+      <a
+        href={href}
+        className={finalClassName}
+        title={title}
+        aria-label={title}
+      >
+        {iconElem} <span>{title}</span>
+      </a>
+    );
+    else if (onNavigate) return (
+      <button
+        onClick={handleClick}
+        className={finalClassName}
+        title={title}
+        aria-label={title}
+      >
+        {iconElem} <span>{title}</span>
+      </button>
+    );
+    else return null;
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/local_settings/page/deprecated_item/index.jsx b/app/javascript/flavours/blobfox/features/local_settings/page/deprecated_item/index.jsx
new file mode 100644
index 00000000000000..0f53952cbe45cf
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/local_settings/page/deprecated_item/index.jsx
@@ -0,0 +1,83 @@
+//  Package imports
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+//  * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
+
+export default class LocalSettingsPageItem extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node.isRequired,
+    id: PropTypes.string.isRequired,
+    options: PropTypes.arrayOf(PropTypes.shape({
+      value: PropTypes.string.isRequired,
+      message: PropTypes.string.isRequired,
+      hint: PropTypes.string,
+    })),
+    value: PropTypes.any,
+    placeholder: PropTypes.string,
+  };
+
+  render () {
+    const { id, options, children, placeholder, value } = this.props;
+
+    if (options && options.length > 0) {
+      const currentValue = value;
+      const optionElems = options && options.length > 0 && options.map((opt) => {
+        let optionId = `${id}--${opt.value}`;
+        return (
+          <label key={id} htmlFor={optionId}>
+            <input
+              type='radio'
+              name={id}
+              id={optionId}
+              value={opt.value}
+              checked={currentValue === opt.value}
+              disabled
+            />
+            {opt.message}
+            {opt.hint && <span className='hint'>{opt.hint}</span>}
+          </label>
+        );
+      });
+      return (
+        <div className='blobfox local-settings__page__item radio_buttons'>
+          <fieldset>
+            <legend>{children}</legend>
+            {optionElems}
+          </fieldset>
+        </div>
+      );
+    } else if (placeholder) {
+      return (
+        <div className='blobfox local-settings__page__item string'>
+          <label htmlFor={id}>
+            <p>{children}</p>
+            <p>
+              <input
+                id={id}
+                type='text'
+                value={value}
+                placeholder={placeholder}
+                disabled
+              />
+            </p>
+          </label>
+        </div>
+      );
+    } else return (
+      <div className='blobfox local-settings__page__item boolean'>
+        <label htmlFor={id}>
+          <input
+            id={id}
+            type='checkbox'
+            checked={value}
+            disabled
+          />
+          {children}
+        </label>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/local_settings/page/index.jsx b/app/javascript/flavours/blobfox/features/local_settings/page/index.jsx
new file mode 100644
index 00000000000000..917839d5c10ac5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/local_settings/page/index.jsx
@@ -0,0 +1,505 @@
+//  Package imports
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+
+//  Our imports
+import { expandSpoilers } from 'flavours/blobfox/initial_state';
+import { preferenceLink } from 'flavours/blobfox/utils/backend_links';
+
+import DeprecatedLocalSettingsPageItem from './deprecated_item';
+import LocalSettingsPageItem from './item';
+
+//  * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
+
+const messages = defineMessages({
+  side_arm_none: { id: 'settings.side_arm.none', defaultMessage: 'None' },
+  side_arm_keep: { id: 'settings.side_arm_reply_mode.keep', defaultMessage: 'Keep its set privacy' },
+  side_arm_copy: { id: 'settings.side_arm_reply_mode.copy', defaultMessage: 'Copy privacy setting of the toot being replied to' },
+  side_arm_restrict: { id: 'settings.side_arm_reply_mode.restrict', defaultMessage: 'Restrict privacy setting to that of the toot being replied to' },
+  regexp: { id: 'settings.content_warnings.regexp', defaultMessage: 'Regular expression' },
+  rewrite_mentions_no: { id: 'settings.rewrite_mentions_no', defaultMessage: 'Do not rewrite mentions' },
+  rewrite_mentions_acct: { id: 'settings.rewrite_mentions_acct', defaultMessage: 'Rewrite with username and domain (when the account is remote)' },
+  rewrite_mentions_username: { id: 'settings.rewrite_mentions_username', defaultMessage:  'Rewrite with username' },
+  pop_in_left: { id: 'settings.pop_in_left', defaultMessage: 'Left' },
+  pop_in_right: { id: 'settings.pop_in_right', defaultMessage:  'Right' },
+  public: { id: 'privacy.public.short', defaultMessage: 'Public' },
+  unlisted: { id: 'privacy.unlisted.short', defaultMessage: 'Unlisted' },
+  private: { id: 'privacy.private.short', defaultMessage: 'Followers only' },
+  direct: { id: 'privacy.direct.short', defaultMessage: 'Mentioned people only' },
+});
+
+class LocalSettingsPage extends PureComponent {
+
+  static propTypes = {
+    index    : PropTypes.number,
+    intl     : PropTypes.object.isRequired,
+    onChange : PropTypes.func.isRequired,
+    settings : ImmutablePropTypes.map.isRequired,
+  };
+
+  pages = [
+    ({ intl, onChange, settings }) => (
+      <div className='blobfox local-settings__page general'>
+        <h1><FormattedMessage id='settings.general' defaultMessage='General' /></h1>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['show_reply_count']}
+          id='mastodon-settings--reply-count'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.show_reply_counter' defaultMessage='Display an estimate of the reply count' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['hicolor_privacy_icons']}
+          id='mastodon-settings--hicolor_privacy_icons'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.hicolor_privacy_icons' defaultMessage='High color privacy icons' />
+          <span className='hint'><FormattedMessage id='settings.hicolor_privacy_icons.hint' defaultMessage='Display privacy icons in bright and easily distinguishable colors' /></span>
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['confirm_boost_missing_media_description']}
+          id='mastodon-settings--confirm_boost_missing_media_description'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.confirm_boost_missing_media_description' defaultMessage='Show confirmation dialog before boosting toots lacking media descriptions' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['tag_misleading_links']}
+          id='mastodon-settings--tag_misleading_links'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.tag_misleading_links' defaultMessage='Tag misleading links' />
+          <span className='hint'><FormattedMessage id='settings.tag_misleading_links.hint' defaultMessage='Add a visual indication with the link target host to every link not mentioning it explicitly' /></span>
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['rewrite_mentions']}
+          id='mastodon-settings--rewrite_mentions'
+          options={[
+            { value: 'no', message: intl.formatMessage(messages.rewrite_mentions_no) },
+            { value: 'acct', message: intl.formatMessage(messages.rewrite_mentions_acct) },
+            { value: 'username', message: intl.formatMessage(messages.rewrite_mentions_username) },
+          ]}
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.rewrite_mentions' defaultMessage='Rewrite mentions in displayed statuses' />
+        </LocalSettingsPageItem>
+        <section>
+          <h2><FormattedMessage id='settings.notifications_opts' defaultMessage='Notifications options' /></h2>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['notifications', 'tab_badge']}
+            id='mastodon-settings--notifications-tab_badge'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.notifications.tab_badge' defaultMessage='Unread notifications badge' />
+            <span className='hint'><FormattedMessage id='settings.notifications.tab_badge.hint' defaultMessage="Display a badge for unread notifications in the column icons when the notifications column isn't open" /></span>
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['notifications', 'favicon_badge']}
+            id='mastodon-settings--notifications-favicon_badge'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.notifications.favicon_badge' defaultMessage='Unread notifications favicon badge' />
+            <span className='hint'><FormattedMessage id='settings.notifications.favicon_badge.hint' defaultMessage='Add a badge for unread notifications to the favicon' /></span>
+          </LocalSettingsPageItem>
+        </section>
+
+        <section>
+          <h2><FormattedMessage id='settings.status_icons' defaultMessage='Toot icons' /></h2>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['status_icons', 'language']}
+            id='mastodon-settings--status-icons-language'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.status_icons_language' defaultMessage='Language indicator' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['status_icons', 'reply']}
+            id='mastodon-settings--status-icons-reply'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.status_icons_reply' defaultMessage='Reply indicator' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['status_icons', 'local_only']}
+            id='mastodon-settings--status-icons-local_only'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.status_icons_local_only' defaultMessage='Local-only indicator' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['status_icons', 'media']}
+            id='mastodon-settings--status-icons-media'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.status_icons_media' defaultMessage='Media and poll indicators' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['status_icons', 'visibility']}
+            id='mastodon-settings--status-icons-visibility'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.status_icons_visibility' defaultMessage='Toot privacy indicator' />
+          </LocalSettingsPageItem>
+        </section>
+        <section>
+          <h2><FormattedMessage id='settings.layout_opts' defaultMessage='Layout options' /></h2>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['stretch']}
+            id='mastodon-settings--stretch'
+            onChange={onChange}
+          >
+            <FormattedMessage id='settings.wide_view' defaultMessage='Wide view (Desktop mode only)' />
+            <span className='hint'><FormattedMessage id='settings.wide_view_hint' defaultMessage='Stretches columns to better fill the available space.' /></span>
+          </LocalSettingsPageItem>
+        </section>
+      </div>
+    ),
+    ({ intl, onChange, settings }) => (
+      <div className='blobfox local-settings__page compose_box_opts'>
+        <h1><FormattedMessage id='settings.compose_box_opts' defaultMessage='Compose box' /></h1>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['always_show_spoilers_field']}
+          id='mastodon-settings--always_show_spoilers_field'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.always_show_spoilers_field' defaultMessage='Always enable the Content Warning field' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['prepend_cw_re']}
+          id='mastodon-settings--prepend_cw_re'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.prepend_cw_re' defaultMessage='Prepend “re: ” to content warnings when replying' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['preselect_on_reply']}
+          id='mastodon-settings--preselect_on_reply'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.preselect_on_reply' defaultMessage='Pre-select usernames on reply' />
+          <span className='hint'><FormattedMessage id='settings.preselect_on_reply_hint' defaultMessage='When replying to a conversation with multiple participants, pre-select usernames past the first' /></span>
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['confirm_missing_media_description']}
+          id='mastodon-settings--confirm_missing_media_description'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.confirm_missing_media_description' defaultMessage='Show confirmation dialog before sending toots lacking media descriptions' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['confirm_before_clearing_draft']}
+          id='mastodon-settings--confirm_before_clearing_draft'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.confirm_before_clearing_draft' defaultMessage='Show confirmation dialog before overwriting the message being composed' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['show_content_type_choice']}
+          id='mastodon-settings--show_content_type_choice'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.show_content_type_choice' defaultMessage='Show content-type choice when authoring toots' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['side_arm']}
+          id='mastodon-settings--side_arm'
+          options={[
+            { value: 'none', message: intl.formatMessage(messages.side_arm_none) },
+            { value: 'direct', message: intl.formatMessage(messages.direct) },
+            { value: 'private', message: intl.formatMessage(messages.private) },
+            { value: 'unlisted', message: intl.formatMessage(messages.unlisted) },
+            { value: 'public', message: intl.formatMessage(messages.public) },
+          ]}
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.side_arm' defaultMessage='Secondary toot button:' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['side_arm_reply_mode']}
+          id='mastodon-settings--side_arm_reply_mode'
+          options={[
+            { value: 'keep', message: intl.formatMessage(messages.side_arm_keep) },
+            { value: 'copy', message: intl.formatMessage(messages.side_arm_copy) },
+            { value: 'restrict', message: intl.formatMessage(messages.side_arm_restrict) },
+          ]}
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.side_arm_reply_mode' defaultMessage='When replying to a toot, the secondary toot button should:' />
+        </LocalSettingsPageItem>
+      </div>
+    ),
+    ({ intl, onChange, settings }) => (
+      <div className='blobfox local-settings__page content_warnings'>
+        <h1><FormattedMessage id='settings.content_warnings' defaultMessage='Content Warnings' /></h1>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['content_warnings', 'shared_state']}
+          id='mastodon-settings--content_warnings-shared_state'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.content_warnings_shared_state' defaultMessage='Show/hide content of all copies at once' />
+          <span className='hint'><FormattedMessage id='settings.content_warnings_shared_state_hint' defaultMessage='Reproduce upstream Mastodon behavior by having the Content Warning button affect all copies of a post at once. This will prevent automatic collapsing of any copy of a toot with unfolded CW' /></span>
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['content_warnings', 'media_outside']}
+          id='mastodon-settings--content_warnings-media_outside'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.content_warnings_media_outside' defaultMessage='Display media attachments outside content warnings' />
+          <span className='hint'><FormattedMessage id='settings.content_warnings_media_outside_hint' defaultMessage='Reproduce upstream Mastodon behavior by having the Content Warning toggle not affect media attachments' /></span>
+        </LocalSettingsPageItem>
+        <section>
+          <h2><FormattedMessage id='settings.content_warnings_unfold_opts' defaultMessage='Auto-unfolding options' /></h2>
+          <DeprecatedLocalSettingsPageItem
+            id='mastodon-settings--content_warnings-auto_unfold'
+            value={expandSpoilers}
+          >
+            <FormattedMessage id='settings.enable_content_warnings_auto_unfold' defaultMessage='Automatically unfold content-warnings' />
+            <span className='hint'>
+              <FormattedMessage
+                id='settings.deprecated_setting'
+                defaultMessage="This setting is now controlled from Mastodon's {settings_page_link}"
+                values={{
+                  settings_page_link: (
+                    <a href={preferenceLink('user_setting_expand_spoilers')}>
+                      <FormattedMessage
+                        id='settings.shared_settings_link'
+                        defaultMessage='user preferences'
+                      />
+                    </a>
+                  ),
+                }}
+              />
+            </span>
+          </DeprecatedLocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['content_warnings', 'filter']}
+            id='mastodon-settings--content_warnings-auto_unfold'
+            onChange={onChange}
+            placeholder={intl.formatMessage(messages.regexp)}
+            disabled={!expandSpoilers}
+          >
+            <FormattedMessage id='settings.content_warnings_filter' defaultMessage='Content warnings to not automatically unfold:' />
+          </LocalSettingsPageItem>
+        </section>
+      </div>
+    ),
+    ({ onChange, settings }) => (
+      <div className='blobfox local-settings__page collapsed'>
+        <h1><FormattedMessage id='settings.collapsed_statuses' defaultMessage='Collapsed toots' /></h1>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['collapsed', 'enabled']}
+          id='mastodon-settings--collapsed-enabled'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.enable_collapsed' defaultMessage='Enable collapsed toots' />
+          <span className='hint'><FormattedMessage id='settings.enable_collapsed_hint' defaultMessage='Collapsed posts have parts of their contents hidden to take up less screen space. This is distinct from the Content Warning feature' /></span>
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['collapsed', 'show_action_bar']}
+          id='mastodon-settings--collapsed-show-action-bar'
+          onChange={onChange}
+          dependsOn={[['collapsed', 'enabled']]}
+        >
+          <FormattedMessage id='settings.show_action_bar' defaultMessage='Show action buttons in collapsed toots' />
+        </LocalSettingsPageItem>
+        <section>
+          <h2><FormattedMessage id='settings.auto_collapse' defaultMessage='Automatic collapsing' /></h2>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'all']}
+            id='mastodon-settings--collapsed-auto-all'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+          >
+            <FormattedMessage id='settings.auto_collapse_all' defaultMessage='Everything' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'notifications']}
+            id='mastodon-settings--collapsed-auto-notifications'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+            dependsOnNot={[['collapsed', 'auto', 'all']]}
+          >
+            <FormattedMessage id='settings.auto_collapse_notifications' defaultMessage='Notifications' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'lengthy']}
+            id='mastodon-settings--collapsed-auto-lengthy'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+            dependsOnNot={[['collapsed', 'auto', 'all']]}
+          >
+            <FormattedMessage id='settings.auto_collapse_lengthy' defaultMessage='Lengthy toots' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'reblogs']}
+            id='mastodon-settings--collapsed-auto-reblogs'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+            dependsOnNot={[['collapsed', 'auto', 'all']]}
+          >
+            <FormattedMessage id='settings.auto_collapse_reblogs' defaultMessage='Boosts' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'replies']}
+            id='mastodon-settings--collapsed-auto-replies'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+            dependsOnNot={[['collapsed', 'auto', 'all']]}
+          >
+            <FormattedMessage id='settings.auto_collapse_replies' defaultMessage='Replies' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'media']}
+            id='mastodon-settings--collapsed-auto-media'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+            dependsOnNot={[['collapsed', 'auto', 'all']]}
+          >
+            <FormattedMessage id='settings.auto_collapse_media' defaultMessage='Toots with media' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'auto', 'height']}
+            id='mastodon-settings--collapsed-auto-height'
+            placeholder='400'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+            dependsOnNot={[['collapsed', 'auto', 'all']]}
+            inputProps={{ type: 'number', min: '200', max: '999' }}
+          >
+            <FormattedMessage id='settings.auto_collapse_height' defaultMessage='Height (in pixels) for a toot to be considered lengthy' />
+          </LocalSettingsPageItem>
+        </section>
+        <section>
+          <h2><FormattedMessage id='settings.image_backgrounds' defaultMessage='Image backgrounds' /></h2>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'backgrounds', 'user_backgrounds']}
+            id='mastodon-settings--collapsed-user-backgrouns'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+          >
+            <FormattedMessage id='settings.image_backgrounds_users' defaultMessage='Give collapsed toots an image background' />
+          </LocalSettingsPageItem>
+          <LocalSettingsPageItem
+            settings={settings}
+            item={['collapsed', 'backgrounds', 'preview_images']}
+            id='mastodon-settings--collapsed-preview-images'
+            onChange={onChange}
+            dependsOn={[['collapsed', 'enabled']]}
+          >
+            <FormattedMessage id='settings.image_backgrounds_media' defaultMessage='Preview collapsed toot media' />
+            <span className='hint'><FormattedMessage id='settings.image_backgrounds_media_hint' defaultMessage='If the post has any media attachment, use the first one as a background' /></span>
+          </LocalSettingsPageItem>
+        </section>
+      </div>
+    ),
+    ({ intl, onChange, settings }) => (
+      <div className='blobfox local-settings__page media'>
+        <h1><FormattedMessage id='settings.media' defaultMessage='Media' /></h1>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['media', 'letterbox']}
+          id='mastodon-settings--media-letterbox'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.media_letterbox' defaultMessage='Letterbox media' />
+          <span className='hint'><FormattedMessage id='settings.media_letterbox_hint' defaultMessage='Scale down and letterbox media to fill the image containers instead of stretching and cropping them' /></span>
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['media', 'fullwidth']}
+          id='mastodon-settings--media-fullwidth'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.media_fullwidth' defaultMessage='Full-width media previews' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['inline_preview_cards']}
+          id='mastodon-settings--inline-preview-cards'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.inline_preview_cards' defaultMessage='Inline preview cards for external links' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['media', 'reveal_behind_cw']}
+          id='mastodon-settings--reveal-behind-cw'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.media_reveal_behind_cw' defaultMessage='Reveal sensitive media behind a CW by default' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['media', 'pop_in_player']}
+          id='mastodon-settings--pop-in-player'
+          onChange={onChange}
+        >
+          <FormattedMessage id='settings.pop_in_player' defaultMessage='Enable pop-in player' />
+        </LocalSettingsPageItem>
+        <LocalSettingsPageItem
+          settings={settings}
+          item={['media', 'pop_in_position']}
+          id='mastodon-settings--pop-in-position'
+          options={[
+            { value: 'left', message: intl.formatMessage(messages.pop_in_left) },
+            { value: 'right', message: intl.formatMessage(messages.pop_in_right) },
+          ]}
+          onChange={onChange}
+          dependsOn={[['media', 'pop_in_player']]}
+        >
+          <FormattedMessage id='settings.pop_in_position' defaultMessage='Pop-in player position:' />
+        </LocalSettingsPageItem>
+      </div>
+    ),
+  ];
+
+  render () {
+    const { pages } = this;
+    const { index, intl, onChange, settings } = this.props;
+    const CurrentPage = pages[index] || pages[0];
+
+    return <CurrentPage intl={intl} onChange={onChange} settings={settings} />;
+  }
+
+}
+
+export default injectIntl(LocalSettingsPage);
diff --git a/app/javascript/flavours/blobfox/features/local_settings/page/item/index.jsx b/app/javascript/flavours/blobfox/features/local_settings/page/item/index.jsx
new file mode 100644
index 00000000000000..82f1c19088f8dc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/local_settings/page/item/index.jsx
@@ -0,0 +1,118 @@
+//  Package imports
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+//  * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
+
+export default class LocalSettingsPageItem extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node.isRequired,
+    dependsOn: PropTypes.array,
+    dependsOnNot: PropTypes.array,
+    id: PropTypes.string.isRequired,
+    item: PropTypes.array.isRequired,
+    onChange: PropTypes.func.isRequired,
+    inputProps: PropTypes.object,
+    options: PropTypes.arrayOf(PropTypes.shape({
+      value: PropTypes.string.isRequired,
+      message: PropTypes.string.isRequired,
+      hint: PropTypes.string,
+    })),
+    settings: ImmutablePropTypes.map.isRequired,
+    placeholder: PropTypes.string,
+    disabled: PropTypes.bool,
+  };
+
+  handleChange = e => {
+    const { target } = e;
+    const { item, onChange, options, placeholder } = this.props;
+    if (options && options.length > 0) onChange(item, target.value);
+    else if (placeholder) onChange(item, target.value);
+    else onChange(item, target.checked);
+  };
+
+  render () {
+    const { handleChange } = this;
+    const { settings, item, id, inputProps, options, children, dependsOn, dependsOnNot, placeholder, disabled } = this.props;
+    let enabled = !disabled;
+
+    if (dependsOn) {
+      for (let i = 0; i < dependsOn.length; i++) {
+        enabled = enabled && settings.getIn(dependsOn[i]);
+      }
+    }
+    if (dependsOnNot) {
+      for (let i = 0; i < dependsOnNot.length; i++) {
+        enabled = enabled && !settings.getIn(dependsOnNot[i]);
+      }
+    }
+
+    if (options && options.length > 0) {
+      const currentValue = settings.getIn(item);
+      const optionElems = options && options.length > 0 && options.map((opt) => {
+        let optionId = `${id}--${opt.value}`;
+        return (
+          <label key={optionId} htmlFor={optionId}>
+            <input
+              type='radio'
+              name={id}
+              id={optionId}
+              value={opt.value}
+              onBlur={handleChange}
+              onChange={handleChange}
+              checked={currentValue === opt.value}
+              disabled={!enabled}
+              {...inputProps}
+            />
+            {opt.message}
+            {opt.hint && <span className='hint'>{opt.hint}</span>}
+          </label>
+        );
+      });
+      return (
+        <div className='blobfox local-settings__page__item radio_buttons'>
+          <fieldset>
+            <legend>{children}</legend>
+            {optionElems}
+          </fieldset>
+        </div>
+      );
+    } else if (placeholder) {
+      return (
+        <div className='blobfox local-settings__page__item string'>
+          <label htmlFor={id}>
+            <p>{children}</p>
+            <p>
+              <input
+                id={id}
+                type='text'
+                value={settings.getIn(item)}
+                placeholder={placeholder}
+                onChange={handleChange}
+                disabled={!enabled}
+                {...inputProps}
+              />
+            </p>
+          </label>
+        </div>
+      );
+    } else return (
+      <div className='blobfox local-settings__page__item boolean'>
+        <label htmlFor={id}>
+          <input
+            id={id}
+            type='checkbox'
+            checked={settings.getIn(item)}
+            onChange={handleChange}
+            disabled={!enabled}
+            {...inputProps}
+          />
+          {children}
+        </label>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/mutes/index.jsx b/app/javascript/flavours/blobfox/features/mutes/index.jsx
new file mode 100644
index 00000000000000..947fe4c9b2712d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/mutes/index.jsx
@@ -0,0 +1,88 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchMutes, expandMutes } from '../../actions/mutes';
+import ColumnBackButtonSlim from '../../components/column_back_button_slim';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import ScrollableList from '../../components/scrollable_list';
+import AccountContainer from '../../containers/account_container';
+import Column from '../ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'column.mutes', defaultMessage: 'Muted users' },
+});
+
+const mapStateToProps = state => ({
+  accountIds: state.getIn(['user_lists', 'mutes', 'items']),
+  hasMore: !!state.getIn(['user_lists', 'mutes', 'next']),
+  isLoading: state.getIn(['user_lists', 'mutes', 'isLoading'], true),
+});
+
+class Mutes extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    accountIds: ImmutablePropTypes.list,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchMutes());
+  }
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandMutes());
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, hasMore, accountIds, multiColumn, isLoading } = this.props;
+
+    if (!accountIds) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    const emptyMessage = <FormattedMessage id='empty_column.mutes' defaultMessage="You haven't muted any users yet." />;
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='volume-off' heading={intl.formatMessage(messages.heading)}>
+        <ColumnBackButtonSlim />
+        <ScrollableList
+          scrollKey='mutes'
+          onLoadMore={this.handleLoadMore}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {accountIds.map(id =>
+            <AccountContainer key={id} id={id} defaultAction='mute' />,
+          )}
+        </ScrollableList>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Mutes));
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/admin_report.jsx b/app/javascript/flavours/blobfox/features/notifications/components/admin_report.jsx
new file mode 100644
index 00000000000000..b19b8d7453f01d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/admin_report.jsx
@@ -0,0 +1,115 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { HotKeys } from 'react-hotkeys';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import Permalink from 'flavours/blobfox/components/permalink';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import NotificationOverlayContainer from '../containers/overlay_container';
+
+import Report from './report';
+
+class AdminReport extends ImmutablePureComponent {
+
+  static propTypes = {
+    hidden: PropTypes.bool,
+    id: PropTypes.string.isRequired,
+    account: ImmutablePropTypes.map.isRequired,
+    notification: ImmutablePropTypes.map.isRequired,
+    unread: PropTypes.bool,
+    report: ImmutablePropTypes.map.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  handleMoveUp = () => {
+    const { notification, onMoveUp } = this.props;
+    onMoveUp(notification.get('id'));
+  };
+
+  handleMoveDown = () => {
+    const { notification, onMoveDown } = this.props;
+    onMoveDown(notification.get('id'));
+  };
+
+  handleOpen = () => {
+    this.handleOpenProfile();
+  };
+
+  handleOpenProfile = () => {
+    const { history, notification } = this.props;
+    history.push(`/@${notification.getIn(['account', 'acct'])}`);
+  };
+
+  handleMention = e => {
+    e.preventDefault();
+
+    const { history, notification, onMention } = this.props;
+    onMention(notification.get('account'), history);
+  };
+
+  getHandlers () {
+    return {
+      moveUp: this.handleMoveUp,
+      moveDown: this.handleMoveDown,
+      open: this.handleOpen,
+      openProfile: this.handleOpenProfile,
+      mention: this.handleMention,
+      reply: this.handleMention,
+    };
+  }
+
+  render () {
+    const { account, notification, unread, report } = this.props;
+
+    if (!report) {
+      return null;
+    }
+
+    //  Links to the display name.
+    const displayName = account.get('display_name_html') || account.get('username');
+    const link = (
+      <bdi><Permalink
+        className='notification__display-name'
+        href={account.get('url')}
+        title={account.get('acct')}
+        to={`/@${account.get('acct')}`}
+        dangerouslySetInnerHTML={{ __html: displayName }}
+      /></bdi>
+    );
+
+    const targetAccount = report.get('target_account');
+    const targetDisplayNameHtml = { __html: targetAccount.get('display_name_html') };
+    const targetLink = <bdi><Permalink className='notification__display-name' href={targetAccount.get('url')} title={targetAccount.get('acct')} to={`/@${targetAccount.get('acct')}`} dangerouslySetInnerHTML={targetDisplayNameHtml} /></bdi>;
+
+    return (
+      <HotKeys handlers={this.getHandlers()}>
+        <div className={classNames('notification notification-admin-report focusable', { unread })} tabIndex={0}>
+          <div className='notification__message'>
+            <div className='notification__favourite-icon-wrapper'>
+              <Icon id='flag' fixedWidth />
+            </div>
+
+            <span title={notification.get('created_at')}>
+              <FormattedMessage id='notification.admin.report' defaultMessage='{name} reported {target}' values={{ name: link, target: targetLink }} />
+            </span>
+          </div>
+
+          <Report account={account} report={notification.get('report')} hidden={this.props.hidden} />
+          <NotificationOverlayContainer notification={notification} />
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default withRouter(AdminReport);
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/admin_signup.jsx b/app/javascript/flavours/blobfox/features/notifications/components/admin_signup.jsx
new file mode 100644
index 00000000000000..39fe15a8aa2eb4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/admin_signup.jsx
@@ -0,0 +1,108 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { HotKeys } from 'react-hotkeys';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import Permalink from 'flavours/blobfox/components/permalink';
+import AccountContainer from 'flavours/blobfox/containers/account_container';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import NotificationOverlayContainer from '../containers/overlay_container';
+
+class NotificationAdminSignup extends ImmutablePureComponent {
+
+  static propTypes = {
+    hidden: PropTypes.bool,
+    id: PropTypes.string.isRequired,
+    account: ImmutablePropTypes.map.isRequired,
+    notification: ImmutablePropTypes.map.isRequired,
+    unread: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  handleMoveUp = () => {
+    const { notification, onMoveUp } = this.props;
+    onMoveUp(notification.get('id'));
+  };
+
+  handleMoveDown = () => {
+    const { notification, onMoveDown } = this.props;
+    onMoveDown(notification.get('id'));
+  };
+
+  handleOpen = () => {
+    this.handleOpenProfile();
+  };
+
+  handleOpenProfile = () => {
+    const { history, notification } = this.props;
+    history.push(`/@${notification.getIn(['account', 'acct'])}`);
+  };
+
+  handleMention = e => {
+    e.preventDefault();
+
+    const { history, notification, onMention } = this.props;
+    onMention(notification.get('account'), history);
+  };
+
+  getHandlers () {
+    return {
+      moveUp: this.handleMoveUp,
+      moveDown: this.handleMoveDown,
+      open: this.handleOpen,
+      openProfile: this.handleOpenProfile,
+      mention: this.handleMention,
+      reply: this.handleMention,
+    };
+  }
+
+  render () {
+    const { account, notification, hidden, unread } = this.props;
+
+    //  Links to the display name.
+    const displayName = account.get('display_name_html') || account.get('username');
+    const link = (
+      <bdi><Permalink
+        className='notification__display-name'
+        href={account.get('url')}
+        title={account.get('acct')}
+        to={`/@${account.get('acct')}`}
+        dangerouslySetInnerHTML={{ __html: displayName }}
+      /></bdi>
+    );
+
+    //  Renders.
+    return (
+      <HotKeys handlers={this.getHandlers()}>
+        <div className={classNames('notification notification-admin-sign-up focusable', { unread })} tabIndex={0}>
+          <div className='notification__message'>
+            <div className='notification__favourite-icon-wrapper'>
+              <Icon fixedWidth id='user-plus' />
+            </div>
+
+            <FormattedMessage
+              id='notification.admin.sign_up'
+              defaultMessage='{name} signed up'
+              values={{ name: link }}
+            />
+          </div>
+
+          <AccountContainer hidden={hidden} id={account.get('id')} withNote={false} />
+          <NotificationOverlayContainer notification={notification} />
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default withRouter(NotificationAdminSignup);
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/clear_column_button.jsx b/app/javascript/flavours/blobfox/features/notifications/components/clear_column_button.jsx
new file mode 100644
index 00000000000000..63121999227fe3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/clear_column_button.jsx
@@ -0,0 +1,20 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+export default class ClearColumnButton extends PureComponent {
+
+  static propTypes = {
+    onClick: PropTypes.func.isRequired,
+  };
+
+  render () {
+    return (
+      <button className='text-btn column-header__setting-btn' tabIndex={0} onClick={this.props.onClick}><Icon id='eraser' /> <FormattedMessage id='notifications.clear' defaultMessage='Clear notifications' /></button>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/column_settings.jsx b/app/javascript/flavours/blobfox/features/notifications/components/column_settings.jsx
new file mode 100644
index 00000000000000..9b6ed0ff123542
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/column_settings.jsx
@@ -0,0 +1,218 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { PERMISSION_MANAGE_USERS, PERMISSION_MANAGE_REPORTS } from 'flavours/blobfox/permissions';
+
+import ClearColumnButton from './clear_column_button';
+import GrantPermissionButton from './grant_permission_button';
+import PillBarButton from './pill_bar_button';
+import SettingToggle from './setting_toggle';
+
+export default class ColumnSettings extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    settings: ImmutablePropTypes.map.isRequired,
+    pushSettings: ImmutablePropTypes.map.isRequired,
+    onChange: PropTypes.func.isRequired,
+    onClear: PropTypes.func.isRequired,
+    onRequestNotificationPermission: PropTypes.func,
+    alertsEnabled: PropTypes.bool,
+    browserSupport: PropTypes.bool,
+    browserPermission: PropTypes.string,
+  };
+
+  onPushChange = (path, checked) => {
+    this.props.onChange(['push', ...path], checked);
+  };
+
+  render () {
+    const { settings, pushSettings, onChange, onClear, alertsEnabled, browserSupport, browserPermission, onRequestNotificationPermission } = this.props;
+
+    const unreadMarkersShowStr = <FormattedMessage id='notifications.column_settings.unread_notifications.highlight' defaultMessage='Highlight unread notifications' />;
+    const filterBarShowStr = <FormattedMessage id='notifications.column_settings.filter_bar.show_bar' defaultMessage='Show filter bar' />;
+    const filterAdvancedStr = <FormattedMessage id='notifications.column_settings.filter_bar.advanced' defaultMessage='Display all categories' />;
+    const alertStr = <FormattedMessage id='notifications.column_settings.alert' defaultMessage='Desktop notifications' />;
+    const showStr = <FormattedMessage id='notifications.column_settings.show' defaultMessage='Show in column' />;
+    const soundStr = <FormattedMessage id='notifications.column_settings.sound' defaultMessage='Play sound' />;
+
+    const showPushSettings = pushSettings.get('browserSupport') && pushSettings.get('isSubscribed');
+    const pushStr = showPushSettings && <FormattedMessage id='notifications.column_settings.push' defaultMessage='Push notifications' />;
+
+    return (
+      <div>
+        {alertsEnabled && browserSupport && browserPermission === 'denied' && (
+          <div className='column-settings__row column-settings__row--with-margin'>
+            <span className='warning-hint'><FormattedMessage id='notifications.permission_denied' defaultMessage='Desktop notifications are unavailable due to previously denied browser permissions request' /></span>
+          </div>
+        )}
+
+        {alertsEnabled && browserSupport && browserPermission === 'default' && (
+          <div className='column-settings__row column-settings__row--with-margin'>
+            <span className='warning-hint'>
+              <FormattedMessage id='notifications.permission_required' defaultMessage='Desktop notifications are unavailable because the required permission has not been granted.' /> <GrantPermissionButton onClick={onRequestNotificationPermission} />
+            </span>
+          </div>
+        )}
+
+        <div className='column-settings__row'>
+          <ClearColumnButton onClick={onClear} />
+        </div>
+
+        <div role='group' aria-labelledby='notifications-unread-markers'>
+          <span id='notifications-unread-markers' className='column-settings__section'>
+            <FormattedMessage id='notifications.column_settings.unread_notifications.category' defaultMessage='Unread notifications' />
+          </span>
+
+          <div className='column-settings__row'>
+            <SettingToggle id='unread-notification-markers' prefix='notifications' settings={settings} settingPath={['showUnread']} onChange={onChange} label={unreadMarkersShowStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-filter-bar'>
+          <span id='notifications-filter-bar' className='column-settings__section'>
+            <FormattedMessage id='notifications.column_settings.filter_bar.category' defaultMessage='Quick filter bar' />
+          </span>
+
+          <div className='column-settings__row'>
+            <SettingToggle id='show-filter-bar' prefix='notifications' settings={settings} settingPath={['quickFilter', 'show']} onChange={onChange} label={filterBarShowStr} />
+            <SettingToggle id='show-filter-bar' prefix='notifications' settings={settings} settingPath={['quickFilter', 'advanced']} onChange={onChange} label={filterAdvancedStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-follow'>
+          <span id='notifications-follow' className='column-settings__section'><FormattedMessage id='notifications.column_settings.follow' defaultMessage='New followers:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'follow']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'follow']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'follow']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'follow']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-follow-request'>
+          <span id='notifications-follow-request' className='column-settings__section'><FormattedMessage id='notifications.column_settings.follow_request' defaultMessage='New follow requests:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'follow_request']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'follow_request']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'follow_request']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'follow_request']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-favourite'>
+          <span id='notifications-favourite' className='column-settings__section'><FormattedMessage id='notifications.column_settings.favourite' defaultMessage='Favorites:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'favourite']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'favourite']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'favourite']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'favourite']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-reaction'>
+          <span id='notifications-reaction' className='column-settings__section'><FormattedMessage id='notifications.column_settings.reaction' defaultMessage='Reactions:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'reaction']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'reaction']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'reaction']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'reaction']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-mention'>
+          <span id='notifications-mention' className='column-settings__section'><FormattedMessage id='notifications.column_settings.mention' defaultMessage='Mentions:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'mention']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'mention']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'mention']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'mention']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-reblog'>
+          <span id='notifications-reblog' className='column-settings__section'><FormattedMessage id='notifications.column_settings.reblog' defaultMessage='Boosts:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'reblog']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'reblog']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'reblog']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'reblog']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-poll'>
+          <span id='notifications-poll' className='column-settings__section'><FormattedMessage id='notifications.column_settings.poll' defaultMessage='Poll results:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'poll']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'poll']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'poll']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'poll']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-status'>
+          <span id='notifications-status' className='column-settings__section'><FormattedMessage id='notifications.column_settings.status' defaultMessage='New posts:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'status']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'status']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'status']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'status']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        <div role='group' aria-labelledby='notifications-update'>
+          <span id='notifications-update' className='column-settings__section'><FormattedMessage id='notifications.column_settings.update' defaultMessage='Edits:' /></span>
+
+          <div className='column-settings__pillbar'>
+            <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'update']} onChange={onChange} label={alertStr} />
+            {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'update']} onChange={this.onPushChange} label={pushStr} />}
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'update']} onChange={onChange} label={showStr} />
+            <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'update']} onChange={onChange} label={soundStr} />
+          </div>
+        </div>
+
+        {((this.context.identity.permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS) && (
+          <div role='group' aria-labelledby='notifications-admin-sign-up'>
+            <span id='notifications-status' className='column-settings__section'><FormattedMessage id='notifications.column_settings.admin.sign_up' defaultMessage='New sign-ups:' /></span>
+
+            <div className='column-settings__pillbar'>
+              <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'admin.sign_up']} onChange={onChange} label={alertStr} />
+              {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'admin.sign_up']} onChange={this.onPushChange} label={pushStr} />}
+              <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'admin.sign_up']} onChange={onChange} label={showStr} />
+              <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'admin.sign_up']} onChange={onChange} label={soundStr} />
+            </div>
+          </div>
+        )}
+
+        {((this.context.identity.permissions & PERMISSION_MANAGE_REPORTS) === PERMISSION_MANAGE_REPORTS) && (
+          <div role='group' aria-labelledby='notifications-admin-report'>
+            <span id='notifications-status' className='column-settings__section'><FormattedMessage id='notifications.column_settings.admin.report' defaultMessage='New reports:' /></span>
+
+            <div className='column-settings__pillbar'>
+              <PillBarButton disabled={browserPermission === 'denied'} prefix='notifications_desktop' settings={settings} settingPath={['alerts', 'admin.report']} onChange={onChange} label={alertStr} />
+              {showPushSettings && <PillBarButton prefix='notifications_push' settings={pushSettings} settingPath={['alerts', 'admin.report']} onChange={this.onPushChange} label={pushStr} />}
+              <PillBarButton prefix='notifications' settings={settings} settingPath={['shows', 'admin.report']} onChange={onChange} label={showStr} />
+              <PillBarButton prefix='notifications' settings={settings} settingPath={['sounds', 'admin.report']} onChange={onChange} label={soundStr} />
+            </div>
+          </div>
+        )}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/filter_bar.jsx b/app/javascript/flavours/blobfox/features/notifications/components/filter_bar.jsx
new file mode 100644
index 00000000000000..0bfbf8afa3c9d0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/filter_bar.jsx
@@ -0,0 +1,121 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+const tooltips = defineMessages({
+  mentions: { id: 'notifications.filter.mentions', defaultMessage: 'Mentions' },
+  favourites: { id: 'notifications.filter.favourites', defaultMessage: 'Favorites' },
+  reactions: { id: 'notifications.filter.reactions', defaultMessage: 'Reactions' },
+  boosts: { id: 'notifications.filter.boosts', defaultMessage: 'Boosts' },
+  polls: { id: 'notifications.filter.polls', defaultMessage: 'Poll results' },
+  follows: { id: 'notifications.filter.follows', defaultMessage: 'Follows' },
+  statuses: { id: 'notifications.filter.statuses', defaultMessage: 'Updates from people you follow' },
+});
+
+class FilterBar extends PureComponent {
+
+  static propTypes = {
+    selectFilter: PropTypes.func.isRequired,
+    selectedFilter: PropTypes.string.isRequired,
+    advancedMode: PropTypes.bool.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  onClick (notificationType) {
+    return () => this.props.selectFilter(notificationType);
+  }
+
+  render () {
+    const { selectedFilter, advancedMode, intl } = this.props;
+    const renderedElement = !advancedMode ? (
+      <div className='notification__filter-bar'>
+        <button
+          className={selectedFilter === 'all' ? 'active' : ''}
+          onClick={this.onClick('all')}
+        >
+          <FormattedMessage
+            id='notifications.filter.all'
+            defaultMessage='All'
+          />
+        </button>
+        <button
+          className={selectedFilter === 'mention' ? 'active' : ''}
+          onClick={this.onClick('mention')}
+        >
+          <FormattedMessage
+            id='notifications.filter.mentions'
+            defaultMessage='Mentions'
+          />
+        </button>
+      </div>
+    ) : (
+      <div className='notification__filter-bar'>
+        <button
+          className={selectedFilter === 'all' ? 'active' : ''}
+          onClick={this.onClick('all')}
+        >
+          <FormattedMessage
+            id='notifications.filter.all'
+            defaultMessage='All'
+          />
+        </button>
+        <button
+          className={selectedFilter === 'mention' ? 'active' : ''}
+          onClick={this.onClick('mention')}
+          title={intl.formatMessage(tooltips.mentions)}
+        >
+          <Icon id='reply-all' fixedWidth />
+        </button>
+        <button
+          className={selectedFilter === 'favourite' ? 'active' : ''}
+          onClick={this.onClick('favourite')}
+          title={intl.formatMessage(tooltips.favourites)}
+        >
+          <Icon id='star' fixedWidth />
+        </button>
+        <button
+          className={selectedFilter === 'reaction' ? 'active' : ''}
+          onClick={this.onClick('reaction')}
+          title={intl.formatMessage(tooltips.reactions)}
+        >
+          <Icon id='plus' fixedWidth />
+        </button>
+        <button
+          className={selectedFilter === 'reblog' ? 'active' : ''}
+          onClick={this.onClick('reblog')}
+          title={intl.formatMessage(tooltips.boosts)}
+        >
+          <Icon id='retweet' fixedWidth />
+        </button>
+        <button
+          className={selectedFilter === 'poll' ? 'active' : ''}
+          onClick={this.onClick('poll')}
+          title={intl.formatMessage(tooltips.polls)}
+        >
+          <Icon id='tasks' fixedWidth />
+        </button>
+        <button
+          className={selectedFilter === 'status' ? 'active' : ''}
+          onClick={this.onClick('status')}
+          title={intl.formatMessage(tooltips.statuses)}
+        >
+          <Icon id='home' fixedWidth />
+        </button>
+        <button
+          className={selectedFilter === 'follow' ? 'active' : ''}
+          onClick={this.onClick('follow')}
+          title={intl.formatMessage(tooltips.follows)}
+        >
+          <Icon id='user-plus' fixedWidth />
+        </button>
+      </div>
+    );
+    return renderedElement;
+  }
+
+}
+
+export default injectIntl(FilterBar);
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/follow.jsx b/app/javascript/flavours/blobfox/features/notifications/components/follow.jsx
new file mode 100644
index 00000000000000..2d04ab9d16ce10
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/follow.jsx
@@ -0,0 +1,108 @@
+import PropTypes from 'prop-types';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { HotKeys } from 'react-hotkeys';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+import Permalink from 'flavours/blobfox/components/permalink';
+import AccountContainer from 'flavours/blobfox/containers/account_container';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import NotificationOverlayContainer from '../containers/overlay_container';
+
+class NotificationFollow extends ImmutablePureComponent {
+
+  static propTypes = {
+    hidden: PropTypes.bool,
+    id: PropTypes.string.isRequired,
+    account: ImmutablePropTypes.map.isRequired,
+    notification: ImmutablePropTypes.map.isRequired,
+    unread: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  handleMoveUp = () => {
+    const { notification, onMoveUp } = this.props;
+    onMoveUp(notification.get('id'));
+  };
+
+  handleMoveDown = () => {
+    const { notification, onMoveDown } = this.props;
+    onMoveDown(notification.get('id'));
+  };
+
+  handleOpen = () => {
+    this.handleOpenProfile();
+  };
+
+  handleOpenProfile = () => {
+    const { history, notification } = this.props;
+    history.push(`/@${notification.getIn(['account', 'acct'])}`);
+  };
+
+  handleMention = e => {
+    e.preventDefault();
+
+    const { history, notification, onMention } = this.props;
+    onMention(notification.get('account'), history);
+  };
+
+  getHandlers () {
+    return {
+      moveUp: this.handleMoveUp,
+      moveDown: this.handleMoveDown,
+      open: this.handleOpen,
+      openProfile: this.handleOpenProfile,
+      mention: this.handleMention,
+      reply: this.handleMention,
+    };
+  }
+
+  render () {
+    const { account, notification, hidden, unread } = this.props;
+
+    //  Links to the display name.
+    const displayName = account.get('display_name_html') || account.get('username');
+    const link = (
+      <bdi><Permalink
+        className='notification__display-name'
+        href={account.get('url')}
+        title={account.get('acct')}
+        to={`/@${account.get('acct')}`}
+        dangerouslySetInnerHTML={{ __html: displayName }}
+      /></bdi>
+    );
+
+    //  Renders.
+    return (
+      <HotKeys handlers={this.getHandlers()}>
+        <div className={classNames('notification notification-follow focusable', { unread })} tabIndex={0}>
+          <div className='notification__message'>
+            <div className='notification__favourite-icon-wrapper'>
+              <Icon fixedWidth id='user-plus' />
+            </div>
+
+            <FormattedMessage
+              id='notification.follow'
+              defaultMessage='{name} followed you'
+              values={{ name: link }}
+            />
+          </div>
+
+          <AccountContainer hidden={hidden} id={account.get('id')} withNote={false} />
+          <NotificationOverlayContainer notification={notification} />
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default withRouter(NotificationFollow);
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/follow_request.jsx b/app/javascript/flavours/blobfox/features/notifications/components/follow_request.jsx
new file mode 100644
index 00000000000000..749169358b4110
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/follow_request.jsx
@@ -0,0 +1,141 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { HotKeys } from 'react-hotkeys';
+
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import { Icon } from 'flavours/blobfox/components/icon';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Permalink from 'flavours/blobfox/components/permalink';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import NotificationOverlayContainer from '../containers/overlay_container';
+
+const messages = defineMessages({
+  authorize: { id: 'follow_request.authorize', defaultMessage: 'Authorize' },
+  reject: { id: 'follow_request.reject', defaultMessage: 'Reject' },
+});
+
+class FollowRequest extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    onAuthorize: PropTypes.func.isRequired,
+    onReject: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    notification: ImmutablePropTypes.map.isRequired,
+    unread: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  handleMoveUp = () => {
+    const { notification, onMoveUp } = this.props;
+    onMoveUp(notification.get('id'));
+  };
+
+  handleMoveDown = () => {
+    const { notification, onMoveDown } = this.props;
+    onMoveDown(notification.get('id'));
+  };
+
+  handleOpen = () => {
+    this.handleOpenProfile();
+  };
+
+  handleOpenProfile = () => {
+    const { history, notification } = this.props;
+    history.push(`/@${notification.getIn(['account', 'acct'])}`);
+  };
+
+  handleMention = e => {
+    e.preventDefault();
+
+    const { history, notification, onMention } = this.props;
+    onMention(notification.get('account'), history);
+  };
+
+  getHandlers () {
+    return {
+      moveUp: this.handleMoveUp,
+      moveDown: this.handleMoveDown,
+      open: this.handleOpen,
+      openProfile: this.handleOpenProfile,
+      mention: this.handleMention,
+      reply: this.handleMention,
+    };
+  }
+
+  render () {
+    const { intl, hidden, account, onAuthorize, onReject, notification, unread } = this.props;
+
+    if (!account) {
+      return <div />;
+    }
+
+    if (hidden) {
+      return (
+        <>
+          {account.get('display_name')}
+          {account.get('username')}
+        </>
+      );
+    }
+
+    //  Links to the display name.
+    const displayName = account.get('display_name_html') || account.get('username');
+    const link = (
+      <bdi><Permalink
+        className='notification__display-name'
+        href={account.get('url')}
+        title={account.get('acct')}
+        to={`/@${account.get('acct')}`}
+        dangerouslySetInnerHTML={{ __html: displayName }}
+      /></bdi>
+    );
+
+    return (
+      <HotKeys handlers={this.getHandlers()}>
+        <div className={classNames('notification notification-follow-request focusable', { unread })} tabIndex={0}>
+          <div className='notification__message'>
+            <div className='notification__favourite-icon-wrapper'>
+              <Icon id='user' fixedWidth />
+            </div>
+
+            <FormattedMessage
+              id='notification.follow_request'
+              defaultMessage='{name} has requested to follow you'
+              values={{ name: link }}
+            />
+          </div>
+
+          <div className='account'>
+            <div className='account__wrapper'>
+              <Permalink key={account.get('id')} className='account__display-name' title={account.get('acct')} href={account.get('url')} to={`/@${account.get('acct')}`}>
+                <div className='account__avatar-wrapper'><Avatar account={account} size={36} /></div>
+                <DisplayName account={account} />
+              </Permalink>
+
+              <div className='account__relationship'>
+                <IconButton title={intl.formatMessage(messages.authorize)} icon='check' onClick={onAuthorize} />
+                <IconButton title={intl.formatMessage(messages.reject)} icon='times' onClick={onReject} />
+              </div>
+            </div>
+          </div>
+
+          <NotificationOverlayContainer notification={notification} />
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(FollowRequest));
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/grant_permission_button.jsx b/app/javascript/flavours/blobfox/features/notifications/components/grant_permission_button.jsx
new file mode 100644
index 00000000000000..cd46d878bbc2e0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/grant_permission_button.jsx
@@ -0,0 +1,20 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+export default class GrantPermissionButton extends PureComponent {
+
+  static propTypes = {
+    onClick: PropTypes.func.isRequired,
+  };
+
+  render () {
+    return (
+      <button className='text-btn column-header__permission-btn' tabIndex={0} onClick={this.props.onClick}>
+        <FormattedMessage id='notifications.grant_permission' defaultMessage='Grant permission.' />
+      </button>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/notification.jsx b/app/javascript/flavours/blobfox/features/notifications/components/notification.jsx
new file mode 100644
index 00000000000000..146562563be009
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/notification.jsx
@@ -0,0 +1,258 @@
+//  Package imports.
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+//  Our imports,
+import StatusContainer from 'flavours/blobfox/containers/status_container';
+
+import NotificationAdminReportContainer from '../containers/admin_report_container';
+import NotificationFollowRequestContainer from '../containers/follow_request_container';
+
+import NotificationAdminSignup from './admin_signup';
+import NotificationFollow from './follow';
+
+export default class Notification extends ImmutablePureComponent {
+
+  static propTypes = {
+    notification: ImmutablePropTypes.map.isRequired,
+    hidden: PropTypes.bool,
+    onMoveUp: PropTypes.func.isRequired,
+    onMoveDown: PropTypes.func.isRequired,
+    onMention: PropTypes.func.isRequired,
+    getScrollPosition: PropTypes.func,
+    updateScrollBottom: PropTypes.func,
+    cacheMediaWidth: PropTypes.func,
+    cachedMediaWidth: PropTypes.number,
+    onUnmount: PropTypes.func,
+    unread: PropTypes.bool,
+  };
+
+  render () {
+    const {
+      hidden,
+      notification,
+      onMoveDown,
+      onMoveUp,
+      onMention,
+      getScrollPosition,
+      updateScrollBottom,
+    } = this.props;
+
+    switch(notification.get('type')) {
+    case 'follow':
+      return (
+        <NotificationFollow
+          hidden={hidden}
+          id={notification.get('id')}
+          account={notification.get('account')}
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          unread={this.props.unread}
+        />
+      );
+    case 'follow_request':
+      return (
+        <NotificationFollowRequestContainer
+          hidden={hidden}
+          id={notification.get('id')}
+          account={notification.get('account')}
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          unread={this.props.unread}
+        />
+      );
+    case 'admin.sign_up':
+      return (
+        <NotificationAdminSignup
+          hidden={hidden}
+          id={notification.get('id')}
+          account={notification.get('account')}
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          unread={this.props.unread}
+        />
+      );
+    case 'admin.report':
+      return (
+        <NotificationAdminReportContainer
+          hidden={hidden}
+          id={notification.get('id')}
+          account={notification.get('account')}
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          unread={this.props.unread}
+        />
+      );
+    case 'mention':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          contextType='notifications'
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    case 'status':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          account={notification.get('account')}
+          prepend='status'
+          muted
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          contextType='notifications'
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    case 'favourite':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          account={notification.get('account')}
+          prepend='favourite'
+          muted
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          contextType='notifications'
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    case 'reaction':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          account={notification.get('account')}
+          prepend='reaction'
+          muted
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    case 'reblog':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          account={notification.get('account')}
+          prepend='reblog'
+          muted
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          contextType='notifications'
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    case 'poll':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          account={notification.get('account')}
+          prepend='poll'
+          muted
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          contextType='notifications'
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    case 'update':
+      return (
+        <StatusContainer
+          containerId={notification.get('id')}
+          hidden={hidden}
+          id={notification.get('status')}
+          account={notification.get('account')}
+          prepend='update'
+          muted
+          notification={notification}
+          onMoveDown={onMoveDown}
+          onMoveUp={onMoveUp}
+          onMention={onMention}
+          contextType='notifications'
+          getScrollPosition={getScrollPosition}
+          updateScrollBottom={updateScrollBottom}
+          cachedMediaWidth={this.props.cachedMediaWidth}
+          cacheMediaWidth={this.props.cacheMediaWidth}
+          onUnmount={this.props.onUnmount}
+          withDismiss
+          unread={this.props.unread}
+        />
+      );
+    default:
+      return null;
+    }
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/notifications_permission_banner.jsx b/app/javascript/flavours/blobfox/features/notifications/components/notifications_permission_banner.jsx
new file mode 100644
index 00000000000000..ea45de786f8d6c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/notifications_permission_banner.jsx
@@ -0,0 +1,51 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { requestBrowserPermission } from 'flavours/blobfox/actions/notifications';
+import { changeSetting } from 'flavours/blobfox/actions/settings';
+import { Button } from 'flavours/blobfox/components/button';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+class NotificationsPermissionBanner extends PureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleClick = () => {
+    this.props.dispatch(requestBrowserPermission());
+  };
+
+  handleClose = () => {
+    this.props.dispatch(changeSetting(['notifications', 'dismissPermissionBanner'], true));
+  };
+
+  render () {
+    const { intl } = this.props;
+
+    return (
+      <div className='notifications-permission-banner'>
+        <div className='notifications-permission-banner__close'>
+          <IconButton icon='times' onClick={this.handleClose} title={intl.formatMessage(messages.close)} />
+        </div>
+
+        <h2><FormattedMessage id='notifications_permission_banner.title' defaultMessage='Never miss a thing' /></h2>
+        <p><FormattedMessage id='notifications_permission_banner.how_to_control' defaultMessage="To receive notifications when Mastodon isn't open, enable desktop notifications. You can control precisely which types of interactions generate desktop notifications through the {icon} button above once they're enabled." values={{ icon: <Icon id='sliders' /> }} /></p>
+        <Button onClick={this.handleClick}><FormattedMessage id='notifications_permission_banner.enable' defaultMessage='Enable desktop notifications' /></Button>
+      </div>
+    );
+  }
+
+}
+
+export default connect()(injectIntl(NotificationsPermissionBanner));
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/overlay.jsx b/app/javascript/flavours/blobfox/features/notifications/components/overlay.jsx
new file mode 100644
index 00000000000000..066f0647f31f06
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/overlay.jsx
@@ -0,0 +1,61 @@
+/**
+ * Notification overlay
+ */
+
+
+//  Package imports.
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const messages = defineMessages({
+  markForDeletion: { id: 'notification.markForDeletion', defaultMessage: 'Mark for deletion' },
+});
+
+class NotificationOverlay extends ImmutablePureComponent {
+
+  static propTypes = {
+    notification    : ImmutablePropTypes.map.isRequired,
+    onMarkForDelete : PropTypes.func.isRequired,
+    show            : PropTypes.bool.isRequired,
+    intl            : PropTypes.object.isRequired,
+  };
+
+  onToggleMark = () => {
+    const mark = !this.props.notification.get('markedForDelete');
+    const id = this.props.notification.get('id');
+    this.props.onMarkForDelete(id, mark);
+  };
+
+  render () {
+    const { notification, show, intl } = this.props;
+
+    const active = notification.get('markedForDelete');
+    const label = intl.formatMessage(messages.markForDeletion);
+
+    return show ? (
+      <div
+        aria-label={label}
+        role='checkbox'
+        aria-checked={active}
+        tabIndex={0}
+        className={`notification__dismiss-overlay ${active ? 'active' : ''}`}
+        onClick={this.onToggleMark}
+      >
+        <div className='wrappy'>
+          <div className='ckbox' aria-hidden='true' title={label}>
+            {active ? (<Icon id='check' />) : ''}
+          </div>
+        </div>
+      </div>
+    ) : null;
+  }
+
+}
+
+export default injectIntl(NotificationOverlay);
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/pill_bar_button.jsx b/app/javascript/flavours/blobfox/features/notifications/components/pill_bar_button.jsx
new file mode 100644
index 00000000000000..633401d6e3102b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/pill_bar_button.jsx
@@ -0,0 +1,43 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+export default class PillBarButton extends PureComponent {
+
+  static propTypes = {
+    prefix: PropTypes.string,
+    settings: ImmutablePropTypes.map.isRequired,
+    settingPath: PropTypes.array.isRequired,
+    label: PropTypes.node.isRequired,
+    onChange: PropTypes.func.isRequired,
+    disabled: PropTypes.bool,
+  };
+
+  onChange = () => {
+    const { settings, settingPath } = this.props;
+    this.props.onChange(settingPath, !settings.getIn(settingPath));
+  };
+
+  render () {
+    const { prefix, settings, settingPath, label, disabled } = this.props;
+    const id = ['setting-pillbar-button', prefix, ...settingPath].filter(Boolean).join('-');
+    const active = settings.getIn(settingPath);
+
+    return (
+      <button
+        key={id}
+        id={id}
+        className={classNames('pillbar-button', { active })}
+        disabled={disabled}
+        onClick={this.onChange}
+        aria-pressed={active}
+      >
+        {label}
+      </button>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/report.jsx b/app/javascript/flavours/blobfox/features/notifications/components/report.jsx
new file mode 100644
index 00000000000000..63b4f18a98f028
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/report.jsx
@@ -0,0 +1,67 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { AvatarOverlay } from 'flavours/blobfox/components/avatar_overlay';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+
+// This needs to be kept in sync with app/models/report.rb
+const messages = defineMessages({
+  openReport: { id: 'report_notification.open', defaultMessage: 'Open report' },
+  other: { id: 'report_notification.categories.other', defaultMessage: 'Other' },
+  spam: { id: 'report_notification.categories.spam', defaultMessage: 'Spam' },
+  legal: { id: 'report_notification.categories.legal', defaultMessage: 'Legal' },
+  violation: { id: 'report_notification.categories.violation', defaultMessage: 'Rule violation' },
+});
+
+class Report extends ImmutablePureComponent {
+
+  static propTypes = {
+    account: ImmutablePropTypes.record.isRequired,
+    report: ImmutablePropTypes.map.isRequired,
+    hidden: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { intl, hidden, report, account } = this.props;
+
+    if (!report) {
+      return null;
+    }
+
+    if (hidden) {
+      return (
+        <>
+          {report.get('id')}
+        </>
+      );
+    }
+
+    return (
+      <div className='notification__report'>
+        <div className='notification__report__avatar'>
+          <AvatarOverlay account={report.get('target_account')} friend={account} />
+        </div>
+
+        <div className='notification__report__details'>
+          <div>
+            <RelativeTimestamp timestamp={report.get('created_at')} short={false} /> · <FormattedMessage id='report_notification.attached_statuses' defaultMessage='{count, plural, one {# post} other {# posts}} attached' values={{ count: report.get('status_ids').size }} />
+            <br />
+            <strong>{intl.formatMessage(messages[report.get('category')])}</strong>
+          </div>
+
+          <div className='notification__report__actions'>
+            <a href={`/admin/reports/${report.get('id')}`} className='button' target='_blank' rel='noopener noreferrer'>{intl.formatMessage(messages.openReport)}</a>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Report);
diff --git a/app/javascript/flavours/blobfox/features/notifications/components/setting_toggle.jsx b/app/javascript/flavours/blobfox/features/notifications/components/setting_toggle.jsx
new file mode 100644
index 00000000000000..2f849c54841031
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/components/setting_toggle.jsx
@@ -0,0 +1,38 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import Toggle from 'react-toggle';
+
+export default class SettingToggle extends PureComponent {
+
+  static propTypes = {
+    prefix: PropTypes.string,
+    settings: ImmutablePropTypes.map.isRequired,
+    settingPath: PropTypes.array.isRequired,
+    label: PropTypes.node.isRequired,
+    meta: PropTypes.node,
+    onChange: PropTypes.func.isRequired,
+    defaultValue: PropTypes.bool,
+    disabled: PropTypes.bool,
+  };
+
+  onChange = ({ target }) => {
+    this.props.onChange(this.props.settingPath, target.checked);
+  };
+
+  render () {
+    const { prefix, settings, settingPath, label, meta, defaultValue, disabled } = this.props;
+    const id = ['setting-toggle', prefix, ...settingPath].filter(Boolean).join('-');
+
+    return (
+      <div className='setting-toggle'>
+        <Toggle disabled={disabled} id={id} checked={settings.getIn(settingPath, defaultValue)} onChange={this.onChange} onKeyDown={this.onKeyDown} />
+        <label htmlFor={id} className='setting-toggle__label'>{label}</label>
+        {meta && <span className='setting-meta__label'>{meta}</span>}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/notifications/containers/admin_report_container.js b/app/javascript/flavours/blobfox/features/notifications/containers/admin_report_container.js
new file mode 100644
index 00000000000000..27a8450a750a24
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/containers/admin_report_container.js
@@ -0,0 +1,15 @@
+import { connect } from 'react-redux';
+
+import { makeGetReport } from 'flavours/blobfox/selectors';
+
+import AdminReport from '../components/admin_report';
+
+const mapStateToProps = (state, { notification }) => {
+  const getReport = makeGetReport();
+
+  return {
+    report: notification.get('report') ? getReport(state, notification.get('report'), notification.getIn(['report', 'target_account', 'id'])) : null,
+  };
+};
+
+export default connect(mapStateToProps)(AdminReport);
diff --git a/app/javascript/flavours/blobfox/features/notifications/containers/column_settings_container.js b/app/javascript/flavours/blobfox/features/notifications/containers/column_settings_container.js
new file mode 100644
index 00000000000000..1e62ed9a5a45b6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/containers/column_settings_container.js
@@ -0,0 +1,78 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { showAlert } from '../../../actions/alerts';
+import { openModal } from '../../../actions/modal';
+import { setFilter, clearNotifications, requestBrowserPermission } from '../../../actions/notifications';
+import { changeAlerts as changePushNotifications } from '../../../actions/push_notifications';
+import { changeSetting } from '../../../actions/settings';
+import ColumnSettings from '../components/column_settings';
+
+const messages = defineMessages({
+  clearMessage: { id: 'notifications.clear_confirmation', defaultMessage: 'Are you sure you want to permanently clear all your notifications?' },
+  clearConfirm: { id: 'notifications.clear', defaultMessage: 'Clear notifications' },
+  permissionDenied: { id: 'notifications.permission_denied_alert', defaultMessage: 'Desktop notifications can\'t be enabled, as browser permission has been denied before' },
+});
+
+const mapStateToProps = state => ({
+  settings: state.getIn(['settings', 'notifications']),
+  pushSettings: state.get('push_notifications'),
+  alertsEnabled: state.getIn(['settings', 'notifications', 'alerts']).includes(true),
+  browserSupport: state.getIn(['notifications', 'browserSupport']),
+  browserPermission: state.getIn(['notifications', 'browserPermission']),
+});
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+
+  onChange (path, checked) {
+    if (path[0] === 'push') {
+      if (checked && typeof window.Notification !== 'undefined' && Notification.permission !== 'granted') {
+        dispatch(requestBrowserPermission((permission) => {
+          if (permission === 'granted') {
+            dispatch(changePushNotifications(path.slice(1), checked));
+          } else {
+            dispatch(showAlert({ message: messages.permissionDenied }));
+          }
+        }));
+      } else {
+        dispatch(changePushNotifications(path.slice(1), checked));
+      }
+    } else if (path[0] === 'quickFilter') {
+      dispatch(changeSetting(['notifications', ...path], checked));
+      dispatch(setFilter('all'));
+    } else if (path[0] === 'alerts' && checked && typeof window.Notification !== 'undefined' && Notification.permission !== 'granted') {
+      if (checked && typeof window.Notification !== 'undefined' && Notification.permission !== 'granted') {
+        dispatch(requestBrowserPermission((permission) => {
+          if (permission === 'granted') {
+            dispatch(changeSetting(['notifications', ...path], checked));
+          } else {
+            dispatch(showAlert({ message: messages.permissionDenied }));
+          }
+        }));
+      } else {
+        dispatch(changeSetting(['notifications', ...path], checked));
+      }
+    } else {
+      dispatch(changeSetting(['notifications', ...path], checked));
+    }
+  },
+
+  onClear () {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.clearMessage),
+        confirm: intl.formatMessage(messages.clearConfirm),
+        onConfirm: () => dispatch(clearNotifications()),
+      },
+    }));
+  },
+
+  onRequestNotificationPermission () {
+    dispatch(requestBrowserPermission());
+  },
+
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(ColumnSettings));
diff --git a/app/javascript/flavours/blobfox/features/notifications/containers/filter_bar_container.js b/app/javascript/flavours/blobfox/features/notifications/containers/filter_bar_container.js
new file mode 100644
index 00000000000000..4e0184cef30534
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/containers/filter_bar_container.js
@@ -0,0 +1,17 @@
+import { connect } from 'react-redux';
+
+import { setFilter } from '../../../actions/notifications';
+import FilterBar from '../components/filter_bar';
+
+const makeMapStateToProps = state => ({
+  selectedFilter: state.getIn(['settings', 'notifications', 'quickFilter', 'active']),
+  advancedMode: state.getIn(['settings', 'notifications', 'quickFilter', 'advanced']),
+});
+
+const mapDispatchToProps = (dispatch) => ({
+  selectFilter (newActiveFilter) {
+    dispatch(setFilter(newActiveFilter));
+  },
+});
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(FilterBar);
diff --git a/app/javascript/flavours/blobfox/features/notifications/containers/follow_request_container.js b/app/javascript/flavours/blobfox/features/notifications/containers/follow_request_container.js
new file mode 100644
index 00000000000000..ebe41844dc2697
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/containers/follow_request_container.js
@@ -0,0 +1,17 @@
+import { connect } from 'react-redux';
+
+import { authorizeFollowRequest, rejectFollowRequest } from 'flavours/blobfox/actions/accounts';
+
+import FollowRequest from '../components/follow_request';
+
+const mapDispatchToProps = (dispatch, { account }) => ({
+  onAuthorize () {
+    dispatch(authorizeFollowRequest(account.get('id')));
+  },
+
+  onReject () {
+    dispatch(rejectFollowRequest(account.get('id')));
+  },
+});
+
+export default connect(null, mapDispatchToProps)(FollowRequest);
diff --git a/app/javascript/flavours/blobfox/features/notifications/containers/notification_container.js b/app/javascript/flavours/blobfox/features/notifications/containers/notification_container.js
new file mode 100644
index 00000000000000..3a507128983b81
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/containers/notification_container.js
@@ -0,0 +1,25 @@
+import { connect } from 'react-redux';
+
+import { mentionCompose } from 'flavours/blobfox/actions/compose';
+import { makeGetNotification } from 'flavours/blobfox/selectors';
+
+import Notification from '../components/notification';
+
+const makeMapStateToProps = () => {
+  const getNotification = makeGetNotification();
+
+  const mapStateToProps = (state, props) => ({
+    notification: getNotification(state, props.notification, props.accountId),
+    notifCleaning: state.getIn(['notifications', 'cleaningMode']),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = dispatch => ({
+  onMention: (account, history) => {
+    dispatch(mentionCompose(account, history));
+  },
+});
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(Notification);
diff --git a/app/javascript/flavours/blobfox/features/notifications/containers/overlay_container.js b/app/javascript/flavours/blobfox/features/notifications/containers/overlay_container.js
new file mode 100644
index 00000000000000..145efb3663f6d4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/containers/overlay_container.js
@@ -0,0 +1,19 @@
+//  Package imports.
+import { connect } from 'react-redux';
+
+//  Our imports.
+import { markNotificationForDelete } from 'flavours/blobfox/actions/notifications';
+
+import NotificationOverlay from '../components/overlay';
+
+const mapDispatchToProps = dispatch => ({
+  onMarkForDelete(id, yes) {
+    dispatch(markNotificationForDelete(id, yes));
+  },
+});
+
+const mapStateToProps = state => ({
+  show: state.getIn(['notifications', 'cleaningMode']),
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(NotificationOverlay);
diff --git a/app/javascript/flavours/blobfox/features/notifications/index.jsx b/app/javascript/flavours/blobfox/features/notifications/index.jsx
new file mode 100644
index 00000000000000..c0cca9debcd56c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/notifications/index.jsx
@@ -0,0 +1,368 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { debounce } from 'lodash';
+
+import { compareId } from 'flavours/blobfox/compare_id';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { NotSignedInIndicator } from 'flavours/blobfox/components/not_signed_in_indicator';
+
+import { addColumn, removeColumn, moveColumn } from '../../actions/columns';
+import { submitMarkers } from '../../actions/markers';
+import {
+  enterNotificationClearingMode,
+  expandNotifications,
+  scrollTopNotifications,
+  loadPending,
+  mountNotifications,
+  unmountNotifications,
+  markNotificationsAsRead,
+} from '../../actions/notifications';
+import Column from '../../components/column';
+import ColumnHeader from '../../components/column_header';
+import { LoadGap } from '../../components/load_gap';
+import ScrollableList from '../../components/scrollable_list';
+import NotificationPurgeButtonsContainer from '../../containers/notification_purge_buttons_container';
+
+import NotificationsPermissionBanner from './components/notifications_permission_banner';
+import ColumnSettingsContainer from './containers/column_settings_container';
+import FilterBarContainer from './containers/filter_bar_container';
+import NotificationContainer from './containers/notification_container';
+
+const messages = defineMessages({
+  title: { id: 'column.notifications', defaultMessage: 'Notifications' },
+  enterNotifCleaning : { id: 'notification_purge.start', defaultMessage: 'Enter notification cleaning mode' },
+  markAsRead : { id: 'notifications.mark_as_read', defaultMessage: 'Mark every notification as read' },
+});
+
+const getExcludedTypes = createSelector([
+  state => state.getIn(['settings', 'notifications', 'shows']),
+], (shows) => {
+  return ImmutableList(shows.filter(item => !item).keys());
+});
+
+const getNotifications = createSelector([
+  state => state.getIn(['settings', 'notifications', 'quickFilter', 'show']),
+  state => state.getIn(['settings', 'notifications', 'quickFilter', 'active']),
+  getExcludedTypes,
+  state => state.getIn(['notifications', 'items']),
+], (showFilterBar, allowedType, excludedTypes, notifications) => {
+  if (!showFilterBar || allowedType === 'all') {
+    // used if user changed the notification settings after loading the notifications from the server
+    // otherwise a list of notifications will come pre-filtered from the backend
+    // we need to turn it off for FilterBar in order not to block ourselves from seeing a specific category
+    return notifications.filterNot(item => item !== null && excludedTypes.includes(item.get('type')));
+  }
+  return notifications.filter(item => item === null || allowedType === item.get('type'));
+});
+
+const mapStateToProps = state => ({
+  showFilterBar: state.getIn(['settings', 'notifications', 'quickFilter', 'show']),
+  notifications: getNotifications(state),
+  localSettings:  state.get('local_settings'),
+  isLoading: state.getIn(['notifications', 'isLoading'], 0) > 0,
+  isUnread: state.getIn(['notifications', 'unread']) > 0 || state.getIn(['notifications', 'pendingItems']).size > 0,
+  hasMore: state.getIn(['notifications', 'hasMore']),
+  numPending: state.getIn(['notifications', 'pendingItems'], ImmutableList()).size,
+  notifCleaningActive: state.getIn(['notifications', 'cleaningMode']),
+  lastReadId: state.getIn(['settings', 'notifications', 'showUnread']) ? state.getIn(['notifications', 'readMarkerId']) : '0',
+  canMarkAsRead: state.getIn(['settings', 'notifications', 'showUnread']) && state.getIn(['notifications', 'readMarkerId']) !== '0' && getNotifications(state).some(item => item !== null && compareId(item.get('id'), state.getIn(['notifications', 'readMarkerId'])) > 0),
+  needsNotificationPermission: state.getIn(['settings', 'notifications', 'alerts']).includes(true) && state.getIn(['notifications', 'browserSupport']) && state.getIn(['notifications', 'browserPermission']) === 'default' && !state.getIn(['settings', 'notifications', 'dismissPermissionBanner']),
+});
+
+/* blobfox */
+const mapDispatchToProps = dispatch => ({
+  onEnterCleaningMode(yes) {
+    dispatch(enterNotificationClearingMode(yes));
+  },
+  dispatch,
+});
+
+class Notifications extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    columnId: PropTypes.string,
+    notifications: ImmutablePropTypes.list.isRequired,
+    showFilterBar: PropTypes.bool.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    isLoading: PropTypes.bool,
+    isUnread: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    hasMore: PropTypes.bool,
+    numPending: PropTypes.number,
+    localSettings: ImmutablePropTypes.map,
+    notifCleaningActive: PropTypes.bool,
+    onEnterCleaningMode: PropTypes.func,
+    lastReadId: PropTypes.string,
+    canMarkAsRead: PropTypes.bool,
+    needsNotificationPermission: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    trackScroll: true,
+  };
+
+  state = {
+    animatingNCD: false,
+  };
+
+  componentDidMount() {
+    this.props.dispatch(mountNotifications());
+  }
+
+  componentWillUnmount () {
+    this.handleLoadOlder.cancel();
+    this.handleScrollToTop.cancel();
+    this.handleScroll.cancel();
+    // this.props.dispatch(scrollTopNotifications(false));
+    this.props.dispatch(unmountNotifications());
+  }
+
+  handleLoadGap = (maxId) => {
+    this.props.dispatch(expandNotifications({ maxId }));
+  };
+
+  handleLoadOlder = debounce(() => {
+    const last = this.props.notifications.last();
+    this.props.dispatch(expandNotifications({ maxId: last && last.get('id') }));
+  }, 300, { leading: true });
+
+  handleLoadPending = () => {
+    this.props.dispatch(loadPending());
+  };
+
+  handleScrollToTop = debounce(() => {
+    this.props.dispatch(scrollTopNotifications(true));
+  }, 100);
+
+  handleScroll = debounce(() => {
+    this.props.dispatch(scrollTopNotifications(false));
+  }, 100);
+
+  handlePin = () => {
+    const { columnId, dispatch } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn('NOTIFICATIONS', {}));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setColumnRef = c => {
+    this.column = c;
+  };
+
+  handleMoveUp = id => {
+    const elementIndex = this.props.notifications.findIndex(item => item !== null && item.get('id') === id) - 1;
+    this._selectChild(elementIndex, true);
+  };
+
+  handleMoveDown = id => {
+    const elementIndex = this.props.notifications.findIndex(item => item !== null && item.get('id') === id) + 1;
+    this._selectChild(elementIndex, false);
+  };
+
+  _selectChild (index, align_top) {
+    const container = this.column.node;
+    const element = container.querySelector(`article:nth-of-type(${index + 1}) .focusable`);
+
+    if (element) {
+      if (align_top && container.scrollTop > element.offsetTop) {
+        element.scrollIntoView(true);
+      } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) {
+        element.scrollIntoView(false);
+      }
+      element.focus();
+    }
+  }
+
+  handleTransitionEndNCD = () => {
+    this.setState({ animatingNCD: false });
+  };
+
+  onEnterCleaningMode = () => {
+    this.setState({ animatingNCD: true });
+    this.props.onEnterCleaningMode(!this.props.notifCleaningActive);
+  };
+
+  handleMarkAsRead = () => {
+    this.props.dispatch(markNotificationsAsRead());
+    this.props.dispatch(submitMarkers({ immediate: true }));
+  };
+
+  render () {
+    const { intl, notifications, isLoading, isUnread, columnId, multiColumn, hasMore, numPending, showFilterBar, lastReadId, canMarkAsRead, needsNotificationPermission } = this.props;
+    const { notifCleaningActive } = this.props;
+    const { animatingNCD } = this.state;
+    const pinned = !!columnId;
+    const emptyMessage = <FormattedMessage id='empty_column.notifications' defaultMessage="You don't have any notifications yet. When other people interact with you, you will see it here." />;
+    const { signedIn } = this.context.identity;
+
+    let scrollableContent = null;
+
+    const filterBarContainer = (signedIn && showFilterBar)
+      ? (<FilterBarContainer />)
+      : null;
+
+    if (isLoading && this.scrollableContent) {
+      scrollableContent = this.scrollableContent;
+    } else if (notifications.size > 0 || hasMore) {
+      scrollableContent = notifications.map((item, index) => item === null ? (
+        <LoadGap
+          key={'gap:' + notifications.getIn([index + 1, 'id'])}
+          disabled={isLoading}
+          maxId={index > 0 ? notifications.getIn([index - 1, 'id']) : null}
+          onClick={this.handleLoadGap}
+        />
+      ) : (
+        <NotificationContainer
+          key={item.get('id')}
+          notification={item}
+          accountId={item.get('account')}
+          onMoveUp={this.handleMoveUp}
+          onMoveDown={this.handleMoveDown}
+          unread={lastReadId !== '0' && compareId(item.get('id'), lastReadId) > 0}
+        />
+      ));
+    } else {
+      scrollableContent = null;
+    }
+
+    this.scrollableContent = scrollableContent;
+
+    let scrollContainer;
+
+    if (signedIn) {
+      scrollContainer = (
+        <ScrollableList
+          scrollKey={`notifications-${columnId}`}
+          trackScroll={!pinned}
+          isLoading={isLoading}
+          showLoading={isLoading && notifications.size === 0}
+          hasMore={hasMore}
+          numPending={numPending}
+          prepend={needsNotificationPermission && <NotificationsPermissionBanner />}
+          alwaysPrepend
+          emptyMessage={emptyMessage}
+          onLoadMore={this.handleLoadOlder}
+          onLoadPending={this.handleLoadPending}
+          onScrollToTop={this.handleScrollToTop}
+          onScroll={this.handleScroll}
+          bindToDocument={!multiColumn}
+        >
+          {scrollableContent}
+        </ScrollableList>
+      );
+    } else {
+      scrollContainer = <NotSignedInIndicator />;
+    }
+
+    const extraButtons = [];
+
+    if (canMarkAsRead) {
+      extraButtons.push(
+        <button
+          key='mark-as-read'
+          aria-label={intl.formatMessage(messages.markAsRead)}
+          title={intl.formatMessage(messages.markAsRead)}
+          onClick={this.handleMarkAsRead}
+          className='column-header__button'
+        >
+          <Icon id='check' />
+        </button>,
+      );
+    }
+
+    const notifCleaningButtonClassName = classNames('column-header__button', {
+      'active': notifCleaningActive,
+    });
+
+    const notifCleaningDrawerClassName = classNames('ncd column-header__collapsible', {
+      'collapsed': !notifCleaningActive,
+      'animating': animatingNCD,
+    });
+
+    const msgEnterNotifCleaning = intl.formatMessage(messages.enterNotifCleaning);
+
+    extraButtons.push(
+      <button
+        key='notif-cleaning'
+        aria-label={msgEnterNotifCleaning}
+        title={msgEnterNotifCleaning}
+        onClick={this.onEnterCleaningMode}
+        className={notifCleaningButtonClassName}
+      >
+        <Icon id='eraser' />
+      </button>,
+    );
+
+    const notifCleaningDrawer = (
+      <div className={notifCleaningDrawerClassName} onTransitionEnd={this.handleTransitionEndNCD}>
+        <div className='column-header__collapsible-inner nopad-drawer'>
+          {(notifCleaningActive || animatingNCD) ? (<NotificationPurgeButtonsContainer />) : null }
+        </div>
+      </div>
+    );
+
+    return (
+      <Column
+        bindToDocument={!multiColumn}
+        ref={this.setColumnRef}
+        extraClasses={this.props.notifCleaningActive ? 'notif-cleaning' : null}
+        label={intl.formatMessage(messages.title)}
+      >
+        <ColumnHeader
+          icon='bell'
+          active={isUnread}
+          title={intl.formatMessage(messages.title)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+          localSettings={this.props.localSettings}
+          extraButton={extraButtons}
+          appendContent={notifCleaningDrawer}
+        >
+          <ColumnSettingsContainer />
+        </ColumnHeader>
+
+        {filterBarContainer}
+        {scrollContainer}
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(Notifications));
diff --git a/app/javascript/flavours/blobfox/features/onboarding/components/arrow_small_right.jsx b/app/javascript/flavours/blobfox/features/onboarding/components/arrow_small_right.jsx
new file mode 100644
index 00000000000000..79b9db383f563d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/onboarding/components/arrow_small_right.jsx
@@ -0,0 +1,7 @@
+const ArrowSmallRight = () => (
+  <svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20' fill='currentColor'>
+    <path fillRule='evenodd' d='M5 10a.75.75 0 01.75-.75h6.638L10.23 7.29a.75.75 0 111.04-1.08l3.5 3.25a.75.75 0 010 1.08l-3.5 3.25a.75.75 0 11-1.04-1.08l2.158-1.96H5.75A.75.75 0 015 10z' clipRule='evenodd' />
+  </svg>
+);
+
+export default ArrowSmallRight;
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/features/onboarding/components/progress_indicator.jsx b/app/javascript/flavours/blobfox/features/onboarding/components/progress_indicator.jsx
new file mode 100644
index 00000000000000..5a2623616ecb3f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/onboarding/components/progress_indicator.jsx
@@ -0,0 +1,28 @@
+import PropTypes from 'prop-types';
+import { Fragment } from 'react';
+
+import classNames from 'classnames';
+
+import { Check } from 'flavours/blobfox/components/check';
+
+
+const ProgressIndicator = ({ steps, completed }) => (
+  <div className='onboarding__progress-indicator'>
+    {(new Array(steps)).fill().map((_, i) => (
+      <Fragment key={i}>
+        {i > 0 && <div className={classNames('onboarding__progress-indicator__line', { active: completed > i })} />}
+
+        <div className={classNames('onboarding__progress-indicator__step', { active: completed > i })}>
+          {completed > i && <Check />}
+        </div>
+      </Fragment>
+    ))}
+  </div>
+);
+
+ProgressIndicator.propTypes = {
+  steps: PropTypes.number.isRequired,
+  completed: PropTypes.number,
+};
+
+export default ProgressIndicator;
diff --git a/app/javascript/flavours/blobfox/features/onboarding/components/step.jsx b/app/javascript/flavours/blobfox/features/onboarding/components/step.jsx
new file mode 100644
index 00000000000000..b904c47ab308f4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/onboarding/components/step.jsx
@@ -0,0 +1,50 @@
+import PropTypes from 'prop-types';
+
+import { Check } from 'flavours/blobfox/components/check';
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+import ArrowSmallRight from './arrow_small_right';
+
+const Step = ({ label, description, icon, completed, onClick, href }) => {
+  const content = (
+    <>
+      <div className='onboarding__steps__item__icon'>
+        <Icon id={icon} />
+      </div>
+
+      <div className='onboarding__steps__item__description'>
+        <h6>{label}</h6>
+        <p>{description}</p>
+      </div>
+
+      <div className={completed ? 'onboarding__steps__item__progress' : 'onboarding__steps__item__go'}>
+        {completed ? <Check /> : <ArrowSmallRight />}
+      </div>
+    </>
+  );
+
+  if (href) {
+    return (
+      <a href={href} onClick={onClick} target='_blank' rel='noopener' className='onboarding__steps__item'>
+        {content}
+      </a>
+    );
+  }
+
+  return (
+    <button onClick={onClick} className='onboarding__steps__item'>
+      {content}
+    </button>
+  );
+};
+
+Step.propTypes = {
+  label: PropTypes.node,
+  description: PropTypes.node,
+  icon: PropTypes.string,
+  completed: PropTypes.bool,
+  href: PropTypes.string,
+  onClick: PropTypes.func,
+};
+
+export default Step;
diff --git a/app/javascript/flavours/blobfox/features/onboarding/follows.jsx b/app/javascript/flavours/blobfox/features/onboarding/follows.jsx
new file mode 100644
index 00000000000000..de753b0e6dbaf8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/onboarding/follows.jsx
@@ -0,0 +1,80 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { fetchSuggestions } from 'flavours/blobfox/actions/suggestions';
+import { markAsPartial } from 'flavours/blobfox/actions/timelines';
+import Column from 'flavours/blobfox/components/column';
+import ColumnBackButton from 'flavours/blobfox/components/column_back_button';
+import { EmptyAccount } from 'flavours/blobfox/components/empty_account';
+import Account from 'flavours/blobfox/containers/account_container';
+
+const mapStateToProps = state => ({
+  suggestions: state.getIn(['suggestions', 'items']),
+  isLoading: state.getIn(['suggestions', 'isLoading']),
+});
+
+class Follows extends PureComponent {
+
+  static propTypes = {
+    onBack: PropTypes.func,
+    dispatch: PropTypes.func.isRequired,
+    suggestions: ImmutablePropTypes.list,
+    isLoading: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(fetchSuggestions(true));
+  }
+
+  componentWillUnmount () {
+    const { dispatch } = this.props;
+    dispatch(markAsPartial('home'));
+  }
+
+  render () {
+    const { onBack, isLoading, suggestions, multiColumn } = this.props;
+
+    let loadedContent;
+
+    if (isLoading) {
+      loadedContent = (new Array(8)).fill().map((_, i) => <EmptyAccount key={i} />);
+    } else if (suggestions.isEmpty()) {
+      loadedContent = <div className='follow-recommendations__empty'><FormattedMessage id='onboarding.follows.empty' defaultMessage='Unfortunately, no results can be shown right now. You can try using search or browsing the explore page to find people to follow, or try again later.' /></div>;
+    } else {
+      loadedContent = suggestions.map(suggestion => <Account id={suggestion.get('account')} key={suggestion.get('account')} withBio />);
+    }
+
+    return (
+      <Column>
+        <ColumnBackButton multiColumn={multiColumn} onClick={onBack} />
+
+        <div className='scrollable privacy-policy'>
+          <div className='column-title'>
+            <h3><FormattedMessage id='onboarding.follows.title' defaultMessage='Popular on Mastodon' /></h3>
+            <p><FormattedMessage id='onboarding.follows.lead' defaultMessage='You curate your own home feed. The more people you follow, the more active and interesting it will be. These profiles may be a good starting point—you can always unfollow them later!' /></p>
+          </div>
+
+          <div className='follow-recommendations'>
+            {loadedContent}
+          </div>
+
+          <p className='onboarding__lead'><FormattedMessage id='onboarding.tips.accounts_from_other_servers' defaultMessage='<strong>Did you know?</strong> Since Mastodon is decentralized, some profiles you come across will be hosted on servers other than yours. And yet you can interact with them seamlessly! Their server is in the second half of their username!' values={{ strong: chunks => <strong>{chunks}</strong> }} /></p>
+
+          <div className='onboarding__footer'>
+            <button className='link-button' onClick={onBack}><FormattedMessage id='onboarding.actions.back' defaultMessage='Take me back' /></button>
+          </div>
+        </div>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(Follows);
diff --git a/app/javascript/flavours/blobfox/features/onboarding/index.jsx b/app/javascript/flavours/blobfox/features/onboarding/index.jsx
new file mode 100644
index 00000000000000..abb1dd7d075ec0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/onboarding/index.jsx
@@ -0,0 +1,149 @@
+import PropTypes from 'prop-types';
+import React from 'react';
+
+import { FormattedMessage, injectIntl, defineMessages } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+import { Link, withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { fetchAccount } from 'flavours/blobfox/actions/accounts';
+import { focusCompose } from 'flavours/blobfox/actions/compose';
+import { closeOnboarding } from 'flavours/blobfox/actions/onboarding';
+import Column from 'flavours/blobfox/features/ui/components/column';
+import { me } from 'flavours/blobfox/initial_state';
+import { makeGetAccount } from 'flavours/blobfox/selectors';
+import { assetHost } from 'flavours/blobfox/utils/config';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+import illustration from 'mastodon/../images/elephant_ui_conversation.svg';
+
+import ArrowSmallRight from './components/arrow_small_right';
+import Step from './components/step';
+import Follows from './follows';
+import Share from './share';
+
+const messages = defineMessages({
+  template: { id: 'onboarding.compose.template', defaultMessage: 'Hello #Mastodon!' },
+});
+
+const mapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  return state => ({
+    account: getAccount(state, me),
+  });
+};
+
+class Onboarding extends ImmutablePureComponent {
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    account: ImmutablePropTypes.record,
+    multiColumn: PropTypes.bool,
+    ...WithRouterPropTypes,
+  };
+
+  state = {
+    step: null,
+    profileClicked: false,
+    shareClicked: false,
+  };
+
+  handleClose = () => {
+    const { dispatch, history } = this.props;
+
+    dispatch(closeOnboarding());
+    history.push('/home');
+  };
+
+  handleProfileClick = () => {
+    this.setState({ profileClicked: true });
+  };
+
+  handleFollowClick = () => {
+    this.setState({ step: 'follows' });
+  };
+
+  handleComposeClick = () => {
+    const { dispatch, intl, history } = this.props;
+
+    dispatch(focusCompose(history, intl.formatMessage(messages.template)));
+  };
+
+  handleShareClick = () => {
+    this.setState({ step: 'share', shareClicked: true });
+  };
+
+  handleBackClick = () => {
+    this.setState({ step: null });
+  };
+
+  handleWindowFocus = debounce(() => {
+    const { dispatch, account } = this.props;
+    dispatch(fetchAccount(account.get('id')));
+  }, 1000, { trailing: true });
+
+  componentDidMount () {
+    window.addEventListener('focus', this.handleWindowFocus, false);
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('focus', this.handleWindowFocus);
+  }
+
+  render () {
+    const { account, multiColumn } = this.props;
+    const { step, shareClicked } = this.state;
+
+    switch(step) {
+    case 'follows':
+      return <Follows onBack={this.handleBackClick} multiColumn={multiColumn} />;
+    case 'share':
+      return <Share onBack={this.handleBackClick} multiColumn={multiColumn} />;
+    }
+
+    return (
+      <Column>
+        <div className='scrollable privacy-policy'>
+          <div className='column-title'>
+            <img src={illustration} alt='' className='onboarding__illustration' />
+            <h3><FormattedMessage id='onboarding.start.title' defaultMessage="You've made it!" /></h3>
+            <p><FormattedMessage id='onboarding.start.lead' defaultMessage="Your new Mastodon account is ready to go. Here's how you can make the most of it:" /></p>
+          </div>
+
+          <div className='onboarding__steps'>
+            <Step onClick={this.handleProfileClick} href='/settings/profile' completed={(!account.get('avatar').endsWith('missing.png')) || (account.get('display_name').length > 0 && account.get('note').length > 0)} icon='address-book-o' label={<FormattedMessage id='onboarding.steps.setup_profile.title' defaultMessage='Customize your profile' />} description={<FormattedMessage id='onboarding.steps.setup_profile.body' defaultMessage='Others are more likely to interact with you with a filled out profile.' />} />
+            <Step onClick={this.handleFollowClick} completed={(account.get('following_count') * 1) >= 7} icon='user-plus' label={<FormattedMessage id='onboarding.steps.follow_people.title' defaultMessage='Find at least {count, plural, one {one person} other {# people}} to follow' values={{ count: 7 }} />} description={<FormattedMessage id='onboarding.steps.follow_people.body' defaultMessage="You curate your own home feed. Let's fill it with interesting people." />} />
+            <Step onClick={this.handleComposeClick} completed={(account.get('statuses_count') * 1) >= 1} icon='pencil-square-o' label={<FormattedMessage id='onboarding.steps.publish_status.title' defaultMessage='Make your first post' />} description={<FormattedMessage id='onboarding.steps.publish_status.body' defaultMessage='Say hello to the world.' values={{ emoji: <img className='emojione' alt='🐘' src={`${assetHost}/emoji/1f418.svg`} /> }} />} />
+            <Step onClick={this.handleShareClick} completed={shareClicked} icon='copy' label={<FormattedMessage id='onboarding.steps.share_profile.title' defaultMessage='Share your profile' />} description={<FormattedMessage id='onboarding.steps.share_profile.body' defaultMessage='Let your friends know how to find you on Mastodon!' />} />
+          </div>
+
+          <p className='onboarding__lead'><FormattedMessage id='onboarding.start.skip' defaultMessage="Don't need help getting started?" /></p>
+
+          <div className='onboarding__links'>
+            <Link to='/explore' className='onboarding__link'>
+              <FormattedMessage id='onboarding.actions.go_to_explore' defaultMessage='Take me to trending' />
+              <ArrowSmallRight />
+            </Link>
+
+            <Link to='/home' className='onboarding__link'>
+              <FormattedMessage id='onboarding.actions.go_to_home' defaultMessage='Take me to my home feed' />
+              <ArrowSmallRight />
+            </Link>
+          </div>
+        </div>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default withRouter(connect(mapStateToProps)(injectIntl(Onboarding)));
diff --git a/app/javascript/flavours/blobfox/features/onboarding/share.jsx b/app/javascript/flavours/blobfox/features/onboarding/share.jsx
new file mode 100644
index 00000000000000..b6d4d0eb02c798
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/onboarding/share.jsx
@@ -0,0 +1,200 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { Link } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import SwipeableViews from 'react-swipeable-views';
+
+import Column from 'flavours/blobfox/components/column';
+import ColumnBackButton from 'flavours/blobfox/components/column_back_button';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { me, domain } from 'flavours/blobfox/initial_state';
+
+import ArrowSmallRight from './components/arrow_small_right';
+
+const messages = defineMessages({
+  shareableMessage: { id: 'onboarding.share.message', defaultMessage: 'I\'m {username} on #Mastodon! Come follow me at {url}' },
+});
+
+const mapStateToProps = state => ({
+  account: state.getIn(['accounts', me]),
+});
+
+class CopyPasteText extends PureComponent {
+
+  static propTypes = {
+    value: PropTypes.string,
+  };
+
+  state = {
+    copied: false,
+    focused: false,
+  };
+
+  setRef = c => {
+    this.input = c;
+  };
+
+  handleInputClick = () => {
+    this.setState({ copied: false });
+    this.input.focus();
+    this.input.select();
+    this.input.setSelectionRange(0, this.props.value.length);
+  };
+
+  handleButtonClick = e => {
+    e.stopPropagation();
+
+    const { value } = this.props;
+    navigator.clipboard.writeText(value);
+    this.input.blur();
+    this.setState({ copied: true });
+    this.timeout = setTimeout(() => this.setState({ copied: false }), 700);
+  };
+
+  handleFocus = () => {
+    this.setState({ focused: true });
+  };
+
+  handleBlur = () => {
+    this.setState({ focused: false });
+  };
+
+  componentWillUnmount () {
+    if (this.timeout) clearTimeout(this.timeout);
+  }
+
+  render () {
+    const { value } = this.props;
+    const { copied, focused } = this.state;
+
+    return (
+      <div className={classNames('copy-paste-text', { copied, focused })} tabIndex='0' role='button' onClick={this.handleInputClick}>
+        <textarea readOnly value={value} ref={this.setRef} onClick={this.handleInputClick} onFocus={this.handleFocus} onBlur={this.handleBlur} />
+
+        <button className='button' onClick={this.handleButtonClick}>
+          <Icon id='copy' /> {copied ? <FormattedMessage id='copypaste.copied' defaultMessage='Copied' /> : <FormattedMessage id='copypaste.copy_to_clipboard' defaultMessage='Copy to clipboard' />}
+        </button>
+      </div>
+    );
+  }
+
+}
+
+class TipCarousel extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node,
+  };
+
+  state = {
+    index: 0,
+  };
+
+  handleSwipe = index => {
+    this.setState({ index });
+  };
+
+  handleChangeIndex = e => {
+    this.setState({ index: Number(e.currentTarget.getAttribute('data-index')) });
+  };
+
+  handleKeyDown = e => {
+    switch(e.key) {
+    case 'ArrowLeft':
+      e.preventDefault();
+      this.setState(({ index }, { children }) => ({ index: Math.abs(index - 1) % children.length }));
+      break;
+    case 'ArrowRight':
+      e.preventDefault();
+      this.setState(({ index }, { children }) => ({ index: (index + 1) % children.length }));
+      break;
+    }
+  };
+
+  render () {
+    const { children } = this.props;
+    const { index } = this.state;
+
+    return (
+      <div className='tip-carousel' tabIndex='0' onKeyDown={this.handleKeyDown}>
+        <SwipeableViews onChangeIndex={this.handleSwipe} index={index} enableMouseEvents tabIndex='-1'>
+          {children}
+        </SwipeableViews>
+
+        <div className='media-modal__pagination'>
+          {children.map((_, i) => (
+            <button key={i} className={classNames('media-modal__page-dot', { active: i === index })} data-index={i} onClick={this.handleChangeIndex}>
+              {i + 1}
+            </button>
+          ))}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+class Share extends PureComponent {
+
+  static propTypes = {
+    onBack: PropTypes.func,
+    account: ImmutablePropTypes.record,
+    multiColumn: PropTypes.bool,
+    intl: PropTypes.object,
+  };
+
+  render () {
+    const { onBack, account, multiColumn, intl } = this.props;
+
+    const url = (new URL(`/@${account.get('username')}`, document.baseURI)).href;
+
+    return (
+      <Column>
+        <ColumnBackButton multiColumn={multiColumn} onClick={onBack} />
+
+        <div className='scrollable privacy-policy'>
+          <div className='column-title'>
+            <h3><FormattedMessage id='onboarding.share.title' defaultMessage='Share your profile' /></h3>
+            <p><FormattedMessage id='onboarding.share.lead' defaultMessage='Let people know how they can find you on Mastodon!' /></p>
+          </div>
+
+          <CopyPasteText value={intl.formatMessage(messages.shareableMessage, { username: `@${account.get('username')}@${domain}`, url })} />
+
+          <TipCarousel>
+            <div><p className='onboarding__lead'><FormattedMessage id='onboarding.tips.verification' defaultMessage='<strong>Did you know?</strong> You can verify your account by putting a link to your Mastodon profile on your own website and adding the website to your profile. No fees or documents necessary!'  values={{ strong: chunks => <strong>{chunks}</strong> }}  /></p></div>
+            <div><p className='onboarding__lead'><FormattedMessage id='onboarding.tips.migration' defaultMessage='<strong>Did you know?</strong> If you feel like {domain} is not a great server choice for you in the future, you can move to another Mastodon server without losing your followers. You can even host your own server!' values={{ domain, strong: chunks => <strong>{chunks}</strong> }} /></p></div>
+            <div><p className='onboarding__lead'><FormattedMessage id='onboarding.tips.2fa' defaultMessage='<strong>Did you know?</strong> You can secure your account by setting up two-factor authentication in your account settings. It works with any TOTP app of your choice, no phone number necessary!'  values={{ strong: chunks => <strong>{chunks}</strong> }}  /></p></div>
+          </TipCarousel>
+
+          <p className='onboarding__lead'><FormattedMessage id='onboarding.share.next_steps' defaultMessage='Possible next steps:' /></p>
+
+          <div className='onboarding__links'>
+            <Link to='/home' className='onboarding__link'>
+              <FormattedMessage id='onboarding.actions.go_to_home' defaultMessage='Take me to my home feed' />
+              <ArrowSmallRight />
+            </Link>
+
+            <Link to='/explore' className='onboarding__link'>
+              <FormattedMessage id='onboarding.actions.go_to_explore' defaultMessage='Take me to trending' />
+              <ArrowSmallRight />
+            </Link>
+          </div>
+
+          <div className='onboarding__footer'>
+            <button className='link-button' onClick={onBack}><FormattedMessage id='onboarding.action.back' defaultMessage='Take me back' /></button>
+          </div>
+        </div>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Share));
diff --git a/app/javascript/flavours/blobfox/features/picture_in_picture/components/footer.jsx b/app/javascript/flavours/blobfox/features/picture_in_picture/components/footer.jsx
new file mode 100644
index 00000000000000..e91282d9529027
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/picture_in_picture/components/footer.jsx
@@ -0,0 +1,231 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { initBoostModal } from 'flavours/blobfox/actions/boosts';
+import { replyCompose } from 'flavours/blobfox/actions/compose';
+import { reblog, favourite, unreblog, unfavourite } from 'flavours/blobfox/actions/interactions';
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import { me, boostModal } from 'flavours/blobfox/initial_state';
+import { makeGetStatus } from 'flavours/blobfox/selectors';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const messages = defineMessages({
+  reply: { id: 'status.reply', defaultMessage: 'Reply' },
+  replyAll: { id: 'status.replyAll', defaultMessage: 'Reply to thread' },
+  reblog: { id: 'status.reblog', defaultMessage: 'Boost' },
+  reblog_private: { id: 'status.reblog_private', defaultMessage: 'Boost with original visibility' },
+  cancel_reblog_private: { id: 'status.cancel_reblog_private', defaultMessage: 'Unboost' },
+  cannot_reblog: { id: 'status.cannot_reblog', defaultMessage: 'This post cannot be boosted' },
+  favourite: { id: 'status.favourite', defaultMessage: 'Favorite' },
+  replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' },
+  replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
+  open: { id: 'status.open', defaultMessage: 'Expand this status' },
+});
+
+const makeMapStateToProps = () => {
+  const getStatus = makeGetStatus();
+
+  const mapStateToProps = (state, { statusId }) => ({
+    status: getStatus(state, { id: statusId }),
+    askReplyConfirmation: state.getIn(['compose', 'text']).trim().length !== 0,
+    showReplyCount: state.getIn(['local_settings', 'show_reply_count']),
+  });
+
+  return mapStateToProps;
+};
+
+class Footer extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    statusId: PropTypes.string.isRequired,
+    status: ImmutablePropTypes.map.isRequired,
+    intl: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    askReplyConfirmation: PropTypes.bool,
+    showReplyCount: PropTypes.bool,
+    withOpenButton: PropTypes.bool,
+    onClose: PropTypes.func,
+    ...WithRouterPropTypes,
+  };
+
+  _performReply = () => {
+    const { dispatch, status, onClose, history } = this.props;
+
+    if (onClose) {
+      onClose(true);
+    }
+
+    dispatch(replyCompose(status, history));
+  };
+
+  handleReplyClick = () => {
+    const { dispatch, askReplyConfirmation, status, intl } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      if (askReplyConfirmation) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: intl.formatMessage(messages.replyMessage),
+            confirm: intl.formatMessage(messages.replyConfirm),
+            onConfirm: this._performReply,
+          },
+        }));
+      } else {
+        this._performReply();
+      }
+    } else {
+      dispatch(openModal({
+        modalType: 'INTERACTION',
+        modalProps: {
+          type: 'reply',
+          accountId: status.getIn(['account', 'id']),
+          url: status.get('uri'),
+        },
+      }));
+    }
+  };
+
+  handleFavouriteClick = () => {
+    const { dispatch, status } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      if (status.get('favourited')) {
+        dispatch(unfavourite(status));
+      } else {
+        dispatch(favourite(status));
+      }
+    } else {
+      dispatch(openModal({
+        modalType: 'INTERACTION',
+        modalProps: {
+          type: 'favourite',
+          accountId: status.getIn(['account', 'id']),
+          url: status.get('uri'),
+        },
+      }));
+    }
+  };
+
+  _performReblog = (status, privacy) => {
+    const { dispatch } = this.props;
+    dispatch(reblog(status, privacy));
+  };
+
+  handleReblogClick = e => {
+    const { dispatch, status } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      if (status.get('reblogged')) {
+        dispatch(unreblog(status));
+      } else if ((e && e.shiftKey) || !boostModal) {
+        this._performReblog(status);
+      } else {
+        dispatch(initBoostModal({ status, onReblog: this._performReblog }));
+      }
+    } else {
+      dispatch(openModal({
+        modalType: 'INTERACTION',
+        modalProps: {
+          type: 'reblog',
+          accountId: status.getIn(['account', 'id']),
+          url: status.get('uri'),
+        },
+      }));
+    }
+  };
+
+  handleOpenClick = e => {
+    if (e.button !== 0 || !history) {
+      return;
+    }
+
+    const { status, onClose } = this.props;
+
+    if (onClose) {
+      onClose();
+    }
+
+    this.props.history.push(`/@${status.getIn(['account', 'acct'])}/${status.get('id')}`);
+  };
+
+  render () {
+    const { status, intl, showReplyCount, withOpenButton } = this.props;
+
+    const publicStatus  = ['public', 'unlisted'].includes(status.get('visibility'));
+    const reblogPrivate = status.getIn(['account', 'id']) === me && status.get('visibility') === 'private';
+
+    let replyIcon, replyTitle;
+
+    if (status.get('in_reply_to_id', null) === null) {
+      replyIcon = 'reply';
+      replyTitle = intl.formatMessage(messages.reply);
+    } else {
+      replyIcon = 'reply-all';
+      replyTitle = intl.formatMessage(messages.replyAll);
+    }
+
+    let reblogTitle = '';
+
+    if (status.get('reblogged')) {
+      reblogTitle = intl.formatMessage(messages.cancel_reblog_private);
+    } else if (publicStatus) {
+      reblogTitle = intl.formatMessage(messages.reblog);
+    } else if (reblogPrivate) {
+      reblogTitle = intl.formatMessage(messages.reblog_private);
+    } else {
+      reblogTitle = intl.formatMessage(messages.cannot_reblog);
+    }
+
+    let replyButton = null;
+    if (showReplyCount) {
+      replyButton = (
+        <IconButton
+          className='status__action-bar-button'
+          title={replyTitle}
+          icon={status.get('in_reply_to_account_id') === status.getIn(['account', 'id']) ? 'reply' : replyIcon}
+          onClick={this.handleReplyClick}
+          counter={status.get('replies_count')}
+          obfuscateCount
+        />
+      );
+    } else {
+      replyButton = (
+        <IconButton
+          className='status__action-bar-button'
+          title={replyTitle}
+          icon={status.get('in_reply_to_account_id') === status.getIn(['account', 'id']) ? 'reply' : replyIcon}
+          onClick={this.handleReplyClick}
+        />
+      );
+    }
+
+    return (
+      <div className='picture-in-picture__footer'>
+        {replyButton}
+        <IconButton className={classNames('status__action-bar-button', { reblogPrivate })} disabled={!publicStatus && !reblogPrivate}  active={status.get('reblogged')} title={reblogTitle} icon='retweet' onClick={this.handleReblogClick} counter={status.get('reblogs_count')} />
+        <IconButton className='status__action-bar-button star-icon' animate active={status.get('favourited')} title={intl.formatMessage(messages.favourite)} icon='star' onClick={this.handleFavouriteClick} counter={status.get('favourites_count')} />
+        {withOpenButton && <IconButton className='status__action-bar-button' title={intl.formatMessage(messages.open)} icon='external-link' onClick={this.handleOpenClick} href={status.get('url')} />}
+      </div>
+    );
+  }
+
+}
+
+export default  withRouter(connect(makeMapStateToProps)(injectIntl(Footer)));
diff --git a/app/javascript/flavours/blobfox/features/picture_in_picture/components/header.jsx b/app/javascript/flavours/blobfox/features/picture_in_picture/components/header.jsx
new file mode 100644
index 00000000000000..887924f33767e8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/picture_in_picture/components/header.jsx
@@ -0,0 +1,50 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+const mapStateToProps = (state, { accountId }) => ({
+  account: state.getIn(['accounts', accountId]),
+});
+
+class Header extends ImmutablePureComponent {
+
+  static propTypes = {
+    accountId: PropTypes.string.isRequired,
+    statusId: PropTypes.string.isRequired,
+    account: ImmutablePropTypes.record.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  render () {
+    const { account, statusId, onClose, intl } = this.props;
+
+    return (
+      <div className='picture-in-picture__header'>
+        <Link to={`/@${account.get('acct')}/${statusId}`} className='picture-in-picture__header__account'>
+          <Avatar account={account} size={36} />
+          <DisplayName account={account} />
+        </Link>
+
+        <IconButton icon='times' onClick={onClose} title={intl.formatMessage(messages.close)} />
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Header));
diff --git a/app/javascript/flavours/blobfox/features/picture_in_picture/index.jsx b/app/javascript/flavours/blobfox/features/picture_in_picture/index.jsx
new file mode 100644
index 00000000000000..9c5d0bb422bd01
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/picture_in_picture/index.jsx
@@ -0,0 +1,93 @@
+import PropTypes from 'prop-types';
+import { Component } from 'react';
+
+import classNames from 'classnames';
+
+import { connect } from 'react-redux';
+
+import { removePictureInPicture } from 'flavours/blobfox/actions/picture_in_picture';
+import Audio from 'flavours/blobfox/features/audio';
+import Video from 'flavours/blobfox/features/video';
+
+import Footer from './components/footer';
+import Header from './components/header';
+
+const mapStateToProps = state => ({
+  ...state.get('picture_in_picture'),
+  left: state.getIn(['local_settings', 'media', 'pop_in_position']) === 'left',
+});
+
+class PictureInPicture extends Component {
+
+  static propTypes = {
+    statusId: PropTypes.string,
+    accountId: PropTypes.string,
+    type: PropTypes.string,
+    src: PropTypes.string,
+    muted: PropTypes.bool,
+    volume: PropTypes.number,
+    currentTime: PropTypes.number,
+    poster: PropTypes.string,
+    backgroundColor: PropTypes.string,
+    foregroundColor: PropTypes.string,
+    accentColor: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    left: PropTypes.bool,
+  };
+
+  handleClose = () => {
+    const { dispatch } = this.props;
+    dispatch(removePictureInPicture());
+  };
+
+  render () {
+    const { type, src, currentTime, accountId, statusId, left } = this.props;
+
+    if (!currentTime) {
+      return null;
+    }
+
+    let player;
+
+    if (type === 'video') {
+      player = (
+        <Video
+          src={src}
+          currentTime={this.props.currentTime}
+          volume={this.props.volume}
+          muted={this.props.muted}
+          autoPlay
+          inline
+          alwaysVisible
+        />
+      );
+    } else if (type === 'audio') {
+      player = (
+        <Audio
+          src={src}
+          currentTime={this.props.currentTime}
+          volume={this.props.volume}
+          muted={this.props.muted}
+          poster={this.props.poster}
+          backgroundColor={this.props.backgroundColor}
+          foregroundColor={this.props.foregroundColor}
+          accentColor={this.props.accentColor}
+          autoPlay
+        />
+      );
+    }
+
+    return (
+      <div className={classNames('picture-in-picture', { left })}>
+        <Header accountId={accountId} statusId={statusId} onClose={this.handleClose} />
+
+        {player}
+
+        <Footer statusId={statusId} />
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(PictureInPicture);
diff --git a/app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/account_container.js b/app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/account_container.js
new file mode 100644
index 00000000000000..841d5baeb4d85c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/account_container.js
@@ -0,0 +1,25 @@
+import { injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { pinAccount, unpinAccount } from 'flavours/blobfox/actions/accounts';
+import Account from 'flavours/blobfox/features/list_editor/components/account';
+import { makeGetAccount } from 'flavours/blobfox/selectors';
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { accountId, added }) => ({
+    account: getAccount(state, accountId),
+    added: typeof added === 'undefined' ? state.getIn(['pinnedAccountsEditor', 'accounts', 'items']).includes(accountId) : added,
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { accountId }) => ({
+  onRemove: () => dispatch(unpinAccount(accountId)),
+  onAdd: () => dispatch(pinAccount(accountId)),
+});
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(Account));
diff --git a/app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/search_container.js b/app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/search_container.js
new file mode 100644
index 00000000000000..095417884afcc7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/pinned_accounts_editor/containers/search_container.js
@@ -0,0 +1,24 @@
+import { injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import Search from 'flavours/blobfox/features/list_editor/components/search';
+
+import {
+  fetchPinnedAccountsSuggestions,
+  clearPinnedAccountsSuggestions,
+  changePinnedAccountsSuggestions,
+} from '../../../actions/accounts';
+
+
+const mapStateToProps = state => ({
+  value: state.getIn(['pinnedAccountsEditor', 'suggestions', 'value']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onSubmit: value => dispatch(fetchPinnedAccountsSuggestions(value)),
+  onClear: () => dispatch(clearPinnedAccountsSuggestions()),
+  onChange: value => dispatch(changePinnedAccountsSuggestions(value)),
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(Search));
diff --git a/app/javascript/flavours/blobfox/features/pinned_accounts_editor/index.jsx b/app/javascript/flavours/blobfox/features/pinned_accounts_editor/index.jsx
new file mode 100644
index 00000000000000..d550984f597ab9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/pinned_accounts_editor/index.jsx
@@ -0,0 +1,82 @@
+import PropTypes from 'prop-types';
+
+import { injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import spring from 'react-motion/lib/spring';
+
+import { fetchPinnedAccounts, clearPinnedAccountsSuggestions, resetPinnedAccountsEditor } from 'flavours/blobfox/actions/accounts';
+import Motion from 'flavours/blobfox/features/ui/util/optional_motion';
+
+import AccountContainer from './containers/account_container';
+import SearchContainer from './containers/search_container';
+
+const mapStateToProps = state => ({
+  accountIds: state.getIn(['pinnedAccountsEditor', 'accounts', 'items']),
+  searchAccountIds: state.getIn(['pinnedAccountsEditor', 'suggestions', 'items']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onInitialize: () => dispatch(fetchPinnedAccounts()),
+  onClear: () => dispatch(clearPinnedAccountsSuggestions()),
+  onReset: () => dispatch(resetPinnedAccountsEditor()),
+});
+
+class PinnedAccountsEditor extends ImmutablePureComponent {
+
+  static propTypes = {
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    onInitialize: PropTypes.func.isRequired,
+    onClear: PropTypes.func.isRequired,
+    onReset: PropTypes.func.isRequired,
+    title: PropTypes.string.isRequired,
+    accountIds: ImmutablePropTypes.list.isRequired,
+    searchAccountIds: ImmutablePropTypes.list.isRequired,
+  };
+
+  componentDidMount () {
+    const { onInitialize } = this.props;
+    onInitialize();
+  }
+
+  componentWillUnmount () {
+    const { onReset } = this.props;
+    onReset();
+  }
+
+  render () {
+    const { accountIds, searchAccountIds, onClear } = this.props;
+    const showSearch = searchAccountIds.size > 0;
+
+    return (
+      <div className='modal-root__modal list-editor'>
+        <h4><FormattedMessage id='endorsed_accounts_editor.endorsed_accounts' defaultMessage='Featured accounts' /></h4>
+
+        <SearchContainer />
+
+        <div className='drawer__pager'>
+          <div className='drawer__inner list-editor__accounts'>
+            {accountIds.map(accountId => <AccountContainer key={accountId} accountId={accountId} added />)}
+          </div>
+
+          {showSearch && <div role='button' tabIndex={-1} className='drawer__backdrop' onClick={onClear} />}
+
+          <Motion defaultStyle={{ x: -100 }} style={{ x: spring(showSearch ? 0 : -100, { stiffness: 210, damping: 20 }) }}>
+            {({ x }) =>
+              (<div className='drawer__inner backdrop' style={{ transform: x === 0 ? null : `translateX(${x}%)`, visibility: x === -100 ? 'hidden' : 'visible' }}>
+                {searchAccountIds.map(accountId => <AccountContainer key={accountId} accountId={accountId} />)}
+              </div>)
+            }
+          </Motion>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(PinnedAccountsEditor));
diff --git a/app/javascript/flavours/blobfox/features/pinned_statuses/index.jsx b/app/javascript/flavours/blobfox/features/pinned_statuses/index.jsx
new file mode 100644
index 00000000000000..287b5be16235c8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/pinned_statuses/index.jsx
@@ -0,0 +1,70 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { getStatusList } from 'flavours/blobfox/selectors';
+
+import { fetchPinnedStatuses } from '../../actions/pin_statuses';
+import ColumnBackButtonSlim from '../../components/column_back_button_slim';
+import StatusList from '../../components/status_list';
+import Column from '../ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'column.pins', defaultMessage: 'Pinned post' },
+});
+
+const mapStateToProps = state => ({
+  statusIds: getStatusList(state, 'pins'),
+  hasMore: !!state.getIn(['status_lists', 'pins', 'next']),
+});
+
+class PinnedStatuses extends ImmutablePureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    statusIds: ImmutablePropTypes.list.isRequired,
+    intl: PropTypes.object.isRequired,
+    hasMore: PropTypes.bool.isRequired,
+    multiColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    this.props.dispatch(fetchPinnedStatuses());
+  }
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  render () {
+    const { intl, statusIds, hasMore, multiColumn } = this.props;
+
+    return (
+      <Column bindToDocument={!multiColumn} icon='thumb-tack' heading={intl.formatMessage(messages.heading)} ref={this.setRef}>
+        <ColumnBackButtonSlim />
+        <StatusList
+          statusIds={statusIds}
+          scrollKey='pinned_statuses'
+          hasMore={hasMore}
+          bindToDocument={!multiColumn}
+        />
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(PinnedStatuses));
diff --git a/app/javascript/flavours/blobfox/features/privacy_policy/index.jsx b/app/javascript/flavours/blobfox/features/privacy_policy/index.jsx
new file mode 100644
index 00000000000000..888037e46ef2a7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/privacy_policy/index.jsx
@@ -0,0 +1,65 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, FormattedDate, injectIntl, defineMessages } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import api from 'flavours/blobfox/api';
+import Column from 'flavours/blobfox/components/column';
+import { Skeleton } from 'flavours/blobfox/components/skeleton';
+
+const messages = defineMessages({
+  title: { id: 'privacy_policy.title', defaultMessage: 'Privacy Policy' },
+});
+
+class PrivacyPolicy extends PureComponent {
+
+  static propTypes = {
+    intl: PropTypes.object,
+    multiColumn: PropTypes.bool,
+  };
+
+  state = {
+    content: null,
+    lastUpdated: null,
+    isLoading: true,
+  };
+
+  componentDidMount () {
+    api().get('/api/v1/instance/privacy_policy').then(({ data }) => {
+      this.setState({ content: data.content, lastUpdated: data.updated_at, isLoading: false });
+    }).catch(() => {
+      this.setState({ isLoading: false });
+    });
+  }
+
+  render () {
+    const { intl, multiColumn } = this.props;
+    const { isLoading, content, lastUpdated } = this.state;
+
+    return (
+      <Column bindToDocument={!multiColumn} label={intl.formatMessage(messages.title)}>
+        <div className='scrollable privacy-policy'>
+          <div className='column-title'>
+            <h3><FormattedMessage id='privacy_policy.title' defaultMessage='Privacy Policy' /></h3>
+            <p><FormattedMessage id='privacy_policy.last_updated' defaultMessage='Last updated {date}' values={{ date: isLoading ? <Skeleton width='10ch' /> : <FormattedDate value={lastUpdated} year='numeric' month='short' day='2-digit' /> }} /></p>
+          </div>
+
+          <div
+            className='privacy-policy__body prose'
+            dangerouslySetInnerHTML={{ __html: content }}
+          />
+        </div>
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='all' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default injectIntl(PrivacyPolicy);
diff --git a/app/javascript/flavours/blobfox/features/public_timeline/components/column_settings.jsx b/app/javascript/flavours/blobfox/features/public_timeline/components/column_settings.jsx
new file mode 100644
index 00000000000000..82684c83685439
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/public_timeline/components/column_settings.jsx
@@ -0,0 +1,46 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import SettingText from '../../../components/setting_text';
+import SettingToggle from '../../notifications/components/setting_toggle';
+
+const messages = defineMessages({
+  filter_regex: { id: 'home.column_settings.filter_regex', defaultMessage: 'Filter out by regular expressions' },
+});
+
+class ColumnSettings extends PureComponent {
+
+  static propTypes = {
+    settings: ImmutablePropTypes.map.isRequired,
+    onChange: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    columnId: PropTypes.string,
+  };
+
+  render () {
+    const { settings, onChange, intl } = this.props;
+
+    return (
+      <div>
+        <div className='column-settings__row'>
+          <SettingToggle settings={settings} settingPath={['other', 'onlyMedia']} onChange={onChange} label={<FormattedMessage id='community.column_settings.media_only' defaultMessage='Media only' />} />
+          <SettingToggle settings={settings} settingPath={['other', 'onlyRemote']} onChange={onChange} label={<FormattedMessage id='community.column_settings.remote_only' defaultMessage='Remote only' />} />
+          {!settings.getIn(['other', 'onlyRemote']) && <SettingToggle settings={settings} settingPath={['other', 'allowLocalOnly']} onChange={onChange} label={<FormattedMessage id='community.column_settings.allow_local_only' defaultMessage='Show local-only toots' />} />}
+        </div>
+
+        <span className='column-settings__section'><FormattedMessage id='home.column_settings.advanced' defaultMessage='Advanced' /></span>
+
+        <div className='column-settings__row'>
+          <SettingText settings={settings} settingPath={['regex', 'body']} onChange={onChange} label={intl.formatMessage(messages.filter_regex)} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/public_timeline/containers/column_settings_container.js b/app/javascript/flavours/blobfox/features/public_timeline/containers/column_settings_container.js
new file mode 100644
index 00000000000000..6476d51ffbf25e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/public_timeline/containers/column_settings_container.js
@@ -0,0 +1,29 @@
+import { connect } from 'react-redux';
+
+import { changeColumnParams } from '../../../actions/columns';
+import { changeSetting } from '../../../actions/settings';
+import ColumnSettings from '../components/column_settings';
+
+const mapStateToProps = (state, { columnId }) => {
+  const uuid = columnId;
+  const columns = state.getIn(['settings', 'columns']);
+  const index = columns.findIndex(c => c.get('uuid') === uuid);
+
+  return {
+    settings: (uuid && index >= 0) ? columns.get(index).get('params') : state.getIn(['settings', 'public']),
+  };
+};
+
+const mapDispatchToProps = (dispatch, { columnId }) => {
+  return {
+    onChange (key, checked) {
+      if (columnId) {
+        dispatch(changeColumnParams(columnId, key, checked));
+      } else {
+        dispatch(changeSetting(['public', ...key], checked));
+      }
+    },
+  };
+};
+
+export default connect(mapStateToProps, mapDispatchToProps)(ColumnSettings);
diff --git a/app/javascript/flavours/blobfox/features/public_timeline/index.jsx b/app/javascript/flavours/blobfox/features/public_timeline/index.jsx
new file mode 100644
index 00000000000000..b5e750851c73b5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/public_timeline/index.jsx
@@ -0,0 +1,171 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import { connect } from 'react-redux';
+
+import { DismissableBanner } from 'flavours/blobfox/components/dismissable_banner';
+import { domain } from 'flavours/blobfox/initial_state';
+
+import { addColumn, removeColumn, moveColumn } from '../../actions/columns';
+import { connectPublicStream } from '../../actions/streaming';
+import { expandPublicTimeline } from '../../actions/timelines';
+import Column from '../../components/column';
+import ColumnHeader from '../../components/column_header';
+import StatusListContainer from '../ui/containers/status_list_container';
+
+import ColumnSettingsContainer from './containers/column_settings_container';
+
+const messages = defineMessages({
+  title: { id: 'column.public', defaultMessage: 'Federated timeline' },
+});
+
+const mapStateToProps = (state, { columnId }) => {
+  const uuid = columnId;
+  const columns = state.getIn(['settings', 'columns']);
+  const index = columns.findIndex(c => c.get('uuid') === uuid);
+  const onlyMedia = (columnId && index >= 0) ? columns.get(index).getIn(['params', 'other', 'onlyMedia']) : state.getIn(['settings', 'public', 'other', 'onlyMedia']);
+  const onlyRemote = (columnId && index >= 0) ? columns.get(index).getIn(['params', 'other', 'onlyRemote']) : state.getIn(['settings', 'public', 'other', 'onlyRemote']);
+  const allowLocalOnly = (columnId && index >= 0) ? columns.get(index).getIn(['params', 'other', 'allowLocalOnly']) : state.getIn(['settings', 'public', 'other', 'allowLocalOnly']);
+  const regex = (columnId && index >= 0) ? columns.get(index).getIn(['params', 'regex', 'body']) : state.getIn(['settings', 'public', 'regex', 'body']);
+  const timelineState = state.getIn(['timelines', `public${onlyRemote ? ':remote' : allowLocalOnly ? ':allow_local_only' : ''}${onlyMedia ? ':media' : ''}`]);
+
+  return {
+    hasUnread: !!timelineState && timelineState.get('unread') > 0,
+    onlyMedia,
+    onlyRemote,
+    allowLocalOnly,
+    regex,
+  };
+};
+
+class PublicTimeline extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static defaultProps = {
+    onlyMedia: false,
+  };
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    columnId: PropTypes.string,
+    multiColumn: PropTypes.bool,
+    hasUnread: PropTypes.bool,
+    onlyMedia: PropTypes.bool,
+    onlyRemote: PropTypes.bool,
+    allowLocalOnly: PropTypes.bool,
+    regex: PropTypes.string,
+  };
+
+  handlePin = () => {
+    const { columnId, dispatch, onlyMedia, onlyRemote, allowLocalOnly } = this.props;
+
+    if (columnId) {
+      dispatch(removeColumn(columnId));
+    } else {
+      dispatch(addColumn(onlyRemote ? 'REMOTE' : 'PUBLIC', { other: { onlyMedia, onlyRemote, allowLocalOnly } }));
+    }
+  };
+
+  handleMove = (dir) => {
+    const { columnId, dispatch } = this.props;
+    dispatch(moveColumn(columnId, dir));
+  };
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  componentDidMount () {
+    const { dispatch, onlyMedia, onlyRemote, allowLocalOnly } = this.props;
+    const { signedIn } = this.context.identity;
+
+    dispatch(expandPublicTimeline({ onlyMedia, onlyRemote, allowLocalOnly }));
+    if (signedIn) {
+      this.disconnect = dispatch(connectPublicStream({ onlyMedia, onlyRemote, allowLocalOnly }));
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { signedIn } = this.context.identity;
+
+    if (prevProps.onlyMedia !== this.props.onlyMedia || prevProps.onlyRemote !== this.props.onlyRemote || prevProps.allowLocalOnly !== this.props.allowLocalOnly) {
+      const { dispatch, onlyMedia, onlyRemote, allowLocalOnly } = this.props;
+
+      if (this.disconnect) {
+        this.disconnect();
+      }
+
+      dispatch(expandPublicTimeline({ onlyMedia, onlyRemote, allowLocalOnly }));
+
+      if (signedIn) {
+        this.disconnect = dispatch(connectPublicStream({ onlyMedia, onlyRemote, allowLocalOnly }));
+      }
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.disconnect) {
+      this.disconnect();
+      this.disconnect = null;
+    }
+  }
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleLoadMore = maxId => {
+    const { dispatch, onlyMedia, onlyRemote, allowLocalOnly } = this.props;
+
+    dispatch(expandPublicTimeline({ maxId, onlyMedia, onlyRemote, allowLocalOnly }));
+  };
+
+  render () {
+    const { intl, columnId, hasUnread, multiColumn, onlyMedia, onlyRemote, allowLocalOnly } = this.props;
+    const pinned = !!columnId;
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setRef} label={intl.formatMessage(messages.title)}>
+        <ColumnHeader
+          icon='globe'
+          active={hasUnread}
+          title={intl.formatMessage(messages.title)}
+          onPin={this.handlePin}
+          onMove={this.handleMove}
+          onClick={this.handleHeaderClick}
+          pinned={pinned}
+          multiColumn={multiColumn}
+        >
+          <ColumnSettingsContainer columnId={columnId} />
+        </ColumnHeader>
+
+        <StatusListContainer
+          prepend={<DismissableBanner id='public_timeline'><FormattedMessage id='dismissable_banner.public_timeline' defaultMessage='These are the most recent public posts from people on the social web that people on {domain} follow.' values={{ domain }} /></DismissableBanner>}
+          timelineId={`public${onlyRemote ? ':remote' : (allowLocalOnly ? ':allow_local_only' : '')}${onlyMedia ? ':media' : ''}`}
+          onLoadMore={this.handleLoadMore}
+          trackScroll={!pinned}
+          scrollKey={`public_timeline-${columnId}`}
+          emptyMessage={<FormattedMessage id='empty_column.public' defaultMessage='There is nothing here! Write something publicly, or manually follow users from other servers to fill it up' />}
+          bindToDocument={!multiColumn}
+          regex={this.props.regex}
+        />
+
+        <Helmet>
+          <title>{intl.formatMessage(messages.title)}</title>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(PublicTimeline));
diff --git a/app/javascript/flavours/blobfox/features/reblogs/index.jsx b/app/javascript/flavours/blobfox/features/reblogs/index.jsx
new file mode 100644
index 00000000000000..d617b4de829715
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/reblogs/index.jsx
@@ -0,0 +1,115 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { Helmet } from 'react-helmet';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { debounce } from 'lodash';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+import { fetchReblogs, expandReblogs } from '../../actions/interactions';
+import ColumnHeader from '../../components/column_header';
+import { LoadingIndicator } from '../../components/loading_indicator';
+import ScrollableList from '../../components/scrollable_list';
+import AccountContainer from '../../containers/account_container';
+import Column from '../ui/components/column';
+
+const messages = defineMessages({
+  heading: { id: 'column.reblogged_by', defaultMessage: 'Boosted by' },
+  refresh: { id: 'refresh', defaultMessage: 'Refresh' },
+});
+
+const mapStateToProps = (state, props) => ({
+  accountIds: state.getIn(['user_lists', 'reblogged_by', props.params.statusId, 'items']),
+  hasMore: !!state.getIn(['user_lists', 'reblogged_by', props.params.statusId, 'next']),
+  isLoading: state.getIn(['user_lists', 'reblogged_by', props.params.statusId, 'isLoading'], true),
+});
+
+class Reblogs extends ImmutablePureComponent {
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    accountIds: ImmutablePropTypes.list,
+    hasMore: PropTypes.bool,
+    isLoading: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  UNSAFE_componentWillMount () {
+    if (!this.props.accountIds) {
+      this.props.dispatch(fetchReblogs(this.props.params.statusId));
+    }
+  }
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  setRef = c => {
+    this.column = c;
+  };
+
+  handleRefresh = () => {
+    this.props.dispatch(fetchReblogs(this.props.params.statusId));
+  };
+
+  handleLoadMore = debounce(() => {
+    this.props.dispatch(expandReblogs(this.props.params.statusId));
+  }, 300, { leading: true });
+
+  render () {
+    const { intl, accountIds, hasMore, isLoading, multiColumn } = this.props;
+
+    if (!accountIds) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    const emptyMessage = <FormattedMessage id='status.reblogs.empty' defaultMessage='No one has boosted this post yet. When someone does, they will show up here.' />;
+
+    return (
+      <Column ref={this.setRef}>
+        <ColumnHeader
+          icon='retweet'
+          title={intl.formatMessage(messages.heading)}
+          onClick={this.handleHeaderClick}
+          showBackButton
+          multiColumn={multiColumn}
+          extraButton={(
+            <button className='column-header__button' title={intl.formatMessage(messages.refresh)} aria-label={intl.formatMessage(messages.refresh)} onClick={this.handleRefresh}><Icon id='refresh' /></button>
+          )}
+        />
+
+        <ScrollableList
+          scrollKey='reblogs'
+          onLoadMore={this.handleLoadMore}
+          hasMore={hasMore}
+          isLoading={isLoading}
+          emptyMessage={emptyMessage}
+          bindToDocument={!multiColumn}
+        >
+          {accountIds.map(id =>
+            <AccountContainer key={id} id={id} withNote={false} />,
+          )}
+        </ScrollableList>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Reblogs));
diff --git a/app/javascript/flavours/blobfox/features/report/category.jsx b/app/javascript/flavours/blobfox/features/report/category.jsx
new file mode 100644
index 00000000000000..ee9382667fd96c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/category.jsx
@@ -0,0 +1,108 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { Button } from 'flavours/blobfox/components/button';
+
+import Option from './components/option';
+
+const messages = defineMessages({
+  dislike: { id: 'report.reasons.dislike', defaultMessage: 'I don\'t like it' },
+  dislike_description: { id: 'report.reasons.dislike_description', defaultMessage: 'It is not something you want to see' },
+  spam: { id: 'report.reasons.spam', defaultMessage: 'It\'s spam' },
+  spam_description: { id: 'report.reasons.spam_description', defaultMessage: 'Malicious links, fake engagement, or repetitive replies' },
+  violation: { id: 'report.reasons.violation', defaultMessage: 'It violates server rules' },
+  violation_description: { id: 'report.reasons.violation_description', defaultMessage: 'You are aware that it breaks specific rules' },
+  other: { id: 'report.reasons.other', defaultMessage: 'It\'s something else' },
+  other_description: { id: 'report.reasons.other_description', defaultMessage: 'The issue does not fit into other categories' },
+  status: { id: 'report.category.title_status', defaultMessage: 'post' },
+  account: { id: 'report.category.title_account', defaultMessage: 'profile' },
+});
+
+const mapStateToProps = state => ({
+  rules: state.getIn(['server', 'server', 'rules'], ImmutableList()),
+});
+
+class Category extends PureComponent {
+
+  static propTypes = {
+    onNextStep: PropTypes.func.isRequired,
+    rules: ImmutablePropTypes.list,
+    category: PropTypes.string,
+    onChangeCategory: PropTypes.func.isRequired,
+    startedFrom: PropTypes.oneOf(['status', 'account']),
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleNextClick = () => {
+    const { onNextStep, category } = this.props;
+
+    switch(category) {
+    case 'dislike':
+      onNextStep('thanks');
+      break;
+    case 'violation':
+      onNextStep('rules');
+      break;
+    default:
+      onNextStep('statuses');
+      break;
+    }
+  };
+
+  handleCategoryToggle = (value, checked) => {
+    const { onChangeCategory } = this.props;
+
+    if (checked) {
+      onChangeCategory(value);
+    }
+  };
+
+  render () {
+    const { category, startedFrom, rules, intl } = this.props;
+
+    const options = rules.size > 0 ? [
+      'spam',
+      'violation',
+      'other',
+    ] : [
+      'spam',
+      'other',
+    ];
+
+    return (
+      <>
+        <h3 className='report-dialog-modal__title'><FormattedMessage id='report.category.title' defaultMessage="Tell us what's going on with this {type}" values={{ type: intl.formatMessage(messages[startedFrom]) }} /></h3>
+        <p className='report-dialog-modal__lead'><FormattedMessage id='report.category.subtitle' defaultMessage='Choose the best match' /></p>
+
+        <div>
+          {options.map(item => (
+            <Option
+              key={item}
+              name='category'
+              value={item}
+              checked={category === item}
+              onToggle={this.handleCategoryToggle}
+              label={intl.formatMessage(messages[item])}
+              description={intl.formatMessage(messages[`${item}_description`])}
+            />
+          ))}
+        </div>
+
+        <div className='flex-spacer' />
+
+        <div className='report-dialog-modal__actions'>
+          <Button onClick={this.handleNextClick} disabled={category === null}><FormattedMessage id='report.next' defaultMessage='Next' /></Button>
+        </div>
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(Category));
diff --git a/app/javascript/flavours/blobfox/features/report/comment.jsx b/app/javascript/flavours/blobfox/features/report/comment.jsx
new file mode 100644
index 00000000000000..5c7c835704e623
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/comment.jsx
@@ -0,0 +1,121 @@
+import PropTypes from 'prop-types';
+import { useCallback, useEffect, useRef } from 'react';
+
+import { useIntl, defineMessages, FormattedMessage } from 'react-intl';
+
+import { OrderedSet, List as ImmutableList } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { shallowEqual } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import Toggle from 'react-toggle';
+
+import { fetchAccount } from 'flavours/blobfox/actions/accounts';
+import { Button } from 'flavours/blobfox/components/button';
+import { useAppDispatch, useAppSelector } from 'flavours/blobfox/store';
+
+const messages = defineMessages({
+  placeholder: { id: 'report.placeholder', defaultMessage: 'Type or paste additional comments' },
+});
+
+const selectRepliedToAccountIds = createSelector(
+  [
+    (state) => state.get('statuses'),
+    (_, statusIds) => statusIds,
+  ],
+  (statusesMap, statusIds) => statusIds.map((statusId) => statusesMap.getIn([statusId, 'in_reply_to_account_id'])),
+  {
+    resultEqualityCheck: shallowEqual,
+  }
+);
+
+const Comment = ({ comment, domain, statusIds, isRemote, isSubmitting, selectedDomains, onSubmit, onChangeComment, onToggleDomain }) => {
+  const intl = useIntl();
+
+  const dispatch = useAppDispatch();
+  const loadedRef = useRef(false);
+
+  const handleClick = useCallback(() => onSubmit(), [onSubmit]);
+  const handleChange = useCallback((e) => onChangeComment(e.target.value), [onChangeComment]);
+  const handleToggleDomain = useCallback(e => onToggleDomain(e.target.value, e.target.checked), [onToggleDomain]);
+
+  const handleKeyDown = useCallback((e) => {
+    if (e.keyCode === 13 && (e.ctrlKey || e.metaKey)) {
+      handleClick();
+    }
+  }, [handleClick]);
+
+  // Memoize accountIds since we don't want it to trigger `useEffect` on each render
+  const accountIds = useAppSelector((state) => domain ? selectRepliedToAccountIds(state, statusIds) : ImmutableList());
+
+  // While we could memoize `availableDomains`, it is pretty inexpensive to recompute
+  const accountsMap = useAppSelector((state) => state.get('accounts'));
+  const availableDomains = domain ? OrderedSet([domain]).union(accountIds.map((accountId) => accountsMap.getIn([accountId, 'acct'], '').split('@')[1]).filter(domain => !!domain)) : OrderedSet();
+
+  useEffect(() => {
+    if (loadedRef.current) {
+      return;
+    }
+
+    loadedRef.current = true;
+
+    // First, pre-select known domains
+    availableDomains.forEach((domain) => {
+      onToggleDomain(domain, true);
+    });
+
+    // Then, fetch missing replied-to accounts
+    const unknownAccounts = OrderedSet(accountIds.filter(accountId => accountId && !accountsMap.has(accountId)));
+    unknownAccounts.forEach((accountId) => {
+      dispatch(fetchAccount(accountId));
+    });
+  });
+
+  return (
+    <>
+      <h3 className='report-dialog-modal__title'><FormattedMessage id='report.comment.title' defaultMessage='Is there anything else you think we should know?' /></h3>
+
+      <textarea
+        className='report-dialog-modal__textarea'
+        placeholder={intl.formatMessage(messages.placeholder)}
+        value={comment}
+        onChange={handleChange}
+        onKeyDown={handleKeyDown}
+        disabled={isSubmitting}
+      />
+
+      {isRemote && (
+        <>
+          <p className='report-dialog-modal__lead'><FormattedMessage id='report.forward_hint' defaultMessage='The account is from another server. Send an anonymized copy of the report there as well?' /></p>
+
+          { availableDomains.map((domain) => (
+            <label className='report-dialog-modal__toggle' key={`toggle-${domain}`}>
+              <Toggle checked={selectedDomains.includes(domain)} disabled={isSubmitting} onChange={handleToggleDomain} value={domain} />
+              <FormattedMessage id='report.forward' defaultMessage='Forward to {target}' values={{ target: domain }} />
+            </label>
+          ))}
+        </>
+      )}
+
+      <div className='flex-spacer' />
+
+      <div className='report-dialog-modal__actions'>
+        <Button onClick={handleClick} disabled={isSubmitting}><FormattedMessage id='report.submit' defaultMessage='Submit report' /></Button>
+      </div>
+    </>
+  );
+};
+
+Comment.propTypes = {
+  comment: PropTypes.string.isRequired,
+  domain: PropTypes.string,
+  statusIds: ImmutablePropTypes.list.isRequired,
+  isRemote: PropTypes.bool,
+  isSubmitting: PropTypes.bool,
+  selectedDomains: ImmutablePropTypes.set.isRequired,
+  onSubmit: PropTypes.func.isRequired,
+  onChangeComment: PropTypes.func.isRequired,
+  onToggleDomain: PropTypes.func.isRequired,
+};
+
+export default Comment;
diff --git a/app/javascript/flavours/blobfox/features/report/components/option.jsx b/app/javascript/flavours/blobfox/features/report/components/option.jsx
new file mode 100644
index 00000000000000..873fb0d8274892
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/components/option.jsx
@@ -0,0 +1,62 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import { Check } from 'flavours/blobfox/components/check';
+
+export default class Option extends PureComponent {
+
+  static propTypes = {
+    name: PropTypes.string.isRequired,
+    value: PropTypes.string.isRequired,
+    checked: PropTypes.bool,
+    label: PropTypes.node,
+    description: PropTypes.node,
+    onToggle: PropTypes.func,
+    multiple: PropTypes.bool,
+    labelComponent: PropTypes.node,
+  };
+
+  handleKeyPress = e => {
+    const { value, checked, onToggle } = this.props;
+
+    if (e.key === 'Enter' || e.key === ' ') {
+      e.stopPropagation();
+      e.preventDefault();
+      onToggle(value, !checked);
+    }
+  };
+
+  handleChange = e => {
+    const { value, onToggle } = this.props;
+    onToggle(value, e.target.checked);
+  };
+
+  render () {
+    const { name, value, checked, label, labelComponent, description, multiple } = this.props;
+
+    return (
+      <label className='dialog-option poll__option selectable'>
+        <input type={multiple ? 'checkbox' : 'radio'} name={name} value={value} checked={checked} onChange={this.handleChange} />
+
+        <span
+          className={classNames('poll__input', { active: checked, checkbox: multiple })}
+          tabIndex={0}
+          role='radio'
+          onKeyPress={this.handleKeyPress}
+          aria-checked={checked}
+          aria-label={label}
+        >{checked && <Check />}</span>
+
+        {labelComponent ? labelComponent : (
+          <span className='poll__option__text'>
+            <strong>{label}</strong>
+            {description}
+          </span>
+        )}
+      </label>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/report/components/status_check_box.jsx b/app/javascript/flavours/blobfox/features/report/components/status_check_box.jsx
new file mode 100644
index 00000000000000..d7c5cd2c1e3299
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/components/status_check_box.jsx
@@ -0,0 +1,67 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import MediaAttachments from 'flavours/blobfox/components/media_attachments';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+import StatusContent from 'flavours/blobfox/components/status_content';
+import VisibilityIcon from 'flavours/blobfox/components/status_visibility_icon';
+
+import Option from './option';
+
+class StatusCheckBox extends PureComponent {
+
+  static propTypes = {
+    id: PropTypes.string.isRequired,
+    status: ImmutablePropTypes.map.isRequired,
+    checked: PropTypes.bool,
+    onToggle: PropTypes.func.isRequired,
+  };
+
+  handleStatusesToggle = (value, checked) => {
+    const { onToggle } = this.props;
+    onToggle(value, checked);
+  };
+
+  render () {
+    const { status, checked } = this.props;
+
+    if (status.get('reblog')) {
+      return null;
+    }
+
+    const labelComponent = (
+      <div className='status-check-box__status poll__option__text'>
+        <div className='detailed-status__display-name'>
+          <div className='detailed-status__display-avatar'>
+            <Avatar account={status.get('account')} size={46} />
+          </div>
+
+          <div>
+            <DisplayName account={status.get('account')} /> · <VisibilityIcon visibility={status.get('visibility')} /><RelativeTimestamp timestamp={status.get('created_at')} />
+          </div>
+        </div>
+
+        <StatusContent status={status} media={<MediaAttachments status={status} visible={false} />} />
+      </div>
+    );
+
+    return (
+      <Option
+        name='status_ids'
+        value={status.get('id')}
+        checked={checked}
+        onToggle={this.handleStatusesToggle}
+        label={status.get('search_index')}
+        labelComponent={labelComponent}
+        multiple
+      />
+    );
+  }
+
+}
+
+export default StatusCheckBox;
diff --git a/app/javascript/flavours/blobfox/features/report/containers/status_check_box_container.js b/app/javascript/flavours/blobfox/features/report/containers/status_check_box_container.js
new file mode 100644
index 00000000000000..123b6f127a6ea4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/containers/status_check_box_container.js
@@ -0,0 +1,17 @@
+import { connect } from 'react-redux';
+
+import { makeGetStatus } from 'flavours/blobfox/selectors';
+
+import StatusCheckBox from '../components/status_check_box';
+
+const makeMapStateToProps = () => {
+  const getStatus = makeGetStatus();
+
+  const mapStateToProps = (state, { id }) => ({
+    status: getStatus(state, { id }),
+  });
+
+  return mapStateToProps;
+};
+
+export default connect(makeMapStateToProps)(StatusCheckBox);
diff --git a/app/javascript/flavours/blobfox/features/report/rules.jsx b/app/javascript/flavours/blobfox/features/report/rules.jsx
new file mode 100644
index 00000000000000..ffb0e29e7feb83
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/rules.jsx
@@ -0,0 +1,69 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { Button } from 'flavours/blobfox/components/button';
+
+import Option from './components/option';
+
+const mapStateToProps = state => ({
+  rules: state.getIn(['server', 'server', 'rules']),
+});
+
+class Rules extends PureComponent {
+
+  static propTypes = {
+    onNextStep: PropTypes.func.isRequired,
+    rules: ImmutablePropTypes.list,
+    selectedRuleIds: ImmutablePropTypes.set.isRequired,
+    onToggle: PropTypes.func.isRequired,
+  };
+
+  handleNextClick = () => {
+    const { onNextStep } = this.props;
+    onNextStep('statuses');
+  };
+
+  handleRulesToggle = (value, checked) => {
+    const { onToggle } = this.props;
+    onToggle(value, checked);
+  };
+
+  render () {
+    const { rules, selectedRuleIds } = this.props;
+
+    return (
+      <>
+        <h3 className='report-dialog-modal__title'><FormattedMessage id='report.rules.title' defaultMessage='Which rules are being violated?' /></h3>
+        <p className='report-dialog-modal__lead'><FormattedMessage id='report.rules.subtitle' defaultMessage='Select all that apply' /></p>
+
+        <div>
+          {rules.map(item => (
+            <Option
+              key={item.get('id')}
+              name='rule_ids'
+              value={item.get('id')}
+              checked={selectedRuleIds.includes(item.get('id'))}
+              onToggle={this.handleRulesToggle}
+              label={item.get('text')}
+              multiple
+            />
+          ))}
+        </div>
+
+        <div className='flex-spacer' />
+
+        <div className='report-dialog-modal__actions'>
+          <Button onClick={this.handleNextClick} disabled={selectedRuleIds.size < 1}><FormattedMessage id='report.next' defaultMessage='Next' /></Button>
+        </div>
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(Rules);
diff --git a/app/javascript/flavours/blobfox/features/report/statuses.jsx b/app/javascript/flavours/blobfox/features/report/statuses.jsx
new file mode 100644
index 00000000000000..5bcf1f795c373a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/statuses.jsx
@@ -0,0 +1,65 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import { OrderedSet } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import { Button } from 'flavours/blobfox/components/button';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import StatusCheckBox from 'flavours/blobfox/features/report/containers/status_check_box_container';
+
+const mapStateToProps = (state, { accountId }) => ({
+  availableStatusIds: OrderedSet(state.getIn(['timelines', `account:${accountId}:with_replies`, 'items'])),
+  isLoading: state.getIn(['timelines', `account:${accountId}:with_replies`, 'isLoading']),
+});
+
+class Statuses extends PureComponent {
+
+  static propTypes = {
+    onNextStep: PropTypes.func.isRequired,
+    accountId: PropTypes.string.isRequired,
+    availableStatusIds: ImmutablePropTypes.set.isRequired,
+    selectedStatusIds: ImmutablePropTypes.set.isRequired,
+    isLoading: PropTypes.bool,
+    onToggle: PropTypes.func.isRequired,
+  };
+
+  handleNextClick = () => {
+    const { onNextStep } = this.props;
+    onNextStep('comment');
+  };
+
+  render () {
+    const { availableStatusIds, selectedStatusIds, onToggle, isLoading } = this.props;
+
+    return (
+      <>
+        <h3 className='report-dialog-modal__title'><FormattedMessage id='report.statuses.title' defaultMessage='Are there any posts that back up this report?' /></h3>
+        <p className='report-dialog-modal__lead'><FormattedMessage id='report.statuses.subtitle' defaultMessage='Select all that apply' /></p>
+
+        <div className='report-dialog-modal__statuses'>
+          {isLoading ? <LoadingIndicator /> : availableStatusIds.union(selectedStatusIds).map(statusId => (
+            <StatusCheckBox
+              id={statusId}
+              key={statusId}
+              checked={selectedStatusIds.includes(statusId)}
+              onToggle={onToggle}
+            />
+          ))}
+        </div>
+
+        <div className='flex-spacer' />
+
+        <div className='report-dialog-modal__actions'>
+          <Button onClick={this.handleNextClick}><FormattedMessage id='report.next' defaultMessage='Next' /></Button>
+        </div>
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(Statuses);
diff --git a/app/javascript/flavours/blobfox/features/report/thanks.jsx b/app/javascript/flavours/blobfox/features/report/thanks.jsx
new file mode 100644
index 00000000000000..aee3e9980134fa
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/report/thanks.jsx
@@ -0,0 +1,88 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import {
+  unfollowAccount,
+  muteAccount,
+  blockAccount,
+} from 'flavours/blobfox/actions/accounts';
+import { Button } from 'flavours/blobfox/components/button';
+
+const mapStateToProps = () => ({});
+
+class Thanks extends PureComponent {
+
+  static propTypes = {
+    submitted: PropTypes.bool,
+    onClose: PropTypes.func.isRequired,
+    account: ImmutablePropTypes.record.isRequired,
+    dispatch: PropTypes.func.isRequired,
+  };
+
+  handleCloseClick = () => {
+    const { onClose } = this.props;
+    onClose();
+  };
+
+  handleUnfollowClick = () => {
+    const { dispatch, account, onClose } = this.props;
+    dispatch(unfollowAccount(account.get('id')));
+    onClose();
+  };
+
+  handleMuteClick = () => {
+    const { dispatch, account, onClose } = this.props;
+    dispatch(muteAccount(account.get('id')));
+    onClose();
+  };
+
+  handleBlockClick = () => {
+    const { dispatch, account, onClose } = this.props;
+    dispatch(blockAccount(account.get('id')));
+    onClose();
+  };
+
+  render () {
+    const { account, submitted } = this.props;
+
+    return (
+      <>
+        <h3 className='report-dialog-modal__title'>{submitted ? <FormattedMessage id='report.thanks.title_actionable' defaultMessage="Thanks for reporting, we'll look into this." /> : <FormattedMessage id='report.thanks.title' defaultMessage="Don't want to see this?" />}</h3>
+        <p className='report-dialog-modal__lead'>{submitted ? <FormattedMessage id='report.thanks.take_action_actionable' defaultMessage='While we review this, you can take action against @{name}:' values={{ name: account.get('username') }} /> : <FormattedMessage id='report.thanks.take_action' defaultMessage='Here are your options for controlling what you see on Mastodon:' />}</p>
+
+        {account.getIn(['relationship', 'following']) && (
+          <>
+            <h4 className='report-dialog-modal__subtitle'><FormattedMessage id='report.unfollow' defaultMessage='Unfollow @{name}' values={{ name: account.get('username') }} /></h4>
+            <p className='report-dialog-modal__lead'><FormattedMessage id='report.unfollow_explanation' defaultMessage='You are following this account. To not see their posts in your home feed anymore, unfollow them.' /></p>
+            <Button secondary onClick={this.handleUnfollowClick}><FormattedMessage id='account.unfollow' defaultMessage='Unfollow' /></Button>
+            <hr />
+          </>
+        )}
+
+        <h4 className='report-dialog-modal__subtitle'><FormattedMessage id='account.mute' defaultMessage='Mute @{name}' values={{ name: account.get('username') }} /></h4>
+        <p className='report-dialog-modal__lead'><FormattedMessage id='report.mute_explanation' defaultMessage='You will not see their posts. They can still follow you and see your posts and will not know that they are muted.' /></p>
+        <Button secondary onClick={this.handleMuteClick}>{!account.getIn(['relationship', 'muting']) ? <FormattedMessage id='report.mute' defaultMessage='Mute' /> : <FormattedMessage id='account.muted' defaultMessage='Muted' />}</Button>
+
+        <hr />
+
+        <h4 className='report-dialog-modal__subtitle'><FormattedMessage id='account.block' defaultMessage='Block @{name}' values={{ name: account.get('username') }} /></h4>
+        <p className='report-dialog-modal__lead'><FormattedMessage id='report.block_explanation' defaultMessage='You will not see their posts. They will not be able to see your posts or follow you. They will be able to tell that they are blocked.' /></p>
+        <Button secondary onClick={this.handleBlockClick}>{!account.getIn(['relationship', 'blocking']) ? <FormattedMessage id='report.block' defaultMessage='Block' /> : <FormattedMessage id='account.blocked' defaultMessage='Blocked' />}</Button>
+
+        <div className='flex-spacer' />
+
+        <div className='report-dialog-modal__actions'>
+          <Button onClick={this.handleCloseClick}><FormattedMessage id='report.close' defaultMessage='Done' /></Button>
+        </div>
+      </>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(Thanks);
diff --git a/app/javascript/flavours/blobfox/features/standalone/compose/index.jsx b/app/javascript/flavours/blobfox/features/standalone/compose/index.jsx
new file mode 100644
index 00000000000000..c36e843f5ada65
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/standalone/compose/index.jsx
@@ -0,0 +1,21 @@
+import { PureComponent } from 'react';
+
+import ComposeFormContainer from '../../compose/containers/compose_form_container';
+import LoadingBarContainer from '../../ui/containers/loading_bar_container';
+import ModalContainer from '../../ui/containers/modal_container';
+import NotificationsContainer from '../../ui/containers/notifications_container';
+
+export default class Compose extends PureComponent {
+
+  render () {
+    return (
+      <div>
+        <ComposeFormContainer autoFocus />
+        <NotificationsContainer />
+        <ModalContainer />
+        <LoadingBarContainer className='loading-bar' />
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/status/components/action_bar.jsx b/app/javascript/flavours/blobfox/features/status/components/action_bar.jsx
new file mode 100644
index 00000000000000..a71eea35e318ef
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/status/components/action_bar.jsx
@@ -0,0 +1,267 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { PERMISSION_MANAGE_USERS, PERMISSION_MANAGE_FEDERATION } from 'flavours/blobfox/permissions';
+import { accountAdminLink, statusAdminLink } from 'flavours/blobfox/utils/backend_links';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { IconButton } from '../../../components/icon_button';
+import DropdownMenuContainer from '../../../containers/dropdown_menu_container';
+import { me, maxReactions } from '../../../initial_state';
+import EmojiPickerDropdown from '../../compose/containers/emoji_picker_dropdown_container';
+
+const messages = defineMessages({
+  delete: { id: 'status.delete', defaultMessage: 'Delete' },
+  redraft: { id: 'status.redraft', defaultMessage: 'Delete & re-draft' },
+  edit: { id: 'status.edit', defaultMessage: 'Edit' },
+  direct: { id: 'status.direct', defaultMessage: 'Privately mention @{name}' },
+  mention: { id: 'status.mention', defaultMessage: 'Mention @{name}' },
+  reply: { id: 'status.reply', defaultMessage: 'Reply' },
+  reblog: { id: 'status.reblog', defaultMessage: 'Boost' },
+  reblog_private: { id: 'status.reblog_private', defaultMessage: 'Boost with original visibility' },
+  cancel_reblog_private: { id: 'status.cancel_reblog_private', defaultMessage: 'Unboost' },
+  cannot_reblog: { id: 'status.cannot_reblog', defaultMessage: 'This post cannot be boosted' },
+  favourite: { id: 'status.favourite', defaultMessage: 'Favorite' },
+  react: { id: 'status.react', defaultMessage: 'React' },
+  bookmark: { id: 'status.bookmark', defaultMessage: 'Bookmark' },
+  more: { id: 'status.more', defaultMessage: 'More' },
+  mute: { id: 'status.mute', defaultMessage: 'Mute @{name}' },
+  muteConversation: { id: 'status.mute_conversation', defaultMessage: 'Mute conversation' },
+  unmuteConversation: { id: 'status.unmute_conversation', defaultMessage: 'Unmute conversation' },
+  block: { id: 'status.block', defaultMessage: 'Block @{name}' },
+  report: { id: 'status.report', defaultMessage: 'Report @{name}' },
+  share: { id: 'status.share', defaultMessage: 'Share' },
+  pin: { id: 'status.pin', defaultMessage: 'Pin on profile' },
+  unpin: { id: 'status.unpin', defaultMessage: 'Unpin from profile' },
+  embed: { id: 'status.embed', defaultMessage: 'Embed' },
+  admin_account: { id: 'status.admin_account', defaultMessage: 'Open moderation interface for @{name}' },
+  admin_status: { id: 'status.admin_status', defaultMessage: 'Open this post in the moderation interface' },
+  admin_domain: { id: 'status.admin_domain', defaultMessage: 'Open moderation interface for {domain}' },
+  copy: { id: 'status.copy', defaultMessage: 'Copy link to post' },
+  openOriginalPage: { id: 'account.open_original_page', defaultMessage: 'Open original page' },
+});
+
+class ActionBar extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    onReply: PropTypes.func.isRequired,
+    onReblog: PropTypes.func.isRequired,
+    onFavourite: PropTypes.func.isRequired,
+    onReactionAdd: PropTypes.func.isRequired,
+    onBookmark: PropTypes.func.isRequired,
+    onDelete: PropTypes.func.isRequired,
+    onEdit: PropTypes.func.isRequired,
+    onDirect: PropTypes.func.isRequired,
+    onMention: PropTypes.func.isRequired,
+    onMute: PropTypes.func,
+    onBlock: PropTypes.func,
+    onMuteConversation: PropTypes.func,
+    onReport: PropTypes.func,
+    onPin: PropTypes.func,
+    onEmbed: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  handleReplyClick = () => {
+    this.props.onReply(this.props.status);
+  };
+
+  handleReblogClick = (e) => {
+    this.props.onReblog(this.props.status, e);
+  };
+
+  handleFavouriteClick = (e) => {
+    this.props.onFavourite(this.props.status, e);
+  };
+
+  handleEmojiPick = data => {
+    this.props.onReactionAdd(this.props.status.get('id'), data.native.replace(/:/g, ''), data.imageUrl);
+  };
+
+  handleBookmarkClick = (e) => {
+    this.props.onBookmark(this.props.status, e);
+  };
+
+  handleDeleteClick = () => {
+    this.props.onDelete(this.props.status, this.props.history);
+  };
+
+  handleRedraftClick = () => {
+    this.props.onDelete(this.props.status, this.props.history, true);
+  };
+
+  handleEditClick = () => {
+    this.props.onEdit(this.props.status, this.props.history);
+  };
+
+  handleDirectClick = () => {
+    this.props.onDirect(this.props.status.get('account'), this.props.history);
+  };
+
+  handleMentionClick = () => {
+    this.props.onMention(this.props.status.get('account'), this.props.history);
+  };
+
+  handleMuteClick = () => {
+    this.props.onMute(this.props.status.get('account'));
+  };
+
+  handleBlockClick = () => {
+    this.props.onBlock(this.props.status);
+  };
+
+  handleConversationMuteClick = () => {
+    this.props.onMuteConversation(this.props.status);
+  };
+
+  handleReport = () => {
+    this.props.onReport(this.props.status);
+  };
+
+  handlePinClick = () => {
+    this.props.onPin(this.props.status);
+  };
+
+  handleShare = () => {
+    navigator.share({
+      url: this.props.status.get('url'),
+    });
+  };
+
+  handleEmbed = () => {
+    this.props.onEmbed(this.props.status);
+  };
+
+  handleCopy = () => {
+    const url = this.props.status.get('url');
+    navigator.clipboard.writeText(url);
+  };
+
+  handleNoOp = () => {}; // hack for reaction add button
+
+  render () {
+    const { status, intl } = this.props;
+    const { signedIn, permissions } = this.context.identity;
+
+    const publicStatus       = ['public', 'unlisted'].includes(status.get('visibility'));
+    const pinnableStatus     = ['public', 'unlisted', 'private'].includes(status.get('visibility'));
+    const mutingConversation = status.get('muted');
+    const writtenByMe        = status.getIn(['account', 'id']) === me;
+    const isRemote           = status.getIn(['account', 'username']) !== status.getIn(['account', 'acct']);
+
+    let menu = [];
+
+    if (publicStatus && isRemote) {
+      menu.push({ text: intl.formatMessage(messages.openOriginalPage), href: status.get('url') });
+    }
+
+    menu.push({ text: intl.formatMessage(messages.copy), action: this.handleCopy });
+
+    if (publicStatus && 'share' in navigator) {
+      menu.push({ text: intl.formatMessage(messages.share), action: this.handleShare });
+    }
+
+    if (publicStatus && (signedIn || !isRemote)) {
+      menu.push({ text: intl.formatMessage(messages.embed), action: this.handleEmbed });
+    }
+
+    if (signedIn) {
+      menu.push(null);
+
+      if (writtenByMe) {
+        if (pinnableStatus) {
+          menu.push({ text: intl.formatMessage(status.get('pinned') ? messages.unpin : messages.pin), action: this.handlePinClick });
+          menu.push(null);
+        }
+
+        menu.push({ text: intl.formatMessage(mutingConversation ? messages.unmuteConversation : messages.muteConversation), action: this.handleConversationMuteClick });
+        menu.push(null);
+        menu.push({ text: intl.formatMessage(messages.edit), action: this.handleEditClick });
+        menu.push({ text: intl.formatMessage(messages.delete), action: this.handleDeleteClick, dangerous: true });
+        menu.push({ text: intl.formatMessage(messages.redraft), action: this.handleRedraftClick, dangerous: true });
+      } else {
+        menu.push({ text: intl.formatMessage(messages.mention, { name: status.getIn(['account', 'username']) }), action: this.handleMentionClick });
+        menu.push({ text: intl.formatMessage(messages.direct, { name: status.getIn(['account', 'username']) }), action: this.handleDirectClick });
+        menu.push(null);
+        menu.push({ text: intl.formatMessage(messages.mute, { name: status.getIn(['account', 'username']) }), action: this.handleMuteClick, dangerous: true });
+        menu.push({ text: intl.formatMessage(messages.block, { name: status.getIn(['account', 'username']) }), action: this.handleBlockClick, dangerous: true });
+        menu.push({ text: intl.formatMessage(messages.report, { name: status.getIn(['account', 'username']) }), action: this.handleReport, dangerous: true });
+        if (((permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS && (accountAdminLink || statusAdminLink)) || (isRemote && (permissions & PERMISSION_MANAGE_FEDERATION) === PERMISSION_MANAGE_FEDERATION)) {
+          menu.push(null);
+          if ((permissions & PERMISSION_MANAGE_USERS) === PERMISSION_MANAGE_USERS) {
+            if (accountAdminLink !== undefined) {
+              menu.push({ text: intl.formatMessage(messages.admin_account, { name: status.getIn(['account', 'username']) }), href: accountAdminLink(status.getIn(['account', 'id'])) });
+            }
+            if (statusAdminLink !== undefined) {
+              menu.push({ text: intl.formatMessage(messages.admin_status), href: statusAdminLink(status.getIn(['account', 'id']), status.get('id')) });
+            }
+          }
+          if (isRemote && (permissions & PERMISSION_MANAGE_FEDERATION) === PERMISSION_MANAGE_FEDERATION) {
+            const domain = status.getIn(['account', 'acct']).split('@')[1];
+            menu.push({ text: intl.formatMessage(messages.admin_domain, { domain: domain }), href: `/admin/instances/${domain}` });
+          }
+        }
+      }
+    }
+
+    const canReact = signedIn && status.get('reactions').filter(r => r.get('count') > 0 && r.get('me')).size < maxReactions;
+    const reactButton = (
+      <IconButton
+        className='plus-icon'
+        onClick={this.handleNoOp} // EmojiPickerDropdown handles that
+        title={intl.formatMessage(messages.react)}
+        disabled={!canReact}
+        icon='plus'
+      />
+    );
+
+    const reblogPrivate = status.getIn(['account', 'id']) === me && status.get('visibility') === 'private';
+
+    let reblogTitle;
+    if (status.get('reblogged')) {
+      reblogTitle = intl.formatMessage(messages.cancel_reblog_private);
+    } else if (publicStatus) {
+      reblogTitle = intl.formatMessage(messages.reblog);
+    } else if (reblogPrivate) {
+      reblogTitle = intl.formatMessage(messages.reblog_private);
+    } else {
+      reblogTitle = intl.formatMessage(messages.cannot_reblog);
+    }
+
+    return (
+      <div className='detailed-status__action-bar'>
+        <div className='detailed-status__button'><IconButton title={intl.formatMessage(messages.reply)} icon={status.get('in_reply_to_id', null) === null ? 'reply' : 'reply-all'} onClick={this.handleReplyClick} /></div>
+        <div className='detailed-status__button'><IconButton className={classNames({ reblogPrivate })} disabled={!publicStatus && !reblogPrivate} active={status.get('reblogged')} title={reblogTitle} icon='retweet' onClick={this.handleReblogClick} /></div>
+        <div className='detailed-status__button'><IconButton className='star-icon' animate active={status.get('favourited')} title={intl.formatMessage(messages.favourite)} icon='star' onClick={this.handleFavouriteClick} /></div>
+        <div className='detailed-status__button'>
+          {
+            signedIn
+              ? <EmojiPickerDropdown onPickEmoji={this.handleEmojiPick} button={reactButton} disabled={!canReact} />
+              : reactButton
+          }
+        </div>
+        <div className='detailed-status__button'><IconButton className='bookmark-icon' disabled={!signedIn} active={status.get('bookmarked')} title={intl.formatMessage(messages.bookmark)} icon='bookmark' onClick={this.handleBookmarkClick} /></div>
+
+        <div className='detailed-status__action-bar-dropdown'>
+          <DropdownMenuContainer size={18} icon='ellipsis-h' items={menu} direction='left' title={intl.formatMessage(messages.more)} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(ActionBar));
diff --git a/app/javascript/flavours/blobfox/features/status/components/card.jsx b/app/javascript/flavours/blobfox/features/status/components/card.jsx
new file mode 100644
index 00000000000000..51e548a1584b24
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/status/components/card.jsx
@@ -0,0 +1,256 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import Immutable from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Blurhash } from 'flavours/blobfox/components/blurhash';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { useBlurhash } from 'flavours/blobfox/initial_state';
+import { decode as decodeIDNA } from 'flavours/blobfox/utils/idna';
+
+const getHostname = url => {
+  const parser = document.createElement('a');
+  parser.href = url;
+  return parser.hostname;
+};
+
+const domParser = new DOMParser();
+
+const addAutoPlay = html => {
+  const document = domParser.parseFromString(html, 'text/html').documentElement;
+  const iframe = document.querySelector('iframe');
+
+  if (iframe) {
+    if (iframe.src.indexOf('?') !== -1) {
+      iframe.src += '&';
+    } else {
+      iframe.src += '?';
+    }
+
+    iframe.src += 'autoplay=1&auto_play=1';
+
+    // DOM parser creates html/body elements around original HTML fragment,
+    // so we need to get innerHTML out of the body and not the entire document
+    return document.querySelector('body').innerHTML;
+  }
+
+  return html;
+};
+
+export default class Card extends PureComponent {
+
+  static propTypes = {
+    card: ImmutablePropTypes.map,
+    onOpenMedia: PropTypes.func.isRequired,
+    compact: PropTypes.bool,
+    sensitive: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    compact: false,
+  };
+
+  state = {
+    previewLoaded: false,
+    embedded: false,
+    revealed: !this.props.sensitive,
+  };
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (!Immutable.is(this.props.card, nextProps.card)) {
+      this.setState({ embedded: false, previewLoaded: false });
+    }
+    if (this.props.sensitive !== nextProps.sensitive) {
+      this.setState({ revealed: !nextProps.sensitive });
+    }
+  }
+
+  componentDidMount () {
+    window.addEventListener('resize', this.handleResize, { passive: true });
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('resize', this.handleResize);
+  }
+
+  handlePhotoClick = () => {
+    const { card, onOpenMedia } = this.props;
+
+    onOpenMedia(
+      Immutable.fromJS([
+        {
+          type: 'image',
+          url: card.get('embed_url'),
+          description: card.get('title'),
+          meta: {
+            original: {
+              width: card.get('width'),
+              height: card.get('height'),
+            },
+          },
+        },
+      ]),
+      0,
+    );
+  };
+
+  handleEmbedClick = () => {
+    const { card } = this.props;
+
+    if (card.get('type') === 'photo') {
+      this.handlePhotoClick();
+    } else {
+      this.setState({ embedded: true });
+    }
+  };
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  handleImageLoad = () => {
+    this.setState({ previewLoaded: true });
+  };
+
+  handleReveal = e => {
+    e.preventDefault();
+    e.stopPropagation();
+    this.setState({ revealed: true });
+  };
+
+  renderVideo () {
+    const { card }  = this.props;
+    const content   = { __html: addAutoPlay(card.get('html')) };
+
+    return (
+      <div
+        ref={this.setRef}
+        className='status-card__image status-card-video'
+        dangerouslySetInnerHTML={content}
+        style={{ aspectRatio: `${card.get('width')} / ${card.get('height')}` }}
+      />
+    );
+  }
+
+  render () {
+    const { card, compact } = this.props;
+    const { embedded, revealed } = this.state;
+
+    if (card === null) {
+      return null;
+    }
+
+    const provider    = card.get('provider_name').length === 0 ? decodeIDNA(getHostname(card.get('url'))) : card.get('provider_name');
+    const horizontal  = (!compact && card.get('width') > card.get('height')) || card.get('type') !== 'link' || embedded;
+    const interactive = card.get('type') !== 'link';
+    const className   = classNames('status-card', { horizontal, compact, interactive });
+    const title       = interactive ? <a className='status-card__title' href={card.get('url')} title={card.get('title')} rel='noopener noreferrer' target='_blank'><strong>{card.get('title')}</strong></a> : <strong className='status-card__title' title={card.get('title')}>{card.get('title')}</strong>;
+    const language    = card.get('language') || '';
+
+    const description = (
+      <div className='status-card__content' lang={language}>
+        {title}
+        {!(horizontal || compact) && <p className='status-card__description' title={card.get('description')}>{card.get('description')}</p>}
+        <span className='status-card__host'>{provider}</span>
+      </div>
+    );
+
+    const thumbnailStyle = {
+      visibility: revealed? null : 'hidden',
+    };
+
+    if (horizontal) {
+      thumbnailStyle.aspectRatio = (compact && !embedded) ? '16 / 9' : `${card.get('width')} / ${card.get('height')}`;
+    }
+
+    let embed     = '';
+    let canvas = (
+      <Blurhash
+        className={classNames('status-card__image-preview', {
+          'status-card__image-preview--hidden': revealed && this.state.previewLoaded,
+        })}
+        hash={card.get('blurhash')}
+        dummy={!useBlurhash}
+      />
+    );
+
+    const thumbnailDescription = card.get('image_description');
+    const thumbnail = <img src={card.get('image')} alt={thumbnailDescription} title={thumbnailDescription} lang={language} style={thumbnailStyle} onLoad={this.handleImageLoad} className='status-card__image-image' />;
+
+    let spoilerButton = (
+      <button type='button' onClick={this.handleReveal} className='spoiler-button__overlay'>
+        <span className='spoiler-button__overlay__label'>
+          <FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' />
+          <span className='spoiler-button__overlay__action'><FormattedMessage id='status.media.show' defaultMessage='Click to show' /></span>
+        </span>
+      </button>
+    );
+
+    spoilerButton = (
+      <div className={classNames('spoiler-button', { 'spoiler-button--minified': revealed })}>
+        {spoilerButton}
+      </div>
+    );
+
+    if (interactive) {
+      if (embedded) {
+        embed = this.renderVideo();
+      } else {
+        let iconVariant = 'play';
+
+        if (card.get('type') === 'photo') {
+          iconVariant = 'search-plus';
+        }
+
+        embed = (
+          <div className='status-card__image'>
+            {canvas}
+            {thumbnail}
+
+            {revealed ? (
+              <div className='status-card__actions'>
+                <div>
+                  <button type='button' onClick={this.handleEmbedClick}><Icon id={iconVariant} /></button>
+                  {horizontal && <a href={card.get('url')} target='_blank' rel='noopener noreferrer'><Icon id='external-link' /></a>}
+                </div>
+              </div>
+            ) : spoilerButton}
+          </div>
+        );
+      }
+
+      return (
+        <div className={className} ref={this.setRef} onClick={revealed ? null : this.handleReveal} role={revealed ? 'button' : null}>
+          {embed}
+          {!compact && description}
+        </div>
+      );
+    } else if (card.get('image')) {
+      embed = (
+        <div className='status-card__image'>
+          {canvas}
+          {thumbnail}
+        </div>
+      );
+    } else {
+      embed = (
+        <div className='status-card__image'>
+          <Icon id='file-text' />
+        </div>
+      );
+    }
+
+    return (
+      <a href={card.get('url')} className={className} target='_blank' rel='noopener noreferrer' ref={this.setRef}>
+        {embed}
+        {description}
+      </a>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/status/components/detailed_status.jsx b/app/javascript/flavours/blobfox/features/status/components/detailed_status.jsx
new file mode 100644
index 00000000000000..2b4c96bc83ce97
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/status/components/detailed_status.jsx
@@ -0,0 +1,362 @@
+import PropTypes from 'prop-types';
+
+import { injectIntl, FormattedDate } from 'react-intl';
+
+import classNames from 'classnames';
+import { Link, withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { AnimatedNumber } from 'flavours/blobfox/components/animated_number';
+import AttachmentList from 'flavours/blobfox/components/attachment_list';
+import EditedTimestamp from 'flavours/blobfox/components/edited_timestamp';
+import { getHashtagBarForStatus } from 'flavours/blobfox/components/hashtag_bar';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import PictureInPicturePlaceholder from 'flavours/blobfox/components/picture_in_picture_placeholder';
+import VisibilityIcon from 'flavours/blobfox/components/status_visibility_icon';
+import PollContainer from 'flavours/blobfox/containers/poll_container';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { Avatar } from '../../../components/avatar';
+import { DisplayName } from '../../../components/display_name';
+import MediaGallery from '../../../components/media_gallery';
+import StatusContent from '../../../components/status_content';
+import StatusReactions from '../../../components/status_reactions';
+import Audio from '../../audio';
+import scheduleIdleTask from '../../ui/util/schedule_idle_task';
+import Video from '../../video';
+
+import Card from './card';
+
+class DetailedStatus extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    status: ImmutablePropTypes.map,
+    settings: ImmutablePropTypes.map.isRequired,
+    onOpenMedia: PropTypes.func.isRequired,
+    onOpenVideo: PropTypes.func.isRequired,
+    onToggleHidden: PropTypes.func,
+    onTranslate: PropTypes.func.isRequired,
+    expanded: PropTypes.bool,
+    measureHeight: PropTypes.bool,
+    onHeightChange: PropTypes.func,
+    domain: PropTypes.string.isRequired,
+    compact: PropTypes.bool,
+    showMedia: PropTypes.bool,
+    pictureInPicture: ImmutablePropTypes.contains({
+      inUse: PropTypes.bool,
+      available: PropTypes.bool,
+    }),
+    onToggleMediaVisibility: PropTypes.func,
+    onReactionAdd: PropTypes.func.isRequired,
+    onReactionRemove: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  state = {
+    height: null,
+  };
+
+  handleAccountClick = (e) => {
+    if (e.button === 0 && !(e.ctrlKey || e.altKey || e.metaKey) && this.props.history) {
+      e.preventDefault();
+      this.props.history.push(`/@${this.props.status.getIn(['account', 'acct'])}`);
+    }
+
+    e.stopPropagation();
+  };
+
+  parseClick = (e, destination) => {
+    if (e.button === 0 && !(e.ctrlKey || e.altKey || e.metaKey) && this.props.history) {
+      e.preventDefault();
+      this.props.history.push(destination);
+    }
+
+    e.stopPropagation();
+  };
+
+  handleOpenVideo = (options) => {
+    this.props.onOpenVideo(this.props.status.getIn(['media_attachments', 0]), options);
+  };
+
+  _measureHeight (heightJustChanged) {
+    if (this.props.measureHeight && this.node) {
+      scheduleIdleTask(() => this.node && this.setState({ height: Math.ceil(this.node.scrollHeight) + 1 }));
+
+      if (this.props.onHeightChange && heightJustChanged) {
+        this.props.onHeightChange();
+      }
+    }
+  }
+
+  setRef = c => {
+    this.node = c;
+    this._measureHeight();
+  };
+
+  componentDidUpdate (prevProps, prevState) {
+    this._measureHeight(prevState.height !== this.state.height);
+  }
+
+  handleChildUpdate = () => {
+    this._measureHeight();
+  };
+
+  handleModalLink = e => {
+    e.preventDefault();
+
+    let href;
+
+    if (e.target.nodeName !== 'A') {
+      href = e.target.parentNode.href;
+    } else {
+      href = e.target.href;
+    }
+
+    window.open(href, 'mastodon-intent', 'width=445,height=600,resizable=no,menubar=no,status=no,scrollbars=yes');
+  };
+
+  handleTranslate = () => {
+    const { onTranslate, status } = this.props;
+    onTranslate(status);
+  };
+
+  render () {
+    const status = (this.props.status && this.props.status.get('reblog')) ? this.props.status.get('reblog') : this.props.status;
+    const outerStyle = { boxSizing: 'border-box' };
+    const { compact, pictureInPicture, expanded, onToggleHidden, settings } = this.props;
+
+    if (!status) {
+      return null;
+    }
+
+    let applicationLink = '';
+    let reblogLink = '';
+    let reblogIcon = 'retweet';
+    let favouriteLink = '';
+    let edited = '';
+
+    //  Depending on user settings, some media are considered as parts of the
+    //  contents (affected by CW) while other will be displayed outside of the
+    //  CW.
+    let contentMedia = [];
+    let contentMediaIcons = [];
+    let extraMedia = [];
+    let extraMediaIcons = [];
+    let media = contentMedia;
+    let mediaIcons = contentMediaIcons;
+
+    if (settings.getIn(['content_warnings', 'media_outside'])) {
+      media = extraMedia;
+      mediaIcons = extraMediaIcons;
+    }
+
+    if (this.props.measureHeight) {
+      outerStyle.height = `${this.state.height}px`;
+    }
+
+    const language = status.getIn(['translation', 'language']) || status.get('language');
+
+    if (pictureInPicture.get('inUse')) {
+      media.push(<PictureInPicturePlaceholder />);
+      mediaIcons.push('video-camera');
+    } else if (status.get('media_attachments').size > 0) {
+      if (status.get('media_attachments').some(item => item.get('type') === 'unknown')) {
+        media.push(<AttachmentList media={status.get('media_attachments')} />);
+      } else if (status.getIn(['media_attachments', 0, 'type']) === 'audio') {
+        const attachment = status.getIn(['media_attachments', 0]);
+        const description = attachment.getIn(['translation', 'description']) || attachment.get('description');
+
+        media.push(
+          <Audio
+            src={attachment.get('url')}
+            alt={description}
+            lang={language}
+            duration={attachment.getIn(['meta', 'original', 'duration'], 0)}
+            poster={attachment.get('preview_url') || status.getIn(['account', 'avatar_static'])}
+            backgroundColor={attachment.getIn(['meta', 'colors', 'background'])}
+            foregroundColor={attachment.getIn(['meta', 'colors', 'foreground'])}
+            accentColor={attachment.getIn(['meta', 'colors', 'accent'])}
+            sensitive={status.get('sensitive')}
+            visible={this.props.showMedia}
+            blurhash={attachment.get('blurhash')}
+            height={150}
+            onToggleVisibility={this.props.onToggleMediaVisibility}
+          />,
+        );
+        mediaIcons.push('music');
+      } else if (status.getIn(['media_attachments', 0, 'type']) === 'video') {
+        const attachment = status.getIn(['media_attachments', 0]);
+        const description = attachment.getIn(['translation', 'description']) || attachment.get('description');
+
+        media.push(
+          <Video
+            preview={attachment.get('preview_url')}
+            frameRate={attachment.getIn(['meta', 'original', 'frame_rate'])}
+            blurhash={attachment.get('blurhash')}
+            src={attachment.get('url')}
+            alt={description}
+            lang={language}
+            inline
+            sensitive={status.get('sensitive')}
+            letterbox={settings.getIn(['media', 'letterbox'])}
+            fullwidth={settings.getIn(['media', 'fullwidth'])}
+            preventPlayback={!expanded}
+            onOpenVideo={this.handleOpenVideo}
+            autoplay
+            visible={this.props.showMedia}
+            onToggleVisibility={this.props.onToggleMediaVisibility}
+          />,
+        );
+        mediaIcons.push('video-camera');
+      } else {
+        media.push(
+          <MediaGallery
+            standalone
+            sensitive={status.get('sensitive')}
+            media={status.get('media_attachments')}
+            lang={language}
+            letterbox={settings.getIn(['media', 'letterbox'])}
+            fullwidth={settings.getIn(['media', 'fullwidth'])}
+            hidden={!expanded}
+            onOpenMedia={this.props.onOpenMedia}
+            visible={this.props.showMedia}
+            onToggleVisibility={this.props.onToggleMediaVisibility}
+          />,
+        );
+        mediaIcons.push('picture-o');
+      }
+    } else if (status.get('card')) {
+      media.push(<Card sensitive={status.get('sensitive')} onOpenMedia={this.props.onOpenMedia} card={status.get('card')} />);
+      mediaIcons.push('link');
+    }
+
+    if (status.get('poll')) {
+      contentMedia.push(<PollContainer pollId={status.get('poll')} lang={status.get('language')} />);
+      contentMediaIcons.push('tasks');
+    }
+
+    if (status.get('application')) {
+      applicationLink = <> · <a className='detailed-status__application' href={status.getIn(['application', 'website'])} target='_blank' rel='noopener noreferrer'>{status.getIn(['application', 'name'])}</a></>;
+    }
+
+    const visibilityLink = <> · <VisibilityIcon visibility={status.get('visibility')} /></>;
+
+    if (status.get('visibility') === 'direct') {
+      reblogIcon = 'envelope';
+    } else if (status.get('visibility') === 'private') {
+      reblogIcon = 'lock';
+    }
+
+    if (!['unlisted', 'public'].includes(status.get('visibility'))) {
+      reblogLink = null;
+    } else if (this.props.history) {
+      reblogLink = (
+        <>
+          {' · '}
+          <Link to={`/@${status.getIn(['account', 'acct'])}/${status.get('id')}/reblogs`} className='detailed-status__link'>
+            <Icon id={reblogIcon} />
+            <span className='detailed-status__reblogs'>
+              <AnimatedNumber value={status.get('reblogs_count')} />
+            </span>
+          </Link>
+        </>
+      );
+    } else {
+      reblogLink = (
+        <>
+          {' · '}
+          <a href={`/interact/${status.get('id')}?type=reblog`} className='detailed-status__link' onClick={this.handleModalLink}>
+            <Icon id={reblogIcon} />
+            <span className='detailed-status__reblogs'>
+              <AnimatedNumber value={status.get('reblogs_count')} />
+            </span>
+          </a>
+        </>
+      );
+    }
+
+    if (this.props.history) {
+      favouriteLink = (
+        <Link to={`/@${status.getIn(['account', 'acct'])}/${status.get('id')}/favourites`} className='detailed-status__link'>
+          <Icon id='star' />
+          <span className='detailed-status__favorites'>
+            <AnimatedNumber value={status.get('favourites_count')} />
+          </span>
+        </Link>
+      );
+    } else {
+      favouriteLink = (
+        <a href={`/interact/${status.get('id')}?type=favourite`} className='detailed-status__link' onClick={this.handleModalLink}>
+          <Icon id='star' />
+          <span className='detailed-status__favorites'>
+            <AnimatedNumber value={status.get('favourites_count')} />
+          </span>
+        </a>
+      );
+    }
+
+    if (status.get('edited_at')) {
+      edited = (
+        <>
+          {' · '}
+          <EditedTimestamp statusId={status.get('id')} timestamp={status.get('edited_at')} />
+        </>
+      );
+    }
+
+    const {statusContentProps, hashtagBar} = getHashtagBarForStatus(status);
+    contentMedia.push(hashtagBar);
+
+    return (
+      <div style={outerStyle}>
+        <div ref={this.setRef} className={classNames('detailed-status', `detailed-status-${status.get('visibility')}`, { compact })} data-status-by={status.getIn(['account', 'acct'])}>
+          <a href={status.getIn(['account', 'url'])} onClick={this.handleAccountClick} className='detailed-status__display-name'>
+            <div className='detailed-status__display-avatar'><Avatar account={status.get('account')} size={48} /></div>
+            <DisplayName account={status.get('account')} localDomain={this.props.domain} />
+          </a>
+
+          <StatusContent
+            status={status}
+            media={contentMedia}
+            extraMedia={extraMedia}
+            mediaIcons={contentMediaIcons}
+            expanded={expanded}
+            collapsed={false}
+            onExpandedToggle={onToggleHidden}
+            onTranslate={this.handleTranslate}
+            parseClick={this.parseClick}
+            onUpdate={this.handleChildUpdate}
+            tagLinks={settings.get('tag_misleading_links')}
+            rewriteMentions={settings.get('rewrite_mentions')}
+            disabled
+            {...statusContentProps}
+          />
+
+          <StatusReactions
+            statusId={status.get('id')}
+            reactions={status.get('reactions')}
+            addReaction={this.props.onReactionAdd}
+            removeReaction={this.props.onReactionRemove}
+            canReact={this.context.identity.signedIn}
+          />
+
+          <div className='detailed-status__meta'>
+            <a className='detailed-status__datetime' href={status.get('url')} target='_blank' rel='noopener noreferrer'>
+              <FormattedDate value={new Date(status.get('created_at'))} hour12={false} year='numeric' month='short' day='2-digit' hour='2-digit' minute='2-digit' />
+            </a>{edited}{visibilityLink}{applicationLink}{reblogLink} · {favouriteLink}
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(DetailedStatus));
diff --git a/app/javascript/flavours/blobfox/features/status/containers/detailed_status_container.js b/app/javascript/flavours/blobfox/features/status/containers/detailed_status_container.js
new file mode 100644
index 00000000000000..40803441d201d2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/status/containers/detailed_status_container.js
@@ -0,0 +1,176 @@
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { showAlertForError } from '../../../actions/alerts';
+import { initBlockModal } from '../../../actions/blocks';
+import { initBoostModal } from '../../../actions/boosts';
+import {
+  replyCompose,
+  mentionCompose,
+  directCompose,
+} from '../../../actions/compose';
+import {
+  reblog,
+  favourite,
+  unreblog,
+  unfavourite,
+  pin,
+  unpin,
+} from '../../../actions/interactions';
+import { openModal } from '../../../actions/modal';
+import { initMuteModal } from '../../../actions/mutes';
+import { initReport } from '../../../actions/reports';
+import {
+  muteStatus,
+  unmuteStatus,
+  deleteStatus,
+} from '../../../actions/statuses';
+import { boostModal, deleteModal } from '../../../initial_state';
+import { makeGetStatus } from '../../../selectors';
+import DetailedStatus from '../components/detailed_status';
+
+const messages = defineMessages({
+  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
+  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
+  redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' },
+  redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this status and re-draft it? Favorites and boosts will be lost, and replies to the original post will be orphaned.' },
+  replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' },
+  replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
+});
+
+const makeMapStateToProps = () => {
+  const getStatus = makeGetStatus();
+
+  const mapStateToProps = (state, props) => ({
+    status: getStatus(state, props),
+    domain: state.getIn(['meta', 'domain']),
+    settings: state.get('local_settings'),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+
+  onReply (status, router) {
+    dispatch((_, getState) => {
+      let state = getState();
+      if (state.getIn(['compose', 'text']).trim().length !== 0) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: intl.formatMessage(messages.replyMessage),
+            confirm: intl.formatMessage(messages.replyConfirm),
+            onConfirm: () => dispatch(replyCompose(status, router)),
+          },
+        }));
+      } else {
+        dispatch(replyCompose(status, router));
+      }
+    });
+  },
+
+  onModalReblog (status, privacy) {
+    dispatch(reblog(status, privacy));
+  },
+
+  onReblog (status, e) {
+    if (status.get('reblogged')) {
+      dispatch(unreblog(status));
+    } else {
+      if (e.shiftKey || !boostModal) {
+        this.onModalReblog(status);
+      } else {
+        dispatch(initBoostModal({ status, onReblog: this.onModalReblog }));
+      }
+    }
+  },
+
+  onFavourite (status) {
+    if (status.get('favourited')) {
+      dispatch(unfavourite(status));
+    } else {
+      dispatch(favourite(status));
+    }
+  },
+
+  onPin (status) {
+    if (status.get('pinned')) {
+      dispatch(unpin(status));
+    } else {
+      dispatch(pin(status));
+    }
+  },
+
+  onEmbed (status) {
+    dispatch(openModal({
+      modalType: 'EMBED',
+      modalProps: {
+        id: status.get('id'),
+        onError: error => dispatch(showAlertForError(error)),
+      },
+    }));
+  },
+
+  onDelete (status, history, withRedraft = false) {
+    if (!deleteModal) {
+      dispatch(deleteStatus(status.get('id'), history, withRedraft));
+    } else {
+      dispatch(openModal({
+        modalType: 'CONFIRM',
+        modalProps: {
+          message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage),
+          confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm),
+          onConfirm: () => dispatch(deleteStatus(status.get('id'), history, withRedraft)),
+        },
+      }));
+    }
+  },
+
+  onDirect (account, router) {
+    dispatch(directCompose(account, router));
+  },
+
+  onMention (account, router) {
+    dispatch(mentionCompose(account, router));
+  },
+
+  onOpenMedia (media, index, lang) {
+    dispatch(openModal({
+      modalType: 'MEDIA',
+      modalProps: { media, index, lang },
+    }));
+  },
+
+  onOpenVideo (media, lang, options) {
+    dispatch(openModal({
+      modalType: 'VIDEO',
+      modalProps: { media, lang, options },
+    }));
+  },
+
+  onBlock (status) {
+    const account = status.get('account');
+    dispatch(initBlockModal(account));
+  },
+
+  onReport (status) {
+    dispatch(initReport(status.get('account'), status));
+  },
+
+  onMute (account) {
+    dispatch(initMuteModal(account));
+  },
+
+  onMuteConversation (status) {
+    if (status.get('muted')) {
+      dispatch(unmuteStatus(status.get('id')));
+    } else {
+      dispatch(muteStatus(status.get('id')));
+    }
+  },
+
+});
+
+export default injectIntl(connect(makeMapStateToProps, mapDispatchToProps)(DetailedStatus));
diff --git a/app/javascript/flavours/blobfox/features/status/index.jsx b/app/javascript/flavours/blobfox/features/status/index.jsx
new file mode 100644
index 00000000000000..0e387e6f237179
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/status/index.jsx
@@ -0,0 +1,807 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+import { withRouter } from 'react-router-dom';
+
+import Immutable from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { HotKeys } from 'react-hotkeys';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { LoadingIndicator } from 'flavours/blobfox/components/loading_indicator';
+import ScrollContainer from 'flavours/blobfox/containers/scroll_container';
+import BundleColumnError from 'flavours/blobfox/features/ui/components/bundle_column_error';
+import { autoUnfoldCW } from 'flavours/blobfox/utils/content_warning';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { initBlockModal } from '../../actions/blocks';
+import { initBoostModal } from '../../actions/boosts';
+import {
+  replyCompose,
+  mentionCompose,
+  directCompose,
+} from '../../actions/compose';
+import {
+  favourite,
+  unfavourite,
+  bookmark,
+  unbookmark,
+  reblog,
+  unreblog,
+  pin,
+  unpin,
+  addReaction,
+  removeReaction,
+} from '../../actions/interactions';
+import { changeLocalSetting } from '../../actions/local_settings';
+import { openModal } from '../../actions/modal';
+import { initMuteModal } from '../../actions/mutes';
+import { initReport } from '../../actions/reports';
+import {
+  fetchStatus,
+  muteStatus,
+  unmuteStatus,
+  deleteStatus,
+  editStatus,
+  hideStatus,
+  revealStatus,
+  translateStatus,
+  undoStatusTranslation,
+} from '../../actions/statuses';
+import ColumnHeader from '../../components/column_header';
+import { textForScreenReader, defaultMediaVisibility } from '../../components/status';
+import StatusContainer from '../../containers/status_container';
+import { boostModal, favouriteModal, deleteModal } from '../../initial_state';
+import { makeGetStatus, makeGetPictureInPicture } from '../../selectors';
+import Column from '../ui/components/column';
+import { attachFullscreenListener, detachFullscreenListener, isFullscreen } from '../ui/util/fullscreen';
+
+import ActionBar from './components/action_bar';
+import DetailedStatus from './components/detailed_status';
+
+
+const messages = defineMessages({
+  deleteConfirm: { id: 'confirmations.delete.confirm', defaultMessage: 'Delete' },
+  deleteMessage: { id: 'confirmations.delete.message', defaultMessage: 'Are you sure you want to delete this status?' },
+  redraftConfirm: { id: 'confirmations.redraft.confirm', defaultMessage: 'Delete & redraft' },
+  redraftMessage: { id: 'confirmations.redraft.message', defaultMessage: 'Are you sure you want to delete this status and re-draft it? Favorites and boosts will be lost, and replies to the original post will be orphaned.' },
+  revealAll: { id: 'status.show_more_all', defaultMessage: 'Show more for all' },
+  hideAll: { id: 'status.show_less_all', defaultMessage: 'Show less for all' },
+  statusTitleWithAttachments: { id: 'status.title.with_attachments', defaultMessage: '{user} posted {attachmentCount, plural, one {an attachment} other {# attachments}}' },
+  detailedStatus: { id: 'status.detailed_status', defaultMessage: 'Detailed conversation view' },
+  replyConfirm: { id: 'confirmations.reply.confirm', defaultMessage: 'Reply' },
+  replyMessage: { id: 'confirmations.reply.message', defaultMessage: 'Replying now will overwrite the message you are currently composing. Are you sure you want to proceed?' },
+  tootHeading: { id: 'account.posts_with_replies', defaultMessage: 'Posts and replies' },
+});
+
+const makeMapStateToProps = () => {
+  const getStatus = makeGetStatus();
+  const getPictureInPicture = makeGetPictureInPicture();
+
+  const getAncestorsIds = createSelector([
+    (_, { id }) => id,
+    state => state.getIn(['contexts', 'inReplyTos']),
+  ], (statusId, inReplyTos) => {
+    let ancestorsIds = Immutable.List();
+    ancestorsIds = ancestorsIds.withMutations(mutable => {
+      let id = statusId;
+
+      while (id && !mutable.includes(id)) {
+        mutable.unshift(id);
+        id = inReplyTos.get(id);
+      }
+    });
+
+    return ancestorsIds;
+  });
+
+  const getDescendantsIds = createSelector([
+    (_, { id }) => id,
+    state => state.getIn(['contexts', 'replies']),
+    state => state.get('statuses'),
+  ], (statusId, contextReplies, statuses) => {
+    let descendantsIds = [];
+    const ids = [statusId];
+
+    while (ids.length > 0) {
+      let id        = ids.pop();
+      const replies = contextReplies.get(id);
+
+      if (statusId !== id) {
+        descendantsIds.push(id);
+      }
+
+      if (replies) {
+        replies.reverse().forEach(reply => {
+          if (!ids.includes(reply) && !descendantsIds.includes(reply) && statusId !== reply) ids.push(reply);
+        });
+      }
+    }
+
+    let insertAt = descendantsIds.findIndex((id) => statuses.get(id).get('in_reply_to_account_id') !== statuses.get(id).get('account'));
+    if (insertAt !== -1) {
+      descendantsIds.forEach((id, idx) => {
+        if (idx > insertAt && statuses.get(id).get('in_reply_to_account_id') === statuses.get(id).get('account')) {
+          descendantsIds.splice(idx, 1);
+          descendantsIds.splice(insertAt, 0, id);
+          insertAt += 1;
+        }
+      });
+    }
+
+    return Immutable.List(descendantsIds);
+  });
+
+  const mapStateToProps = (state, props) => {
+    const status = getStatus(state, { id: props.params.statusId });
+
+    let ancestorsIds   = Immutable.List();
+    let descendantsIds = Immutable.List();
+
+    if (status) {
+      ancestorsIds   = getAncestorsIds(state, { id: status.get('in_reply_to_id') });
+      descendantsIds = getDescendantsIds(state, { id: status.get('id') });
+    }
+
+    return {
+      isLoading: state.getIn(['statuses', props.params.statusId, 'isLoading']),
+      status,
+      ancestorsIds,
+      descendantsIds,
+      settings: state.get('local_settings'),
+      askReplyConfirmation: state.getIn(['local_settings', 'confirm_before_clearing_draft']) && state.getIn(['compose', 'text']).trim().length !== 0,
+      domain: state.getIn(['meta', 'domain']),
+      pictureInPicture: getPictureInPicture(state, { id: props.params.statusId }),
+    };
+  };
+
+  return mapStateToProps;
+};
+
+const truncate = (str, num) => {
+  const arr = Array.from(str);
+  if (arr.length > num) {
+    return arr.slice(0, num).join('') + '…';
+  } else {
+    return str;
+  }
+};
+
+const titleFromStatus = (intl, status) => {
+  const displayName = status.getIn(['account', 'display_name']);
+  const username = status.getIn(['account', 'username']);
+  const user = displayName.trim().length === 0 ? username : displayName;
+  const text = status.get('search_index');
+  const attachmentCount = status.get('media_attachments').size;
+
+  return text ? `${user}: "${truncate(text, 30)}"` : intl.formatMessage(messages.statusTitleWithAttachments, { user, attachmentCount });
+};
+
+class Status extends ImmutablePureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    params: PropTypes.object.isRequired,
+    dispatch: PropTypes.func.isRequired,
+    status: ImmutablePropTypes.map,
+    isLoading: PropTypes.bool,
+    settings: ImmutablePropTypes.map.isRequired,
+    ancestorsIds: ImmutablePropTypes.list.isRequired,
+    descendantsIds: ImmutablePropTypes.list.isRequired,
+    intl: PropTypes.object.isRequired,
+    askReplyConfirmation: PropTypes.bool,
+    multiColumn: PropTypes.bool,
+    domain: PropTypes.string.isRequired,
+    pictureInPicture: ImmutablePropTypes.contains({
+      inUse: PropTypes.bool,
+      available: PropTypes.bool,
+    }),
+    ...WithRouterPropTypes
+  };
+
+  state = {
+    fullscreen: false,
+    isExpanded: undefined,
+    threadExpanded: undefined,
+    statusId: undefined,
+    loadedStatusId: undefined,
+    showMedia: undefined,
+    revealBehindCW: undefined,
+  };
+
+  componentDidMount () {
+    attachFullscreenListener(this.onFullScreenChange);
+    this.props.dispatch(fetchStatus(this.props.params.statusId));
+    this._scrollStatusIntoView();
+  }
+
+  static getDerivedStateFromProps(props, state) {
+    let update = {};
+    let updated = false;
+
+    if (props.params.statusId && state.statusId !== props.params.statusId) {
+      props.dispatch(fetchStatus(props.params.statusId));
+      update.threadExpanded = undefined;
+      update.statusId = props.params.statusId;
+      updated = true;
+    }
+
+    const revealBehindCW = props.settings.getIn(['media', 'reveal_behind_cw']);
+    if (revealBehindCW !== state.revealBehindCW) {
+      update.revealBehindCW = revealBehindCW;
+      if (revealBehindCW) update.showMedia = defaultMediaVisibility(props.status, props.settings);
+      updated = true;
+    }
+
+    if (props.status && state.loadedStatusId !== props.status.get('id')) {
+      update.showMedia = defaultMediaVisibility(props.status, props.settings);
+      update.loadedStatusId = props.status.get('id');
+      update.isExpanded = autoUnfoldCW(props.settings, props.status);
+      updated = true;
+    }
+
+    return updated ? update : null;
+  }
+
+  handleToggleHidden = () => {
+    const { status } = this.props;
+
+    if (this.props.settings.getIn(['content_warnings', 'shared_state'])) {
+      if (status.get('hidden')) {
+        this.props.dispatch(revealStatus(status.get('id')));
+      } else {
+        this.props.dispatch(hideStatus(status.get('id')));
+      }
+    } else if (this.props.status.get('spoiler_text')) {
+      this.setExpansion(!this.state.isExpanded);
+    }
+  };
+
+  handleToggleMediaVisibility = () => {
+    this.setState({ showMedia: !this.state.showMedia });
+  };
+
+  handleModalFavourite = (status) => {
+    this.props.dispatch(favourite(status));
+  };
+
+  handleFavouriteClick = (status, e) => {
+    const { dispatch } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      if (status.get('favourited')) {
+        dispatch(unfavourite(status));
+      } else {
+        if ((e && e.shiftKey) || !favouriteModal) {
+          this.handleModalFavourite(status);
+        } else {
+          dispatch(openModal({
+            modalType: 'FAVOURITE',
+            modalProps: {
+              status,
+              onFavourite: this.handleModalFavourite,
+            },
+          }));
+        }
+      }
+    } else {
+      dispatch(openModal({
+        modalType: 'INTERACTION',
+        modalProps: {
+          type: 'favourite',
+          accountId: status.getIn(['account', 'id']),
+          url: status.get('uri'),
+        },
+      }));
+    }
+  };
+
+  handleReactionAdd = (statusId, name, url) => {
+    const { dispatch } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      dispatch(addReaction(statusId, name, url));
+    }
+  };
+
+  handleReactionRemove = (statusId, name) => {
+    this.props.dispatch(removeReaction(statusId, name));
+  };
+
+  handlePin = (status) => {
+    if (status.get('pinned')) {
+      this.props.dispatch(unpin(status));
+    } else {
+      this.props.dispatch(pin(status));
+    }
+  };
+
+  handleReplyClick = (status) => {
+    const { askReplyConfirmation, dispatch, intl } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      if (askReplyConfirmation) {
+        dispatch(openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: intl.formatMessage(messages.replyMessage),
+            confirm: intl.formatMessage(messages.replyConfirm),
+            onDoNotAsk: () => dispatch(changeLocalSetting(['confirm_before_clearing_draft'], false)),
+            onConfirm: () => dispatch(replyCompose(status, this.props.history)),
+          },
+        }));
+      } else {
+        dispatch(replyCompose(status, this.props.history));
+      }
+    } else {
+      dispatch(openModal({
+        modalType: 'INTERACTION',
+        modalProps: {
+          type: 'reply',
+          accountId: status.getIn(['account', 'id']),
+          url: status.get('uri'),
+        },
+      }));
+    }
+  };
+
+  handleModalReblog = (status, privacy) => {
+    const { dispatch } = this.props;
+
+    if (status.get('reblogged')) {
+      dispatch(unreblog(status));
+    } else {
+      dispatch(reblog(status, privacy));
+    }
+  };
+
+  handleReblogClick = (status, e) => {
+    const { settings, dispatch } = this.props;
+    const { signedIn } = this.context.identity;
+
+    if (signedIn) {
+      if (settings.get('confirm_boost_missing_media_description') && status.get('media_attachments').some(item => !item.get('description')) && !status.get('reblogged')) {
+        dispatch(initBoostModal({ status, onReblog: this.handleModalReblog, missingMediaDescription: true }));
+      } else if ((e && e.shiftKey) || !boostModal) {
+        this.handleModalReblog(status);
+      } else {
+        dispatch(initBoostModal({ status, onReblog: this.handleModalReblog }));
+      }
+    } else {
+      dispatch(openModal({
+        modalType: 'INTERACTION',
+        modalProps: {
+          type: 'reblog',
+          accountId: status.getIn(['account', 'id']),
+          url: status.get('uri'),
+        },
+      }));
+    }
+  };
+
+  handleBookmarkClick = (status) => {
+    if (status.get('bookmarked')) {
+      this.props.dispatch(unbookmark(status));
+    } else {
+      this.props.dispatch(bookmark(status));
+    }
+  };
+
+  handleDeleteClick = (status, history, withRedraft = false) => {
+    const { dispatch, intl } = this.props;
+
+    if (!deleteModal) {
+      dispatch(deleteStatus(status.get('id'), history, withRedraft));
+    } else {
+      dispatch(openModal({
+        modalType: 'CONFIRM',
+        modalProps: {
+          message: intl.formatMessage(withRedraft ? messages.redraftMessage : messages.deleteMessage),
+          confirm: intl.formatMessage(withRedraft ? messages.redraftConfirm : messages.deleteConfirm),
+          onConfirm: () => dispatch(deleteStatus(status.get('id'), history, withRedraft)),
+        },
+      }));
+    }
+  };
+
+  handleEditClick = (status, history) => {
+    this.props.dispatch(editStatus(status.get('id'), history));
+  };
+
+  handleDirectClick = (account, history) => {
+    this.props.dispatch(directCompose(account, history));
+  };
+
+  handleMentionClick = (account, history) => {
+    this.props.dispatch(mentionCompose(account, history));
+  };
+
+  handleOpenMedia = (media, index, lang) => {
+    this.props.dispatch(openModal({
+      modalType: 'MEDIA',
+      modalProps: { statusId: this.props.status.get('id'), media, index, lang },
+    }));
+  };
+
+  handleOpenVideo = (media, lang, options) => {
+    this.props.dispatch(openModal({
+      modalType: 'VIDEO',
+      modalProps: { statusId: this.props.status.get('id'), media, lang, options },
+    }));
+  };
+
+  handleHotkeyOpenMedia = e => {
+    const { status } = this.props;
+
+    e.preventDefault();
+
+    if (status.get('media_attachments').size > 0) {
+      if (status.getIn(['media_attachments', 0, 'type']) === 'video') {
+        this.handleOpenVideo(status.getIn(['media_attachments', 0]), { startTime: 0 });
+      } else {
+        this.handleOpenMedia(status.get('media_attachments'), 0);
+      }
+    }
+  };
+
+  handleMuteClick = (account) => {
+    this.props.dispatch(initMuteModal(account));
+  };
+
+  handleConversationMuteClick = (status) => {
+    if (status.get('muted')) {
+      this.props.dispatch(unmuteStatus(status.get('id')));
+    } else {
+      this.props.dispatch(muteStatus(status.get('id')));
+    }
+  };
+
+  handleToggleAll = () => {
+    const { status, ancestorsIds, descendantsIds, settings } = this.props;
+    const statusIds = [status.get('id')].concat(ancestorsIds.toJS(), descendantsIds.toJS());
+    let { isExpanded } = this.state;
+
+    if (settings.getIn(['content_warnings', 'shared_state']))
+      isExpanded = !status.get('hidden');
+
+    if (!isExpanded) {
+      this.props.dispatch(revealStatus(statusIds));
+    } else {
+      this.props.dispatch(hideStatus(statusIds));
+    }
+
+    this.setState({ isExpanded: !isExpanded, threadExpanded: !isExpanded });
+  };
+
+  handleTranslate = status => {
+    const { dispatch } = this.props;
+
+    if (status.get('translation')) {
+      dispatch(undoStatusTranslation(status.get('id'), status.get('poll')));
+    } else {
+      dispatch(translateStatus(status.get('id')));
+    }
+  };
+
+  handleBlockClick = (status) => {
+    const { dispatch } = this.props;
+    const account = status.get('account');
+    dispatch(initBlockModal(account));
+  };
+
+  handleReport = (status) => {
+    this.props.dispatch(initReport(status.get('account'), status));
+  };
+
+  handleEmbed = (status) => {
+    this.props.dispatch(openModal({
+      modalType: 'EMBED',
+      modalProps: { id: status.get('id') },
+    }));
+  };
+
+  handleHotkeyToggleSensitive = () => {
+    this.handleToggleMediaVisibility();
+  };
+
+  handleHotkeyMoveUp = () => {
+    this.handleMoveUp(this.props.status.get('id'));
+  };
+
+  handleHotkeyMoveDown = () => {
+    this.handleMoveDown(this.props.status.get('id'));
+  };
+
+  handleHotkeyReply = e => {
+    e.preventDefault();
+    this.handleReplyClick(this.props.status);
+  };
+
+  handleHotkeyFavourite = () => {
+    this.handleFavouriteClick(this.props.status);
+  };
+
+  handleHotkeyBoost = () => {
+    this.handleReblogClick(this.props.status);
+  };
+
+  handleHotkeyBookmark = () => {
+    this.handleBookmarkClick(this.props.status);
+  };
+
+  handleHotkeyMention = e => {
+    e.preventDefault();
+    this.handleMentionClick(this.props.status);
+  };
+
+  handleHotkeyOpenProfile = () => {
+    this.props.history.push(`/@${this.props.status.getIn(['account', 'acct'])}`);
+  };
+
+  handleMoveUp = id => {
+    const { status, ancestorsIds, descendantsIds } = this.props;
+
+    if (id === status.get('id')) {
+      this._selectChild(ancestorsIds.size - 1, true);
+    } else {
+      let index = ancestorsIds.indexOf(id);
+
+      if (index === -1) {
+        index = descendantsIds.indexOf(id);
+        this._selectChild(ancestorsIds.size + index, true);
+      } else {
+        this._selectChild(index - 1, true);
+      }
+    }
+  };
+
+  handleMoveDown = id => {
+    const { status, ancestorsIds, descendantsIds } = this.props;
+
+    if (id === status.get('id')) {
+      this._selectChild(ancestorsIds.size + 1, false);
+    } else {
+      let index = ancestorsIds.indexOf(id);
+
+      if (index === -1) {
+        index = descendantsIds.indexOf(id);
+        this._selectChild(ancestorsIds.size + index + 2, false);
+      } else {
+        this._selectChild(index + 1, false);
+      }
+    }
+  };
+
+  _selectChild (index, align_top) {
+    const container = this.node;
+    const element = container.querySelectorAll('.focusable')[index];
+
+    if (element) {
+      if (align_top && container.scrollTop > element.offsetTop) {
+        element.scrollIntoView(true);
+      } else if (!align_top && container.scrollTop + container.clientHeight < element.offsetTop + element.offsetHeight) {
+        element.scrollIntoView(false);
+      }
+      element.focus();
+    }
+  }
+
+  handleHeaderClick = () => {
+    this.column.scrollTop();
+  };
+
+  renderChildren (list, ancestors) {
+    const { params: { statusId } } = this.props;
+
+    return list.map((id, i) => (
+      <StatusContainer
+        key={id}
+        id={id}
+        expanded={this.state.threadExpanded}
+        onMoveUp={this.handleMoveUp}
+        onMoveDown={this.handleMoveDown}
+        contextType='thread'
+        previousId={i > 0 ? list.get(i - 1) : undefined}
+        nextId={list.get(i + 1) || (ancestors && statusId)}
+        rootId={statusId}
+      />
+    ));
+  }
+
+  setExpansion = value => {
+    this.setState({ isExpanded: value });
+  };
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  setColumnRef = c => {
+    this.column = c;
+  };
+
+  _scrollStatusIntoView () {
+    const { status, multiColumn } = this.props;
+
+    if (status) {
+      window.requestAnimationFrame(() => {
+        this.node?.querySelector('.detailed-status__wrapper')?.scrollIntoView(true);
+
+        // In the single-column interface, `scrollIntoView` will put the post behind the header,
+        // so compensate for that.
+        if (!multiColumn) {
+          const offset = document.querySelector('.column-header__wrapper')?.getBoundingClientRect()?.bottom;
+          if (offset) {
+            const scrollingElement = document.scrollingElement || document.body;
+            scrollingElement.scrollBy(0, -offset);
+          }
+        }
+      });
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    const { status, ancestorsIds } = this.props;
+
+    if (status && (ancestorsIds.size > prevProps.ancestorsIds.size || prevProps.status?.get('id') !== status.get('id'))) {
+      this._scrollStatusIntoView();
+    }
+  }
+
+  componentWillUnmount () {
+    detachFullscreenListener(this.onFullScreenChange);
+  }
+
+  onFullScreenChange = () => {
+    this.setState({ fullscreen: isFullscreen() });
+  };
+
+  shouldUpdateScroll = (prevRouterProps, { location }) => {
+    // Do not change scroll when opening a modal
+    if (location.state?.mastodonModalKey !== prevRouterProps?.location?.state?.mastodonModalKey) {
+      return false;
+    }
+
+    // Scroll to focused post if it is loaded
+    const child = this.node?.querySelector('.detailed-status__wrapper');
+    if (child) {
+      return [0, child.offsetTop];
+    }
+
+    // Do not scroll otherwise, `componentDidUpdate` will take care of that
+    return false;
+  };
+
+  render () {
+    let ancestors, descendants;
+    const { isLoading, status, settings, ancestorsIds, descendantsIds, intl, domain, multiColumn, pictureInPicture } = this.props;
+    const { fullscreen } = this.state;
+
+    if (isLoading) {
+      return (
+        <Column>
+          <LoadingIndicator />
+        </Column>
+      );
+    }
+
+    if (status === null) {
+      return (
+        <BundleColumnError multiColumn={multiColumn} errorType='routing' />
+      );
+    }
+
+    const isExpanded = settings.getIn(['content_warnings', 'shared_state']) ? !status.get('hidden') : this.state.isExpanded;
+
+    if (ancestorsIds && ancestorsIds.size > 0) {
+      ancestors = <>{this.renderChildren(ancestorsIds, true)}</>;
+    }
+
+    if (descendantsIds && descendantsIds.size > 0) {
+      descendants = <>{this.renderChildren(descendantsIds)}</>;
+    }
+
+    const isLocal = status.getIn(['account', 'acct'], '').indexOf('@') === -1;
+    const isIndexable = !status.getIn(['account', 'noindex']);
+
+    const handlers = {
+      moveUp: this.handleHotkeyMoveUp,
+      moveDown: this.handleHotkeyMoveDown,
+      reply: this.handleHotkeyReply,
+      favourite: this.handleHotkeyFavourite,
+      boost: this.handleHotkeyBoost,
+      bookmark: this.handleHotkeyBookmark,
+      mention: this.handleHotkeyMention,
+      openProfile: this.handleHotkeyOpenProfile,
+      toggleHidden: this.handleToggleHidden,
+      toggleSensitive: this.handleHotkeyToggleSensitive,
+      openMedia: this.handleHotkeyOpenMedia,
+    };
+
+    return (
+      <Column bindToDocument={!multiColumn} ref={this.setColumnRef} label={intl.formatMessage(messages.detailedStatus)}>
+        <ColumnHeader
+          icon='comment'
+          title={intl.formatMessage(messages.tootHeading)}
+          onClick={this.handleHeaderClick}
+          showBackButton
+          multiColumn={multiColumn}
+          extraButton={(
+            <button type='button' className='column-header__button' title={intl.formatMessage(!isExpanded ? messages.revealAll : messages.hideAll)} aria-label={intl.formatMessage(!isExpanded ? messages.revealAll : messages.hideAll)} onClick={this.handleToggleAll}><Icon id={!isExpanded ? 'eye-slash' : 'eye'} /></button>
+          )}
+        />
+
+        <ScrollContainer scrollKey='thread' shouldUpdateScroll={this.shouldUpdateScroll}>
+          <div className={classNames('scrollable', { fullscreen })} ref={this.setRef}>
+            {ancestors}
+
+            <HotKeys handlers={handlers}>
+              <div className={classNames('focusable', 'detailed-status__wrapper', `detailed-status__wrapper-${status.get('visibility')}`)} tabIndex={0} aria-label={textForScreenReader(intl, status, false, isExpanded)}>
+                <DetailedStatus
+                  key={`details-${status.get('id')}`}
+                  status={status}
+                  settings={settings}
+                  onOpenVideo={this.handleOpenVideo}
+                  onOpenMedia={this.handleOpenMedia}
+                  onReactionAdd={this.handleReactionAdd}
+                  onReactionRemove={this.handleReactionRemove}
+                  expanded={isExpanded}
+                  onToggleHidden={this.handleToggleHidden}
+                  onTranslate={this.handleTranslate}
+                  domain={domain}
+                  showMedia={this.state.showMedia}
+                  onToggleMediaVisibility={this.handleToggleMediaVisibility}
+                  pictureInPicture={pictureInPicture}
+                />
+
+                <ActionBar
+                  key={`action-bar-${status.get('id')}`}
+                  status={status}
+                  onReply={this.handleReplyClick}
+                  onFavourite={this.handleFavouriteClick}
+                  onReactionAdd={this.handleReactionAdd}
+                  onReblog={this.handleReblogClick}
+                  onBookmark={this.handleBookmarkClick}
+                  onDelete={this.handleDeleteClick}
+                  onEdit={this.handleEditClick}
+                  onDirect={this.handleDirectClick}
+                  onMention={this.handleMentionClick}
+                  onMute={this.handleMuteClick}
+                  onMuteConversation={this.handleConversationMuteClick}
+                  onBlock={this.handleBlockClick}
+                  onReport={this.handleReport}
+                  onPin={this.handlePin}
+                  onEmbed={this.handleEmbed}
+                />
+              </div>
+            </HotKeys>
+
+            {descendants}
+          </div>
+        </ScrollContainer>
+
+        <Helmet>
+          <title>{titleFromStatus(intl, status)}</title>
+          <meta name='robots' content={(isLocal && isIndexable) ? 'all' : 'noindex'} />
+          <link rel='canonical' href={status.get('url')} />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(connect(makeMapStateToProps)(Status)));
diff --git a/app/javascript/flavours/blobfox/features/subscribed_languages_modal/index.jsx b/app/javascript/flavours/blobfox/features/subscribed_languages_modal/index.jsx
new file mode 100644
index 00000000000000..31308c6ce822fa
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/subscribed_languages_modal/index.jsx
@@ -0,0 +1,127 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import { is, List as ImmutableList, Set as ImmutableSet } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { followAccount } from 'flavours/blobfox/actions/accounts';
+import { Button } from 'flavours/blobfox/components/button';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Option from 'flavours/blobfox/features/report/components/option';
+import { languages as preloadedLanguages } from 'flavours/blobfox/initial_state';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+const getAccountLanguages = createSelector([
+  (state, accountId) => state.getIn(['timelines', `account:${accountId}`, 'items'], ImmutableList()),
+  state => state.get('statuses'),
+], (statusIds, statuses) =>
+  new ImmutableSet(statusIds.map(statusId => statuses.get(statusId)).filter(status => !status.get('reblog')).map(status => status.get('language'))));
+
+const mapStateToProps = (state, { accountId }) => ({
+  acct: state.getIn(['accounts', accountId, 'acct']),
+  availableLanguages: getAccountLanguages(state, accountId),
+  selectedLanguages: ImmutableSet(state.getIn(['relationships', accountId, 'languages']) || ImmutableList()),
+});
+
+const mapDispatchToProps = (dispatch, { accountId }) => ({
+
+  onSubmit (languages) {
+    dispatch(followAccount(accountId, { languages }));
+  },
+
+});
+
+class SubscribedLanguagesModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    accountId: PropTypes.string.isRequired,
+    acct: PropTypes.string.isRequired,
+    availableLanguages: ImmutablePropTypes.setOf(PropTypes.string),
+    selectedLanguages: ImmutablePropTypes.setOf(PropTypes.string),
+    onClose: PropTypes.func.isRequired,
+    languages: PropTypes.arrayOf(PropTypes.arrayOf(PropTypes.string)),
+    intl: PropTypes.object.isRequired,
+    submit: PropTypes.func.isRequired,
+  };
+
+  static defaultProps = {
+    languages: preloadedLanguages,
+  };
+
+  state = {
+    selectedLanguages: this.props.selectedLanguages,
+  };
+
+  handleLanguageToggle = (value, checked) => {
+    const { selectedLanguages } = this.state;
+
+    if (checked) {
+      this.setState({ selectedLanguages: selectedLanguages.add(value) });
+    } else {
+      this.setState({ selectedLanguages: selectedLanguages.delete(value) });
+    }
+  };
+
+  handleSubmit = () => {
+    this.props.onSubmit(this.state.selectedLanguages.toArray());
+    this.props.onClose();
+  };
+
+  renderItem (value) {
+    const language = this.props.languages.find(language => language[0] === value);
+    const checked = this.state.selectedLanguages.includes(value);
+
+    if (!language) {
+      return null;
+    }
+
+    return (
+      <Option
+        key={value}
+        name='languages'
+        value={value}
+        label={language[1]}
+        checked={checked}
+        onToggle={this.handleLanguageToggle}
+        multiple
+      />
+    );
+  }
+
+  render () {
+    const { acct, availableLanguages, selectedLanguages, intl, onClose } = this.props;
+
+    return (
+      <div className='modal-root__modal report-dialog-modal'>
+        <div className='report-modal__target'>
+          <IconButton className='report-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={20} />
+          <FormattedMessage id='subscribed_languages.target' defaultMessage='Change subscribed languages for {target}' values={{ target: <strong>{acct}</strong> }} />
+        </div>
+
+        <div className='report-dialog-modal__container'>
+          <p className='report-dialog-modal__lead'><FormattedMessage id='subscribed_languages.lead' defaultMessage='Only posts in selected languages will appear on your home and list timelines after the change. Select none to receive posts in all languages.' /></p>
+
+          <div>
+            {availableLanguages.union(selectedLanguages).delete(null).map(value => this.renderItem(value))}
+          </div>
+
+          <div className='flex-spacer' />
+
+          <div className='report-dialog-modal__actions'>
+            <Button disabled={is(this.state.selectedLanguages, this.props.selectedLanguages)} onClick={this.handleSubmit}><FormattedMessage id='subscribed_languages.save' defaultMessage='Save changes' /></Button>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(SubscribedLanguagesModal));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/actions_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/actions_modal.jsx
new file mode 100644
index 00000000000000..6b58f9423b2c3a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/actions_modal.jsx
@@ -0,0 +1,94 @@
+import PropTypes from 'prop-types';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+import StatusContent from 'flavours/blobfox/components/status_content';
+
+import { IconButton } from '../../../components/icon_button';
+
+export default class ActionsModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    status: ImmutablePropTypes.map,
+    onClick: PropTypes.func,
+    actions: PropTypes.arrayOf(PropTypes.shape({
+      active: PropTypes.bool,
+      href: PropTypes.string,
+      icon: PropTypes.string,
+      meta: PropTypes.string,
+      name: PropTypes.string,
+      text: PropTypes.string,
+    })),
+    renderItemContents: PropTypes.func,
+  };
+
+  renderAction = (action, i) => {
+    if (action === null) {
+      return <li key={`sep-${i}`} className='dropdown-menu__separator' />;
+    }
+
+    const { icon = null, text, meta = null, active = false, href = '#' } = action;
+    let contents = this.props.renderItemContents && this.props.renderItemContents(action, i);
+
+    if (!contents) {
+      contents = (
+        <>
+          {icon && <IconButton title={text} icon={icon} role='presentation' tabIndex={-1} inverted />}
+          <div>
+            <div className={classNames({ 'actions-modal__item-label': !!meta })}>{text}</div>
+            <div>{meta}</div>
+          </div>
+        </>
+      );
+    }
+
+    return (
+      <li key={`${text}-${i}`}>
+        <a href={href} target='_blank' rel='noopener noreferrer' onClick={this.props.onClick} data-index={i} className={classNames('link', { active })}>
+          {contents}
+        </a>
+      </li>
+    );
+  };
+
+  render () {
+    const status = this.props.status && (
+      <div className='status light'>
+        <div className='boost-modal__status-header'>
+          <div className='boost-modal__status-time'>
+            <a href={this.props.status.get('url')} className='status__relative-time' target='_blank' rel='noopener noreferrer'>
+              <RelativeTimestamp timestamp={this.props.status.get('created_at')} />
+            </a>
+          </div>
+
+          <a href={this.props.status.getIn(['account', 'url'])} className='status__display-name' rel='noopener noreferrer'>
+            <div className='status__avatar'>
+              <Avatar account={this.props.status.get('account')} size={48} />
+            </div>
+
+            <DisplayName account={this.props.status.get('account')} />
+          </a>
+        </div>
+
+        <StatusContent status={this.props.status} />
+      </div>
+    );
+
+    return (
+      <div className='modal-root__modal actions-modal'>
+        {status}
+
+        <ul className={classNames({ 'with-status': !!status })}>
+          {this.props.actions.map(this.renderAction)}
+        </ul>
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/audio_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/audio_modal.jsx
new file mode 100644
index 00000000000000..afdbade68ef987
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/audio_modal.jsx
@@ -0,0 +1,61 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import Audio from 'flavours/blobfox/features/audio';
+import Footer from 'flavours/blobfox/features/picture_in_picture/components/footer';
+
+const mapStateToProps = (state, { statusId }) => ({
+  status: state.getIn(['statuses', statusId]),
+  accountStaticAvatar: state.getIn(['accounts', state.getIn(['statuses', statusId, 'account']), 'avatar_static']),
+});
+
+class AudioModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    media: ImmutablePropTypes.map.isRequired,
+    statusId: PropTypes.string.isRequired,
+    status: ImmutablePropTypes.map.isRequired,
+    accountStaticAvatar: PropTypes.string.isRequired,
+    options: PropTypes.shape({
+      autoPlay: PropTypes.bool,
+    }),
+    onClose: PropTypes.func.isRequired,
+    onChangeBackgroundColor: PropTypes.func.isRequired,
+  };
+
+  render () {
+    const { media, status, accountStaticAvatar, onClose } = this.props;
+    const options = this.props.options || {};
+    const language = status.getIn(['translation', 'language']) || status.get('language');
+    const description = media.getIn(['translation', 'description']) || media.get('description');
+
+    return (
+      <div className='modal-root__modal audio-modal'>
+        <div className='audio-modal__container'>
+          <Audio
+            src={media.get('url')}
+            alt={description}
+            lang={language}
+            duration={media.getIn(['meta', 'original', 'duration'], 0)}
+            height={150}
+            poster={media.get('preview_url') || accountStaticAvatar}
+            backgroundColor={media.getIn(['meta', 'colors', 'background'])}
+            foregroundColor={media.getIn(['meta', 'colors', 'foreground'])}
+            accentColor={media.getIn(['meta', 'colors', 'accent'])}
+            autoPlay={options.autoPlay}
+          />
+        </div>
+
+        <div className='media-modal__overlay'>
+          {status && <Footer statusId={status.get('id')} withOpenButton onClose={onClose} />}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, null, null, { forwardRef: true })(AudioModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/block_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/block_modal.jsx
new file mode 100644
index 00000000000000..cfac692324aa23
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/block_modal.jsx
@@ -0,0 +1,100 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { blockAccount } from '../../../actions/accounts';
+import { closeModal } from '../../../actions/modal';
+import { initReport } from '../../../actions/reports';
+import { Button } from '../../../components/button';
+import { makeGetAccount } from '../../../selectors';
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = state => ({
+    account: getAccount(state, state.getIn(['blocks', 'new', 'account_id'])),
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = dispatch => {
+  return {
+    onConfirm(account) {
+      dispatch(blockAccount(account.get('id')));
+    },
+
+    onBlockAndReport(account) {
+      dispatch(blockAccount(account.get('id')));
+      dispatch(initReport(account));
+    },
+
+    onClose() {
+      dispatch(closeModal({
+        modalType: undefined,
+        ignoreFocus: false,
+      }));
+    },
+  };
+};
+
+class BlockModal extends PureComponent {
+
+  static propTypes = {
+    account: PropTypes.object.isRequired,
+    onClose: PropTypes.func.isRequired,
+    onBlockAndReport: PropTypes.func.isRequired,
+    onConfirm: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleClick = () => {
+    this.props.onClose();
+    this.props.onConfirm(this.props.account);
+  };
+
+  handleSecondary = () => {
+    this.props.onClose();
+    this.props.onBlockAndReport(this.props.account);
+  };
+
+  handleCancel = () => {
+    this.props.onClose();
+  };
+
+  render () {
+    const { account } = this.props;
+
+    return (
+      <div className='modal-root__modal block-modal'>
+        <div className='block-modal__container'>
+          <p>
+            <FormattedMessage
+              id='confirmations.block.message'
+              defaultMessage='Are you sure you want to block {name}?'
+              values={{ name: <strong>@{account.get('acct')}</strong> }}
+            />
+          </p>
+        </div>
+
+        <div className='block-modal__action-bar'>
+          <Button onClick={this.handleCancel} className='block-modal__cancel-button'>
+            <FormattedMessage id='confirmation_modal.cancel' defaultMessage='Cancel' />
+          </Button>
+          <Button onClick={this.handleSecondary} className='confirmation-modal__secondary-button'>
+            <FormattedMessage id='confirmations.block.block_and_report' defaultMessage='Block & Report' />
+          </Button>
+          <Button onClick={this.handleClick} autoFocus>
+            <FormattedMessage id='confirmations.block.confirm' defaultMessage='Block' />
+          </Button>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(injectIntl(BlockModal));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/boost_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/boost_modal.jsx
new file mode 100644
index 00000000000000..8a1ebe77b4c187
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/boost_modal.jsx
@@ -0,0 +1,131 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { changeBoostPrivacy } from 'flavours/blobfox/actions/boosts';
+import AttachmentList from 'flavours/blobfox/components/attachment_list';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import VisibilityIcon from 'flavours/blobfox/components/status_visibility_icon';
+import PrivacyDropdown from 'flavours/blobfox/features/compose/components/privacy_dropdown';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { Avatar } from '../../../components/avatar';
+import { Button } from '../../../components/button';
+import { DisplayName } from '../../../components/display_name';
+import { RelativeTimestamp } from '../../../components/relative_timestamp';
+import StatusContent from '../../../components/status_content';
+
+const messages = defineMessages({
+  cancel_reblog: { id: 'status.cancel_reblog_private', defaultMessage: 'Unboost' },
+  reblog: { id: 'status.reblog', defaultMessage: 'Boost' },
+});
+
+const mapStateToProps = state => {
+  return {
+    privacy: state.getIn(['boosts', 'new', 'privacy']),
+  };
+};
+
+const mapDispatchToProps = dispatch => {
+  return {
+    onChangeBoostPrivacy(value) {
+      dispatch(changeBoostPrivacy(value));
+    },
+  };
+};
+
+class BoostModal extends ImmutablePureComponent {
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    onReblog: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    missingMediaDescription: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  handleReblog = () => {
+    this.props.onReblog(this.props.status, this.props.privacy);
+    this.props.onClose();
+  };
+
+  handleAccountClick = (e) => {
+    if (e.button === 0) {
+      e.preventDefault();
+      this.props.onClose();
+      this.props.history.push(`/@${this.props.status.getIn(['account', 'acct'])}`);
+    }
+  };
+
+  _findContainer = () => {
+    return document.getElementsByClassName('modal-root__container')[0];
+  };
+
+  render () {
+    const { status, missingMediaDescription, privacy, intl } = this.props;
+    const buttonText = status.get('reblogged') ? messages.cancel_reblog : messages.reblog;
+
+    return (
+      <div className='modal-root__modal boost-modal'>
+        <div className='boost-modal__container'>
+          <div className={classNames('status', `status-${status.get('visibility')}`, 'light')}>
+            <div className='boost-modal__status-header'>
+              <div className='boost-modal__status-time'>
+                <a href={status.get('url')} className='status__relative-time' target='_blank' rel='noopener noreferrer'>
+                  <VisibilityIcon visibility={status.get('visibility')} />
+                  <RelativeTimestamp timestamp={status.get('created_at')} /></a>
+              </div>
+
+              <a onClick={this.handleAccountClick} href={status.getIn(['account', 'url'])} className='status__display-name'>
+                <div className='status__avatar'>
+                  <Avatar account={status.get('account')} size={48} />
+                </div>
+
+                <DisplayName account={status.get('account')} />
+              </a>
+            </div>
+
+            <StatusContent status={status} />
+
+            {status.get('media_attachments').size > 0 && (
+              <AttachmentList
+                compact
+                media={status.get('media_attachments')}
+              />
+            )}
+          </div>
+        </div>
+
+        <div className='boost-modal__action-bar'>
+          <div>
+            { missingMediaDescription ?
+              <FormattedMessage id='boost_modal.missing_description' defaultMessage='This toot contains some media without description' />
+              :
+              <FormattedMessage id='boost_modal.combo' defaultMessage='You can press {combo} to skip this next time' values={{ combo: <span>Shift + <Icon id='retweet' /></span> }} />
+            }
+          </div>
+
+          {status.get('visibility') !== 'private' && !status.get('reblogged') && (
+            <PrivacyDropdown
+              noDirect
+              value={privacy}
+              container={this._findContainer}
+              onChange={this.props.onChangeBoostPrivacy}
+            />
+          )}
+          <Button text={intl.formatMessage(buttonText)} onClick={this.handleReblog} autoFocus />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(connect(mapStateToProps, mapDispatchToProps)(injectIntl(BoostModal)));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/bundle.jsx b/app/javascript/flavours/blobfox/features/ui/components/bundle.jsx
new file mode 100644
index 00000000000000..15c4220b3483ad
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/bundle.jsx
@@ -0,0 +1,106 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+const emptyComponent = () => null;
+const noop = () => { };
+
+class Bundle extends PureComponent {
+
+  static propTypes = {
+    fetchComponent: PropTypes.func.isRequired,
+    loading: PropTypes.func,
+    error: PropTypes.func,
+    children: PropTypes.func.isRequired,
+    renderDelay: PropTypes.number,
+    onFetch: PropTypes.func,
+    onFetchSuccess: PropTypes.func,
+    onFetchFail: PropTypes.func,
+  };
+
+  static defaultProps = {
+    loading: emptyComponent,
+    error: emptyComponent,
+    renderDelay: 0,
+    onFetch: noop,
+    onFetchSuccess: noop,
+    onFetchFail: noop,
+  };
+
+  static cache = new Map;
+
+  state = {
+    mod: undefined,
+    forceRender: false,
+  };
+
+  UNSAFE_componentWillMount() {
+    this.load(this.props);
+  }
+
+  UNSAFE_componentWillReceiveProps(nextProps) {
+    if (nextProps.fetchComponent !== this.props.fetchComponent) {
+      this.load(nextProps);
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.timeout) {
+      clearTimeout(this.timeout);
+    }
+  }
+
+  load = (props) => {
+    const { fetchComponent, onFetch, onFetchSuccess, onFetchFail, renderDelay } = props || this.props;
+    const cachedMod = Bundle.cache.get(fetchComponent);
+
+    if (fetchComponent === undefined) {
+      this.setState({ mod: null });
+      return Promise.resolve();
+    }
+
+    onFetch();
+
+    if (cachedMod) {
+      this.setState({ mod: cachedMod.default });
+      onFetchSuccess();
+      return Promise.resolve();
+    }
+
+    this.setState({ mod: undefined });
+
+    if (renderDelay !== 0) {
+      this.timestamp = new Date();
+      this.timeout = setTimeout(() => this.setState({ forceRender: true }), renderDelay);
+    }
+
+    return fetchComponent()
+      .then((mod) => {
+        Bundle.cache.set(fetchComponent, mod);
+        this.setState({ mod: mod.default });
+        onFetchSuccess();
+      })
+      .catch((error) => {
+        this.setState({ mod: null });
+        onFetchFail(error);
+      });
+  };
+
+  render() {
+    const { loading: Loading, error: Error, children, renderDelay } = this.props;
+    const { mod, forceRender } = this.state;
+    const elapsed = this.timestamp ? (new Date() - this.timestamp) : renderDelay;
+
+    if (mod === undefined) {
+      return (elapsed >= renderDelay || forceRender) ? <Loading /> : null;
+    }
+
+    if (mod === null) {
+      return <Error onRetry={this.load} />;
+    }
+
+    return children(mod);
+  }
+
+}
+
+export default Bundle;
diff --git a/app/javascript/flavours/blobfox/features/ui/components/bundle_column_error.jsx b/app/javascript/flavours/blobfox/features/ui/components/bundle_column_error.jsx
new file mode 100644
index 00000000000000..59e8ffda043917
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/bundle_column_error.jsx
@@ -0,0 +1,166 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { Helmet } from 'react-helmet';
+import { Link } from 'react-router-dom';
+
+import { Button } from 'flavours/blobfox/components/button';
+import Column from 'flavours/blobfox/components/column';
+import { autoPlayGif } from 'flavours/blobfox/initial_state';
+
+class GIF extends PureComponent {
+
+  static propTypes = {
+    src: PropTypes.string.isRequired,
+    staticSrc: PropTypes.string.isRequired,
+    className: PropTypes.string,
+    animate: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    animate: autoPlayGif,
+  };
+
+  state = {
+    hovering: false,
+  };
+
+  handleMouseEnter = () => {
+    const { animate } = this.props;
+
+    if (!animate) {
+      this.setState({ hovering: true });
+    }
+  };
+
+  handleMouseLeave = () => {
+    const { animate } = this.props;
+
+    if (!animate) {
+      this.setState({ hovering: false });
+    }
+  };
+
+  render () {
+    const { src, staticSrc, className, animate } = this.props;
+    const { hovering } = this.state;
+
+    return (
+      <img
+        className={className}
+        src={(hovering || animate) ? src : staticSrc}
+        alt=''
+        role='presentation'
+        onMouseEnter={this.handleMouseEnter}
+        onMouseLeave={this.handleMouseLeave}
+      />
+    );
+  }
+
+}
+
+class CopyButton extends PureComponent {
+
+  static propTypes = {
+    children: PropTypes.node.isRequired,
+    value: PropTypes.string.isRequired,
+  };
+
+  state = {
+    copied: false,
+  };
+
+  handleClick = () => {
+    const { value } = this.props;
+    navigator.clipboard.writeText(value);
+    this.setState({ copied: true });
+    this.timeout = setTimeout(() => this.setState({ copied: false }), 700);
+  };
+
+  componentWillUnmount () {
+    if (this.timeout) clearTimeout(this.timeout);
+  }
+
+  render () {
+    const { children } = this.props;
+    const { copied } = this.state;
+
+    return (
+      <Button onClick={this.handleClick} className={copied ? 'copied' : 'copyable'}>{copied ? <FormattedMessage id='copypaste.copied' defaultMessage='Copied' /> : children}</Button>
+    );
+  }
+
+}
+
+class BundleColumnError extends PureComponent {
+
+  static propTypes = {
+    errorType: PropTypes.oneOf(['routing', 'network', 'error']),
+    onRetry: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+    multiColumn: PropTypes.bool,
+    stacktrace: PropTypes.string,
+  };
+
+  static defaultProps = {
+    errorType: 'routing',
+  };
+
+  handleRetry = () => {
+    const { onRetry } = this.props;
+
+    if (onRetry) {
+      onRetry();
+    }
+  };
+
+  render () {
+    const { errorType, multiColumn, stacktrace } = this.props;
+
+    let title, body;
+
+    switch(errorType) {
+    case 'routing':
+      title = <FormattedMessage id='bundle_column_error.routing.title' defaultMessage='404' />;
+      body = <FormattedMessage id='bundle_column_error.routing.body' defaultMessage='The requested page could not be found. Are you sure the URL in the address bar is correct?' />;
+      break;
+    case 'network':
+      title = <FormattedMessage id='bundle_column_error.network.title' defaultMessage='Network error' />;
+      body = <FormattedMessage id='bundle_column_error.network.body' defaultMessage='There was an error when trying to load this page. This could be due to a temporary problem with your internet connection or this server.' />;
+      break;
+    case 'error':
+      title = <FormattedMessage id='bundle_column_error.error.title' defaultMessage='Oh, no!' />;
+      body = <FormattedMessage id='bundle_column_error.error.body' defaultMessage='The requested page could not be rendered. It could be due to a bug in our code, or a browser compatibility issue.' />;
+      break;
+    }
+
+    return (
+      <Column bindToDocument={!multiColumn}>
+        <div className='error-column'>
+          <GIF src='/oops.gif' staticSrc='/oops.png' className='error-column__image' />
+
+          <div className='error-column__message'>
+            <h1>{title}</h1>
+            <p>{body}</p>
+
+            <div className='error-column__message__actions'>
+              {errorType === 'network' && <Button onClick={this.handleRetry}><FormattedMessage id='bundle_column_error.retry' defaultMessage='Try again' /></Button>}
+              {errorType === 'error' && <CopyButton value={stacktrace}><FormattedMessage id='bundle_column_error.copy_stacktrace' defaultMessage='Copy error report' /></CopyButton>}
+              <Link to='/' className={classNames('button', { 'button-tertiary': errorType !== 'routing' })}><FormattedMessage id='bundle_column_error.return' defaultMessage='Go back home' /></Link>
+            </div>
+          </div>
+        </div>
+
+        <Helmet>
+          <meta name='robots' content='noindex' />
+        </Helmet>
+      </Column>
+    );
+  }
+
+}
+
+export default injectIntl(BundleColumnError);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/bundle_modal_error.jsx b/app/javascript/flavours/blobfox/features/ui/components/bundle_modal_error.jsx
new file mode 100644
index 00000000000000..67dba3ce0cefcc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/bundle_modal_error.jsx
@@ -0,0 +1,54 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { IconButton } from '../../../components/icon_button';
+
+const messages = defineMessages({
+  error: { id: 'bundle_modal_error.message', defaultMessage: 'Something went wrong while loading this component.' },
+  retry: { id: 'bundle_modal_error.retry', defaultMessage: 'Try again' },
+  close: { id: 'bundle_modal_error.close', defaultMessage: 'Close' },
+});
+
+class BundleModalError extends PureComponent {
+
+  static propTypes = {
+    onRetry: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleRetry = () => {
+    this.props.onRetry();
+  };
+
+  render () {
+    const { onClose, intl: { formatMessage } } = this.props;
+
+    // Keep the markup in sync with <ModalLoading />
+    // (make sure they have the same dimensions)
+    return (
+      <div className='modal-root__modal error-modal'>
+        <div className='error-modal__body'>
+          <IconButton title={formatMessage(messages.retry)} icon='refresh' onClick={this.handleRetry} size={64} />
+          {formatMessage(messages.error)}
+        </div>
+
+        <div className='error-modal__footer'>
+          <div>
+            <button
+              onClick={onClose}
+              className='error-modal__nav onboarding-modal__skip'
+            >
+              {formatMessage(messages.close)}
+            </button>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(BundleModalError);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/column.jsx b/app/javascript/flavours/blobfox/features/ui/components/column.jsx
new file mode 100644
index 00000000000000..6e8ff93e190f3b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/column.jsx
@@ -0,0 +1,78 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { debounce } from 'lodash';
+
+import { isMobile } from '../../../is_mobile';
+import { scrollTop } from '../../../scroll';
+
+import ColumnHeader from './column_header';
+
+export default class Column extends PureComponent {
+
+  static propTypes = {
+    heading: PropTypes.string,
+    icon: PropTypes.string,
+    children: PropTypes.node,
+    active: PropTypes.bool,
+    hideHeadingOnMobile: PropTypes.bool,
+    name: PropTypes.string,
+    bindToDocument: PropTypes.bool,
+  };
+
+  handleHeaderClick = () => {
+    const scrollable = this.props.bindToDocument ? document.scrollingElement : this.node.querySelector('.scrollable');
+
+    if (!scrollable) {
+      return;
+    }
+
+    this._interruptScrollAnimation = scrollTop(scrollable);
+  };
+
+  scrollTop () {
+    const scrollable = this.props.bindToDocument ? document.scrollingElement : this.node.querySelector('.scrollable');
+
+    if (!scrollable) {
+      return;
+    }
+
+    this._interruptScrollAnimation = scrollTop(scrollable);
+  }
+
+
+  handleScroll = debounce(() => {
+    if (typeof this._interruptScrollAnimation !== 'undefined') {
+      this._interruptScrollAnimation();
+    }
+  }, 200);
+
+  setRef = (c) => {
+    this.node = c;
+  };
+
+  render () {
+    const { heading, icon, children, active, hideHeadingOnMobile, name } = this.props;
+
+    const showHeading = heading && (!hideHeadingOnMobile || (hideHeadingOnMobile && !isMobile(window.innerWidth)));
+
+    const columnHeaderId = showHeading && heading.replace(/ /g, '-');
+    const header = showHeading && (
+      <ColumnHeader icon={icon} active={active} type={heading} onClick={this.handleHeaderClick} columnHeaderId={columnHeaderId} />
+    );
+    return (
+      <div
+        ref={this.setRef}
+        role='region'
+        data-column={name}
+        aria-labelledby={columnHeaderId}
+        className='column'
+        onScroll={this.handleScroll}
+      >
+        {header}
+        {children}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/column_header.jsx b/app/javascript/flavours/blobfox/features/ui/components/column_header.jsx
new file mode 100644
index 00000000000000..5d535f2e4fb0fe
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/column_header.jsx
@@ -0,0 +1,40 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import { Icon }  from 'flavours/blobfox/components/icon';
+
+export default class ColumnHeader extends PureComponent {
+
+  static propTypes = {
+    icon: PropTypes.string,
+    type: PropTypes.string,
+    active: PropTypes.bool,
+    onClick: PropTypes.func,
+    columnHeaderId: PropTypes.string,
+  };
+
+  handleClick = () => {
+    this.props.onClick();
+  };
+
+  render () {
+    const { icon, type, active, columnHeaderId } = this.props;
+    let iconElement = '';
+
+    if (icon) {
+      iconElement = <Icon id={icon} fixedWidth className='column-header__icon' />;
+    }
+
+    return (
+      <h1 className={classNames('column-header', { active })} id={columnHeaderId || null}>
+        <button onClick={this.handleClick}>
+          {iconElement}
+          {type}
+        </button>
+      </h1>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/column_link.jsx b/app/javascript/flavours/blobfox/features/ui/components/column_link.jsx
new file mode 100644
index 00000000000000..04c09632a576ca
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/column_link.jsx
@@ -0,0 +1,57 @@
+import PropTypes from 'prop-types';
+
+import classNames from 'classnames';
+import { NavLink } from 'react-router-dom';
+
+import { Icon } from 'flavours/blobfox/components/icon';
+
+const ColumnLink = ({ icon, text, to, onClick, href, method, badge, transparent, ...other }) => {
+  const className = classNames('column-link', { 'column-link--transparent': transparent });
+  const badgeElement = typeof badge !== 'undefined' ? <span className='column-link__badge'>{badge}</span> : null;
+  const iconElement = typeof icon === 'string' ? <Icon id={icon} fixedWidth className='column-link__icon' /> : icon;
+
+  if (href) {
+    return (
+      <a href={href} className={className} data-method={method} title={text} {...other}>
+        {iconElement}
+        <span>{text}</span>
+        {badgeElement}
+      </a>
+    );
+  } else if (to) {
+    return (
+      <NavLink to={to} className={className} title={text} {...other}>
+        {iconElement}
+        <span>{text}</span>
+        {badgeElement}
+      </NavLink>
+    );
+  } else {
+    const handleOnClick = (e) => {
+      e.preventDefault();
+      e.stopPropagation();
+      return onClick(e);
+    };
+    return (
+      // eslint-disable-next-line jsx-a11y/anchor-is-valid -- intentional to have the same look and feel as other menu items
+      <a href='#' onClick={onClick && handleOnClick} className={className} title={text} {...other} tabIndex={0}>
+        {iconElement}
+        <span>{text}</span>
+        {badgeElement}
+      </a>
+    );
+  }
+};
+
+ColumnLink.propTypes = {
+  icon: PropTypes.oneOfType([PropTypes.string, PropTypes.node]).isRequired,
+  text: PropTypes.string.isRequired,
+  to: PropTypes.string,
+  onClick: PropTypes.func,
+  href: PropTypes.string,
+  method: PropTypes.string,
+  badge: PropTypes.node,
+  transparent: PropTypes.bool,
+};
+
+export default ColumnLink;
diff --git a/app/javascript/flavours/blobfox/features/ui/components/column_loading.jsx b/app/javascript/flavours/blobfox/features/ui/components/column_loading.jsx
new file mode 100644
index 00000000000000..d00da6a66f9bc0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/column_loading.jsx
@@ -0,0 +1,32 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import Column from 'flavours/blobfox/components/column';
+import ColumnHeader from 'flavours/blobfox/components/column_header';
+
+export default class ColumnLoading extends ImmutablePureComponent {
+
+  static propTypes = {
+    title: PropTypes.oneOfType([PropTypes.node, PropTypes.string]),
+    icon: PropTypes.string,
+    multiColumn: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    title: '',
+    icon: '',
+  };
+
+  render() {
+    let { title, icon, multiColumn } = this.props;
+
+    return (
+      <Column>
+        <ColumnHeader icon={icon} title={title} multiColumn={multiColumn} focusable={false} placeholder />
+        <div className='scrollable' />
+      </Column>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/column_subheading.jsx b/app/javascript/flavours/blobfox/features/ui/components/column_subheading.jsx
new file mode 100644
index 00000000000000..e970a0bfdd0e1e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/column_subheading.jsx
@@ -0,0 +1,15 @@
+import PropTypes from 'prop-types';
+
+const ColumnSubheading = ({ text }) => {
+  return (
+    <div className='column-subheading'>
+      {text}
+    </div>
+  );
+};
+
+ColumnSubheading.propTypes = {
+  text: PropTypes.string.isRequired,
+};
+
+export default ColumnSubheading;
diff --git a/app/javascript/flavours/blobfox/features/ui/components/columns_area.jsx b/app/javascript/flavours/blobfox/features/ui/components/columns_area.jsx
new file mode 100644
index 00000000000000..05a02ae6ceea9c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/columns_area.jsx
@@ -0,0 +1,180 @@
+import PropTypes from 'prop-types';
+import { Children, cloneElement } from 'react';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import { supportsPassiveEvents } from 'detect-passive-events';
+
+import { scrollRight } from '../../../scroll';
+import BundleContainer from '../containers/bundle_container';
+import {
+  Compose,
+  Notifications,
+  HomeTimeline,
+  CommunityTimeline,
+  PublicTimeline,
+  HashtagTimeline,
+  DirectTimeline,
+  FavouritedStatuses,
+  BookmarkedStatuses,
+  ListTimeline,
+  Directory,
+} from '../util/async-components';
+
+import BundleColumnError from './bundle_column_error';
+import ColumnLoading from './column_loading';
+import ComposePanel from './compose_panel';
+import DrawerLoading from './drawer_loading';
+import NavigationPanel from './navigation_panel';
+
+const componentMap = {
+  'COMPOSE': Compose,
+  'HOME': HomeTimeline,
+  'NOTIFICATIONS': Notifications,
+  'PUBLIC': PublicTimeline,
+  'REMOTE': PublicTimeline,
+  'COMMUNITY': CommunityTimeline,
+  'HASHTAG': HashtagTimeline,
+  'DIRECT': DirectTimeline,
+  'FAVOURITES': FavouritedStatuses,
+  'BOOKMARKS': BookmarkedStatuses,
+  'LIST': ListTimeline,
+  'DIRECTORY': Directory,
+};
+
+export default class ColumnsArea extends ImmutablePureComponent {
+  static propTypes = {
+    columns: ImmutablePropTypes.list.isRequired,
+    singleColumn: PropTypes.bool,
+    children: PropTypes.node,
+    openSettings: PropTypes.func,
+  };
+
+  // Corresponds to (max-width: $no-gap-breakpoint + 285px - 1px) in SCSS
+  mediaQuery = 'matchMedia' in window && window.matchMedia('(max-width: 1174px)');
+
+  state = {
+    renderComposePanel: !(this.mediaQuery && this.mediaQuery.matches),
+  };
+
+  componentDidMount() {
+    if (!this.props.singleColumn) {
+      this.node.addEventListener('wheel', this.handleWheel, supportsPassiveEvents ? { passive: true } : false);
+    }
+
+    if (this.mediaQuery) {
+      if (this.mediaQuery.addEventListener) {
+        this.mediaQuery.addEventListener('change', this.handleLayoutChange);
+      } else {
+        this.mediaQuery.addListener(this.handleLayoutChange);
+      }
+      this.setState({ renderComposePanel: !this.mediaQuery.matches });
+    }
+
+    this.isRtlLayout = document.getElementsByTagName('body')[0].classList.contains('rtl');
+  }
+
+  UNSAFE_componentWillUpdate(nextProps) {
+    if (this.props.singleColumn !== nextProps.singleColumn && nextProps.singleColumn) {
+      this.node.removeEventListener('wheel', this.handleWheel);
+    }
+  }
+
+  componentDidUpdate(prevProps) {
+    if (this.props.singleColumn !== prevProps.singleColumn && !this.props.singleColumn) {
+      this.node.addEventListener('wheel', this.handleWheel, supportsPassiveEvents ? { passive: true } : false);
+    }
+  }
+
+  componentWillUnmount () {
+    if (!this.props.singleColumn) {
+      this.node.removeEventListener('wheel', this.handleWheel);
+    }
+
+    if (this.mediaQuery) {
+      if (this.mediaQuery.removeEventListener) {
+        this.mediaQuery.removeEventListener('change', this.handleLayoutChange);
+      } else {
+        this.mediaQuery.removeListener(this.handleLayoutChange);
+      }
+    }
+  }
+
+  handleChildrenContentChange() {
+    if (!this.props.singleColumn) {
+      const modifier = this.isRtlLayout ? -1 : 1;
+      this._interruptScrollAnimation = scrollRight(this.node, (this.node.scrollWidth - window.innerWidth) * modifier);
+    }
+  }
+
+  handleLayoutChange = (e) => {
+    this.setState({ renderComposePanel: !e.matches });
+  };
+
+  handleWheel = () => {
+    if (typeof this._interruptScrollAnimation !== 'function') {
+      return;
+    }
+
+    this._interruptScrollAnimation();
+  };
+
+  setRef = (node) => {
+    this.node = node;
+  };
+
+  renderLoading = columnId => () => {
+    return columnId === 'COMPOSE' ? <DrawerLoading /> : <ColumnLoading multiColumn />;
+  };
+
+  renderError = (props) => {
+    return <BundleColumnError multiColumn errorType='network' {...props} />;
+  };
+
+  render () {
+    const { columns, children, singleColumn, openSettings } = this.props;
+    const { renderComposePanel } = this.state;
+
+    if (singleColumn) {
+      return (
+        <div className='columns-area__panels'>
+          <div className='columns-area__panels__pane columns-area__panels__pane--compositional'>
+            <div className='columns-area__panels__pane__inner'>
+              {renderComposePanel && <ComposePanel />}
+            </div>
+          </div>
+
+          <div className='columns-area__panels__main'>
+            <div className='tabs-bar__wrapper'><div id='tabs-bar__portal' /></div>
+            <div className='columns-area columns-area--mobile'>{children}</div>
+          </div>
+
+          <div className='columns-area__panels__pane columns-area__panels__pane--start columns-area__panels__pane--navigational'>
+            <div className='columns-area__panels__pane__inner'>
+              <NavigationPanel onOpenSettings={openSettings} />
+            </div>
+          </div>
+        </div>
+      );
+    }
+
+    return (
+      <div className='columns-area' ref={this.setRef}>
+        {columns.map(column => {
+          const params = column.get('params', null) === null ? null : column.get('params').toJS();
+          const other  = params && params.other ? params.other : {};
+
+          return (
+            <BundleContainer key={column.get('uuid')} fetchComponent={componentMap[column.get('id')]} loading={this.renderLoading(column.get('id'))} error={this.renderError}>
+              {SpecificComponent => <SpecificComponent columnId={column.get('uuid')} params={params} multiColumn {...other} />}
+            </BundleContainer>
+          );
+        })}
+
+        {Children.map(children, child => cloneElement(child, { multiColumn: true }))}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/compare_history_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/compare_history_modal.jsx
new file mode 100644
index 00000000000000..3cc75328076965
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/compare_history_modal.jsx
@@ -0,0 +1,110 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import { connect } from 'react-redux';
+
+import escapeTextContentForBrowser from 'escape-html';
+
+import { closeModal } from 'flavours/blobfox/actions/modal';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import InlineAccount from 'flavours/blobfox/components/inline_account';
+import MediaAttachments from 'flavours/blobfox/components/media_attachments';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+import emojify from 'flavours/blobfox/features/emoji/emoji';
+
+const mapStateToProps = (state, { statusId }) => ({
+  language: state.getIn(['statuses', statusId, 'language']),
+  versions: state.getIn(['history', statusId, 'items']),
+});
+
+const mapDispatchToProps = dispatch => ({
+
+  onClose() {
+    dispatch(closeModal({
+      modalType: undefined,
+      ignoreFocus: false,
+    }));
+  },
+
+});
+
+class CompareHistoryModal extends PureComponent {
+
+  static propTypes = {
+    onClose: PropTypes.func.isRequired,
+    index: PropTypes.number.isRequired,
+    statusId: PropTypes.string.isRequired,
+    language: PropTypes.string.isRequired,
+    versions: ImmutablePropTypes.list.isRequired,
+  };
+
+  render () {
+    const { index, versions, language, onClose } = this.props;
+    const currentVersion = versions.get(index);
+
+    const emojiMap = currentVersion.get('emojis').reduce((obj, emoji) => {
+      obj[`:${emoji.get('shortcode')}:`] = emoji.toJS();
+      return obj;
+    }, {});
+
+    const content = { __html: emojify(currentVersion.get('content'), emojiMap) };
+    const spoilerContent = { __html: emojify(escapeTextContentForBrowser(currentVersion.get('spoiler_text')), emojiMap) };
+
+    const formattedDate = <RelativeTimestamp timestamp={currentVersion.get('created_at')} short={false} />;
+    const formattedName = <InlineAccount accountId={currentVersion.get('account')} />;
+
+    const label = currentVersion.get('original') ? (
+      <FormattedMessage id='status.history.created' defaultMessage='{name} created {date}' values={{ name: formattedName, date: formattedDate }} />
+    ) : (
+      <FormattedMessage id='status.history.edited' defaultMessage='{name} edited {date}' values={{ name: formattedName, date: formattedDate }} />
+    );
+
+    return (
+      <div className='modal-root__modal compare-history-modal'>
+        <div className='report-modal__target'>
+          <IconButton className='report-modal__close' icon='times' onClick={onClose} size={20} />
+          {label}
+        </div>
+
+        <div className='compare-history-modal__container'>
+          <div className='status__content'>
+            {currentVersion.get('spoiler_text').length > 0 && (
+              <>
+                <div className='translate' dangerouslySetInnerHTML={spoilerContent} lang={language} />
+                <hr />
+              </>
+            )}
+
+            <div className='status__content__text status__content__text--visible translate' dangerouslySetInnerHTML={content} lang={language} />
+
+            {!!currentVersion.get('poll') && (
+              <div className='poll'>
+                <ul>
+                  {currentVersion.getIn(['poll', 'options']).map(option => (
+                    <li key={option.get('title')}>
+                      <span className='poll__input disabled' />
+
+                      <span
+                        className='poll__option__text translate'
+                        dangerouslySetInnerHTML={{ __html: emojify(escapeTextContentForBrowser(option.get('title')), emojiMap) }}
+                        lang={language}
+                      />
+                    </li>
+                  ))}
+                </ul>
+              </div>
+            )}
+
+            <MediaAttachments status={currentVersion} lang={language} />
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(CompareHistoryModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/compose_panel.jsx b/app/javascript/flavours/blobfox/features/ui/components/compose_panel.jsx
new file mode 100644
index 00000000000000..c6cda04da767f5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/compose_panel.jsx
@@ -0,0 +1,62 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { connect } from 'react-redux';
+
+import { mountCompose, unmountCompose } from 'flavours/blobfox/actions/compose';
+import ServerBanner from 'flavours/blobfox/components/server_banner';
+import ComposeFormContainer from 'flavours/blobfox/features/compose/containers/compose_form_container';
+import NavigationContainer from 'flavours/blobfox/features/compose/containers/navigation_container';
+import SearchContainer from 'flavours/blobfox/features/compose/containers/search_container';
+
+import LinkFooter from './link_footer';
+
+class ComposePanel extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object.isRequired,
+  };
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(mountCompose());
+  }
+
+  componentWillUnmount () {
+    const { dispatch } = this.props;
+    dispatch(unmountCompose());
+  }
+
+  render() {
+    const { signedIn } = this.context.identity;
+
+    return (
+      <div className='compose-panel'>
+        <SearchContainer openInRoute />
+
+        {!signedIn && (
+          <>
+            <ServerBanner />
+            <div className='flex-spacer' />
+          </>
+        )}
+
+        {signedIn && (
+          <>
+            <NavigationContainer />
+            <ComposeFormContainer singleColumn />
+          </>
+        )}
+
+        <LinkFooter />
+      </div>
+    );
+  }
+
+}
+
+export default connect()(ComposePanel);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/confirmation_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/confirmation_modal.jsx
new file mode 100644
index 00000000000000..51a501595bb6e0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/confirmation_modal.jsx
@@ -0,0 +1,83 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { injectIntl, FormattedMessage } from 'react-intl';
+
+import { Button } from '../../../components/button';
+
+class ConfirmationModal extends PureComponent {
+
+  static propTypes = {
+    message: PropTypes.node.isRequired,
+    confirm: PropTypes.string.isRequired,
+    onClose: PropTypes.func.isRequired,
+    onConfirm: PropTypes.func.isRequired,
+    secondary: PropTypes.string,
+    onSecondary: PropTypes.func,
+    closeWhenConfirm: PropTypes.bool,
+    onDoNotAsk: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+  };
+
+  static defaultProps = {
+    closeWhenConfirm: true,
+  };
+
+  handleClick = () => {
+    if (this.props.closeWhenConfirm) {
+      this.props.onClose();
+    }
+    this.props.onConfirm();
+    if (this.props.onDoNotAsk && this.doNotAskCheckbox.checked) {
+      this.props.onDoNotAsk();
+    }
+  };
+
+  handleSecondary = () => {
+    this.props.onClose();
+    this.props.onSecondary();
+  };
+
+  handleCancel = () => {
+    this.props.onClose();
+  };
+
+  setDoNotAskRef = (c) => {
+    this.doNotAskCheckbox = c;
+  };
+
+  render () {
+    const { message, confirm, secondary, onDoNotAsk } = this.props;
+
+    return (
+      <div className='modal-root__modal confirmation-modal'>
+        <div className='confirmation-modal__container'>
+          {message}
+        </div>
+
+        <div>
+          { onDoNotAsk && (
+            <div className='confirmation-modal__do_not_ask_again'>
+              <input type='checkbox' id='confirmation-modal__do_not_ask_again-checkbox' ref={this.setDoNotAskRef} />
+              <label htmlFor='confirmation-modal__do_not_ask_again-checkbox'>
+                <FormattedMessage id='confirmation_modal.do_not_ask_again' defaultMessage='Do not ask for confirmation again' />
+              </label>
+            </div>
+          )}
+          <div className='confirmation-modal__action-bar'>
+            <Button onClick={this.handleCancel} className='confirmation-modal__cancel-button'>
+              <FormattedMessage id='confirmation_modal.cancel' defaultMessage='Cancel' />
+            </Button>
+            {secondary !== undefined && (
+              <Button text={secondary} onClick={this.handleSecondary} className='confirmation-modal__secondary-button' />
+            )}
+            <Button text={confirm} onClick={this.handleClick} autoFocus />
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ConfirmationModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/deprecated_settings_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/deprecated_settings_modal.jsx
new file mode 100644
index 00000000000000..90b0084b175dc0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/deprecated_settings_modal.jsx
@@ -0,0 +1,82 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+
+import { Button } from 'flavours/blobfox/components/button';
+import { Icon } from 'flavours/blobfox/components/icon';
+import illustration from 'flavours/blobfox/images/logo_warn_blobfox.svg';
+import { preferenceLink } from 'flavours/blobfox/utils/backend_links';
+
+const messages = defineMessages({
+  discardChanges: { id: 'confirmations.deprecated_settings.confirm', defaultMessage: 'Use Mastodon preferences' },
+  user_setting_expand_spoilers: { id: 'settings.enable_content_warnings_auto_unfold', defaultMessage: 'Automatically unfold content-warnings' },
+  user_setting_disable_swiping: { id: 'settings.swipe_to_change_columns', defaultMessage: 'Allow swiping to change columns (Mobile only)' },
+});
+
+class DeprecatedSettingsModal extends PureComponent {
+
+  static propTypes = {
+    settings: ImmutablePropTypes.list.isRequired,
+    onClose: PropTypes.func.isRequired,
+    onConfirm: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleClick = () => {
+    this.props.onConfirm();
+    this.props.onClose();
+  };
+
+  render () {
+    const { settings, intl } = this.props;
+
+    return (
+      <div className='modal-root__modal confirmation-modal'>
+        <div className='confirmation-modal__container'>
+
+          <img src={illustration} className='modal-warning' alt='' />
+
+          <FormattedMessage
+            id='confirmations.deprecated_settings.message'
+            defaultMessage='Some of the blobfox-soc device-specific {app_settings} you are using have been replaced by Mastodon {preferences} and will be overriden:'
+            values={{
+              app_settings: (
+                <strong className='deprecated-settings-label'>
+                  <Icon id='cogs' /> <FormattedMessage id='navigation_bar.app_settings' defaultMessage='App settings' />
+                </strong>
+              ),
+              preferences: (
+                <strong className='deprecated-settings-label'>
+                  <Icon id='cog' /> <FormattedMessage id='navigation_bar.preferences' defaultMessage='Preferences' />
+                </strong>
+              ),
+            }}
+          />
+
+          <div className='deprecated-settings-info'>
+            <ul>
+              { settings.map((setting_name) => (
+                <li key={setting_name}>
+                  <a href={preferenceLink(setting_name)}><FormattedMessage {...messages[setting_name]} /></a>
+                </li>
+              )) }
+            </ul>
+          </div>
+        </div>
+
+        <div>
+          <div className='confirmation-modal__action-bar'>
+            <div />
+            <Button text={intl.formatMessage(messages.discardChanges)} onClick={this.handleClick} autoFocus />
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(DeprecatedSettingsModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/disabled_account_banner.jsx b/app/javascript/flavours/blobfox/features/ui/components/disabled_account_banner.jsx
new file mode 100644
index 00000000000000..94fbce30dc8d00
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/disabled_account_banner.jsx
@@ -0,0 +1,99 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import { connect } from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { disabledAccountId, movedToAccountId, domain } from 'flavours/blobfox/initial_state';
+import { logOut } from 'flavours/blobfox/utils/log_out';
+
+const messages = defineMessages({
+  logoutMessage: { id: 'confirmations.logout.message', defaultMessage: 'Are you sure you want to log out?' },
+  logoutConfirm: { id: 'confirmations.logout.confirm', defaultMessage: 'Log out' },
+});
+
+const mapStateToProps = (state) => ({
+  disabledAcct: state.getIn(['accounts', disabledAccountId, 'acct']),
+  movedToAcct: movedToAccountId ? state.getIn(['accounts', movedToAccountId, 'acct']) : undefined,
+});
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onLogout () {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.logoutMessage),
+        confirm: intl.formatMessage(messages.logoutConfirm),
+        closeWhenConfirm: false,
+        onConfirm: () => logOut(),
+      },
+    }));
+  },
+});
+
+class DisabledAccountBanner extends PureComponent {
+
+  static propTypes = {
+    disabledAcct: PropTypes.string.isRequired,
+    movedToAcct: PropTypes.string,
+    onLogout: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleLogOutClick = e => {
+    e.preventDefault();
+    e.stopPropagation();
+
+    this.props.onLogout();
+
+    return false;
+  };
+
+  render () {
+    const { disabledAcct, movedToAcct } = this.props;
+
+    const disabledAccountLink = (
+      <Link to={`/@${disabledAcct}`}>
+        {disabledAcct}@{domain}
+      </Link>
+    );
+
+    return (
+      <div className='sign-in-banner'>
+        <p>
+          {movedToAcct ? (
+            <FormattedMessage
+              id='moved_to_account_banner.text'
+              defaultMessage='Your account {disabledAccount} is currently disabled because you moved to {movedToAccount}.'
+              values={{
+                disabledAccount: disabledAccountLink,
+                movedToAccount: <Link to={`/@${movedToAcct}`}>{movedToAcct.includes('@') ? movedToAcct : `${movedToAcct}@${domain}`}</Link>,
+              }}
+            />
+          ) : (
+            <FormattedMessage
+              id='disabled_account_banner.text'
+              defaultMessage='Your account {disabledAccount} is currently disabled.'
+              values={{
+                disabledAccount: disabledAccountLink,
+              }}
+            />
+          )}
+        </p>
+        <a href='/auth/edit' className='button button--block'>
+          <FormattedMessage id='disabled_account_banner.account_settings' defaultMessage='Account settings' />
+        </a>
+        <button type='button' className='button button--block button-tertiary' onClick={this.handleLogOutClick}>
+          <FormattedMessage id='confirmations.logout.confirm' defaultMessage='Log out' />
+        </button>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(DisabledAccountBanner));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/doodle_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/doodle_modal.jsx
new file mode 100644
index 00000000000000..b4c3c905dc0cf6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/doodle_modal.jsx
@@ -0,0 +1,619 @@
+import PropTypes from 'prop-types';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import Atrament from 'atrament'; // the doodling library
+import { debounce, mapValues } from 'lodash';
+
+import { doodleSet, uploadCompose } from 'flavours/blobfox/actions/compose';
+import { Button } from 'flavours/blobfox/components/button';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+// palette nicked from MyPaint, CC0
+const palette = [
+  ['rgb(  0,    0,    0)', 'Black'],
+  ['rgb( 38,   38,   38)', 'Gray 15'],
+  ['rgb( 77,   77,   77)', 'Grey 30'],
+  ['rgb(128,  128,  128)', 'Grey 50'],
+  ['rgb(171,  171,  171)', 'Grey 67'],
+  ['rgb(217,  217,  217)', 'Grey 85'],
+  ['rgb(255,  255,  255)', 'White'],
+  ['rgb(128,    0,    0)', 'Maroon'],
+  ['rgb(209,    0,    0)', 'English-red'],
+  ['rgb(255,   54,   34)', 'Tomato'],
+  ['rgb(252,   60,    3)', 'Orange-red'],
+  ['rgb(255,  140,  105)', 'Salmon'],
+  ['rgb(252,  232,   32)', 'Cadium-yellow'],
+  ['rgb(243,  253,   37)', 'Lemon yellow'],
+  ['rgb(121,    5,   35)', 'Dark crimson'],
+  ['rgb(169,   32,   62)', 'Deep carmine'],
+  ['rgb(255,  140,    0)', 'Orange'],
+  ['rgb(255,  168,   18)', 'Dark tangerine'],
+  ['rgb(217,  144,   88)', 'Persian orange'],
+  ['rgb(194,  178,  128)', 'Sand'],
+  ['rgb(255,  229,  180)', 'Peach'],
+  ['rgb(100,   54,   46)', 'Bole'],
+  ['rgb(108,   41,   52)', 'Dark cordovan'],
+  ['rgb(163,   65,   44)', 'Chestnut'],
+  ['rgb(228,  136,  100)', 'Dark salmon'],
+  ['rgb(255,  195,  143)', 'Apricot'],
+  ['rgb(255,  219,  188)', 'Unbleached silk'],
+  ['rgb(242,  227,  198)', 'Straw'],
+  ['rgb( 53,   19,   13)', 'Bistre'],
+  ['rgb( 84,   42,   14)', 'Dark chocolate'],
+  ['rgb(102,   51,   43)', 'Burnt sienna'],
+  ['rgb(184,   66,    0)', 'Sienna'],
+  ['rgb(216,  153,   12)', 'Yellow ochre'],
+  ['rgb(210,  180,  140)', 'Tan'],
+  ['rgb(232,  204,  144)', 'Dark wheat'],
+  ['rgb(  0,   49,   83)', 'Prussian blue'],
+  ['rgb( 48,   69,  119)', 'Dark grey blue'],
+  ['rgb(  0,   71,  171)', 'Cobalt blue'],
+  ['rgb( 31,  117,  254)', 'Blue'],
+  ['rgb(120,  180,  255)', 'Bright french blue'],
+  ['rgb(171,  200,  255)', 'Bright steel blue'],
+  ['rgb(208,  231,  255)', 'Ice blue'],
+  ['rgb( 30,   51,   58)', 'Medium jungle green'],
+  ['rgb( 47,   79,   79)', 'Dark slate grey'],
+  ['rgb( 74,  104,   93)', 'Dark grullo green'],
+  ['rgb(  0,  128,  128)', 'Teal'],
+  ['rgb( 67,  170,  176)', 'Turquoise'],
+  ['rgb(109,  174,  199)', 'Cerulean frost'],
+  ['rgb(173,  217,  186)', 'Tiffany green'],
+  ['rgb( 22,   34,   29)', 'Gray-asparagus'],
+  ['rgb( 36,   48,   45)', 'Medium dark teal'],
+  ['rgb( 74,  104,   93)', 'Xanadu'],
+  ['rgb(119,  198,  121)', 'Mint'],
+  ['rgb(175,  205,  182)', 'Timberwolf'],
+  ['rgb(185,  245,  246)', 'Celeste'],
+  ['rgb(193,  255,  234)', 'Aquamarine'],
+  ['rgb( 29,   52,   35)', 'Cal Poly Pomona'],
+  ['rgb(  1,   68,   33)', 'Forest green'],
+  ['rgb( 42,  128,    0)', 'Napier green'],
+  ['rgb(128,  128,    0)', 'Olive'],
+  ['rgb( 65,  156,  105)', 'Sea green'],
+  ['rgb(189,  246,   29)', 'Green-yellow'],
+  ['rgb(231,  244,  134)', 'Bright chartreuse'],
+  ['rgb(138,   23,  137)', 'Purple'],
+  ['rgb( 78,   39,  138)', 'Violet'],
+  ['rgb(193,   75,  110)', 'Dark thulian pink'],
+  ['rgb(222,   49,   99)', 'Cerise'],
+  ['rgb(255,   20,  147)', 'Deep pink'],
+  ['rgb(255,  102,  204)', 'Rose pink'],
+  ['rgb(255,  203,  219)', 'Pink'],
+  ['rgb(255,  255,  255)', 'White'],
+  ['rgb(229,   17,    1)', 'RGB Red'],
+  ['rgb(  0,  255,    0)', 'RGB Green'],
+  ['rgb(  0,    0,  255)', 'RGB Blue'],
+  ['rgb(  0,  255,  255)', 'CMYK Cyan'],
+  ['rgb(255,    0,  255)', 'CMYK Magenta'],
+  ['rgb(255,  255,    0)', 'CMYK Yellow'],
+];
+
+// re-arrange to the right order for display
+let palReordered = [];
+for (let row = 0; row < 7; row++) {
+  for (let col = 0; col < 11; col++) {
+    palReordered.push(palette[col * 7 + row]);
+  }
+  palReordered.push(null); // null indicates a <br />
+}
+
+// Utility for converting base64 image to binary for upload
+// https://stackoverflow.com/questions/35940290/how-to-convert-base64-string-to-javascript-file-object-like-as-from-file-input-f
+function dataURLtoFile(dataurl, filename) {
+  let arr = dataurl.split(','), mime = arr[0].match(/:(.*?);/)[1],
+    bstr = atob(arr[1]), n = bstr.length, u8arr = new Uint8Array(n);
+  while(n--){
+    u8arr[n] = bstr.charCodeAt(n);
+  }
+  return new File([u8arr], filename, { type: mime });
+}
+/** Doodle canvas size options */
+const DOODLE_SIZES = {
+  normal: [500, 500, 'Square 500'],
+  tootbanner: [702, 330, 'Tootbanner'],
+  s640x480: [640, 480, '640×480 - 480p'],
+  s800x600: [800, 600, '800×600 - SVGA'],
+  s720x480: [720, 405, '720x405 - 16:9'],
+};
+
+
+const mapStateToProps = state => ({
+  options: state.getIn(['compose', 'doodle']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  /**
+   * Set options in the redux store
+   * @param {Object} opts
+   */
+  setOpt: (opts) => dispatch(doodleSet(opts)),
+  /**
+   * Submit doodle for upload
+   * @param {File} file
+   */
+  submit: (file) => dispatch(uploadCompose([file])),
+});
+
+/**
+ * Doodling dialog with drawing canvas
+ *
+ * Keyboard shortcuts:
+ * - Delete: Clear screen, fill with background color
+ * - Backspace, Ctrl+Z: Undo one step
+ * - Ctrl held while drawing: Use background color
+ * - Shift held while clicking screen: Use fill tool
+ *
+ * Palette:
+ * - Left mouse button: pick foreground
+ * - Ctrl + left mouse button: pick background
+ * - Right mouse button: pick background
+ */
+class DoodleModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    options: ImmutablePropTypes.map,
+    onClose: PropTypes.func.isRequired,
+    setOpt: PropTypes.func.isRequired,
+    submit: PropTypes.func.isRequired,
+  };
+
+  //region Option getters/setters
+
+  /** Foreground color */
+  get fg () {
+    return this.props.options.get('fg');
+  }
+  set fg (value) {
+    this.props.setOpt({ fg: value });
+  }
+
+  /** Background color */
+  get bg () {
+    return this.props.options.get('bg');
+  }
+  set bg (value) {
+    this.props.setOpt({ bg: value });
+  }
+
+  /** Swap Fg and Bg for drawing */
+  get swapped () {
+    return this.props.options.get('swapped');
+  }
+  set swapped (value) {
+    this.props.setOpt({ swapped: value });
+  }
+
+  /** Mode - 'draw' or 'fill' */
+  get mode () {
+    return this.props.options.get('mode');
+  }
+  set mode (value) {
+    this.props.setOpt({ mode: value });
+  }
+
+  /** Base line weight */
+  get weight () {
+    return this.props.options.get('weight');
+  }
+  set weight (value) {
+    this.props.setOpt({ weight: value });
+  }
+
+  /** Drawing opacity */
+  get opacity () {
+    return this.props.options.get('opacity');
+  }
+  set opacity (value) {
+    this.props.setOpt({ opacity: value });
+  }
+
+  /** Adaptive stroke - change width with speed */
+  get adaptiveStroke () {
+    return this.props.options.get('adaptiveStroke');
+  }
+  set adaptiveStroke (value) {
+    this.props.setOpt({ adaptiveStroke: value });
+  }
+
+  /** Smoothing (for mouse drawing) */
+  get smoothing () {
+    return this.props.options.get('smoothing');
+  }
+  set smoothing (value) {
+    this.props.setOpt({ smoothing: value });
+  }
+
+  /** Size preset */
+  get size () {
+    return this.props.options.get('size');
+  }
+  set size (value) {
+    this.props.setOpt({ size: value });
+  }
+
+  //endregion
+
+  /**
+   * Key up handler
+   * @param {KeyboardEvent} e
+   */
+  handleKeyUp = (e) => {
+    if (e.target.nodeName === 'INPUT') return;
+
+    if (e.key === 'Delete') {
+      e.preventDefault();
+      this.handleClearBtn();
+      return;
+    }
+
+    if (e.key === 'Backspace' || (e.key === 'z' && (e.ctrlKey || e.metaKey))) {
+      e.preventDefault();
+      this.undo();
+    }
+
+    if (e.key === 'Control' || e.key === 'Meta') {
+      this.controlHeld = false;
+      this.swapped = false;
+    }
+
+    if (e.key === 'Shift') {
+      this.shiftHeld = false;
+      this.mode = 'draw';
+    }
+  };
+
+  /**
+   * Key down handler
+   * @param {KeyboardEvent} e
+   */
+  handleKeyDown = (e) => {
+    if (e.key === 'Control' || e.key === 'Meta') {
+      this.controlHeld = true;
+      this.swapped = true;
+    }
+
+    if (e.key === 'Shift') {
+      this.shiftHeld = true;
+      this.mode = 'fill';
+    }
+  };
+
+  /**
+   * Component installed in the DOM, do some initial set-up
+   */
+  componentDidMount () {
+    this.controlHeld = false;
+    this.shiftHeld = false;
+    this.swapped = false;
+    window.addEventListener('keyup', this.handleKeyUp, false);
+    window.addEventListener('keydown', this.handleKeyDown, false);
+  }
+
+  /**
+   * Tear component down
+   */
+  componentWillUnmount () {
+    window.removeEventListener('keyup', this.handleKeyUp, false);
+    window.removeEventListener('keydown', this.handleKeyDown, false);
+    if (this.sketcher) this.sketcher.destroy();
+  }
+
+  /**
+   * Set reference to the canvas element.
+   * This is called during component init
+   * @param {HTMLCanvasElement} elem - canvas element
+   */
+  setCanvasRef = (elem) => {
+    this.canvas = elem;
+    if (elem) {
+      elem.addEventListener('dirty', () => {
+        this.saveUndo();
+        this.sketcher._dirty = false;
+      });
+
+      elem.addEventListener('click', () => {
+        // sketcher bug - does not fire dirty on fill
+        if (this.mode === 'fill') {
+          this.saveUndo();
+        }
+      });
+
+      // prevent context menu
+      elem.addEventListener('contextmenu', (e) => {
+        e.preventDefault();
+      });
+
+      elem.addEventListener('mousedown', (e) => {
+        if (e.button === 2) {
+          this.swapped = true;
+        }
+      });
+
+      elem.addEventListener('mouseup', (e) => {
+        if (e.button === 2) {
+          this.swapped = this.controlHeld;
+        }
+      });
+
+      this.initSketcher(elem);
+      this.mode = 'draw'; // Reset mode - it's confusing if left at 'fill'
+    }
+  };
+
+  /**
+   * Set up the sketcher instance
+   * @param {HTMLCanvasElement | null} canvas - canvas element. Null if we're just resizing
+   */
+  initSketcher (canvas = null) {
+    const sizepreset = DOODLE_SIZES[this.size];
+
+    if (this.sketcher) this.sketcher.destroy();
+    this.sketcher = new Atrament(canvas || this.canvas, sizepreset[0], sizepreset[1]);
+
+    if (canvas) {
+      this.ctx = this.sketcher.context;
+      this.updateSketcherSettings();
+    }
+
+    this.clearScreen();
+  }
+
+  /**
+   * Done button handler
+   */
+  onDoneButton = () => {
+    const dataUrl = this.sketcher.toImage();
+    const file = dataURLtoFile(dataUrl, 'doodle.png');
+    this.props.submit(file);
+    this.props.onClose(); // close dialog
+  };
+
+  /**
+   * Cancel button handler
+   */
+  onCancelButton = () => {
+    if (this.undos.length > 1 && !confirm('Discard doodle? All changes will be lost!')) {
+      return;
+    }
+
+    this.props.onClose(); // close dialog
+  };
+
+  /**
+   * Update sketcher options based on state
+   */
+  updateSketcherSettings () {
+    if (!this.sketcher) return;
+
+    if (this.oldSize !== this.size) this.initSketcher();
+
+    this.sketcher.color = (this.swapped ? this.bg : this.fg);
+    this.sketcher.opacity = this.opacity;
+    this.sketcher.weight = this.weight;
+    this.sketcher.mode = this.mode;
+    this.sketcher.smoothing = this.smoothing;
+    this.sketcher.adaptiveStroke = this.adaptiveStroke;
+
+    this.oldSize = this.size;
+  }
+
+  /**
+   * Fill screen with background color
+   */
+  clearScreen = () => {
+    this.ctx.fillStyle = this.bg;
+    this.ctx.fillRect(-1, -1, this.canvas.width+2, this.canvas.height+2);
+    this.undos = [];
+
+    this.doSaveUndo();
+  };
+
+  /**
+   * Undo one step
+   */
+  undo = () => {
+    if (this.undos.length > 1) {
+      this.undos.pop();
+      const buf = this.undos.pop();
+
+      this.sketcher.clear();
+      this.ctx.putImageData(buf, 0, 0);
+      this.doSaveUndo();
+    }
+  };
+
+  /**
+   * Save canvas content into the undo buffer immediately
+   */
+  doSaveUndo = () => {
+    this.undos.push(this.ctx.getImageData(0, 0, this.canvas.width, this.canvas.height));
+  };
+
+  /**
+   * Called on each canvas change.
+   * Saves canvas content to the undo buffer after some period of inactivity.
+   */
+  saveUndo = debounce(() => {
+    this.doSaveUndo();
+  }, 100);
+
+  /**
+   * Palette left click.
+   * Selects Fg color (or Bg, if Control/Meta is held)
+   * @param {MouseEvent<HTMLButtonElement>} e - event
+   */
+  onPaletteClick = (e) => {
+    const c = e.target.dataset.color;
+
+    if (this.controlHeld) {
+      this.bg = c;
+    } else {
+      this.fg = c;
+    }
+
+    e.target.blur();
+    e.preventDefault();
+  };
+
+  /**
+   * Palette right click.
+   * Selects Bg color
+   * @param {MouseEvent<HTMLButtonElement>} e - event
+   */
+  onPaletteRClick = (e) => {
+    this.bg = e.target.dataset.color;
+    e.target.blur();
+    e.preventDefault();
+  };
+
+  /**
+   * Handle click on the Draw mode button
+   * @param {MouseEvent<HTMLButtonElement>} e - event
+   */
+  setModeDraw = (e) => {
+    this.mode = 'draw';
+    e.target.blur();
+  };
+
+  /**
+   * Handle click on the Fill mode button
+   * @param {MouseEvent<HTMLButtonElement>} e - event
+   */
+  setModeFill = (e) => {
+    this.mode = 'fill';
+    e.target.blur();
+  };
+
+  /**
+   * Handle click on Smooth checkbox
+   * @param {ChangeEvent<HTMLInputElement>} e - event
+   */
+  tglSmooth = (e) => {
+    this.smoothing = !this.smoothing;
+    e.target.blur();
+  };
+
+  /**
+   * Handle click on Adaptive checkbox
+   * @param {ChangeEvent<HTMLInputElement>} e - event
+   */
+  tglAdaptive = (e) => {
+    this.adaptiveStroke = !this.adaptiveStroke;
+    e.target.blur();
+  };
+
+  /**
+   * Handle change of the Weight input field
+   * @param {ChangeEvent<HTMLInputElement>} e - event
+   */
+  setWeight = (e) => {
+    this.weight = +e.target.value || 1;
+  };
+
+  /**
+   * Set size - clalback from the select box
+   * @param {ChangeEvent<HTMLSelectElement>} e - event
+   */
+  changeSize = (e) => {
+    let newSize = e.target.value;
+    if (newSize === this.oldSize) return;
+
+    if (this.undos.length > 1 && !confirm('Change canvas size? This will erase your current drawing!')) {
+      return;
+    }
+
+    this.size = newSize;
+  };
+
+  handleClearBtn = () => {
+    if (this.undos.length > 1 && !confirm('Clear canvas? This will erase your current drawing!')) {
+      return;
+    }
+
+    this.clearScreen();
+  };
+
+  /**
+   * Render the component
+   */
+  render () {
+    this.updateSketcherSettings();
+
+    return (
+      <div className='modal-root__modal doodle-modal'>
+        <div className='doodle-modal__container'>
+          <canvas ref={this.setCanvasRef} />
+        </div>
+
+        <div className='doodle-modal__action-bar'>
+          <div className='doodle-toolbar'>
+            <Button text='Done' onClick={this.onDoneButton} />
+            <Button text='Cancel' onClick={this.onCancelButton} />
+          </div>
+          <div className='filler' />
+          <div className='doodle-toolbar with-inputs'>
+            <div>
+              <label htmlFor='dd_smoothing'>Smoothing</label>
+              <span className='val'>
+                <input type='checkbox' id='dd_smoothing' onChange={this.tglSmooth} checked={this.smoothing} />
+              </span>
+            </div>
+            <div>
+              <label htmlFor='dd_adaptive'>Adaptive</label>
+              <span className='val'>
+                <input type='checkbox' id='dd_adaptive' onChange={this.tglAdaptive} checked={this.adaptiveStroke} />
+              </span>
+            </div>
+            <div>
+              <label htmlFor='dd_weight'>Weight</label>
+              <span className='val'>
+                <input type='number' min={1} id='dd_weight' value={this.weight} onChange={this.setWeight} />
+              </span>
+            </div>
+            <div>
+              <select aria-label='Canvas size' onInput={this.changeSize} defaultValue={this.size}>
+                { Object.values(mapValues(DOODLE_SIZES, (val, k) =>
+                  <option key={k} value={k}>{val[2]}</option>,
+                )) }
+              </select>
+            </div>
+          </div>
+          <div className='doodle-toolbar'>
+            <IconButton icon='pencil' title='Draw' label='Draw' onClick={this.setModeDraw} size={18} active={this.mode === 'draw'} inverted />
+            <IconButton icon='bath' title='Fill' label='Fill' onClick={this.setModeFill} size={18} active={this.mode === 'fill'} inverted />
+            <IconButton icon='undo' title='Undo' label='Undo' onClick={this.undo} size={18} inverted />
+            <IconButton icon='trash' title='Clear' label='Clear' onClick={this.handleClearBtn} size={18} inverted />
+          </div>
+          <div className='doodle-palette'>
+            {
+              palReordered.map((c, i) =>
+                c === null ?
+                  <br key={i} /> :
+                  <button
+                    key={i}
+                    style={{ backgroundColor: c[0] }}
+                    onClick={this.onPaletteClick}
+                    onContextMenu={this.onPaletteRClick}
+                    data-color={c[0]}
+                    title={c[1]}
+                    className={classNames({
+                      'foreground': this.fg === c[0],
+                      'background': this.bg === c[0],
+                    })}
+                  />,
+              )
+            }
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(DoodleModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/drawer_loading.jsx b/app/javascript/flavours/blobfox/features/ui/components/drawer_loading.jsx
new file mode 100644
index 00000000000000..11072ad98cc076
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/drawer_loading.jsx
@@ -0,0 +1,9 @@
+const DrawerLoading = () => (
+  <div className='drawer'>
+    <div className='drawer__pager'>
+      <div className='drawer__inner' />
+    </div>
+  </div>
+);
+
+export default DrawerLoading;
diff --git a/app/javascript/flavours/blobfox/features/ui/components/embed_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/embed_modal.jsx
new file mode 100644
index 00000000000000..f357059d76d53b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/embed_modal.jsx
@@ -0,0 +1,100 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import api from 'flavours/blobfox/api';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+class EmbedModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    id: PropTypes.string.isRequired,
+    onClose: PropTypes.func.isRequired,
+    onError: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    loading: false,
+    oembed: null,
+  };
+
+  componentDidMount () {
+    const { id } = this.props;
+
+    this.setState({ loading: true });
+
+    api().get(`/api/web/embeds/${id}`).then(res => {
+      this.setState({ loading: false, oembed: res.data });
+
+      const iframeDocument = this.iframe.contentWindow.document;
+
+      iframeDocument.open();
+      iframeDocument.write(res.data.html);
+      iframeDocument.close();
+
+      iframeDocument.body.style.margin = 0;
+      this.iframe.width  = iframeDocument.body.scrollWidth;
+      this.iframe.height = iframeDocument.body.scrollHeight;
+    }).catch(error => {
+      this.props.onError(error);
+    });
+  }
+
+  setIframeRef = c =>  {
+    this.iframe = c;
+  };
+
+  handleTextareaClick = (e) => {
+    e.target.select();
+  };
+
+  render () {
+    const { intl, onClose } = this.props;
+    const { oembed } = this.state;
+
+    return (
+      <div className='modal-root__modal report-modal embed-modal'>
+        <div className='report-modal__target'>
+          <IconButton className='media-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={16} />
+          <FormattedMessage id='status.embed' defaultMessage='Embed' />
+        </div>
+
+        <div className='report-modal__container embed-modal__container' style={{ display: 'block' }}>
+          <p className='hint'>
+            <FormattedMessage id='embed.instructions' defaultMessage='Embed this status on your website by copying the code below.' />
+          </p>
+
+          <input
+            type='text'
+            className='embed-modal__html'
+            readOnly
+            value={oembed && oembed.html || ''}
+            onClick={this.handleTextareaClick}
+          />
+
+          <p className='hint'>
+            <FormattedMessage id='embed.preview' defaultMessage='Here is what it will look like:' />
+          </p>
+
+          <iframe
+            className='embed-modal__iframe'
+            frameBorder='0'
+            ref={this.setIframeRef}
+            sandbox='allow-scripts allow-same-origin'
+            title='preview'
+          />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(EmbedModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/favourite_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/favourite_modal.jsx
new file mode 100644
index 00000000000000..47bcd6a612c744
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/favourite_modal.jsx
@@ -0,0 +1,94 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+import { withRouter } from 'react-router-dom';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import AttachmentList from 'flavours/blobfox/components/attachment_list';
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { Button } from 'flavours/blobfox/components/button';
+import { DisplayName } from 'flavours/blobfox/components/display_name';
+import { Icon } from 'flavours/blobfox/components/icon';
+import { RelativeTimestamp } from 'flavours/blobfox/components/relative_timestamp';
+import StatusContent from 'flavours/blobfox/components/status_content';
+import VisibilityIcon from 'flavours/blobfox/components/status_visibility_icon';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+const messages = defineMessages({
+  favourite: { id: 'status.favourite', defaultMessage: 'Favorite' },
+});
+
+class FavouriteModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    status: ImmutablePropTypes.map.isRequired,
+    onFavourite: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    ...WithRouterPropTypes,
+  };
+
+  handleFavourite = () => {
+    this.props.onFavourite(this.props.status);
+    this.props.onClose();
+  };
+
+  handleAccountClick = (e) => {
+    if (e.button === 0) {
+      e.preventDefault();
+      this.props.onClose();
+      this.props.history.push(`/@${this.props.status.getIn(['account', 'acct'])}`);
+    }
+  };
+
+  render () {
+    const { status, intl } = this.props;
+
+    return (
+      <div className='modal-root__modal boost-modal'>
+        <div className='boost-modal__container'>
+          <div className={classNames('status', `status-${status.get('visibility')}`, 'light')}>
+            <div className='boost-modal__status-header'>
+              <div className='boost-modal__status-time'>
+                <a href={status.get('url')} className='status__relative-time' target='_blank' rel='noopener noreferrer'>
+                  <VisibilityIcon visibility={status.get('visibility')} />
+                  <RelativeTimestamp timestamp={status.get('created_at')} />
+                </a>
+              </div>
+
+              <a onClick={this.handleAccountClick} href={status.getIn(['account', 'url'])} className='status__display-name'>
+                <div className='status__avatar'>
+                  <Avatar account={status.get('account')} size={48} />
+                </div>
+
+                <DisplayName account={status.get('account')} />
+
+              </a>
+            </div>
+
+            <StatusContent status={status} />
+
+            {status.get('media_attachments').size > 0 && (
+              <AttachmentList
+                compact
+                media={status.get('media_attachments')}
+              />
+            )}
+          </div>
+        </div>
+
+        <div className='boost-modal__action-bar'>
+          <div><FormattedMessage id='favourite_modal.combo' defaultMessage='You can press {combo} to skip this next time' values={{ combo: <span>Shift + <Icon id='star' /></span> }} /></div>
+          <Button text={intl.formatMessage(messages.favourite)} onClick={this.handleFavourite} autoFocus />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default withRouter(injectIntl(FavouriteModal));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/filter_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/filter_modal.jsx
new file mode 100644
index 00000000000000..7d66e150e49d04
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/filter_modal.jsx
@@ -0,0 +1,136 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { fetchFilters, createFilter, createFilterStatus } from 'flavours/blobfox/actions/filters';
+import { fetchStatus } from 'flavours/blobfox/actions/statuses';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import AddedToFilter from 'flavours/blobfox/features/filters/added_to_filter';
+import SelectFilter from 'flavours/blobfox/features/filters/select_filter';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+class FilterModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    statusId: PropTypes.string.isRequired,
+    contextType: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    step: 'select',
+    filterId: null,
+    isSubmitting: false,
+    isSubmitted: false,
+  };
+
+  handleNewFilterSuccess = (result) => {
+    this.handleSelectFilter(result.id);
+  };
+
+  handleSuccess = () => {
+    const { dispatch, statusId } = this.props;
+    dispatch(fetchStatus(statusId, true));
+    this.setState({ isSubmitting: false, isSubmitted: true, step: 'submitted' });
+  };
+
+  handleFail = () => {
+    this.setState({ isSubmitting: false });
+  };
+
+  handleNextStep = step => {
+    this.setState({ step });
+  };
+
+  handleSelectFilter = (filterId) => {
+    const { dispatch, statusId } = this.props;
+
+    this.setState({ isSubmitting: true, filterId });
+
+    dispatch(createFilterStatus({
+      filter_id: filterId,
+      status_id: statusId,
+    }, this.handleSuccess, this.handleFail));
+  };
+
+  handleNewFilter = (title) => {
+    const { dispatch } = this.props;
+
+    this.setState({ isSubmitting: true });
+
+    dispatch(createFilter({
+      title,
+      context: ['home', 'notifications', 'public', 'thread', 'account'],
+      action: 'warn',
+    }, this.handleNewFilterSuccess, this.handleFail));
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+
+    dispatch(fetchFilters());
+  }
+
+  render () {
+    const {
+      intl,
+      statusId,
+      contextType,
+      onClose,
+    } = this.props;
+
+    const {
+      step,
+      filterId,
+    } = this.state;
+
+    let stepComponent;
+
+    switch(step) {
+    case 'select':
+      stepComponent = (
+        <SelectFilter
+          contextType={contextType}
+          onSelectFilter={this.handleSelectFilter}
+          onNewFilter={this.handleNewFilter}
+        />
+      );
+      break;
+    case 'create':
+      stepComponent = null;
+      break;
+    case 'submitted':
+      stepComponent = (
+        <AddedToFilter
+          contextType={contextType}
+          filterId={filterId}
+          statusId={statusId}
+          onClose={onClose}
+        />
+      );
+    }
+
+    return (
+      <div className='modal-root__modal report-dialog-modal'>
+        <div className='report-modal__target'>
+          <IconButton className='report-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={20} />
+          <FormattedMessage id='filter_modal.title.status' defaultMessage='Filter a post' />
+        </div>
+
+        <div className='report-dialog-modal__container'>
+          {stepComponent}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect()(injectIntl(FilterModal));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/focal_point_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/focal_point_modal.jsx
new file mode 100644
index 00000000000000..6d75f2acab94f8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/focal_point_modal.jsx
@@ -0,0 +1,426 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import Textarea from 'react-textarea-autosize';
+import { length } from 'stringz';
+// eslint-disable-next-line import/extensions
+import tesseractWorkerPath from 'tesseract.js/dist/worker.min.js';
+// eslint-disable-next-line import/no-extraneous-dependencies
+import tesseractCorePath from 'tesseract.js-core/tesseract-core.wasm.js';
+
+import { Button } from 'flavours/blobfox/components/button';
+import { GIFV } from 'flavours/blobfox/components/gifv';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Audio from 'flavours/blobfox/features/audio';
+import CharacterCounter from 'flavours/blobfox/features/compose/components/character_counter';
+import UploadProgress from 'flavours/blobfox/features/compose/components/upload_progress';
+import { Tesseract as fetchTesseract } from 'flavours/blobfox/features/ui/util/async-components';
+import { me } from 'flavours/blobfox/initial_state';
+import { assetHost } from 'flavours/blobfox/utils/config';
+
+import { changeUploadCompose, uploadThumbnail, onChangeMediaDescription, onChangeMediaFocus } from '../../../actions/compose';
+import Video, { getPointerPosition } from '../../video';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+  apply: { id: 'upload_modal.apply', defaultMessage: 'Apply' },
+  applying: { id: 'upload_modal.applying', defaultMessage: 'Applying…' },
+  placeholder: { id: 'upload_modal.description_placeholder', defaultMessage: 'A quick brown fox jumps over the lazy dog' },
+  chooseImage: { id: 'upload_modal.choose_image', defaultMessage: 'Choose image' },
+  discardMessage: { id: 'confirmations.discard_edit_media.message', defaultMessage: 'You have unsaved changes to the media description or preview, discard them anyway?' },
+  discardConfirm: { id: 'confirmations.discard_edit_media.confirm', defaultMessage: 'Discard' },
+});
+
+const mapStateToProps = (state, { id }) => ({
+  media: state.getIn(['compose', 'media_attachments']).find(item => item.get('id') === id),
+  account: state.getIn(['accounts', me]),
+  isUploadingThumbnail: state.getIn(['compose', 'isUploadingThumbnail']),
+  description: state.getIn(['compose', 'media_modal', 'description']),
+  lang: state.getIn(['compose', 'language']),
+  focusX: state.getIn(['compose', 'media_modal', 'focusX']),
+  focusY: state.getIn(['compose', 'media_modal', 'focusY']),
+  dirty: state.getIn(['compose', 'media_modal', 'dirty']),
+  is_changing_upload: state.getIn(['compose', 'is_changing_upload']),
+});
+
+const mapDispatchToProps = (dispatch, { id }) => ({
+
+  onSave: (description, x, y) => {
+    dispatch(changeUploadCompose(id, { description, focus: `${x.toFixed(2)},${y.toFixed(2)}` }));
+  },
+
+  onChangeDescription: (description) => {
+    dispatch(onChangeMediaDescription(description));
+  },
+
+  onChangeFocus: (focusX, focusY) => {
+    dispatch(onChangeMediaFocus(focusX, focusY));
+  },
+
+  onSelectThumbnail: files => {
+    dispatch(uploadThumbnail(id, files[0]));
+  },
+
+});
+
+const removeExtraLineBreaks = str => str.replace(/\n\n/g, '******')
+  .replace(/\n/g, ' ')
+  .replace(/\*\*\*\*\*\*/g, '\n\n');
+
+class ImageLoader extends PureComponent {
+
+  static propTypes = {
+    src: PropTypes.string.isRequired,
+    width: PropTypes.number,
+    height: PropTypes.number,
+  };
+
+  state = {
+    loading: true,
+  };
+
+  componentDidMount() {
+    const image = new Image();
+    image.addEventListener('load', () => this.setState({ loading: false }));
+    image.src = this.props.src;
+  }
+
+  render () {
+    const { loading } = this.state;
+
+    if (loading) {
+      return <canvas width={this.props.width} height={this.props.height} />;
+    } else {
+      return <img {...this.props} alt='' />;
+    }
+  }
+
+}
+
+class FocalPointModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    media: ImmutablePropTypes.map.isRequired,
+    account: ImmutablePropTypes.record.isRequired,
+    isUploadingThumbnail: PropTypes.bool,
+    onSave: PropTypes.func.isRequired,
+    onChangeDescription: PropTypes.func.isRequired,
+    onChangeFocus: PropTypes.func.isRequired,
+    onSelectThumbnail: PropTypes.func.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    dragging: false,
+    dirty: false,
+    progress: 0,
+    loading: true,
+    ocrStatus: '',
+  };
+
+  componentWillUnmount () {
+    document.removeEventListener('mousemove', this.handleMouseMove);
+    document.removeEventListener('mouseup', this.handleMouseUp);
+  }
+
+  handleMouseDown = e => {
+    document.addEventListener('mousemove', this.handleMouseMove);
+    document.addEventListener('mouseup', this.handleMouseUp);
+
+    this.updatePosition(e);
+    this.setState({ dragging: true });
+  };
+
+  handleTouchStart = e => {
+    document.addEventListener('touchmove', this.handleMouseMove);
+    document.addEventListener('touchend', this.handleTouchEnd);
+
+    this.updatePosition(e);
+    this.setState({ dragging: true });
+  };
+
+  handleMouseMove = e => {
+    this.updatePosition(e);
+  };
+
+  handleMouseUp = () => {
+    document.removeEventListener('mousemove', this.handleMouseMove);
+    document.removeEventListener('mouseup', this.handleMouseUp);
+
+    this.setState({ dragging: false });
+  };
+
+  handleTouchEnd = () => {
+    document.removeEventListener('touchmove', this.handleMouseMove);
+    document.removeEventListener('touchend', this.handleTouchEnd);
+
+    this.setState({ dragging: false });
+  };
+
+  updatePosition = e => {
+    const { x, y } = getPointerPosition(this.node, e);
+    const focusX   = (x - .5) *  2;
+    const focusY   = (y - .5) * -2;
+
+    this.props.onChangeFocus(focusX, focusY);
+  };
+
+  handleChange = e => {
+    this.props.onChangeDescription(e.target.value);
+  };
+
+  handleKeyDown = (e) => {
+    if (e.keyCode === 13 && (e.ctrlKey || e.metaKey)) {
+      e.preventDefault();
+      e.stopPropagation();
+      this.props.onChangeDescription(e.target.value);
+      this.handleSubmit();
+    }
+  };
+
+  handleSubmit = () => {
+    this.props.onSave(this.props.description, this.props.focusX, this.props.focusY);
+  };
+
+  getCloseConfirmationMessage = () => {
+    const { intl, dirty } = this.props;
+
+    if (dirty) {
+      return {
+        message: intl.formatMessage(messages.discardMessage),
+        confirm: intl.formatMessage(messages.discardConfirm),
+      };
+    } else {
+      return null;
+    }
+  };
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  handleTextDetection = () => {
+    this._detectText();
+  };
+
+  _detectText = (refreshCache = false) => {
+    const { media } = this.props;
+
+    this.setState({ detecting: true });
+
+    fetchTesseract().then(({ createWorker }) => {
+      const worker = createWorker({
+        workerPath: tesseractWorkerPath,
+        corePath: tesseractCorePath,
+        langPath: `${assetHost}/ocr/lang-data`,
+        logger: ({ status, progress }) => {
+          if (status === 'recognizing text') {
+            this.setState({ ocrStatus: 'detecting', progress });
+          } else {
+            this.setState({ ocrStatus: 'preparing', progress });
+          }
+        },
+        cacheMethod: refreshCache ? 'refresh' : 'write',
+      });
+
+      let media_url = media.get('url');
+
+      if (window.URL && URL.createObjectURL) {
+        try {
+          media_url = URL.createObjectURL(media.get('file'));
+        } catch (error) {
+          console.error(error);
+        }
+      }
+
+      return (async () => {
+        await worker.load();
+        await worker.loadLanguage('eng');
+        await worker.initialize('eng');
+        const { data: { text } } = await worker.recognize(media_url);
+        this.setState({ detecting: false });
+        this.props.onChangeDescription(removeExtraLineBreaks(text));
+        await worker.terminate();
+      })().catch((e) => {
+        if (refreshCache) {
+          throw e;
+        } else {
+          this._detectText(true);
+        }
+      });
+    }).catch((e) => {
+      console.error(e);
+      this.setState({ detecting: false });
+    });
+  };
+
+  handleThumbnailChange = e => {
+    if (e.target.files.length > 0) {
+      this.props.onSelectThumbnail(e.target.files);
+    }
+  };
+
+  setFileInputRef = c => {
+    this.fileInput = c;
+  };
+
+  handleFileInputClick = () => {
+    this.fileInput.click();
+  };
+
+  render () {
+    const { media, intl, account, onClose, isUploadingThumbnail, description, lang, focusX, focusY, dirty, is_changing_upload } = this.props;
+    const { dragging, detecting, progress, ocrStatus } = this.state;
+    const x = (focusX /  2) + .5;
+    const y = (focusY / -2) + .5;
+
+    const width  = media.getIn(['meta', 'original', 'width']) || null;
+    const height = media.getIn(['meta', 'original', 'height']) || null;
+    const focals = ['image', 'gifv'].includes(media.get('type'));
+    const thumbnailable = ['audio', 'video'].includes(media.get('type'));
+
+    const previewRatio  = 16/9;
+    const previewWidth  = 200;
+    const previewHeight = previewWidth / previewRatio;
+
+    let descriptionLabel = null;
+
+    if (media.get('type') === 'audio') {
+      descriptionLabel = <FormattedMessage id='upload_form.audio_description' defaultMessage='Describe for people who are hard of hearing' />;
+    } else if (media.get('type') === 'video') {
+      descriptionLabel = <FormattedMessage id='upload_form.video_description' defaultMessage='Describe for people who are deaf, hard of hearing, blind or have low vision' />;
+    } else {
+      descriptionLabel = <FormattedMessage id='upload_form.description' defaultMessage='Describe for people who are blind or have low vision' />;
+    }
+
+    let ocrMessage = '';
+    if (ocrStatus === 'detecting') {
+      ocrMessage = <FormattedMessage id='upload_modal.analyzing_picture' defaultMessage='Analyzing picture…' />;
+    } else {
+      ocrMessage = <FormattedMessage id='upload_modal.preparing_ocr' defaultMessage='Preparing OCR…' />;
+    }
+
+    return (
+      <div className='modal-root__modal report-modal' style={{ maxWidth: 960 }}>
+        <div className='report-modal__target'>
+          <IconButton className='report-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={20} />
+          <FormattedMessage id='upload_modal.edit_media' defaultMessage='Edit media' />
+        </div>
+
+        <div className='report-modal__container'>
+          <div className='report-modal__comment'>
+            {focals && <p><FormattedMessage id='upload_modal.hint' defaultMessage='Click or drag the circle on the preview to choose the focal point which will always be in view on all thumbnails.' /></p>}
+
+            {thumbnailable && (
+              <>
+                <label className='setting-text-label' htmlFor='upload-modal__thumbnail'><FormattedMessage id='upload_form.thumbnail' defaultMessage='Change thumbnail' /></label>
+
+                <Button disabled={isUploadingThumbnail || !media.get('unattached')} text={intl.formatMessage(messages.chooseImage)} onClick={this.handleFileInputClick} />
+
+                <label>
+                  <span style={{ display: 'none' }}>{intl.formatMessage(messages.chooseImage)}</span>
+
+                  <input
+                    id='upload-modal__thumbnail'
+                    ref={this.setFileInputRef}
+                    type='file'
+                    accept='image/png,image/jpeg'
+                    onChange={this.handleThumbnailChange}
+                    style={{ display: 'none' }}
+                    disabled={isUploadingThumbnail || is_changing_upload}
+                  />
+                </label>
+
+                <hr className='setting-divider' />
+              </>
+            )}
+
+            <label className='setting-text-label' htmlFor='upload-modal__description'>
+              {descriptionLabel}
+            </label>
+
+            <div className='setting-text__wrapper'>
+              <Textarea
+                id='upload-modal__description'
+                className='setting-text light'
+                value={detecting ? '…' : description}
+                lang={lang}
+                onChange={this.handleChange}
+                onKeyDown={this.handleKeyDown}
+                disabled={detecting || is_changing_upload}
+                autoFocus
+              />
+
+              <div className='setting-text__modifiers'>
+                <UploadProgress progress={progress * 100} active={detecting} icon='file-text-o' message={ocrMessage} />
+              </div>
+            </div>
+
+            <div className='setting-text__toolbar'>
+              <button disabled={detecting || media.get('type') !== 'image' || is_changing_upload} className='link-button' onClick={this.handleTextDetection}><FormattedMessage id='upload_modal.detect_text' defaultMessage='Detect text from picture' /></button>
+              <CharacterCounter max={10000} text={detecting ? '' : description} />
+            </div>
+
+            <Button disabled={!dirty || detecting || isUploadingThumbnail || length(description) > 10000 || is_changing_upload} text={intl.formatMessage(is_changing_upload ? messages.applying : messages.apply)} onClick={this.handleSubmit} />
+          </div>
+
+          <div className='focal-point-modal__content'>
+            {focals && (
+              <div className={classNames('focal-point', { dragging })} ref={this.setRef} onMouseDown={this.handleMouseDown} onTouchStart={this.handleTouchStart}>
+                {media.get('type') === 'image' && <ImageLoader src={media.get('url')} width={width} height={height} alt='' />}
+                {media.get('type') === 'gifv' && <GIFV src={media.get('url')} key={media.get('url')} width={width} height={height} />}
+
+                <div className='focal-point__preview'>
+                  <strong><FormattedMessage id='upload_modal.preview_label' defaultMessage='Preview ({ratio})' values={{ ratio: '16:9' }} /></strong>
+                  <div style={{ width: previewWidth, height: previewHeight, backgroundImage: `url(${media.get('preview_url')})`, backgroundSize: 'cover', backgroundPosition: `${x * 100}% ${y * 100}%` }} />
+                </div>
+
+                <div className='focal-point__reticle' style={{ top: `${y * 100}%`, left: `${x * 100}%` }} />
+                <div className='focal-point__overlay' />
+              </div>
+            )}
+
+            {media.get('type') === 'video' && (
+              <Video
+                preview={media.get('preview_url')}
+                frameRate={media.getIn(['meta', 'original', 'frame_rate'])}
+                blurhash={media.get('blurhash')}
+                src={media.get('url')}
+                detailed
+                inline
+                editable
+              />
+            )}
+
+            {media.get('type') === 'audio' && (
+              <Audio
+                src={media.get('url')}
+                duration={media.getIn(['meta', 'original', 'duration'], 0)}
+                height={150}
+                poster={media.get('preview_url') || account.get('avatar_static')}
+                backgroundColor={media.getIn(['meta', 'colors', 'background'])}
+                foregroundColor={media.getIn(['meta', 'colors', 'foreground'])}
+                accentColor={media.getIn(['meta', 'colors', 'accent'])}
+                editable
+              />
+            )}
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps, null, {
+  forwardRef: true,
+})(injectIntl(FocalPointModal, { forwardRef: true }));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/follow_requests_column_link.jsx b/app/javascript/flavours/blobfox/features/ui/components/follow_requests_column_link.jsx
new file mode 100644
index 00000000000000..d2481f0d4d49d0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/follow_requests_column_link.jsx
@@ -0,0 +1,54 @@
+import PropTypes from 'prop-types';
+import { Component } from 'react';
+
+import { injectIntl, defineMessages } from 'react-intl';
+
+import { List as ImmutableList } from 'immutable';
+import { connect } from 'react-redux';
+
+import { fetchFollowRequests } from 'flavours/blobfox/actions/accounts';
+import { IconWithBadge } from 'flavours/blobfox/components/icon_with_badge';
+import ColumnLink from 'flavours/blobfox/features/ui/components/column_link';
+
+const messages = defineMessages({
+  text: { id: 'navigation_bar.follow_requests', defaultMessage: 'Follow requests' },
+});
+
+const mapStateToProps = state => ({
+  count: state.getIn(['user_lists', 'follow_requests', 'items'], ImmutableList()).size,
+});
+
+class FollowRequestsColumnLink extends Component {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    count: PropTypes.number.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+
+    dispatch(fetchFollowRequests());
+  }
+
+  render () {
+    const { count, intl } = this.props;
+
+    if (count === 0) {
+      return null;
+    }
+
+    return (
+      <ColumnLink
+        transparent
+        to='/follow_requests'
+        icon={<IconWithBadge className='column-link__icon' id='user-plus' count={count} />}
+        text={intl.formatMessage(messages.text)}
+      />
+    );
+  }
+
+}
+
+export default injectIntl(connect(mapStateToProps)(FollowRequestsColumnLink));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/header.jsx b/app/javascript/flavours/blobfox/features/ui/components/header.jsx
new file mode 100644
index 00000000000000..6ef773b4d30418
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/header.jsx
@@ -0,0 +1,124 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import { Link, withRouter } from 'react-router-dom';
+
+import { connect } from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { fetchServer } from 'flavours/blobfox/actions/server';
+import { Avatar } from 'flavours/blobfox/components/avatar';
+import { Icon } from 'flavours/blobfox/components/icon';
+import { WordmarkLogo, SymbolLogo } from 'flavours/blobfox/components/logo';
+import Permalink from 'flavours/blobfox/components/permalink';
+import { registrationsOpen, me, sso_redirect } from 'flavours/blobfox/initial_state';
+
+const Account = connect(state => ({
+  account: state.getIn(['accounts', me]),
+}))(({ account }) => (
+  <Permalink href={account.get('url')} to={`/@${account.get('acct')}`} title={account.get('acct')}>
+    <Avatar account={account} size={35} />
+  </Permalink>
+));
+
+const messages = defineMessages({
+  search: { id: 'navigation_bar.search', defaultMessage: 'Search' },
+});
+
+const mapStateToProps = (state) => ({
+  signupUrl: state.getIn(['server', 'server', 'registrations', 'url'], null) || '/auth/sign_up',
+});
+
+const mapDispatchToProps = (dispatch) => ({
+  openClosedRegistrationsModal() {
+    dispatch(openModal({ modalType: 'CLOSED_REGISTRATIONS' }));
+  },
+  dispatchServer() {
+    dispatch(fetchServer());
+  }
+});
+
+class Header extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    openClosedRegistrationsModal: PropTypes.func,
+    location: PropTypes.object,
+    signupUrl: PropTypes.string.isRequired,
+    dispatchServer: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+  };
+
+  componentDidMount () {
+    const { dispatchServer } = this.props;
+    dispatchServer();
+  }
+
+  render () {
+    const { signedIn } = this.context.identity;
+    const { location, openClosedRegistrationsModal, signupUrl, intl } = this.props;
+
+    let content;
+
+    if (signedIn) {
+      content = (
+        <>
+          {location.pathname !== '/search' && <Link to='/search' className='button button-secondary' aria-label={intl.formatMessage(messages.search)}><Icon id='search' /></Link>}
+          {location.pathname !== '/publish' && <Link to='/publish' className='button button-secondary'><FormattedMessage id='compose_form.publish_form' defaultMessage='New post' /></Link>}
+          <Account />
+        </>
+      );
+    } else {
+
+      if (sso_redirect) {
+        content = (
+          <a href={sso_redirect} data-method='post' className='button button--block button-tertiary'><FormattedMessage id='sign_in_banner.sso_redirect' defaultMessage='Login or Register' /></a>
+        );
+      } else {
+        let signupButton;
+
+        if (registrationsOpen) {
+          signupButton = (
+            <a href={signupUrl} className='button'>
+              <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+            </a>
+          );
+        } else {
+          signupButton = (
+            <button className='button' onClick={openClosedRegistrationsModal}>
+              <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+            </button>
+          );
+        }
+
+        content = (
+          <>
+            {signupButton}
+            <a href='/auth/sign_in' className='button button-tertiary'><FormattedMessage id='sign_in_banner.sign_in' defaultMessage='Login' /></a>
+          </>
+        );
+      }
+    }
+
+    return (
+      <div className='ui__header'>
+        <Link to='/' className='ui__header__logo'>
+          <WordmarkLogo />
+          <SymbolLogo />
+        </Link>
+
+        <div className='ui__header__links'>
+          {content}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(withRouter(connect(mapStateToProps, mapDispatchToProps)(Header)));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/image_loader.jsx b/app/javascript/flavours/blobfox/features/ui/components/image_loader.jsx
new file mode 100644
index 00000000000000..9dabc621b427e4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/image_loader.jsx
@@ -0,0 +1,174 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import classNames from 'classnames';
+
+import { LoadingBar } from 'react-redux-loading-bar';
+
+import ZoomableImage from './zoomable_image';
+
+export default class ImageLoader extends PureComponent {
+
+  static propTypes = {
+    alt: PropTypes.string,
+    lang: PropTypes.string,
+    src: PropTypes.string.isRequired,
+    previewSrc: PropTypes.string,
+    width: PropTypes.number,
+    height: PropTypes.number,
+    onClick: PropTypes.func,
+    zoomButtonHidden: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    alt: '',
+    lang: '',
+    width: null,
+    height: null,
+  };
+
+  state = {
+    loading: true,
+    error: false,
+    width: null,
+  };
+
+  removers = [];
+  canvas = null;
+
+  get canvasContext() {
+    if (!this.canvas) {
+      return null;
+    }
+    this._canvasContext = this._canvasContext || this.canvas.getContext('2d');
+    return this._canvasContext;
+  }
+
+  componentDidMount () {
+    this.loadImage(this.props);
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (this.props.src !== nextProps.src) {
+      this.loadImage(nextProps);
+    }
+  }
+
+  componentWillUnmount () {
+    this.removeEventListeners();
+  }
+
+  loadImage (props) {
+    this.removeEventListeners();
+    this.setState({ loading: true, error: false });
+    Promise.all([
+      props.previewSrc && this.loadPreviewCanvas(props),
+      this.hasSize() && this.loadOriginalImage(props),
+    ].filter(Boolean))
+      .then(() => {
+        this.setState({ loading: false, error: false });
+        this.clearPreviewCanvas();
+      })
+      .catch(() => this.setState({ loading: false, error: true }));
+  }
+
+  loadPreviewCanvas = ({ previewSrc, width, height }) => new Promise((resolve, reject) => {
+    const image = new Image();
+    const removeEventListeners = () => {
+      image.removeEventListener('error', handleError);
+      image.removeEventListener('load', handleLoad);
+    };
+    const handleError = () => {
+      removeEventListeners();
+      reject();
+    };
+    const handleLoad = () => {
+      removeEventListeners();
+      this.canvasContext.drawImage(image, 0, 0, width, height);
+      resolve();
+    };
+    image.addEventListener('error', handleError);
+    image.addEventListener('load', handleLoad);
+    image.src = previewSrc;
+    this.removers.push(removeEventListeners);
+  });
+
+  clearPreviewCanvas () {
+    const { width, height } = this.canvas;
+    this.canvasContext.clearRect(0, 0, width, height);
+  }
+
+  loadOriginalImage = ({ src }) => new Promise((resolve, reject) => {
+    const image = new Image();
+    const removeEventListeners = () => {
+      image.removeEventListener('error', handleError);
+      image.removeEventListener('load', handleLoad);
+    };
+    const handleError = () => {
+      removeEventListeners();
+      reject();
+    };
+    const handleLoad = () => {
+      removeEventListeners();
+      resolve();
+    };
+    image.addEventListener('error', handleError);
+    image.addEventListener('load', handleLoad);
+    image.src = src;
+    this.removers.push(removeEventListeners);
+  });
+
+  removeEventListeners () {
+    this.removers.forEach(listeners => listeners());
+    this.removers = [];
+  }
+
+  hasSize () {
+    const { width, height } = this.props;
+    return typeof width === 'number' && typeof height === 'number';
+  }
+
+  setCanvasRef = c => {
+    this.canvas = c;
+    if (c) this.setState({ width: c.offsetWidth });
+  };
+
+  render () {
+    const { alt, lang, src, width, height, onClick } = this.props;
+    const { loading } = this.state;
+
+    const className = classNames('image-loader', {
+      'image-loader--loading': loading,
+      'image-loader--amorphous': !this.hasSize(),
+    });
+
+    return (
+      <div className={className}>
+        {loading ? (
+          <>
+            <div className='loading-bar__container' style={{ width: this.state.width || width }}>
+              <LoadingBar className='loading-bar' loading={1} />
+            </div>
+            <canvas
+              className='image-loader__preview-canvas'
+              ref={this.setCanvasRef}
+              width={width}
+              height={height}
+            />
+          </>
+        ) : (
+          <ZoomableImage
+            alt={alt}
+            lang={lang}
+            src={src}
+            onClick={onClick}
+            width={width}
+            height={height}
+            zoomButtonHidden={this.props.zoomButtonHidden}
+          />
+        )}
+      </div>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/image_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/image_modal.jsx
new file mode 100644
index 00000000000000..a3106d0ac18c3e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/image_modal.jsx
@@ -0,0 +1,64 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+import ImageLoader from './image_loader';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+class ImageModal extends PureComponent {
+
+  static propTypes = {
+    src: PropTypes.string.isRequired,
+    alt: PropTypes.string.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  state = {
+    navigationHidden: false,
+  };
+
+  toggleNavigation = () => {
+    this.setState(prevState => ({
+      navigationHidden: !prevState.navigationHidden,
+    }));
+  };
+
+  render () {
+    const { intl, src, alt, onClose } = this.props;
+    const { navigationHidden } = this.state;
+
+    const navigationClassName = classNames('media-modal__navigation', {
+      'media-modal__navigation--hidden': navigationHidden,
+    });
+
+    return (
+      <div className='modal-root__modal media-modal'>
+        <div className='media-modal__closer' role='presentation' onClick={onClose} >
+          <ImageLoader
+            src={src}
+            width={400}
+            height={400}
+            alt={alt}
+            onClick={this.toggleNavigation}
+          />
+        </div>
+
+        <div className={navigationClassName}>
+          <IconButton className='media-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={40} />
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(ImageModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/link_footer.jsx b/app/javascript/flavours/blobfox/features/ui/components/link_footer.jsx
new file mode 100644
index 00000000000000..d9e80497f371c4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/link_footer.jsx
@@ -0,0 +1,111 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage, defineMessages, injectIntl } from 'react-intl';
+
+import { Link } from 'react-router-dom';
+
+import { connect } from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { domain, version, source_url, statusPageUrl, profile_directory as profileDirectory } from 'flavours/blobfox/initial_state';
+import { PERMISSION_INVITE_USERS } from 'flavours/blobfox/permissions';
+import { logOut } from 'flavours/blobfox/utils/log_out';
+
+const messages = defineMessages({
+  logoutMessage: { id: 'confirmations.logout.message', defaultMessage: 'Are you sure you want to log out?' },
+  logoutConfirm: { id: 'confirmations.logout.confirm', defaultMessage: 'Log out' },
+});
+
+const mapDispatchToProps = (dispatch, { intl }) => ({
+  onLogout () {
+    dispatch(openModal({
+      modalType: 'CONFIRM',
+      modalProps: {
+        message: intl.formatMessage(messages.logoutMessage),
+        confirm: intl.formatMessage(messages.logoutConfirm),
+        closeWhenConfirm: false,
+        onConfirm: () => logOut(),
+      },
+    }));
+  },
+});
+
+class LinkFooter extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    multiColumn: PropTypes.bool,
+    onLogout: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+  };
+
+  handleLogoutClick = e => {
+    e.preventDefault();
+    e.stopPropagation();
+
+    this.props.onLogout();
+
+    return false;
+  };
+
+  render () {
+    const { signedIn, permissions } = this.context.identity;
+    const { multiColumn } = this.props;
+
+    const canInvite = signedIn && ((permissions & PERMISSION_INVITE_USERS) === PERMISSION_INVITE_USERS);
+    const canProfileDirectory = profileDirectory;
+
+    const DividingCircle = <span aria-hidden>{' · '}</span>;
+
+    return (
+      <div className='link-footer'>
+        <p>
+          <strong>{domain}</strong>:
+          {' '}
+          <Link to='/about' target={multiColumn ? '_blank' : undefined}><FormattedMessage id='footer.about' defaultMessage='About' /></Link>
+          {statusPageUrl && (
+            <>
+              {DividingCircle}
+              <a href={statusPageUrl} target='_blank' rel='noopener'><FormattedMessage id='footer.status' defaultMessage='Status' /></a>
+            </>
+          )}
+          {canInvite && (
+            <>
+              {DividingCircle}
+              <a href='/invites' target='_blank'><FormattedMessage id='footer.invite' defaultMessage='Invite people' /></a>
+            </>
+          )}
+          {canProfileDirectory && (
+            <>
+              {DividingCircle}
+              <Link to='/directory'><FormattedMessage id='footer.directory' defaultMessage='Profiles directory' /></Link>
+            </>
+          )}
+          {DividingCircle}
+          <Link to='/privacy-policy' target={multiColumn ? '_blank' : undefined}><FormattedMessage id='footer.privacy_policy' defaultMessage='Privacy policy' /></Link>
+        </p>
+
+        <p>
+          <strong>Mastodon</strong>:
+          {' '}
+          <a href='https://joinmastodon.org' target='_blank'><FormattedMessage id='footer.about' defaultMessage='About' /></a>
+          {DividingCircle}
+          <a href='https://joinmastodon.org/apps' target='_blank'><FormattedMessage id='footer.get_app' defaultMessage='Get the app' /></a>
+          {DividingCircle}
+          <Link to='/keyboard-shortcuts'><FormattedMessage id='footer.keyboard_shortcuts' defaultMessage='Keyboard shortcuts' /></Link>
+          {DividingCircle}
+          <a href={source_url} rel='noopener noreferrer' target='_blank'><FormattedMessage id='footer.source_code' defaultMessage='View source code' /></a>
+          {DividingCircle}
+          <span className='version'>v{version}</span>
+        </p>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(connect(null, mapDispatchToProps)(LinkFooter));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/list_panel.jsx b/app/javascript/flavours/blobfox/features/ui/components/list_panel.jsx
new file mode 100644
index 00000000000000..aeae19018172d2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/list_panel.jsx
@@ -0,0 +1,56 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { fetchLists } from 'flavours/blobfox/actions/lists';
+
+import ColumnLink from './column_link';
+
+const getOrderedLists = createSelector([state => state.get('lists')], lists => {
+  if (!lists) {
+    return lists;
+  }
+
+  return lists.toList().filter(item => !!item).sort((a, b) => a.get('title').localeCompare(b.get('title'))).take(4);
+});
+
+const mapStateToProps = state => ({
+  lists: getOrderedLists(state),
+});
+
+class ListPanel extends ImmutablePureComponent {
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    lists: ImmutablePropTypes.list,
+  };
+
+  componentDidMount () {
+    const { dispatch } = this.props;
+    dispatch(fetchLists());
+  }
+
+  render () {
+    const { lists } = this.props;
+
+    if (!lists || lists.isEmpty()) {
+      return null;
+    }
+
+    return (
+      <div className='list-panel'>
+        <hr />
+
+        {lists.map(list => (
+          <ColumnLink icon='list-ul' key={list.get('id')} strict text={list.get('title')} to={`/lists/${list.get('id')}`} transparent />
+        ))}
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(ListPanel);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/media_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/media_modal.jsx
new file mode 100644
index 00000000000000..77d4b7ae792f9b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/media_modal.jsx
@@ -0,0 +1,261 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+
+import ReactSwipeableViews from 'react-swipeable-views';
+
+import { getAverageFromBlurhash } from 'flavours/blobfox/blurhash';
+import { GIFV } from 'flavours/blobfox/components/gifv';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Footer from 'flavours/blobfox/features/picture_in_picture/components/footer';
+import Video from 'flavours/blobfox/features/video';
+import { disableSwiping } from 'flavours/blobfox/initial_state';
+
+import ImageLoader from './image_loader';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+  previous: { id: 'lightbox.previous', defaultMessage: 'Previous' },
+  next: { id: 'lightbox.next', defaultMessage: 'Next' },
+});
+
+class MediaModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    media: ImmutablePropTypes.list.isRequired,
+    statusId: PropTypes.string,
+    lang: PropTypes.string,
+    index: PropTypes.number.isRequired,
+    onClose: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    onChangeBackgroundColor: PropTypes.func.isRequired,
+    currentTime: PropTypes.number,
+    autoPlay: PropTypes.bool,
+    volume: PropTypes.number,
+  };
+
+  state = {
+    index: null,
+    navigationHidden: false,
+    zoomButtonHidden: false,
+  };
+
+  handleSwipe = (index) => {
+    this.setState({ index: index % this.props.media.size });
+  };
+
+  handleTransitionEnd = () => {
+    this.setState({
+      zoomButtonHidden: false,
+    });
+  };
+
+  handleNextClick = () => {
+    this.setState({
+      index: (this.getIndex() + 1) % this.props.media.size,
+      zoomButtonHidden: true,
+    });
+  };
+
+  handlePrevClick = () => {
+    this.setState({
+      index: (this.props.media.size + this.getIndex() - 1) % this.props.media.size,
+      zoomButtonHidden: true,
+    });
+  };
+
+  handleChangeIndex = (e) => {
+    const index = Number(e.currentTarget.getAttribute('data-index'));
+
+    this.setState({
+      index: index % this.props.media.size,
+      zoomButtonHidden: true,
+    });
+  };
+
+  handleKeyDown = (e) => {
+    switch(e.key) {
+    case 'ArrowLeft':
+      this.handlePrevClick();
+      e.preventDefault();
+      e.stopPropagation();
+      break;
+    case 'ArrowRight':
+      this.handleNextClick();
+      e.preventDefault();
+      e.stopPropagation();
+      break;
+    }
+  };
+
+  componentDidMount () {
+    window.addEventListener('keydown', this.handleKeyDown, false);
+
+    this._sendBackgroundColor();
+  }
+
+  componentDidUpdate (prevProps, prevState) {
+    if (prevState.index !== this.state.index) {
+      this._sendBackgroundColor();
+    }
+  }
+
+  _sendBackgroundColor () {
+    const { media, onChangeBackgroundColor } = this.props;
+    const index = this.getIndex();
+    const blurhash = media.getIn([index, 'blurhash']);
+
+    if (blurhash) {
+      const backgroundColor = getAverageFromBlurhash(blurhash);
+      onChangeBackgroundColor(backgroundColor);
+    }
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('keydown', this.handleKeyDown);
+
+    this.props.onChangeBackgroundColor(null);
+  }
+
+  getIndex () {
+    return this.state.index !== null ? this.state.index : this.props.index;
+  }
+
+  toggleNavigation = () => {
+    this.setState(prevState => ({
+      navigationHidden: !prevState.navigationHidden,
+    }));
+  };
+
+  render () {
+    const { media, statusId, lang, intl, onClose } = this.props;
+    const { navigationHidden } = this.state;
+
+    const index = this.getIndex();
+
+    const leftNav  = media.size > 1 && <button tabIndex={0} className='media-modal__nav media-modal__nav--left' onClick={this.handlePrevClick} aria-label={intl.formatMessage(messages.previous)}><Icon id='chevron-left' fixedWidth /></button>;
+    const rightNav = media.size > 1 && <button tabIndex={0} className='media-modal__nav  media-modal__nav--right' onClick={this.handleNextClick} aria-label={intl.formatMessage(messages.next)}><Icon id='chevron-right' fixedWidth /></button>;
+
+    const content = media.map((image) => {
+      const width  = image.getIn(['meta', 'original', 'width']) || null;
+      const height = image.getIn(['meta', 'original', 'height']) || null;
+      const description = image.getIn(['translation', 'description']) || image.get('description');
+
+      if (image.get('type') === 'image') {
+        return (
+          <ImageLoader
+            previewSrc={image.get('preview_url')}
+            src={image.get('url')}
+            width={width}
+            height={height}
+            alt={description}
+            lang={lang}
+            key={image.get('url')}
+            onClick={this.toggleNavigation}
+            zoomButtonHidden={this.state.zoomButtonHidden}
+          />
+        );
+      } else if (image.get('type') === 'video') {
+        const { currentTime, autoPlay, volume } = this.props;
+
+        return (
+          <Video
+            preview={image.get('preview_url')}
+            blurhash={image.get('blurhash')}
+            src={image.get('url')}
+            width={image.get('width')}
+            height={image.get('height')}
+            frameRate={image.getIn(['meta', 'original', 'frame_rate'])}
+            currentTime={currentTime || 0}
+            autoPlay={autoPlay || false}
+            volume={volume || 1}
+            onCloseVideo={onClose}
+            detailed
+            alt={description}
+            lang={lang}
+            key={image.get('url')}
+          />
+        );
+      } else if (image.get('type') === 'gifv') {
+        return (
+          <GIFV
+            src={image.get('url')}
+            width={width}
+            height={height}
+            key={image.get('url')}
+            alt={description}
+            lang={lang}
+            onClick={this.toggleNavigation}
+          />
+        );
+      }
+
+      return null;
+    }).toArray();
+
+    // you can't use 100vh, because the viewport height is taller
+    // than the visible part of the document in some mobile
+    // browsers when it's address bar is visible.
+    // https://developers.google.com/web/updates/2016/12/url-bar-resizing
+    const swipeableViewsStyle = {
+      width: '100%',
+      height: '100%',
+    };
+
+    const containerStyle = {
+      alignItems: 'center', // center vertically
+    };
+
+    const navigationClassName = classNames('media-modal__navigation', {
+      'media-modal__navigation--hidden': navigationHidden,
+    });
+
+    let pagination;
+
+    if (media.size > 1) {
+      pagination = media.map((item, i) => (
+        <button key={i} className={classNames('media-modal__page-dot', { active: i === index })} data-index={i} onClick={this.handleChangeIndex}>
+          {i + 1}
+        </button>
+      ));
+    }
+
+    return (
+      <div className='modal-root__modal media-modal'>
+        <div className='media-modal__closer' role='presentation' onClick={onClose} >
+          <ReactSwipeableViews
+            style={swipeableViewsStyle}
+            containerStyle={containerStyle}
+            onChangeIndex={this.handleSwipe}
+            onTransitionEnd={this.handleTransitionEnd}
+            index={index}
+            disabled={disableSwiping}
+          >
+            {content}
+          </ReactSwipeableViews>
+        </div>
+
+        <div className={navigationClassName}>
+          <IconButton className='media-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={40} />
+
+          {leftNav}
+          {rightNav}
+
+          <div className='media-modal__overlay'>
+            {pagination && <ul className='media-modal__pagination'>{pagination}</ul>}
+            {statusId && <Footer statusId={statusId} withOpenButton onClose={onClose} />}
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(MediaModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/modal_loading.jsx b/app/javascript/flavours/blobfox/features/ui/components/modal_loading.jsx
new file mode 100644
index 00000000000000..7d19e73513336d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/modal_loading.jsx
@@ -0,0 +1,18 @@
+import { LoadingIndicator } from '../../../components/loading_indicator';
+
+// Keep the markup in sync with <BundleModalError />
+// (make sure they have the same dimensions)
+const ModalLoading = () => (
+  <div className='modal-root__modal error-modal'>
+    <div className='error-modal__body'>
+      <LoadingIndicator />
+    </div>
+    <div className='error-modal__footer'>
+      <div>
+        <button className='error-modal__nav onboarding-modal__skip' />
+      </div>
+    </div>
+  </div>
+);
+
+export default ModalLoading;
diff --git a/app/javascript/flavours/blobfox/features/ui/components/modal_root.jsx b/app/javascript/flavours/blobfox/features/ui/components/modal_root.jsx
new file mode 100644
index 00000000000000..81c63cab9fc401
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/modal_root.jsx
@@ -0,0 +1,142 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { Helmet } from 'react-helmet';
+
+import Base from 'flavours/blobfox/components/modal_root';
+import {
+  MuteModal,
+  BlockModal,
+  ReportModal,
+  SettingsModal,
+  EmbedModal,
+  ListEditor,
+  ListAdder,
+  PinnedAccountsEditor,
+  CompareHistoryModal,
+  FilterModal,
+  InteractionModal,
+  SubscribedLanguagesModal,
+  ClosedRegistrationsModal,
+} from 'flavours/blobfox/features/ui/util/async-components';
+import { getScrollbarWidth } from 'flavours/blobfox/utils/scrollbar';
+
+import BundleContainer from '../containers/bundle_container';
+
+import ActionsModal from './actions_modal';
+import AudioModal from './audio_modal';
+import BoostModal from './boost_modal';
+import BundleModalError from './bundle_modal_error';
+import ConfirmationModal from './confirmation_modal';
+import DeprecatedSettingsModal from './deprecated_settings_modal';
+import DoodleModal from './doodle_modal';
+import FavouriteModal from './favourite_modal';
+import FocalPointModal from './focal_point_modal';
+import ImageModal from './image_modal';
+import MediaModal from './media_modal';
+import ModalLoading from './modal_loading';
+import VideoModal from './video_modal';
+
+export const MODAL_COMPONENTS = {
+  'MEDIA': () => Promise.resolve({ default: MediaModal }),
+  'VIDEO': () => Promise.resolve({ default: VideoModal }),
+  'AUDIO': () => Promise.resolve({ default: AudioModal }),
+  'IMAGE': () => Promise.resolve({ default: ImageModal }),
+  'BOOST': () => Promise.resolve({ default: BoostModal }),
+  'FAVOURITE': () => Promise.resolve({ default: FavouriteModal }),
+  'DOODLE': () => Promise.resolve({ default: DoodleModal }),
+  'CONFIRM': () => Promise.resolve({ default: ConfirmationModal }),
+  'MUTE': MuteModal,
+  'BLOCK': BlockModal,
+  'REPORT': ReportModal,
+  'SETTINGS': SettingsModal,
+  'DEPRECATED_SETTINGS': () => Promise.resolve({ default: DeprecatedSettingsModal }),
+  'ACTIONS': () => Promise.resolve({ default: ActionsModal }),
+  'EMBED': EmbedModal,
+  'LIST_EDITOR': ListEditor,
+  'FOCAL_POINT': () => Promise.resolve({ default: FocalPointModal }),
+  'LIST_ADDER': ListAdder,
+  'PINNED_ACCOUNTS_EDITOR': PinnedAccountsEditor,
+  'COMPARE_HISTORY': CompareHistoryModal,
+  'FILTER': FilterModal,
+  'SUBSCRIBED_LANGUAGES': SubscribedLanguagesModal,
+  'INTERACTION': InteractionModal,
+  'CLOSED_REGISTRATIONS': ClosedRegistrationsModal,
+};
+
+export default class ModalRoot extends PureComponent {
+
+  static propTypes = {
+    type: PropTypes.string,
+    props: PropTypes.object,
+    onClose: PropTypes.func.isRequired,
+    ignoreFocus: PropTypes.bool,
+  };
+
+  state = {
+    backgroundColor: null,
+  };
+
+  componentDidUpdate () {
+    if (this.props.type) {
+      document.body.classList.add('with-modals--active');
+      document.documentElement.style.marginRight = `${getScrollbarWidth()}px`;
+    } else {
+      document.body.classList.remove('with-modals--active');
+      document.documentElement.style.marginRight = '0';
+    }
+  }
+
+  setBackgroundColor = color => {
+    this.setState({ backgroundColor: color });
+  };
+
+  renderLoading = modalId => () => {
+    return ['MEDIA', 'VIDEO', 'BOOST', 'FAVOURITE', 'DOODLE', 'CONFIRM', 'ACTIONS'].indexOf(modalId) === -1 ? <ModalLoading /> : null;
+  };
+
+  renderError = (props) => {
+    const { onClose } = this.props;
+
+    return <BundleModalError {...props} onClose={onClose} />;
+  };
+
+  handleClose = (ignoreFocus = false) => {
+    const { onClose } = this.props;
+    const message = this._modal?.getCloseConfirmationMessage?.();
+    onClose(message, ignoreFocus);
+  };
+
+  setModalRef = (c) => {
+    this._modal = c;
+  };
+
+  // prevent closing of modal when clicking the overlay
+  noop = () => {};
+
+  render () {
+    const { type, props, ignoreFocus } = this.props;
+    const { backgroundColor } = this.state;
+    const visible = !!type;
+
+    return (
+      <Base backgroundColor={backgroundColor} onClose={props && props.noClose ? this.noop : this.handleClose} noEsc={props ? props.noEsc : false} ignoreFocus={ignoreFocus}>
+        {visible && (
+          <>
+            <BundleContainer fetchComponent={MODAL_COMPONENTS[type]} loading={this.renderLoading(type)} error={this.renderError} renderDelay={200}>
+              {(SpecificComponent) => {
+                const ref = typeof SpecificComponent !== 'function' ? this.setModalRef : undefined;
+                return <SpecificComponent {...props} onChangeBackgroundColor={this.setBackgroundColor} onClose={this.handleClose} ref={ref} />;
+              }}
+            </BundleContainer>
+
+            <Helmet>
+              <meta name='robots' content='noindex' />
+            </Helmet>
+          </>
+        )}
+      </Base>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/mute_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/mute_modal.jsx
new file mode 100644
index 00000000000000..2d95cabef84db1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/mute_modal.jsx
@@ -0,0 +1,139 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import Toggle from 'react-toggle';
+
+import { muteAccount } from '../../../actions/accounts';
+import { closeModal } from '../../../actions/modal';
+import { toggleHideNotifications, changeMuteDuration } from '../../../actions/mutes';
+import { Button } from '../../../components/button';
+
+const messages = defineMessages({
+  minutes: { id: 'intervals.full.minutes', defaultMessage: '{number, plural, one {# minute} other {# minutes}}' },
+  hours: { id: 'intervals.full.hours', defaultMessage: '{number, plural, one {# hour} other {# hours}}' },
+  days: { id: 'intervals.full.days', defaultMessage: '{number, plural, one {# day} other {# days}}' },
+  indefinite: { id: 'mute_modal.indefinite', defaultMessage: 'Indefinite' },
+});
+
+const mapStateToProps = state => {
+  return {
+    account: state.getIn(['mutes', 'new', 'account']),
+    notifications: state.getIn(['mutes', 'new', 'notifications']),
+    muteDuration: state.getIn(['mutes', 'new', 'duration']),
+  };
+};
+
+const mapDispatchToProps = dispatch => {
+  return {
+    onConfirm(account, notifications, muteDuration) {
+      dispatch(muteAccount(account.get('id'), notifications, muteDuration));
+    },
+
+    onClose() {
+      dispatch(closeModal({
+        modalType: undefined,
+        ignoreFocus: false,
+      }));
+    },
+
+    onToggleNotifications() {
+      dispatch(toggleHideNotifications());
+    },
+
+    onChangeMuteDuration(e) {
+      dispatch(changeMuteDuration(e.target.value));
+    },
+  };
+};
+
+class MuteModal extends PureComponent {
+
+  static propTypes = {
+    account: PropTypes.object.isRequired,
+    notifications: PropTypes.bool.isRequired,
+    onClose: PropTypes.func.isRequired,
+    onConfirm: PropTypes.func.isRequired,
+    onToggleNotifications: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    muteDuration: PropTypes.number.isRequired,
+    onChangeMuteDuration: PropTypes.func.isRequired,
+  };
+
+  handleClick = () => {
+    this.props.onClose();
+    this.props.onConfirm(this.props.account, this.props.notifications, this.props.muteDuration);
+  };
+
+  handleCancel = () => {
+    this.props.onClose();
+  };
+
+  toggleNotifications = () => {
+    this.props.onToggleNotifications();
+  };
+
+  changeMuteDuration = (e) => {
+    this.props.onChangeMuteDuration(e);
+  };
+
+  render () {
+    const { account, notifications, muteDuration, intl } = this.props;
+
+    return (
+      <div className='modal-root__modal mute-modal'>
+        <div className='mute-modal__container'>
+          <p>
+            <FormattedMessage
+              id='confirmations.mute.message'
+              defaultMessage='Are you sure you want to mute {name}?'
+              values={{ name: <strong>@{account.get('acct')}</strong> }}
+            />
+          </p>
+          <p className='mute-modal__explanation'>
+            <FormattedMessage
+              id='confirmations.mute.explanation'
+              defaultMessage='This will hide posts from them and posts mentioning them, but it will still allow them to see your posts and follow you.'
+            />
+          </p>
+          <div className='setting-toggle'>
+            <Toggle id='mute-modal__hide-notifications-checkbox' checked={notifications} onChange={this.toggleNotifications} />
+            <label className='setting-toggle__label' htmlFor='mute-modal__hide-notifications-checkbox'>
+              <FormattedMessage id='mute_modal.hide_notifications' defaultMessage='Hide notifications from this user?' />
+            </label>
+          </div>
+          <div>
+            <span><FormattedMessage id='mute_modal.duration' defaultMessage='Duration' />: </span>
+
+            {/* eslint-disable-next-line jsx-a11y/no-onchange */}
+            <select value={muteDuration} onChange={this.changeMuteDuration}>
+              <option value={0}>{intl.formatMessage(messages.indefinite)}</option>
+              <option value={300}>{intl.formatMessage(messages.minutes, { number: 5 })}</option>
+              <option value={1800}>{intl.formatMessage(messages.minutes, { number: 30 })}</option>
+              <option value={3600}>{intl.formatMessage(messages.hours, { number: 1 })}</option>
+              <option value={21600}>{intl.formatMessage(messages.hours, { number: 6 })}</option>
+              <option value={86400}>{intl.formatMessage(messages.days, { number: 1 })}</option>
+              <option value={259200}>{intl.formatMessage(messages.days, { number: 3 })}</option>
+              <option value={604800}>{intl.formatMessage(messages.days, { number: 7 })}</option>
+            </select>
+          </div>
+        </div>
+
+        <div className='mute-modal__action-bar'>
+          <Button onClick={this.handleCancel} className='mute-modal__cancel-button'>
+            <FormattedMessage id='confirmation_modal.cancel' defaultMessage='Cancel' />
+          </Button>
+          <Button onClick={this.handleClick} autoFocus>
+            <FormattedMessage id='confirmations.mute.confirm' defaultMessage='Mute' />
+          </Button>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, mapDispatchToProps)(injectIntl(MuteModal));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/navigation_panel.jsx b/app/javascript/flavours/blobfox/features/ui/components/navigation_panel.jsx
new file mode 100644
index 00000000000000..5ed79d0c2b154c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/navigation_panel.jsx
@@ -0,0 +1,127 @@
+import PropTypes from 'prop-types';
+import { Component } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { NavigationPortal } from 'flavours/blobfox/components/navigation_portal';
+import { timelinePreview, trendsEnabled } from 'flavours/blobfox/initial_state';
+import { transientSingleColumn } from 'flavours/blobfox/is_mobile';
+import { preferencesLink } from 'flavours/blobfox/utils/backend_links';
+
+import ColumnLink from './column_link';
+import DisabledAccountBanner from './disabled_account_banner';
+import FollowRequestsColumnLink from './follow_requests_column_link';
+import ListPanel from './list_panel';
+import NotificationsCounterIcon from './notifications_counter_icon';
+import SignInBanner from './sign_in_banner';
+
+const messages = defineMessages({
+  home: { id: 'tabs_bar.home', defaultMessage: 'Home' },
+  notifications: { id: 'tabs_bar.notifications', defaultMessage: 'Notifications' },
+  explore: { id: 'explore.title', defaultMessage: 'Explore' },
+  firehose: { id: 'column.firehose', defaultMessage: 'Live feeds' },
+  direct: { id: 'navigation_bar.direct', defaultMessage: 'Private mentions' },
+  favourites: { id: 'navigation_bar.favourites', defaultMessage: 'Favorites' },
+  bookmarks: { id: 'navigation_bar.bookmarks', defaultMessage: 'Bookmarks' },
+  lists: { id: 'navigation_bar.lists', defaultMessage: 'Lists' },
+  preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
+  followsAndFollowers: { id: 'navigation_bar.follows_and_followers', defaultMessage: 'Follows and followers' },
+  about: { id: 'navigation_bar.about', defaultMessage: 'About' },
+  search: { id: 'navigation_bar.search', defaultMessage: 'Search' },
+  advancedInterface: { id: 'navigation_bar.advanced_interface', defaultMessage: 'Open in advanced web interface' },
+  openedInClassicInterface: { id: 'navigation_bar.opened_in_classic_interface', defaultMessage: 'Posts, accounts, and other specific pages are opened by default in the classic web interface.' },
+  app_settings: { id: 'navigation_bar.app_settings', defaultMessage: 'App settings' },
+});
+
+class NavigationPanel extends Component {
+
+  static contextTypes = {
+    identity: PropTypes.object.isRequired,
+  };
+
+  static propTypes = {
+    intl: PropTypes.object.isRequired,
+    onOpenSettings: PropTypes.func,
+  };
+
+  isFirehoseActive = (match, location) => {
+    return match || location.pathname.startsWith('/public');
+  };
+
+  render () {
+    const { intl, onOpenSettings } = this.props;
+    const { signedIn, disabledAccountId } = this.context.identity;
+
+    let banner = undefined;
+
+    if(transientSingleColumn)
+      banner = (<div className='switch-to-advanced'>
+        {intl.formatMessage(messages.openedInClassicInterface)}
+        {" "}
+        <a href={`/deck${location.pathname}`} className='switch-to-advanced__toggle'>
+          {intl.formatMessage(messages.advancedInterface)}
+        </a>
+      </div>);
+
+    return (
+      <div className='navigation-panel'>
+        {banner &&
+          <div className='navigation-panel__banner'>
+            {banner}
+          </div>
+        }
+
+        {signedIn && (
+          <>
+            <ColumnLink transparent to='/home' icon='home' text={intl.formatMessage(messages.home)} />
+            <ColumnLink transparent to='/notifications' icon={<NotificationsCounterIcon className='column-link__icon' />} text={intl.formatMessage(messages.notifications)} />
+            <FollowRequestsColumnLink />
+          </>
+        )}
+
+        {trendsEnabled ? (
+          <ColumnLink transparent to='/explore' icon='hashtag' text={intl.formatMessage(messages.explore)} />
+        ) : (
+          <ColumnLink transparent to='/search' icon='search' text={intl.formatMessage(messages.search)} />
+        )}
+
+        {(signedIn || timelinePreview) && (
+          <ColumnLink transparent to='/public/local' isActive={this.isFirehoseActive} icon='globe' text={intl.formatMessage(messages.firehose)} />
+        )}
+
+        {!signedIn && (
+          <div className='navigation-panel__sign-in-banner'>
+            <hr />
+            { disabledAccountId ? <DisabledAccountBanner /> : <SignInBanner /> }
+          </div>
+        )}
+
+        {signedIn && (
+          <>
+            <ColumnLink transparent to='/conversations' icon='at' text={intl.formatMessage(messages.direct)} />
+            <ColumnLink transparent to='/bookmarks' icon='bookmark' text={intl.formatMessage(messages.bookmarks)} />
+            <ColumnLink transparent to='/favourites' icon='star' text={intl.formatMessage(messages.favourites)} />
+            <ColumnLink transparent to='/lists' icon='list-ul' text={intl.formatMessage(messages.lists)} />
+
+            <ListPanel />
+
+            <hr />
+
+            {!!preferencesLink && <ColumnLink transparent href={preferencesLink} icon='cog' text={intl.formatMessage(messages.preferences)} />}
+            <ColumnLink transparent onClick={onOpenSettings} icon='cogs' text={intl.formatMessage(messages.app_settings)} />
+          </>
+        )}
+
+        <div className='navigation-panel__legal'>
+          <hr />
+          <ColumnLink transparent to='/about' icon='ellipsis-h' text={intl.formatMessage(messages.about)} />
+        </div>
+
+        <NavigationPortal />
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(NavigationPanel);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/notifications_counter_icon.js b/app/javascript/flavours/blobfox/features/ui/components/notifications_counter_icon.js
new file mode 100644
index 00000000000000..1509a0962ee93d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/notifications_counter_icon.js
@@ -0,0 +1,10 @@
+import { connect } from 'react-redux';
+
+import { IconWithBadge } from 'flavours/blobfox/components/icon_with_badge';
+
+const mapStateToProps = state => ({
+  count: state.getIn(['local_settings', 'notifications', 'tab_badge']) ? state.getIn(['notifications', 'unread']) : 0,
+  id: 'bell',
+});
+
+export default connect(mapStateToProps)(IconWithBadge);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/report_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/report_modal.jsx
new file mode 100644
index 00000000000000..79029da022d5ca
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/report_modal.jsx
@@ -0,0 +1,227 @@
+import PropTypes from 'prop-types';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import { OrderedSet } from 'immutable';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { fetchRelationships } from 'flavours/blobfox/actions/accounts';
+import { submitReport } from 'flavours/blobfox/actions/reports';
+import { fetchServer } from 'flavours/blobfox/actions/server';
+import { expandAccountTimeline } from 'flavours/blobfox/actions/timelines';
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+import Category from 'flavours/blobfox/features/report/category';
+import Comment from 'flavours/blobfox/features/report/comment';
+import Rules from 'flavours/blobfox/features/report/rules';
+import Statuses from 'flavours/blobfox/features/report/statuses';
+import Thanks from 'flavours/blobfox/features/report/thanks';
+import { makeGetAccount } from 'flavours/blobfox/selectors';
+
+const messages = defineMessages({
+  close: { id: 'lightbox.close', defaultMessage: 'Close' },
+});
+
+const makeMapStateToProps = () => {
+  const getAccount = makeGetAccount();
+
+  const mapStateToProps = (state, { accountId }) => ({
+    account: getAccount(state, accountId),
+  });
+
+  return mapStateToProps;
+};
+
+class ReportModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    accountId: PropTypes.string.isRequired,
+    statusId: PropTypes.string,
+    dispatch: PropTypes.func.isRequired,
+    intl: PropTypes.object.isRequired,
+    account: ImmutablePropTypes.record.isRequired,
+  };
+
+  state = {
+    step: 'category',
+    selectedStatusIds: OrderedSet(this.props.statusId ? [this.props.statusId] : []),
+    selectedDomains: OrderedSet(),
+    comment: '',
+    category: null,
+    selectedRuleIds: OrderedSet(),
+    isSubmitting: false,
+    isSubmitted: false,
+  };
+
+  handleSubmit = () => {
+    const { dispatch, accountId } = this.props;
+    const { selectedStatusIds, selectedDomains, comment, category, selectedRuleIds } = this.state;
+
+    this.setState({ isSubmitting: true });
+
+    dispatch(submitReport({
+      account_id: accountId,
+      status_ids: selectedStatusIds.toArray(),
+      forward_to_domains: selectedDomains.toArray(),
+      comment,
+      forward: selectedDomains.size > 0,
+      category,
+      rule_ids: selectedRuleIds.toArray(),
+    }, this.handleSuccess, this.handleFail));
+  };
+
+  handleSuccess = () => {
+    this.setState({ isSubmitting: false, isSubmitted: true, step: 'thanks' });
+  };
+
+  handleFail = () => {
+    this.setState({ isSubmitting: false });
+  };
+
+  handleStatusToggle = (statusId, checked) => {
+    const { selectedStatusIds } = this.state;
+
+    if (checked) {
+      this.setState({ selectedStatusIds: selectedStatusIds.add(statusId) });
+    } else {
+      this.setState({ selectedStatusIds: selectedStatusIds.remove(statusId) });
+    }
+  };
+
+  handleDomainToggle = (domain, checked) => {
+    if (checked) {
+      this.setState((state) => ({ selectedDomains: state.selectedDomains.add(domain) }));
+    } else {
+      this.setState((state) => ({ selectedDomains: state.selectedDomains.remove(domain) }));
+    }
+  };
+
+  handleRuleToggle = (ruleId, checked) => {
+    if (checked) {
+      this.setState((state) => ({ selectedRuleIds: state.selectedRuleIds.add(ruleId) }));
+    } else {
+      this.setState((state) => ({ selectedRuleIds: state.selectedRuleIds.remove(ruleId) }));
+    }
+  };
+
+  handleChangeCategory = category => {
+    this.setState({ category });
+  };
+
+  handleChangeComment = comment => {
+    this.setState({ comment });
+  };
+
+  handleNextStep = step => {
+    this.setState({ step });
+  };
+
+  componentDidMount () {
+    const { dispatch, accountId } = this.props;
+
+    dispatch(fetchRelationships([accountId]));
+    dispatch(expandAccountTimeline(accountId, { withReplies: true }));
+    dispatch(fetchServer());
+  }
+
+  render () {
+    const {
+      accountId,
+      account,
+      intl,
+      onClose,
+    } = this.props;
+
+    if (!account) {
+      return null;
+    }
+
+    const {
+      step,
+      selectedStatusIds,
+      selectedRuleIds,
+      selectedDomains,
+      comment,
+      category,
+      isSubmitting,
+      isSubmitted,
+    } = this.state;
+
+    const domain   = account.get('acct').split('@')[1];
+    const isRemote = !!domain;
+
+    let stepComponent;
+
+    switch(step) {
+    case 'category':
+      stepComponent = (
+        <Category
+          onNextStep={this.handleNextStep}
+          startedFrom={this.props.statusId ? 'status' : 'account'}
+          category={category}
+          onChangeCategory={this.handleChangeCategory}
+        />
+      );
+      break;
+    case 'rules':
+      stepComponent = (
+        <Rules
+          onNextStep={this.handleNextStep}
+          selectedRuleIds={selectedRuleIds}
+          onToggle={this.handleRuleToggle}
+        />
+      );
+      break;
+    case 'statuses':
+      stepComponent = (
+        <Statuses
+          onNextStep={this.handleNextStep}
+          accountId={accountId}
+          selectedStatusIds={selectedStatusIds}
+          onToggle={this.handleStatusToggle}
+        />
+      );
+      break;
+    case 'comment':
+      stepComponent = (
+        <Comment
+          onSubmit={this.handleSubmit}
+          isSubmitting={isSubmitting}
+          isRemote={isRemote}
+          comment={comment}
+          domain={domain}
+          onChangeComment={this.handleChangeComment}
+          statusIds={selectedStatusIds}
+          selectedDomains={selectedDomains}
+          onToggleDomain={this.handleDomainToggle}
+        />
+      );
+      break;
+    case 'thanks':
+      stepComponent = (
+        <Thanks
+          submitted={isSubmitted}
+          account={account}
+          onClose={onClose}
+        />
+      );
+    }
+
+    return (
+      <div className='modal-root__modal report-dialog-modal'>
+        <div className='report-modal__target'>
+          <IconButton className='report-modal__close' title={intl.formatMessage(messages.close)} icon='times' onClick={onClose} size={20} />
+          <FormattedMessage id='report.target' defaultMessage='Report {target}' values={{ target: <strong>{account.get('acct')}</strong> }} />
+        </div>
+
+        <div className='report-dialog-modal__container'>
+          {stepComponent}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(makeMapStateToProps)(injectIntl(ReportModal));
diff --git a/app/javascript/flavours/blobfox/features/ui/components/sign_in_banner.jsx b/app/javascript/flavours/blobfox/features/ui/components/sign_in_banner.jsx
new file mode 100644
index 00000000000000..069e45a4f9d645
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/sign_in_banner.jsx
@@ -0,0 +1,54 @@
+import { useCallback } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+import { registrationsOpen, sso_redirect } from 'flavours/blobfox/initial_state';
+import { useAppDispatch, useAppSelector } from 'flavours/blobfox/store';
+
+const SignInBanner = () => {
+  const dispatch = useAppDispatch();
+
+  const openClosedRegistrationsModal = useCallback(
+    () => dispatch(openModal({ modalType: 'CLOSED_REGISTRATIONS' })),
+    [dispatch],
+  );
+
+  let signupButton;
+
+  const signupUrl = useAppSelector((state) => state.getIn(['server', 'server', 'registrations', 'url'], null) || '/auth/sign_up');
+
+  if (sso_redirect) {
+    return (
+      <div className='sign-in-banner'>
+        <p><FormattedMessage id='sign_in_banner.text' defaultMessage='Login to follow profiles or hashtags, favorite, share and reply to posts. You can also interact from your account on a different server.' /></p>
+        <a href={sso_redirect} data-method='post' className='button button--block button-tertiary'><FormattedMessage id='sign_in_banner.sso_redirect' defaultMessage='Login or Register' /></a>
+      </div>
+    );
+  }
+
+  if (registrationsOpen) {
+    signupButton = (
+      <a href={signupUrl} className='button button--block'>
+        <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+      </a>
+    );
+  } else {
+    signupButton = (
+      <button className='button button--block' onClick={openClosedRegistrationsModal}>
+        <FormattedMessage id='sign_in_banner.create_account' defaultMessage='Create account' />
+      </button>
+    );
+  }
+
+  return (
+    <div className='sign-in-banner'>
+      <p><FormattedMessage id='sign_in_banner.text' defaultMessage='Login to follow profiles or hashtags, favorite, share and reply to posts. You can also interact from your account on a different server.' /></p>
+      {signupButton}
+      <a href='/auth/sign_in' className='button button--block button-tertiary'><FormattedMessage id='sign_in_banner.sign_in' defaultMessage='Login' /></a>
+    </div>
+  );
+};
+
+export default SignInBanner;
diff --git a/app/javascript/flavours/blobfox/features/ui/components/upload_area.jsx b/app/javascript/flavours/blobfox/features/ui/components/upload_area.jsx
new file mode 100644
index 00000000000000..b2702d35efdfd6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/upload_area.jsx
@@ -0,0 +1,55 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { FormattedMessage } from 'react-intl';
+
+import spring from 'react-motion/lib/spring';
+
+import Motion from '../util/optional_motion';
+
+export default class UploadArea extends PureComponent {
+
+  static propTypes = {
+    active: PropTypes.bool,
+    onClose: PropTypes.func,
+  };
+
+  handleKeyUp = (e) => {
+    const keyCode = e.keyCode;
+    if (this.props.active) {
+      switch(keyCode) {
+      case 27:
+        e.preventDefault();
+        e.stopPropagation();
+        this.props.onClose();
+        break;
+      }
+    }
+  };
+
+  componentDidMount () {
+    window.addEventListener('keyup', this.handleKeyUp, false);
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('keyup', this.handleKeyUp);
+  }
+
+  render () {
+    const { active } = this.props;
+
+    return (
+      <Motion defaultStyle={{ backgroundOpacity: 0, backgroundScale: 0.95 }} style={{ backgroundOpacity: spring(active ? 1 : 0, { stiffness: 150, damping: 15 }), backgroundScale: spring(active ? 1 : 0.95, { stiffness: 200, damping: 3 }) }}>
+        {({ backgroundOpacity, backgroundScale }) => (
+          <div className='upload-area' style={{ visibility: active ? 'visible' : 'hidden', opacity: backgroundOpacity }}>
+            <div className='upload-area__drop'>
+              <div className='upload-area__background' style={{ transform: `scale(${backgroundScale})` }} />
+              <div className='upload-area__content'><FormattedMessage id='upload_area.title' defaultMessage='Drag & drop to upload' /></div>
+            </div>
+          </div>
+        )}
+      </Motion>
+    );
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/components/video_modal.jsx b/app/javascript/flavours/blobfox/features/ui/components/video_modal.jsx
new file mode 100644
index 00000000000000..7bed641cd968c1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/video_modal.jsx
@@ -0,0 +1,74 @@
+import PropTypes from 'prop-types';
+
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { connect } from 'react-redux';
+
+import { getAverageFromBlurhash } from 'flavours/blobfox/blurhash';
+import Footer from 'flavours/blobfox/features/picture_in_picture/components/footer';
+import Video from 'flavours/blobfox/features/video';
+
+const mapStateToProps = (state, { statusId }) => ({
+  status: state.getIn(['statuses', statusId]),
+});
+
+class VideoModal extends ImmutablePureComponent {
+
+  static propTypes = {
+    media: ImmutablePropTypes.map.isRequired,
+    statusId: PropTypes.string,
+    status: ImmutablePropTypes.map,
+    options: PropTypes.shape({
+      startTime: PropTypes.number,
+      autoPlay: PropTypes.bool,
+      defaultVolume: PropTypes.number,
+    }),
+    onClose: PropTypes.func.isRequired,
+    onChangeBackgroundColor: PropTypes.func.isRequired,
+  };
+
+  componentDidMount () {
+    const { media, onChangeBackgroundColor } = this.props;
+
+    const backgroundColor = getAverageFromBlurhash(media.get('blurhash'));
+
+    if (backgroundColor) {
+      onChangeBackgroundColor(backgroundColor);
+    }
+  }
+
+  render () {
+    const { media, status, onClose } = this.props;
+    const options = this.props.options || {};
+    const language = status.getIn(['translation', 'language']) || status.get('language');
+    const description = media.getIn(['translation', 'description']) || media.get('description');
+
+    return (
+      <div className='modal-root__modal video-modal'>
+        <div className='video-modal__container'>
+          <Video
+            preview={media.get('preview_url')}
+            frameRate={media.getIn(['meta', 'original', 'frame_rate'])}
+            blurhash={media.get('blurhash')}
+            src={media.get('url')}
+            currentTime={options.startTime}
+            autoPlay={options.autoPlay}
+            volume={options.defaultVolume}
+            onCloseVideo={onClose}
+            autoFocus
+            detailed
+            alt={description}
+            lang={language}
+          />
+        </div>
+
+        <div className='media-modal__overlay'>
+          {status && <Footer statusId={status.get('id')} withOpenButton onClose={onClose} />}
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps, null, null, { forwardRef: true })(VideoModal);
diff --git a/app/javascript/flavours/blobfox/features/ui/components/zoomable_image.jsx b/app/javascript/flavours/blobfox/features/ui/components/zoomable_image.jsx
new file mode 100644
index 00000000000000..7ec3a3d6cd7f5a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/components/zoomable_image.jsx
@@ -0,0 +1,456 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl } from 'react-intl';
+
+import { IconButton } from 'flavours/blobfox/components/icon_button';
+
+const messages = defineMessages({
+  compress: { id: 'lightbox.compress', defaultMessage: 'Compress image view box' },
+  expand: { id: 'lightbox.expand', defaultMessage: 'Expand image view box' },
+});
+
+const MIN_SCALE = 1;
+const MAX_SCALE = 4;
+const NAV_BAR_HEIGHT = 66;
+
+const getMidpoint = (p1, p2) => ({
+  x: (p1.clientX + p2.clientX) / 2,
+  y: (p1.clientY + p2.clientY) / 2,
+});
+
+const getDistance = (p1, p2) =>
+  Math.sqrt(Math.pow(p1.clientX - p2.clientX, 2) + Math.pow(p1.clientY - p2.clientY, 2));
+
+const clamp = (min, max, value) => Math.min(max, Math.max(min, value));
+
+// Normalizing mousewheel speed across browsers
+// copy from: https://github.com/facebookarchive/fixed-data-table/blob/master/src/vendor_upstream/dom/normalizeWheel.js
+const normalizeWheel = event => {
+  // Reasonable defaults
+  const PIXEL_STEP = 10;
+  const LINE_HEIGHT = 40;
+  const PAGE_HEIGHT = 800;
+
+  let sX = 0,
+    sY = 0, // spinX, spinY
+    pX = 0,
+    pY = 0; // pixelX, pixelY
+
+  // Legacy
+  if ('detail' in event) {
+    sY = event.detail;
+  }
+  if ('wheelDelta' in event) {
+    sY = -event.wheelDelta / 120;
+  }
+  if ('wheelDeltaY' in event) {
+    sY = -event.wheelDeltaY / 120;
+  }
+  if ('wheelDeltaX' in event) {
+    sX = -event.wheelDeltaX / 120;
+  }
+
+  // side scrolling on FF with DOMMouseScroll
+  if ('axis' in event && event.axis === event.HORIZONTAL_AXIS) {
+    sX = sY;
+    sY = 0;
+  }
+
+  pX = sX * PIXEL_STEP;
+  pY = sY * PIXEL_STEP;
+
+  if ('deltaY' in event) {
+    pY = event.deltaY;
+  }
+  if ('deltaX' in event) {
+    pX = event.deltaX;
+  }
+
+  if ((pX || pY) && event.deltaMode) {
+    if (event.deltaMode === 1) { // delta in LINE units
+      pX *= LINE_HEIGHT;
+      pY *= LINE_HEIGHT;
+    } else { // delta in PAGE units
+      pX *= PAGE_HEIGHT;
+      pY *= PAGE_HEIGHT;
+    }
+  }
+
+  // Fall-back if spin cannot be determined
+  if (pX && !sX) {
+    sX = (pX < 1) ? -1 : 1;
+  }
+  if (pY && !sY) {
+    sY = (pY < 1) ? -1 : 1;
+  }
+
+  return {
+    spinX: sX,
+    spinY: sY,
+    pixelX: pX,
+    pixelY: pY,
+  };
+};
+
+class ZoomableImage extends PureComponent {
+
+  static propTypes = {
+    alt: PropTypes.string,
+    lang: PropTypes.string,
+    src: PropTypes.string.isRequired,
+    width: PropTypes.number,
+    height: PropTypes.number,
+    onClick: PropTypes.func,
+    zoomButtonHidden: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+  };
+
+  static defaultProps = {
+    alt: '',
+    lang: '',
+    width: null,
+    height: null,
+  };
+
+  state = {
+    scale: MIN_SCALE,
+    zoomMatrix: {
+      type: null, // 'width' 'height'
+      fullScreen: null, // bool
+      rate: null, // full screen scale rate
+      clientWidth: null,
+      clientHeight: null,
+      offsetWidth: null,
+      offsetHeight: null,
+      clientHeightFixed: null,
+      scrollTop: null,
+      scrollLeft: null,
+      translateX: null,
+      translateY: null,
+    },
+    zoomState: 'expand', // 'expand' 'compress'
+    navigationHidden: false,
+    dragPosition: { top: 0, left: 0, x: 0, y: 0 },
+    dragged: false,
+    lockScroll: { x: 0, y: 0 },
+    lockTranslate: { x: 0, y: 0 },
+  };
+
+  removers = [];
+  container = null;
+  image = null;
+  lastTouchEndTime = 0;
+  lastDistance = 0;
+
+  componentDidMount () {
+    let handler = this.handleTouchStart;
+    this.container.addEventListener('touchstart', handler);
+    this.removers.push(() => this.container.removeEventListener('touchstart', handler));
+    handler = this.handleTouchMove;
+    // on Chrome 56+, touch event listeners will default to passive
+    // https://www.chromestatus.com/features/5093566007214080
+    this.container.addEventListener('touchmove', handler, { passive: false });
+    this.removers.push(() => this.container.removeEventListener('touchend', handler));
+
+    handler = this.mouseDownHandler;
+    this.container.addEventListener('mousedown', handler);
+    this.removers.push(() => this.container.removeEventListener('mousedown', handler));
+
+    handler = this.mouseWheelHandler;
+    this.container.addEventListener('wheel', handler);
+    this.removers.push(() => this.container.removeEventListener('wheel', handler));
+    // Old Chrome
+    this.container.addEventListener('mousewheel', handler);
+    this.removers.push(() => this.container.removeEventListener('mousewheel', handler));
+    // Old Firefox
+    this.container.addEventListener('DOMMouseScroll', handler);
+    this.removers.push(() => this.container.removeEventListener('DOMMouseScroll', handler));
+
+    this.initZoomMatrix();
+  }
+
+  componentWillUnmount () {
+    this.removeEventListeners();
+  }
+
+  componentDidUpdate () {
+    this.setState({ zoomState: this.state.scale >= this.state.zoomMatrix.rate ? 'compress' : 'expand' });
+
+    if (this.state.scale === MIN_SCALE) {
+      this.container.style.removeProperty('cursor');
+    }
+  }
+
+  UNSAFE_componentWillReceiveProps () {
+    // reset when slide to next image
+    if (this.props.zoomButtonHidden) {
+      this.setState({
+        scale: MIN_SCALE,
+        lockTranslate: { x: 0, y: 0 },
+      }, () => {
+        this.container.scrollLeft = 0;
+        this.container.scrollTop = 0;
+      });
+    }
+  }
+
+  removeEventListeners () {
+    this.removers.forEach(listeners => listeners());
+    this.removers = [];
+  }
+
+  mouseWheelHandler = e => {
+    e.preventDefault();
+
+    const event = normalizeWheel(e);
+
+    if (this.state.zoomMatrix.type === 'width') {
+      // full width, scroll vertical
+      this.container.scrollTop = Math.max(this.container.scrollTop + event.pixelY, this.state.lockScroll.y);
+    } else {
+      // full height, scroll horizontal
+      this.container.scrollLeft = Math.max(this.container.scrollLeft + event.pixelY, this.state.lockScroll.x);
+    }
+
+    // lock horizontal scroll
+    this.container.scrollLeft = Math.max(this.container.scrollLeft + event.pixelX, this.state.lockScroll.x);
+  };
+
+  mouseDownHandler = e => {
+    this.container.style.cursor = 'grabbing';
+    this.container.style.userSelect = 'none';
+
+    this.setState({ dragPosition: {
+      left: this.container.scrollLeft,
+      top: this.container.scrollTop,
+      // Get the current mouse position
+      x: e.clientX,
+      y: e.clientY,
+    } });
+
+    this.image.addEventListener('mousemove', this.mouseMoveHandler);
+    this.image.addEventListener('mouseup', this.mouseUpHandler);
+  };
+
+  mouseMoveHandler = e => {
+    const dx = e.clientX - this.state.dragPosition.x;
+    const dy = e.clientY - this.state.dragPosition.y;
+
+    this.container.scrollLeft = Math.max(this.state.dragPosition.left - dx, this.state.lockScroll.x);
+    this.container.scrollTop = Math.max(this.state.dragPosition.top - dy, this.state.lockScroll.y);
+
+    this.setState({ dragged: true });
+  };
+
+  mouseUpHandler = () => {
+    this.container.style.cursor = 'grab';
+    this.container.style.removeProperty('user-select');
+
+    this.image.removeEventListener('mousemove', this.mouseMoveHandler);
+    this.image.removeEventListener('mouseup', this.mouseUpHandler);
+  };
+
+  handleTouchStart = e => {
+    if (e.touches.length !== 2) return;
+
+    this.lastDistance = getDistance(...e.touches);
+  };
+
+  handleTouchMove = e => {
+    const { scrollTop, scrollHeight, clientHeight } = this.container;
+    if (e.touches.length === 1 && scrollTop !== scrollHeight - clientHeight) {
+      // prevent propagating event to MediaModal
+      e.stopPropagation();
+      return;
+    }
+    if (e.touches.length !== 2) return;
+
+    e.preventDefault();
+    e.stopPropagation();
+
+    const distance = getDistance(...e.touches);
+    const midpoint = getMidpoint(...e.touches);
+    const _MAX_SCALE = Math.max(MAX_SCALE, this.state.zoomMatrix.rate);
+    const scale = clamp(MIN_SCALE, _MAX_SCALE, this.state.scale * distance / this.lastDistance);
+
+    this.zoom(scale, midpoint);
+
+    this.lastMidpoint = midpoint;
+    this.lastDistance = distance;
+  };
+
+  zoom(nextScale, midpoint) {
+    const { scale, zoomMatrix } = this.state;
+    const { scrollLeft, scrollTop } = this.container;
+
+    // math memo:
+    // x = (scrollLeft + midpoint.x) / scrollWidth
+    // x' = (nextScrollLeft + midpoint.x) / nextScrollWidth
+    // scrollWidth = clientWidth * scale
+    // scrollWidth' = clientWidth * nextScale
+    // Solve x = x' for nextScrollLeft
+    const nextScrollLeft = (scrollLeft + midpoint.x) * nextScale / scale - midpoint.x;
+    const nextScrollTop = (scrollTop + midpoint.y) * nextScale / scale - midpoint.y;
+
+    this.setState({ scale: nextScale }, () => {
+      this.container.scrollLeft = nextScrollLeft;
+      this.container.scrollTop = nextScrollTop;
+      // reset the translateX/Y constantly
+      if (nextScale < zoomMatrix.rate) {
+        this.setState({
+          lockTranslate: {
+            x: zoomMatrix.fullScreen ? 0 : zoomMatrix.translateX * ((nextScale - MIN_SCALE) / (zoomMatrix.rate - MIN_SCALE)),
+            y: zoomMatrix.fullScreen ? 0 : zoomMatrix.translateY * ((nextScale - MIN_SCALE) / (zoomMatrix.rate - MIN_SCALE)),
+          },
+        });
+      }
+    });
+  }
+
+  handleClick = e => {
+    // don't propagate event to MediaModal
+    e.stopPropagation();
+    const dragged = this.state.dragged;
+    this.setState({ dragged: false });
+    if (dragged) return;
+    const handler = this.props.onClick;
+    if (handler) handler();
+    this.setState({ navigationHidden: !this.state.navigationHidden });
+  };
+
+  handleMouseDown = e => {
+    e.preventDefault();
+  };
+
+  initZoomMatrix = () => {
+    const { width, height } = this.props;
+    const { clientWidth, clientHeight } = this.container;
+    const { offsetWidth, offsetHeight } = this.image;
+    const clientHeightFixed = clientHeight - NAV_BAR_HEIGHT;
+
+    const type = width / height < clientWidth / clientHeightFixed ? 'width' : 'height';
+    const fullScreen = type === 'width' ?  width > clientWidth : height > clientHeightFixed;
+    const rate = type === 'width' ? Math.min(clientWidth, width) / offsetWidth : Math.min(clientHeightFixed, height) / offsetHeight;
+    const scrollTop = type === 'width' ?  (clientHeight - offsetHeight) / 2 - NAV_BAR_HEIGHT : (clientHeightFixed - offsetHeight) / 2;
+    const scrollLeft = (clientWidth - offsetWidth) / 2;
+    const translateX = type === 'width' ? (width - offsetWidth) / (2 * rate) : 0;
+    const translateY = type === 'height' ? (height - offsetHeight) / (2 * rate) : 0;
+
+    this.setState({
+      zoomMatrix: {
+        type: type,
+        fullScreen: fullScreen,
+        rate: rate,
+        clientWidth: clientWidth,
+        clientHeight: clientHeight,
+        offsetWidth: offsetWidth,
+        offsetHeight: offsetHeight,
+        clientHeightFixed: clientHeightFixed,
+        scrollTop: scrollTop,
+        scrollLeft: scrollLeft,
+        translateX: translateX,
+        translateY: translateY,
+      },
+    });
+  };
+
+  handleZoomClick = e => {
+    e.preventDefault();
+    e.stopPropagation();
+
+    const { scale, zoomMatrix } = this.state;
+
+    if ( scale >= zoomMatrix.rate ) {
+      this.setState({
+        scale: MIN_SCALE,
+        lockScroll: {
+          x: 0,
+          y: 0,
+        },
+        lockTranslate: {
+          x: 0,
+          y: 0,
+        },
+      }, () => {
+        this.container.scrollLeft = 0;
+        this.container.scrollTop = 0;
+      });
+    } else {
+      this.setState({
+        scale: zoomMatrix.rate,
+        lockScroll: {
+          x: zoomMatrix.scrollLeft,
+          y: zoomMatrix.scrollTop,
+        },
+        lockTranslate: {
+          x: zoomMatrix.fullScreen ? 0 : zoomMatrix.translateX,
+          y: zoomMatrix.fullScreen ? 0 : zoomMatrix.translateY,
+        },
+      }, () => {
+        this.container.scrollLeft = zoomMatrix.scrollLeft;
+        this.container.scrollTop = zoomMatrix.scrollTop;
+      });
+    }
+
+    this.container.style.cursor = 'grab';
+    this.container.style.removeProperty('user-select');
+  };
+
+  setContainerRef = c => {
+    this.container = c;
+  };
+
+  setImageRef = c => {
+    this.image = c;
+  };
+
+  render () {
+    const { alt, lang, src, width, height, intl } = this.props;
+    const { scale, lockTranslate } = this.state;
+    const overflow = scale === MIN_SCALE ? 'hidden' : 'scroll';
+    const zoomButtonShouldHide = this.state.navigationHidden || this.props.zoomButtonHidden || this.state.zoomMatrix.rate <= MIN_SCALE ? 'media-modal__zoom-button--hidden' : '';
+    const zoomButtonTitle = this.state.zoomState === 'compress' ? intl.formatMessage(messages.compress) : intl.formatMessage(messages.expand);
+
+    return (
+      <>
+        <IconButton
+          className={`media-modal__zoom-button ${zoomButtonShouldHide}`}
+          title={zoomButtonTitle}
+          icon={this.state.zoomState}
+          onClick={this.handleZoomClick}
+          size={40}
+          style={{
+            fontSize: '30px', /* Fontawesome's fa-compress fa-expand is larger than fa-close */
+          }}
+        />
+        <div
+          className='zoomable-image'
+          ref={this.setContainerRef}
+          style={{ overflow }}
+        >
+          <img
+            role='presentation'
+            ref={this.setImageRef}
+            alt={alt}
+            title={alt}
+            lang={lang}
+            src={src}
+            width={width}
+            height={height}
+            style={{
+              transform: `scale(${scale}) translate(-${lockTranslate.x}px, -${lockTranslate.y}px)`,
+              transformOrigin: '0 0',
+            }}
+            draggable={false}
+            onClick={this.handleClick}
+            onMouseDown={this.handleMouseDown}
+          />
+        </div>
+      </>
+    );
+  }
+
+}
+
+export default injectIntl(ZoomableImage);
diff --git a/app/javascript/flavours/blobfox/features/ui/containers/bundle_container.js b/app/javascript/flavours/blobfox/features/ui/containers/bundle_container.js
new file mode 100644
index 00000000000000..6a476fe2483cf3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/containers/bundle_container.js
@@ -0,0 +1,18 @@
+import { connect } from 'react-redux';
+
+import { fetchBundleRequest, fetchBundleSuccess, fetchBundleFail } from '../../../actions/bundles';
+import Bundle from '../components/bundle';
+
+const mapDispatchToProps = dispatch => ({
+  onFetch () {
+    dispatch(fetchBundleRequest());
+  },
+  onFetchSuccess () {
+    dispatch(fetchBundleSuccess());
+  },
+  onFetchFail (error) {
+    dispatch(fetchBundleFail(error));
+  },
+});
+
+export default connect(null, mapDispatchToProps)(Bundle);
diff --git a/app/javascript/flavours/blobfox/features/ui/containers/columns_area_container.js b/app/javascript/flavours/blobfox/features/ui/containers/columns_area_container.js
new file mode 100644
index 00000000000000..76c9a3e28cc13b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/containers/columns_area_container.js
@@ -0,0 +1,22 @@
+import { connect } from 'react-redux';
+
+import { openModal } from 'flavours/blobfox/actions/modal';
+
+import ColumnsArea from '../components/columns_area';
+
+const mapStateToProps = state => ({
+  columns: state.getIn(['settings', 'columns']),
+});
+
+const mapDispatchToProps = dispatch => ({
+  openSettings (e) {
+    e.preventDefault();
+    e.stopPropagation();
+    dispatch(openModal({
+      modalType: 'SETTINGS',
+      modalProps: {},
+    }));
+  },
+});
+
+export default connect(mapStateToProps, mapDispatchToProps, null, { forwardRef: true })(ColumnsArea);
diff --git a/app/javascript/flavours/blobfox/features/ui/containers/loading_bar_container.js b/app/javascript/flavours/blobfox/features/ui/containers/loading_bar_container.js
new file mode 100644
index 00000000000000..7efdac55ee393f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/containers/loading_bar_container.js
@@ -0,0 +1,9 @@
+import { connect }    from 'react-redux';
+
+import LoadingBar from 'react-redux-loading-bar';
+
+const mapStateToProps = (state, ownProps) => ({
+  loading: state.get('loadingBar')[ownProps.scope || 'default'],
+});
+
+export default connect(mapStateToProps)(LoadingBar.WrappedComponent);
diff --git a/app/javascript/flavours/blobfox/features/ui/containers/modal_container.js b/app/javascript/flavours/blobfox/features/ui/containers/modal_container.js
new file mode 100644
index 00000000000000..1c3872cd50436f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/containers/modal_container.js
@@ -0,0 +1,36 @@
+import { connect } from 'react-redux';
+
+import { openModal, closeModal } from '../../../actions/modal';
+import ModalRoot from '../components/modal_root';
+
+const mapStateToProps = state => ({
+  ignoreFocus: state.getIn(['modal', 'ignoreFocus']),
+  type: state.getIn(['modal', 'stack', 0, 'modalType'], null),
+  props: state.getIn(['modal', 'stack', 0, 'modalProps'], {}),
+});
+
+const mapDispatchToProps = dispatch => ({
+  onClose (confirmationMessage, ignoreFocus = false) {
+    if (confirmationMessage) {
+      dispatch(
+        openModal({
+          modalType: 'CONFIRM',
+          modalProps: {
+            message: confirmationMessage.message,
+            confirm: confirmationMessage.confirm,
+            onConfirm: () => dispatch(closeModal({
+              modalType: undefined,
+              ignoreFocus: { ignoreFocus },
+            })),
+          } }),
+      );
+    } else {
+      dispatch(closeModal({
+        modalType: undefined,
+        ignoreFocus: { ignoreFocus },
+      }));
+    }
+  },
+});
+
+export default connect(mapStateToProps, mapDispatchToProps)(ModalRoot);
diff --git a/app/javascript/flavours/blobfox/features/ui/containers/notifications_container.js b/app/javascript/flavours/blobfox/features/ui/containers/notifications_container.js
new file mode 100644
index 00000000000000..3d60cfdad1b24a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/containers/notifications_container.js
@@ -0,0 +1,33 @@
+import { injectIntl } from 'react-intl';
+
+import { connect } from 'react-redux';
+
+import { NotificationStack } from 'react-notification';
+
+import { dismissAlert } from '../../../actions/alerts';
+import { getAlerts } from '../../../selectors';
+
+const formatIfNeeded = (intl, message, values) => {
+  if (typeof message === 'object') {
+    return intl.formatMessage(message, values);
+  }
+
+  return message;
+};
+
+const mapStateToProps = (state, { intl }) => ({
+  notifications: getAlerts(state).map(alert => ({
+    ...alert,
+    action: formatIfNeeded(intl, alert.action, alert.values),
+    title: formatIfNeeded(intl, alert.title, alert.values),
+    message: formatIfNeeded(intl, alert.message, alert.values),
+  })),
+});
+
+const mapDispatchToProps = (dispatch) => ({
+  onDismiss (alert) {
+    dispatch(dismissAlert(alert));
+  },
+});
+
+export default injectIntl(connect(mapStateToProps, mapDispatchToProps)(NotificationStack));
diff --git a/app/javascript/flavours/blobfox/features/ui/containers/status_list_container.js b/app/javascript/flavours/blobfox/features/ui/containers/status_list_container.js
new file mode 100644
index 00000000000000..f34d099b24015f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/containers/status_list_container.js
@@ -0,0 +1,89 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+import { connect } from 'react-redux';
+import { createSelector } from 'reselect';
+
+import { debounce } from 'lodash';
+
+import { scrollTopTimeline, loadPending } from '../../../actions/timelines';
+import StatusList from '../../../components/status_list';
+import { me } from '../../../initial_state';
+
+const getRegex = createSelector([
+  (state, { regex }) => regex,
+], (rawRegex) => {
+  let regex = null;
+
+  try {
+    regex = rawRegex && new RegExp(rawRegex.trim(), 'i');
+  } catch (e) {
+    // Bad regex, don't affect filters
+  }
+  return regex;
+});
+
+const makeGetStatusIds = (pending = false) => createSelector([
+  (state, { type }) => state.getIn(['settings', type], ImmutableMap()),
+  (state, { type }) => state.getIn(['timelines', type, pending ? 'pendingItems' : 'items'], ImmutableList()),
+  (state)           => state.get('statuses'),
+  getRegex,
+], (columnSettings, statusIds, statuses, regex) => {
+  return statusIds.filter(id => {
+    if (id === null) return true;
+
+    const statusForId = statuses.get(id);
+    let showStatus    = true;
+
+    if (statusForId.get('account') === me) return true;
+
+    if (columnSettings.getIn(['shows', 'reblog']) === false) {
+      showStatus = showStatus && statusForId.get('reblog') === null;
+    }
+
+    if (columnSettings.getIn(['shows', 'reply']) === false) {
+      showStatus = showStatus && (statusForId.get('in_reply_to_id') === null || statusForId.get('in_reply_to_account_id') === me);
+    }
+
+    if (columnSettings.getIn(['shows', 'direct']) === false) {
+      showStatus = showStatus && statusForId.get('visibility') !== 'direct';
+    }
+
+    if (showStatus && regex) {
+      const searchIndex = statusForId.get('reblog') ? statuses.getIn([statusForId.get('reblog'), 'search_index']) : statusForId.get('search_index');
+      showStatus = !regex.test(searchIndex);
+    }
+
+    return showStatus;
+  });
+});
+
+const makeMapStateToProps = () => {
+  const getStatusIds = makeGetStatusIds();
+  const getPendingStatusIds = makeGetStatusIds(true);
+
+  const mapStateToProps = (state, { timelineId, regex }) => ({
+    statusIds: getStatusIds(state, { type: timelineId, regex }),
+    lastId:    state.getIn(['timelines', timelineId, 'items'])?.last(),
+    isLoading: state.getIn(['timelines', timelineId, 'isLoading'], true),
+    isPartial: state.getIn(['timelines', timelineId, 'isPartial'], false),
+    hasMore:   state.getIn(['timelines', timelineId, 'hasMore']),
+    numPending: getPendingStatusIds(state, { type: timelineId }).size,
+  });
+
+  return mapStateToProps;
+};
+
+const mapDispatchToProps = (dispatch, { timelineId }) => ({
+
+  onScrollToTop: debounce(() => {
+    dispatch(scrollTopTimeline(timelineId, true));
+  }, 100),
+
+  onScroll: debounce(() => {
+    dispatch(scrollTopTimeline(timelineId, false));
+  }, 100),
+
+  onLoadPending: () => dispatch(loadPending(timelineId)),
+
+});
+
+export default connect(makeMapStateToProps, mapDispatchToProps)(StatusList);
diff --git a/app/javascript/flavours/blobfox/features/ui/index.jsx b/app/javascript/flavours/blobfox/features/ui/index.jsx
new file mode 100644
index 00000000000000..dc27f86d4251e8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/index.jsx
@@ -0,0 +1,675 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, FormattedMessage, injectIntl } from 'react-intl';
+
+import classNames from 'classnames';
+import { Redirect, Route, withRouter } from 'react-router-dom';
+
+import { connect } from 'react-redux';
+
+import Favico from 'favico.js';
+import { debounce } from 'lodash';
+import { HotKeys } from 'react-hotkeys';
+
+import { changeLayout } from 'flavours/blobfox/actions/app';
+import { synchronouslySubmitMarkers, submitMarkers, fetchMarkers } from 'flavours/blobfox/actions/markers';
+import { INTRODUCTION_VERSION } from 'flavours/blobfox/actions/onboarding';
+import PermaLink from 'flavours/blobfox/components/permalink';
+import PictureInPicture from 'flavours/blobfox/features/picture_in_picture';
+import { layoutFromWindow } from 'flavours/blobfox/is_mobile';
+import { WithRouterPropTypes } from 'flavours/blobfox/utils/react_router';
+
+import { uploadCompose, resetCompose, changeComposeSpoilerness } from '../../actions/compose';
+import { clearHeight } from '../../actions/height_cache';
+import { expandNotifications, notificationsSetVisibility } from '../../actions/notifications';
+import { fetchServer, fetchServerTranslationLanguages } from '../../actions/server';
+import { expandHomeTimeline } from '../../actions/timelines';
+import initialState, { me, owner, singleUserMode, trendsEnabled, trendsAsLanding } from '../../initial_state';
+
+import BundleColumnError from './components/bundle_column_error';
+import Header from './components/header';
+import UploadArea from './components/upload_area';
+import ColumnsAreaContainer from './containers/columns_area_container';
+import LoadingBarContainer from './containers/loading_bar_container';
+import ModalContainer from './containers/modal_container';
+import NotificationsContainer from './containers/notifications_container';
+import {
+  Compose,
+  Status,
+  GettingStarted,
+  KeyboardShortcuts,
+  Firehose,
+  AccountTimeline,
+  AccountGallery,
+  HomeTimeline,
+  Followers,
+  Following,
+  Reblogs,
+  Favourites,
+  DirectTimeline,
+  HashtagTimeline,
+  Notifications,
+  FollowRequests,
+  FavouritedStatuses,
+  BookmarkedStatuses,
+  FollowedTags,
+  ListTimeline,
+  Blocks,
+  DomainBlocks,
+  Mutes,
+  PinnedStatuses,
+  Lists,
+  GettingStartedMisc,
+  Directory,
+  Explore,
+  Onboarding,
+  About,
+  PrivacyPolicy,
+} from './util/async-components';
+import { WrappedSwitch, WrappedRoute } from './util/react_router_helpers';
+
+// Dummy import, to make sure that <Status /> ends up in the application bundle.
+// Without this it ends up in ~8 very commonly used bundles.
+import '../../components/status';
+
+const messages = defineMessages({
+  beforeUnload: { id: 'ui.beforeunload', defaultMessage: 'Your draft will be lost if you leave Mastodon.' },
+});
+
+const mapStateToProps = state => ({
+  layout: state.getIn(['meta', 'layout']),
+  hasComposingText: state.getIn(['compose', 'text']).trim().length !== 0,
+  hasMediaAttachments: state.getIn(['compose', 'media_attachments']).size > 0,
+  canUploadMore: !state.getIn(['compose', 'media_attachments']).some(x => ['audio', 'video'].includes(x.get('type'))) && state.getIn(['compose', 'media_attachments']).size < 4,
+  isWide: state.getIn(['local_settings', 'stretch']),
+  dropdownMenuIsOpen: state.dropdownMenu.openId !== null,
+  unreadNotifications: state.getIn(['notifications', 'unread']),
+  showFaviconBadge: state.getIn(['local_settings', 'notifications', 'favicon_badge']),
+  hicolorPrivacyIcons: state.getIn(['local_settings', 'hicolor_privacy_icons']),
+  moved: state.getIn(['accounts', me, 'moved']) && state.getIn(['accounts', state.getIn(['accounts', me, 'moved'])]),
+  firstLaunch: state.getIn(['settings', 'introductionVersion'], 0) < INTRODUCTION_VERSION,
+  username: state.getIn(['accounts', me, 'username']),
+});
+
+const keyMap = {
+  help: '?',
+  new: 'n',
+  search: 's',
+  forceNew: 'option+n',
+  toggleComposeSpoilers: 'option+x',
+  focusColumn: ['1', '2', '3', '4', '5', '6', '7', '8', '9'],
+  reply: 'r',
+  favourite: 'f',
+  boost: 'b',
+  mention: 'm',
+  open: ['enter', 'o'],
+  openProfile: 'p',
+  moveDown: ['down', 'j'],
+  moveUp: ['up', 'k'],
+  back: 'backspace',
+  goToHome: 'g h',
+  goToNotifications: 'g n',
+  goToLocal: 'g l',
+  goToFederated: 'g t',
+  goToDirect: 'g d',
+  goToStart: 'g s',
+  goToFavourites: 'g f',
+  goToPinned: 'g p',
+  goToProfile: 'g u',
+  goToBlocked: 'g b',
+  goToMuted: 'g m',
+  goToRequests: 'g r',
+  toggleHidden: 'x',
+  bookmark: 'd',
+  toggleCollapse: 'shift+x',
+  toggleSensitive: 'h',
+  openMedia: 'e',
+};
+
+class SwitchingColumnsArea extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object,
+  };
+
+  static propTypes = {
+    children: PropTypes.node,
+    location: PropTypes.object,
+    singleColumn: PropTypes.bool,
+  };
+
+  UNSAFE_componentWillMount () {
+    if (this.props.singleColumn) {
+      document.body.classList.toggle('layout-single-column', true);
+      document.body.classList.toggle('layout-multiple-columns', false);
+    } else {
+      document.body.classList.toggle('layout-single-column', false);
+      document.body.classList.toggle('layout-multiple-columns', true);
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    if (![this.props.location.pathname, '/'].includes(prevProps.location.pathname)) {
+      this.node.handleChildrenContentChange();
+    }
+
+    if (prevProps.singleColumn !== this.props.singleColumn) {
+      document.body.classList.toggle('layout-single-column', this.props.singleColumn);
+      document.body.classList.toggle('layout-multiple-columns', !this.props.singleColumn);
+    }
+  }
+
+  setRef = c => {
+    if (c) {
+      this.node = c;
+    }
+  };
+
+  render () {
+    const { children, singleColumn } = this.props;
+    const { signedIn } = this.context.identity;
+    const pathName = this.props.location.pathname;
+
+    let redirect;
+
+    if (signedIn) {
+      if (singleColumn) {
+        redirect = <Redirect from='/' to='/home' exact />;
+      } else {
+        redirect = <Redirect from='/' to='/deck/getting-started' exact />;
+      }
+    } else if (singleUserMode && owner && initialState?.accounts[owner]) {
+      redirect = <Redirect from='/' to={`/@${initialState.accounts[owner].username}`} exact />;
+    } else if (trendsEnabled && trendsAsLanding) {
+      redirect = <Redirect from='/' to='/explore' exact />;
+    } else {
+      redirect = <Redirect from='/' to='/about' exact />;
+    }
+
+    return (
+      <ColumnsAreaContainer ref={this.setRef} singleColumn={singleColumn}>
+        <WrappedSwitch>
+          {redirect}
+
+          {singleColumn ? <Redirect from='/deck' to='/home' exact /> : null}
+          {singleColumn && pathName.startsWith('/deck/') ? <Redirect from={pathName} to={pathName.slice(5)} /> : null}
+          {/* Redirect old bookmarks (without /deck) with home-like routes to the advanced interface */}
+          {!singleColumn && pathName === '/getting-started' ? <Redirect from='/getting-started' to='/deck/getting-started' exact /> : null}
+          {!singleColumn && pathName === '/home' ? <Redirect from='/home' to='/deck/getting-started' exact /> : null}
+
+          <WrappedRoute path='/getting-started' component={GettingStarted} content={children} />
+          <WrappedRoute path='/keyboard-shortcuts' component={KeyboardShortcuts} content={children} />
+          <WrappedRoute path='/about' component={About} content={children} />
+          <WrappedRoute path='/privacy-policy' component={PrivacyPolicy} content={children} />
+
+          <WrappedRoute path={['/home', '/timelines/home']} component={HomeTimeline} content={children} />
+          <Redirect from='/timelines/public' to='/public' exact />
+          <Redirect from='/timelines/public/local' to='/public/local' exact />
+          <WrappedRoute path='/public' exact component={Firehose} componentParams={{ feedType: 'public' }} content={children} />
+          <WrappedRoute path='/public/local' exact component={Firehose} componentParams={{ feedType: 'community' }} content={children} />
+          <WrappedRoute path='/public/remote' exact component={Firehose} componentParams={{ feedType: 'public:remote' }} content={children} />
+          <WrappedRoute path={['/conversations', '/timelines/direct']} component={DirectTimeline} content={children} />
+          <WrappedRoute path='/tags/:id' component={HashtagTimeline} content={children} />
+          <WrappedRoute path='/lists/:id' component={ListTimeline} content={children} />
+          <WrappedRoute path='/notifications' component={Notifications} content={children} />
+          <WrappedRoute path='/favourites' component={FavouritedStatuses} content={children} />
+
+          <WrappedRoute path='/bookmarks' component={BookmarkedStatuses} content={children} />
+          <WrappedRoute path='/pinned' component={PinnedStatuses} content={children} />
+
+          <WrappedRoute path='/start' exact component={Onboarding} content={children} />
+          <WrappedRoute path='/directory' component={Directory} content={children} />
+          <WrappedRoute path={['/explore', '/search']} component={Explore} content={children} />
+          <WrappedRoute path={['/publish', '/statuses/new']} component={Compose} content={children} />
+
+          <WrappedRoute path={['/@:acct', '/accounts/:id']} exact component={AccountTimeline} content={children} />
+          <WrappedRoute path='/@:acct/tagged/:tagged?' exact component={AccountTimeline} content={children} />
+          <WrappedRoute path={['/@:acct/with_replies', '/accounts/:id/with_replies']} component={AccountTimeline} content={children} componentParams={{ withReplies: true }} />
+          <WrappedRoute path={['/accounts/:id/followers', '/users/:acct/followers', '/@:acct/followers']} component={Followers} content={children} />
+          <WrappedRoute path={['/accounts/:id/following', '/users/:acct/following', '/@:acct/following']} component={Following} content={children} />
+          <WrappedRoute path={['/@:acct/media', '/accounts/:id/media']} component={AccountGallery} content={children} />
+          <WrappedRoute path='/@:acct/:statusId' exact component={Status} content={children} />
+          <WrappedRoute path='/@:acct/:statusId/reblogs' component={Reblogs} content={children} />
+          <WrappedRoute path='/@:acct/:statusId/favourites' component={Favourites} content={children} />
+
+          {/* Legacy routes, cannot be easily factored with other routes because they share a param name */}
+          <WrappedRoute path='/timelines/tag/:id' component={HashtagTimeline} content={children} />
+          <WrappedRoute path='/timelines/list/:id' component={ListTimeline} content={children} />
+          <WrappedRoute path='/statuses/:statusId' exact component={Status} content={children} />
+          <WrappedRoute path='/statuses/:statusId/reblogs' component={Reblogs} content={children} />
+          <WrappedRoute path='/statuses/:statusId/favourites' component={Favourites} content={children} />
+
+          <WrappedRoute path='/follow_requests' component={FollowRequests} content={children} />
+          <WrappedRoute path='/blocks' component={Blocks} content={children} />
+          <WrappedRoute path='/domain_blocks' component={DomainBlocks} content={children} />
+          <WrappedRoute path='/followed_tags' component={FollowedTags} content={children} />
+          <WrappedRoute path='/mutes' component={Mutes} content={children} />
+          <WrappedRoute path='/lists' component={Lists} content={children} />
+          <WrappedRoute path='/getting-started-misc' component={GettingStartedMisc} content={children} />
+
+          <Route component={BundleColumnError} />
+        </WrappedSwitch>
+      </ColumnsAreaContainer>
+    );
+  }
+
+}
+
+class UI extends PureComponent {
+
+  static contextTypes = {
+    identity: PropTypes.object.isRequired,
+  };
+
+  static propTypes = {
+    dispatch: PropTypes.func.isRequired,
+    children: PropTypes.node,
+    isWide: PropTypes.bool,
+    systemFontUi: PropTypes.bool,
+    isComposing: PropTypes.bool,
+    hasComposingText: PropTypes.bool,
+    hasMediaAttachments: PropTypes.bool,
+    canUploadMore: PropTypes.bool,
+    intl: PropTypes.object.isRequired,
+    dropdownMenuIsOpen: PropTypes.bool,
+    unreadNotifications: PropTypes.number,
+    showFaviconBadge: PropTypes.bool,
+    hicolorPrivacyIcons: PropTypes.bool,
+    moved: PropTypes.map,
+    layout: PropTypes.string.isRequired,
+    firstLaunch: PropTypes.bool,
+    username: PropTypes.string,
+    ...WithRouterPropTypes,
+  };
+
+  state = {
+    draggingOver: false,
+  };
+
+  handleBeforeUnload = e => {
+    const { intl, dispatch, hasComposingText, hasMediaAttachments } = this.props;
+
+    dispatch(synchronouslySubmitMarkers());
+
+    if (hasComposingText || hasMediaAttachments) {
+      // Setting returnValue to any string causes confirmation dialog.
+      // Many browsers no longer display this text to users,
+      // but we set user-friendly message for other browsers, e.g. Edge.
+      e.returnValue = intl.formatMessage(messages.beforeUnload);
+    }
+  };
+
+  handleVisibilityChange = () => {
+    const visibility = !document[this.visibilityHiddenProp];
+    this.props.dispatch(notificationsSetVisibility(visibility));
+    if (visibility) {
+      this.props.dispatch(submitMarkers({ immediate: true }));
+    }
+  };
+
+  handleDragEnter = (e) => {
+    e.preventDefault();
+
+    if (!this.dragTargets) {
+      this.dragTargets = [];
+    }
+
+    if (this.dragTargets.indexOf(e.target) === -1) {
+      this.dragTargets.push(e.target);
+    }
+
+    if (e.dataTransfer && Array.from(e.dataTransfer.types).includes('Files') && this.props.canUploadMore && this.context.identity.signedIn) {
+      this.setState({ draggingOver: true });
+    }
+  };
+
+  handleDragOver = (e) => {
+    if (this.dataTransferIsText(e.dataTransfer)) return false;
+
+    e.preventDefault();
+    e.stopPropagation();
+
+    try {
+      e.dataTransfer.dropEffect = 'copy';
+    } catch (err) {
+
+    }
+
+    return false;
+  };
+
+  handleDrop = (e) => {
+    if (this.dataTransferIsText(e.dataTransfer)) return;
+
+    e.preventDefault();
+
+    this.setState({ draggingOver: false });
+    this.dragTargets = [];
+
+    if (e.dataTransfer && e.dataTransfer.files.length >= 1 && this.props.canUploadMore && this.context.identity.signedIn) {
+      this.props.dispatch(uploadCompose(e.dataTransfer.files));
+    }
+  };
+
+  handleDragLeave = (e) => {
+    e.preventDefault();
+    e.stopPropagation();
+
+    this.dragTargets = this.dragTargets.filter(el => el !== e.target && this.node.contains(el));
+
+    if (this.dragTargets.length > 0) {
+      return;
+    }
+
+    this.setState({ draggingOver: false });
+  };
+
+  dataTransferIsText = (dataTransfer) => {
+    return (dataTransfer && Array.from(dataTransfer.types).filter((type) => type === 'text/plain').length === 1);
+  };
+
+  closeUploadModal = () => {
+    this.setState({ draggingOver: false });
+  };
+
+  handleServiceWorkerPostMessage = ({ data }) => {
+    if (data.type === 'navigate') {
+      this.props.history.push(data.path);
+    } else {
+      console.warn('Unknown message type:', data.type);
+    }
+  };
+
+  handleLayoutChange = debounce(() => {
+    this.props.dispatch(clearHeight()); // The cached heights are no longer accurate, invalidate
+  }, 500, {
+    trailing: true,
+  });
+
+  handleResize = () => {
+    const layout = layoutFromWindow();
+
+    if (layout !== this.props.layout) {
+      this.handleLayoutChange.cancel();
+      this.props.dispatch(changeLayout({ layout }));
+    } else {
+      this.handleLayoutChange();
+    }
+  };
+
+  componentDidMount () {
+    const { signedIn } = this.context.identity;
+
+    window.addEventListener('beforeunload', this.handleBeforeUnload, false);
+    window.addEventListener('resize', this.handleResize, { passive: true });
+
+    document.addEventListener('dragenter', this.handleDragEnter, false);
+    document.addEventListener('dragover', this.handleDragOver, false);
+    document.addEventListener('drop', this.handleDrop, false);
+    document.addEventListener('dragleave', this.handleDragLeave, false);
+    document.addEventListener('dragend', this.handleDragEnd, false);
+
+    if ('serviceWorker' in  navigator) {
+      navigator.serviceWorker.addEventListener('message', this.handleServiceWorkerPostMessage);
+    }
+
+    this.favicon = new Favico({ animation:'none' });
+
+    if (signedIn) {
+      this.props.dispatch(fetchMarkers());
+      this.props.dispatch(expandHomeTimeline());
+      this.props.dispatch(expandNotifications());
+      this.props.dispatch(fetchServerTranslationLanguages());
+
+      setTimeout(() => this.props.dispatch(fetchServer()), 3000);
+    }
+
+    this.hotkeys.__mousetrap__.stopCallback = (e, element) => {
+      return ['TEXTAREA', 'SELECT', 'INPUT'].includes(element.tagName);
+    };
+
+    if (typeof document.hidden !== 'undefined') { // Opera 12.10 and Firefox 18 and later support
+      this.visibilityHiddenProp = 'hidden';
+      this.visibilityChange = 'visibilitychange';
+    } else if (typeof document.msHidden !== 'undefined') {
+      this.visibilityHiddenProp = 'msHidden';
+      this.visibilityChange = 'msvisibilitychange';
+    } else if (typeof document.webkitHidden !== 'undefined') {
+      this.visibilityHiddenProp = 'webkitHidden';
+      this.visibilityChange = 'webkitvisibilitychange';
+    }
+
+    if (this.visibilityChange !== undefined) {
+      document.addEventListener(this.visibilityChange, this.handleVisibilityChange, false);
+      this.handleVisibilityChange();
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    if (this.props.unreadNotifications !== prevProps.unreadNotifications ||
+        this.props.showFaviconBadge !== prevProps.showFaviconBadge) {
+      if (this.favicon) {
+        try {
+          this.favicon.badge(this.props.showFaviconBadge ? this.props.unreadNotifications : 0);
+        } catch (err) {
+          console.error(err);
+        }
+      }
+    }
+  }
+
+  componentWillUnmount () {
+    if (this.visibilityChange !== undefined) {
+      document.removeEventListener(this.visibilityChange, this.handleVisibilityChange);
+    }
+
+    window.removeEventListener('beforeunload', this.handleBeforeUnload);
+    window.removeEventListener('resize', this.handleResize);
+
+    document.removeEventListener('dragenter', this.handleDragEnter);
+    document.removeEventListener('dragover', this.handleDragOver);
+    document.removeEventListener('drop', this.handleDrop);
+    document.removeEventListener('dragleave', this.handleDragLeave);
+    document.removeEventListener('dragend', this.handleDragEnd);
+  }
+
+  setRef = c => {
+    this.node = c;
+  };
+
+  handleHotkeyNew = e => {
+    e.preventDefault();
+
+    const element = this.node.querySelector('.compose-form__autosuggest-wrapper textarea');
+
+    if (element) {
+      element.focus();
+    }
+  };
+
+  handleHotkeySearch = e => {
+    e.preventDefault();
+
+    const element = this.node.querySelector('.search__input');
+
+    if (element) {
+      element.focus();
+    }
+  };
+
+  handleHotkeyForceNew = e => {
+    this.handleHotkeyNew(e);
+    this.props.dispatch(resetCompose());
+  };
+
+  handleHotkeyToggleComposeSpoilers = e => {
+    e.preventDefault();
+    this.props.dispatch(changeComposeSpoilerness());
+  };
+
+  handleHotkeyFocusColumn = e => {
+    const index  = (e.key * 1) + 1; // First child is drawer, skip that
+    const column = this.node.querySelector(`.column:nth-child(${index})`);
+    if (!column) return;
+    const container = column.querySelector('.scrollable');
+
+    if (container) {
+      const status = container.querySelector('.focusable');
+
+      if (status) {
+        if (container.scrollTop > status.offsetTop) {
+          status.scrollIntoView(true);
+        }
+        status.focus();
+      }
+    }
+  };
+
+  handleHotkeyBack = () => {
+    const { history } = this.props;
+
+    if (history.location?.state?.fromMastodon) {
+      history.goBack();
+    } else {
+      history.push('/');
+    }
+  };
+
+  setHotkeysRef = c => {
+    this.hotkeys = c;
+  };
+
+  handleHotkeyToggleHelp = () => {
+    if (this.props.location.pathname === '/keyboard-shortcuts') {
+      this.props.history.goBack();
+    } else {
+      this.props.history.push('/keyboard-shortcuts');
+    }
+  };
+
+  handleHotkeyGoToHome = () => {
+    this.props.history.push('/home');
+  };
+
+  handleHotkeyGoToNotifications = () => {
+    this.props.history.push('/notifications');
+  };
+
+  handleHotkeyGoToLocal = () => {
+    this.props.history.push('/public/local');
+  };
+
+  handleHotkeyGoToFederated = () => {
+    this.props.history.push('/public');
+  };
+
+  handleHotkeyGoToDirect = () => {
+    this.props.history.push('/conversations');
+  };
+
+  handleHotkeyGoToStart = () => {
+    this.props.history.push('/getting-started');
+  };
+
+  handleHotkeyGoToFavourites = () => {
+    this.props.history.push('/favourites');
+  };
+
+  handleHotkeyGoToPinned = () => {
+    this.props.history.push('/pinned');
+  };
+
+  handleHotkeyGoToProfile = () => {
+    this.props.history.push(`/@${this.props.username}`);
+  };
+
+  handleHotkeyGoToBlocked = () => {
+    this.props.history.push('/blocks');
+  };
+
+  handleHotkeyGoToMuted = () => {
+    this.props.history.push('/mutes');
+  };
+
+  handleHotkeyGoToRequests = () => {
+    this.props.history.push('/follow_requests');
+  };
+
+  render () {
+    const { draggingOver } = this.state;
+    const { children, isWide, location, dropdownMenuIsOpen, layout, moved } = this.props;
+
+    const columnsClass = layout => {
+      switch (layout) {
+      case 'single':
+        return 'single-column';
+      case 'multiple':
+        return 'multi-columns';
+      default:
+        return 'auto-columns';
+      }
+    };
+
+    const className = classNames('ui', columnsClass(layout), {
+      'wide': isWide,
+      'system-font': this.props.systemFontUi,
+      'hicolor-privacy-icons': this.props.hicolorPrivacyIcons,
+    });
+
+    const handlers = {
+      help: this.handleHotkeyToggleHelp,
+      new: this.handleHotkeyNew,
+      search: this.handleHotkeySearch,
+      forceNew: this.handleHotkeyForceNew,
+      toggleComposeSpoilers: this.handleHotkeyToggleComposeSpoilers,
+      focusColumn: this.handleHotkeyFocusColumn,
+      back: this.handleHotkeyBack,
+      goToHome: this.handleHotkeyGoToHome,
+      goToNotifications: this.handleHotkeyGoToNotifications,
+      goToLocal: this.handleHotkeyGoToLocal,
+      goToFederated: this.handleHotkeyGoToFederated,
+      goToDirect: this.handleHotkeyGoToDirect,
+      goToStart: this.handleHotkeyGoToStart,
+      goToFavourites: this.handleHotkeyGoToFavourites,
+      goToPinned: this.handleHotkeyGoToPinned,
+      goToProfile: this.handleHotkeyGoToProfile,
+      goToBlocked: this.handleHotkeyGoToBlocked,
+      goToMuted: this.handleHotkeyGoToMuted,
+      goToRequests: this.handleHotkeyGoToRequests,
+    };
+
+    return (
+      <HotKeys keyMap={keyMap} handlers={handlers} ref={this.setHotkeysRef} attach={window} focused>
+        <div className={className} ref={this.setRef} style={{ pointerEvents: dropdownMenuIsOpen ? 'none' : null }}>
+          {moved && (<div className='flash-message alert'>
+            <FormattedMessage
+              id='moved_to_warning'
+              defaultMessage='This account is marked as moved to {moved_to_link}, and may thus not accept new follows.'
+              values={{ moved_to_link: (
+                <PermaLink href={moved.get('url')} to={`/@${moved.get('acct')}`}>
+                  @{moved.get('acct')}
+                </PermaLink>
+              ) }}
+            />
+          </div>)}
+
+          <Header />
+
+          <SwitchingColumnsArea location={location} singleColumn={layout === 'mobile' || layout === 'single-column'}>
+            {children}
+          </SwitchingColumnsArea>
+
+          {layout !== 'mobile' && <PictureInPicture />}
+          <NotificationsContainer />
+          <LoadingBarContainer className='loading-bar' />
+          <ModalContainer />
+          <UploadArea active={draggingOver} onClose={this.closeUploadModal} />
+        </div>
+      </HotKeys>
+    );
+  }
+
+}
+
+export default connect(mapStateToProps)(injectIntl(withRouter(UI)));
diff --git a/app/javascript/flavours/blobfox/features/ui/util/async-components.js b/app/javascript/flavours/blobfox/features/ui/util/async-components.js
new file mode 100644
index 00000000000000..c2b4042e5a4ed4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/async-components.js
@@ -0,0 +1,203 @@
+export function EmojiPicker () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/emoji_picker" */'flavours/blobfox/features/emoji/emoji_picker');
+}
+
+export function Compose () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/compose" */'flavours/blobfox/features/compose');
+}
+
+export function Notifications () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/notifications" */'flavours/blobfox/features/notifications');
+}
+
+export function HomeTimeline () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/home_timeline" */'flavours/blobfox/features/home_timeline');
+}
+
+export function PublicTimeline () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/public_timeline" */'flavours/blobfox/features/public_timeline');
+}
+
+export function CommunityTimeline () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/community_timeline" */'flavours/blobfox/features/community_timeline');
+}
+
+export function Firehose () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/firehose" */'../../firehose');
+}
+
+export function HashtagTimeline () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/hashtag_timeline" */'flavours/blobfox/features/hashtag_timeline');
+}
+
+export function ListTimeline () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/list_timeline" */'flavours/blobfox/features/list_timeline');
+}
+
+export function Lists () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/lists" */'flavours/blobfox/features/lists');
+}
+
+export function ListEditor () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/list_editor" */'flavours/blobfox/features/list_editor');
+}
+
+export function PinnedAccountsEditor () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/pinned_accounts_editor" */'flavours/blobfox/features/pinned_accounts_editor');
+}
+
+export function DirectTimeline() {
+  return import(/* webpackChunkName: "flavours/blobfox/async/direct_timeline" */'flavours/blobfox/features/direct_timeline');
+}
+
+export function Status () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/status" */'flavours/blobfox/features/status');
+}
+
+export function GettingStarted () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/getting_started" */'flavours/blobfox/features/getting_started');
+}
+
+export function KeyboardShortcuts () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/keyboard_shortcuts" */'flavours/blobfox/features/keyboard_shortcuts');
+}
+
+export function PinnedStatuses () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/pinned_statuses" */'flavours/blobfox/features/pinned_statuses');
+}
+
+export function AccountTimeline () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/account_timeline" */'flavours/blobfox/features/account_timeline');
+}
+
+export function AccountGallery () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/account_gallery" */'flavours/blobfox/features/account_gallery');
+}
+
+export function Followers () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/followers" */'flavours/blobfox/features/followers');
+}
+
+export function Following () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/following" */'flavours/blobfox/features/following');
+}
+
+export function Reblogs () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/reblogs" */'flavours/blobfox/features/reblogs');
+}
+
+export function Favourites () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/favourites" */'flavours/blobfox/features/favourites');
+}
+
+export function FollowRequests () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/follow_requests" */'flavours/blobfox/features/follow_requests');
+}
+
+export function FavouritedStatuses () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/favourited_statuses" */'flavours/blobfox/features/favourited_statuses');
+}
+
+export function FollowedTags () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/followed_tags" */'flavours/blobfox/features/followed_tags');
+}
+
+export function BookmarkedStatuses () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/bookmarked_statuses" */'flavours/blobfox/features/bookmarked_statuses');
+}
+
+export function Blocks () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/blocks" */'flavours/blobfox/features/blocks');
+}
+
+export function DomainBlocks () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/domain_blocks" */'flavours/blobfox/features/domain_blocks');
+}
+
+export function Mutes () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/mutes" */'flavours/blobfox/features/mutes');
+}
+
+export function MuteModal () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/mute_modal" */'flavours/blobfox/features/ui/components/mute_modal');
+}
+
+export function BlockModal () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/block_modal" */'flavours/blobfox/features/ui/components/block_modal');
+}
+
+export function ReportModal () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/report_modal" */'flavours/blobfox/features/ui/components/report_modal');
+}
+
+export function SettingsModal () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/settings_modal" */'flavours/blobfox/features/local_settings');
+}
+
+export function MediaGallery () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/media_gallery" */'flavours/blobfox/components/media_gallery');
+}
+
+export function Video () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/video" */'flavours/blobfox/features/video');
+}
+
+export function Audio () {
+  return import(/* webpackChunkName: "features/blobfox/async/audio" */'flavours/blobfox/features/audio');
+}
+
+export function EmbedModal () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/embed_modal" */'flavours/blobfox/features/ui/components/embed_modal');
+}
+
+export function GettingStartedMisc () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/getting_started_misc" */'flavours/blobfox/features/getting_started_misc');
+}
+
+export function ListAdder () {
+  return import(/* webpackChunkName: "features/blobfox/async/list_adder" */'flavours/blobfox/features/list_adder');
+}
+
+export function Tesseract () {
+  return import(/*webpackChunkName: "tesseract" */'tesseract.js');
+}
+
+export function Directory () {
+  return import(/* webpackChunkName: "features/blobfox/async/directory" */'flavours/blobfox/features/directory');
+}
+
+export function Onboarding () {
+  return import(/* webpackChunkName: "features/blobfox/async/onboarding" */'flavours/blobfox/features/onboarding');
+}
+
+export function CompareHistoryModal () {
+  return import(/*webpackChunkName: "flavours/blobfox/async/compare_history_modal" */'flavours/blobfox/features/ui/components/compare_history_modal');
+}
+
+export function FilterModal () {
+  return import(/*webpackChunkName: "flavours/blobfox/async/filter_modal" */'flavours/blobfox/features/ui/components/filter_modal');
+}
+
+export function Explore () {
+  return import(/* webpackChunkName: "flavours/blobfox/async/explore" */'flavours/blobfox/features/explore');
+}
+
+export function InteractionModal () {
+  return import(/*webpackChunkName: "flavours/blobfox/async/modals/interaction_modal" */'flavours/blobfox/features/interaction_modal');
+}
+
+export function SubscribedLanguagesModal () {
+  return import(/*webpackChunkName: "flavours/blobfox/async/modals/subscribed_languages_modal" */'flavours/blobfox/features/subscribed_languages_modal');
+}
+
+export function ClosedRegistrationsModal () {
+  return import(/*webpackChunkName: "flavours/blobfox/async/modals/closed_registrations_modal" */'flavours/blobfox/features/closed_registrations_modal');
+}
+
+export function About () {
+  return import(/*webpackChunkName: "features/blobfox/async/about" */'flavours/blobfox/features/about');
+}
+
+export function PrivacyPolicy () {
+  return import(/*webpackChunkName: "features/blobfox/async/privacy_policy" */'flavours/blobfox/features/privacy_policy');
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/util/fullscreen.js b/app/javascript/flavours/blobfox/features/ui/util/fullscreen.js
new file mode 100644
index 00000000000000..cf5d0cf98d0cd6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/fullscreen.js
@@ -0,0 +1,46 @@
+// APIs for normalizing fullscreen operations. Note that Edge uses
+// the WebKit-prefixed APIs currently (as of Edge 16).
+
+export const isFullscreen = () => document.fullscreenElement ||
+  document.webkitFullscreenElement ||
+  document.mozFullScreenElement;
+
+export const exitFullscreen = () => {
+  if (document.exitFullscreen) {
+    document.exitFullscreen();
+  } else if (document.webkitExitFullscreen) {
+    document.webkitExitFullscreen();
+  } else if (document.mozCancelFullScreen) {
+    document.mozCancelFullScreen();
+  }
+};
+
+export const requestFullscreen = el => {
+  if (el.requestFullscreen) {
+    el.requestFullscreen();
+  } else if (el.webkitRequestFullscreen) {
+    el.webkitRequestFullscreen();
+  } else if (el.mozRequestFullScreen) {
+    el.mozRequestFullScreen();
+  }
+};
+
+export const attachFullscreenListener = (listener) => {
+  if ('onfullscreenchange' in document) {
+    document.addEventListener('fullscreenchange', listener);
+  } else if ('onwebkitfullscreenchange' in document) {
+    document.addEventListener('webkitfullscreenchange', listener);
+  } else if ('onmozfullscreenchange' in document) {
+    document.addEventListener('mozfullscreenchange', listener);
+  }
+};
+
+export const detachFullscreenListener = (listener) => {
+  if ('onfullscreenchange' in document) {
+    document.removeEventListener('fullscreenchange', listener);
+  } else if ('onwebkitfullscreenchange' in document) {
+    document.removeEventListener('webkitfullscreenchange', listener);
+  } else if ('onmozfullscreenchange' in document) {
+    document.removeEventListener('mozfullscreenchange', listener);
+  }
+};
diff --git a/app/javascript/flavours/blobfox/features/ui/util/get_rect_from_entry.js b/app/javascript/flavours/blobfox/features/ui/util/get_rect_from_entry.js
new file mode 100644
index 00000000000000..c266cd7dce7374
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/get_rect_from_entry.js
@@ -0,0 +1,21 @@
+
+// Get the bounding client rect from an IntersectionObserver entry.
+// This is to work around a bug in Chrome: https://crbug.com/737228
+
+let hasBoundingRectBug;
+
+function getRectFromEntry(entry) {
+  if (typeof hasBoundingRectBug !== 'boolean') {
+    const boundingRect = entry.target.getBoundingClientRect();
+    const observerRect = entry.boundingClientRect;
+    hasBoundingRectBug = boundingRect.height !== observerRect.height ||
+      boundingRect.top !== observerRect.top ||
+      boundingRect.width !== observerRect.width ||
+      boundingRect.bottom !== observerRect.bottom ||
+      boundingRect.left !== observerRect.left ||
+      boundingRect.right !== observerRect.right;
+  }
+  return hasBoundingRectBug ? entry.target.getBoundingClientRect() : entry.boundingClientRect;
+}
+
+export default getRectFromEntry;
diff --git a/app/javascript/flavours/blobfox/features/ui/util/intersection_observer_wrapper.js b/app/javascript/flavours/blobfox/features/ui/util/intersection_observer_wrapper.js
new file mode 100644
index 00000000000000..2b24c65831d87c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/intersection_observer_wrapper.js
@@ -0,0 +1,57 @@
+// Wrapper for IntersectionObserver in order to make working with it
+// a bit easier. We also follow this performance advice:
+// "If you need to observe multiple elements, it is both possible and
+// advised to observe multiple elements using the same IntersectionObserver
+// instance by calling observe() multiple times."
+// https://developers.google.com/web/updates/2016/04/intersectionobserver
+
+class IntersectionObserverWrapper {
+
+  callbacks = {};
+  observerBacklog = [];
+  observer = null;
+
+  connect (options) {
+    const onIntersection = (entries) => {
+      entries.forEach(entry => {
+        const id = entry.target.getAttribute('data-id');
+        if (this.callbacks[id]) {
+          this.callbacks[id](entry);
+        }
+      });
+    };
+
+    this.observer = new IntersectionObserver(onIntersection, options);
+    this.observerBacklog.forEach(([ id, node, callback ]) => {
+      this.observe(id, node, callback);
+    });
+    this.observerBacklog = null;
+  }
+
+  observe (id, node, callback) {
+    if (!this.observer) {
+      this.observerBacklog.push([ id, node, callback ]);
+    } else {
+      this.callbacks[id] = callback;
+      this.observer.observe(node);
+    }
+  }
+
+  unobserve (id, node) {
+    if (this.observer) {
+      delete this.callbacks[id];
+      this.observer.unobserve(node);
+    }
+  }
+
+  disconnect () {
+    if (this.observer) {
+      this.callbacks = {};
+      this.observer.disconnect();
+      this.observer = null;
+    }
+  }
+
+}
+
+export default IntersectionObserverWrapper;
diff --git a/app/javascript/flavours/blobfox/features/ui/util/optional_motion.js b/app/javascript/flavours/blobfox/features/ui/util/optional_motion.js
new file mode 100644
index 00000000000000..0b6d4d97f7967f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/optional_motion.js
@@ -0,0 +1,7 @@
+import Motion from 'react-motion/lib/Motion';
+
+import { reduceMotion } from '../../../initial_state';
+
+import ReducedMotion from './reduced_motion';
+
+export default reduceMotion ? ReducedMotion : Motion;
diff --git a/app/javascript/flavours/blobfox/features/ui/util/react_router_helpers.jsx b/app/javascript/flavours/blobfox/features/ui/util/react_router_helpers.jsx
new file mode 100644
index 00000000000000..c0ee31bf68041b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/react_router_helpers.jsx
@@ -0,0 +1,105 @@
+import PropTypes from 'prop-types';
+import { Component, cloneElement, Children } from 'react';
+
+import { Switch, Route, useLocation } from 'react-router-dom';
+
+import StackTrace from 'stacktrace-js';
+
+import BundleColumnError from '../components/bundle_column_error';
+import ColumnLoading from '../components/column_loading';
+import BundleContainer from '../containers/bundle_container';
+
+// Small wrapper to pass multiColumn to the route components
+export const WrappedSwitch = ({ multiColumn, children }) => {
+  const  location = useLocation();
+
+  const decklessLocation = multiColumn && location.pathname.startsWith('/deck')
+    ? {...location, pathname: location.pathname.slice(5)}
+    : location;
+
+  return (
+    <Switch location={decklessLocation}>
+      {Children.map(children, child => child ? cloneElement(child, { multiColumn }) : null)}
+    </Switch>
+  );
+};
+
+
+WrappedSwitch.propTypes = {
+  multiColumn: PropTypes.bool,
+  children: PropTypes.node,
+};
+
+// Small Wrapper to extract the params from the route and pass
+// them to the rendered component, together with the content to
+// be rendered inside (the children)
+export class WrappedRoute extends Component {
+
+  static propTypes = {
+    component: PropTypes.func.isRequired,
+    content: PropTypes.node,
+    multiColumn: PropTypes.bool,
+    componentParams: PropTypes.object,
+  };
+
+  static defaultProps = {
+    componentParams: {},
+  };
+
+  static getDerivedStateFromError () {
+    return {
+      hasError: true,
+    };
+  }
+
+  state = {
+    hasError: false,
+    stacktrace: '',
+  };
+
+  componentDidCatch (error) {
+    StackTrace.fromError(error).then(stackframes => {
+      this.setState({ stacktrace: error.toString() + '\n' + stackframes.map(frame => frame.toString()).join('\n') });
+    }).catch(err => {
+      console.error(err);
+    });
+  }
+
+  renderComponent = ({ match }) => {
+    const { component, content, multiColumn, componentParams } = this.props;
+    const { hasError, stacktrace } = this.state;
+
+    if (hasError) {
+      return (
+        <BundleColumnError
+          stacktrace={stacktrace}
+          multiColumn={multiColumn}
+          errorType='error'
+        />
+      );
+    }
+
+    return (
+      <BundleContainer fetchComponent={component} loading={this.renderLoading} error={this.renderError}>
+        {Component => <Component params={match.params} multiColumn={multiColumn} {...componentParams}>{content}</Component>}
+      </BundleContainer>
+    );
+  };
+
+  renderLoading = () => {
+    const { multiColumn } = this.props;
+
+    return <ColumnLoading multiColumn={multiColumn} />;
+  };
+
+  renderError = (props) => {
+    return <BundleColumnError {...props} errorType='network' />;
+  };
+
+  render () {
+    const { component: Component, content, ...rest } = this.props;
+
+    return <Route {...rest} render={this.renderComponent} />;
+  }
+
+}
diff --git a/app/javascript/flavours/blobfox/features/ui/util/reduced_motion.jsx b/app/javascript/flavours/blobfox/features/ui/util/reduced_motion.jsx
new file mode 100644
index 00000000000000..fd044497f80279
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/reduced_motion.jsx
@@ -0,0 +1,45 @@
+// Like react-motion's Motion, but reduces all animations to cross-fades
+// for the benefit of users with motion sickness.
+import PropTypes from 'prop-types';
+import { Component } from 'react';
+
+import Motion from 'react-motion/lib/Motion';
+
+const stylesToKeep = ['opacity', 'backgroundOpacity'];
+
+const extractValue = (value) => {
+  // This is either an object with a "val" property or it's a number
+  return (typeof value === 'object' && value && 'val' in value) ? value.val : value;
+};
+
+class ReducedMotion extends Component {
+
+  static propTypes = {
+    defaultStyle: PropTypes.object,
+    style: PropTypes.object,
+    children: PropTypes.func,
+  };
+
+  render() {
+
+    const { style, defaultStyle, children } = this.props;
+
+    Object.keys(style).forEach(key => {
+      if (stylesToKeep.includes(key)) {
+        return;
+      }
+      // If it's setting an x or height or scale or some other value, we need
+      // to preserve the end-state value without actually animating it
+      style[key] = defaultStyle[key] = extractValue(style[key]);
+    });
+
+    return (
+      <Motion style={style} defaultStyle={defaultStyle}>
+        {children}
+      </Motion>
+    );
+  }
+
+}
+
+export default ReducedMotion;
diff --git a/app/javascript/flavours/blobfox/features/ui/util/schedule_idle_task.js b/app/javascript/flavours/blobfox/features/ui/util/schedule_idle_task.js
new file mode 100644
index 00000000000000..b04d4a8eefa3b7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/ui/util/schedule_idle_task.js
@@ -0,0 +1,29 @@
+// Wrapper to call requestIdleCallback() to schedule low-priority work.
+// See https://developer.mozilla.org/en-US/docs/Web/API/Background_Tasks_API
+// for a good breakdown of the concepts behind this.
+
+import Queue from 'tiny-queue';
+
+const taskQueue = new Queue();
+let runningRequestIdleCallback = false;
+
+function runTasks(deadline) {
+  while (taskQueue.length && deadline.timeRemaining() > 0) {
+    taskQueue.shift()();
+  }
+  if (taskQueue.length) {
+    requestIdleCallback(runTasks);
+  } else {
+    runningRequestIdleCallback = false;
+  }
+}
+
+function scheduleIdleTask(task) {
+  taskQueue.push(task);
+  if (!runningRequestIdleCallback) {
+    runningRequestIdleCallback = true;
+    requestIdleCallback(runTasks);
+  }
+}
+
+export default scheduleIdleTask;
diff --git a/app/javascript/flavours/blobfox/features/video/index.jsx b/app/javascript/flavours/blobfox/features/video/index.jsx
new file mode 100644
index 00000000000000..7fb70f33c135d1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/features/video/index.jsx
@@ -0,0 +1,665 @@
+import PropTypes from 'prop-types';
+import { PureComponent } from 'react';
+
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+
+import classNames from 'classnames';
+
+import { is } from 'immutable';
+
+import { throttle } from 'lodash';
+
+import { Blurhash } from 'flavours/blobfox/components/blurhash';
+import { Icon }  from 'flavours/blobfox/components/icon';
+import { playerSettings } from 'flavours/blobfox/settings';
+
+import { displayMedia, useBlurhash } from '../../initial_state';
+import { isFullscreen, requestFullscreen, exitFullscreen } from '../ui/util/fullscreen';
+
+const messages = defineMessages({
+  play: { id: 'video.play', defaultMessage: 'Play' },
+  pause: { id: 'video.pause', defaultMessage: 'Pause' },
+  mute: { id: 'video.mute', defaultMessage: 'Mute sound' },
+  unmute: { id: 'video.unmute', defaultMessage: 'Unmute sound' },
+  hide: { id: 'video.hide', defaultMessage: 'Hide video' },
+  expand: { id: 'video.expand', defaultMessage: 'Expand video' },
+  close: { id: 'video.close', defaultMessage: 'Close video' },
+  fullscreen: { id: 'video.fullscreen', defaultMessage: 'Full screen' },
+  exit_fullscreen: { id: 'video.exit_fullscreen', defaultMessage: 'Exit full screen' },
+});
+
+export const formatTime = secondsNum => {
+  let hours   = Math.floor(secondsNum / 3600);
+  let minutes = Math.floor((secondsNum - (hours * 3600)) / 60);
+  let seconds = secondsNum - (hours * 3600) - (minutes * 60);
+
+  if (hours   < 10) hours   = '0' + hours;
+  if (minutes < 10) minutes = '0' + minutes;
+  if (seconds < 10) seconds = '0' + seconds;
+
+  return (hours === '00' ? '' : `${hours}:`) + `${minutes}:${seconds}`;
+};
+
+export const findElementPosition = el => {
+  let box;
+
+  if (el.getBoundingClientRect && el.parentNode) {
+    box = el.getBoundingClientRect();
+  }
+
+  if (!box) {
+    return {
+      left: 0,
+      top: 0,
+    };
+  }
+
+  const docEl = document.documentElement;
+  const body  = document.body;
+
+  const clientLeft = docEl.clientLeft || body.clientLeft || 0;
+  const scrollLeft = window.pageXOffset || body.scrollLeft;
+  const left       = (box.left + scrollLeft) - clientLeft;
+
+  const clientTop = docEl.clientTop || body.clientTop || 0;
+  const scrollTop = window.pageYOffset || body.scrollTop;
+  const top       = (box.top + scrollTop) - clientTop;
+
+  return {
+    left: Math.round(left),
+    top: Math.round(top),
+  };
+};
+
+export const getPointerPosition = (el, event) => {
+  const position = {};
+  const box = findElementPosition(el);
+  const boxW = el.offsetWidth;
+  const boxH = el.offsetHeight;
+  const boxY = box.top;
+  const boxX = box.left;
+
+  let pageY = event.pageY;
+  let pageX = event.pageX;
+
+  if (event.changedTouches) {
+    pageX = event.changedTouches[0].pageX;
+    pageY = event.changedTouches[0].pageY;
+  }
+
+  position.y = Math.max(0, Math.min(1, (pageY - boxY) / boxH));
+  position.x = Math.max(0, Math.min(1, (pageX - boxX) / boxW));
+
+  return position;
+};
+
+export const fileNameFromURL = str => {
+  const url      = new URL(str);
+  const pathname = url.pathname;
+  const index    = pathname.lastIndexOf('/');
+
+  return pathname.slice(index + 1);
+};
+
+class Video extends PureComponent {
+
+  static propTypes = {
+    preview: PropTypes.string,
+    frameRate: PropTypes.string,
+    src: PropTypes.string.isRequired,
+    alt: PropTypes.string,
+    lang: PropTypes.string,
+    sensitive: PropTypes.bool,
+    currentTime: PropTypes.number,
+    onOpenVideo: PropTypes.func,
+    onCloseVideo: PropTypes.func,
+    detailed: PropTypes.bool,
+    inline: PropTypes.bool,
+    editable: PropTypes.bool,
+    alwaysVisible: PropTypes.bool,
+    visible: PropTypes.bool,
+    letterbox: PropTypes.bool,
+    fullwidth: PropTypes.bool,
+    preventPlayback: PropTypes.bool,
+    onToggleVisibility: PropTypes.func,
+    deployPictureInPicture: PropTypes.func,
+    intl: PropTypes.object.isRequired,
+    blurhash: PropTypes.string,
+    autoPlay: PropTypes.bool,
+    volume: PropTypes.number,
+    muted: PropTypes.bool,
+    componentIndex: PropTypes.number,
+    autoFocus: PropTypes.bool,
+  };
+
+  static defaultProps = {
+    frameRate: '25',
+  };
+
+  state = {
+    currentTime: 0,
+    duration: 0,
+    volume: 0.5,
+    paused: true,
+    dragging: false,
+    fullscreen: false,
+    hovered: false,
+    muted: false,
+    revealed: this.props.visible !== undefined ? this.props.visible : (displayMedia !== 'hide_all' && !this.props.sensitive || displayMedia === 'show_all'),
+  };
+
+  setPlayerRef = c => {
+    this.player = c;
+  };
+
+  setVideoRef = c => {
+    this.video = c;
+
+    if (this.video) {
+      this.setState({ volume: this.video.volume, muted: this.video.muted });
+    }
+  };
+
+  setSeekRef = c => {
+    this.seek = c;
+  };
+
+  setVolumeRef = c => {
+    this.volume = c;
+  };
+
+  handleClickRoot = e => e.stopPropagation();
+
+  handlePlay = () => {
+    this.setState({ paused: false });
+    this._updateTime();
+  };
+
+  handlePause = () => {
+    this.setState({ paused: true });
+  };
+
+  _updateTime () {
+    requestAnimationFrame(() => {
+      if (!this.video) return;
+
+      this.handleTimeUpdate();
+
+      if (!this.state.paused) {
+        this._updateTime();
+      }
+    });
+  }
+
+  handleTimeUpdate = () => {
+    this.setState({
+      currentTime: this.video.currentTime,
+      duration:this.video.duration,
+    });
+  };
+
+  handleVolumeMouseDown = e => {
+    document.addEventListener('mousemove', this.handleMouseVolSlide, true);
+    document.addEventListener('mouseup', this.handleVolumeMouseUp, true);
+    document.addEventListener('touchmove', this.handleMouseVolSlide, true);
+    document.addEventListener('touchend', this.handleVolumeMouseUp, true);
+
+    this.handleMouseVolSlide(e);
+
+    e.preventDefault();
+    e.stopPropagation();
+  };
+
+  handleVolumeMouseUp = () => {
+    document.removeEventListener('mousemove', this.handleMouseVolSlide, true);
+    document.removeEventListener('mouseup', this.handleVolumeMouseUp, true);
+    document.removeEventListener('touchmove', this.handleMouseVolSlide, true);
+    document.removeEventListener('touchend', this.handleVolumeMouseUp, true);
+  };
+
+  handleMouseVolSlide = throttle(e => {
+    const { x } = getPointerPosition(this.volume, e);
+
+    if(!isNaN(x)) {
+      this.setState((state) => ({ volume: x, muted: state.muted && x === 0 }), () => {
+        this._syncVideoToVolumeState(x);
+        this._saveVolumeState(x);
+      });
+    }
+  }, 15);
+
+  handleMouseDown = e => {
+    document.addEventListener('mousemove', this.handleMouseMove, true);
+    document.addEventListener('mouseup', this.handleMouseUp, true);
+    document.addEventListener('touchmove', this.handleMouseMove, true);
+    document.addEventListener('touchend', this.handleMouseUp, true);
+
+    this.setState({ dragging: true });
+    this.video.pause();
+    this.handleMouseMove(e);
+
+    e.preventDefault();
+    e.stopPropagation();
+  };
+
+  handleMouseUp = () => {
+    document.removeEventListener('mousemove', this.handleMouseMove, true);
+    document.removeEventListener('mouseup', this.handleMouseUp, true);
+    document.removeEventListener('touchmove', this.handleMouseMove, true);
+    document.removeEventListener('touchend', this.handleMouseUp, true);
+
+    this.setState({ dragging: false });
+    this.video.play();
+  };
+
+  handleMouseMove = throttle(e => {
+    const { x } = getPointerPosition(this.seek, e);
+    const currentTime = this.video.duration * x;
+
+    if (!isNaN(currentTime)) {
+      this.setState({ currentTime }, () => {
+        this.video.currentTime = currentTime;
+      });
+    }
+  }, 15);
+
+  seekBy (time) {
+    const currentTime = this.video.currentTime + time;
+
+    if (!isNaN(currentTime)) {
+      this.setState({ currentTime }, () => {
+        this.video.currentTime = currentTime;
+      });
+    }
+  }
+
+  handleVideoKeyDown = e => {
+    // On the video element or the seek bar, we can safely use the space bar
+    // for playback control because there are no buttons to press
+
+    if (e.key === ' ') {
+      e.preventDefault();
+      e.stopPropagation();
+      this.togglePlay();
+    }
+  };
+
+  handleKeyDown = e => {
+    const frameTime = 1 / this.getFrameRate();
+
+    switch(e.key) {
+    case 'k':
+      e.preventDefault();
+      e.stopPropagation();
+      this.togglePlay();
+      break;
+    case 'm':
+      e.preventDefault();
+      e.stopPropagation();
+      this.toggleMute();
+      break;
+    case 'f':
+      e.preventDefault();
+      e.stopPropagation();
+      this.toggleFullscreen();
+      break;
+    case 'j':
+      e.preventDefault();
+      e.stopPropagation();
+      this.seekBy(-10);
+      break;
+    case 'l':
+      e.preventDefault();
+      e.stopPropagation();
+      this.seekBy(10);
+      break;
+    case ',':
+      e.preventDefault();
+      e.stopPropagation();
+      this.seekBy(-frameTime);
+      break;
+    case '.':
+      e.preventDefault();
+      e.stopPropagation();
+      this.seekBy(frameTime);
+      break;
+    }
+
+    // If we are in fullscreen mode, we don't want any hotkeys
+    // interacting with the UI that's not visible
+
+    if (this.state.fullscreen) {
+      e.preventDefault();
+      e.stopPropagation();
+
+      if (e.key === 'Escape') {
+        exitFullscreen();
+      }
+    }
+  };
+
+  togglePlay = () => {
+    if (this.state.paused) {
+      this.setState({ paused: false }, () => this.video.play());
+    } else {
+      this.setState({ paused: true }, () => this.video.pause());
+    }
+  };
+
+  toggleFullscreen = () => {
+    if (isFullscreen()) {
+      exitFullscreen();
+    } else {
+      requestFullscreen(this.player);
+    }
+  };
+
+  componentDidMount () {
+    document.addEventListener('fullscreenchange', this.handleFullscreenChange, true);
+    document.addEventListener('webkitfullscreenchange', this.handleFullscreenChange, true);
+    document.addEventListener('mozfullscreenchange', this.handleFullscreenChange, true);
+    document.addEventListener('MSFullscreenChange', this.handleFullscreenChange, true);
+
+    window.addEventListener('scroll', this.handleScroll);
+
+    this._syncVideoFromLocalStorage();
+  }
+
+  componentWillUnmount () {
+    window.removeEventListener('scroll', this.handleScroll);
+
+    document.removeEventListener('fullscreenchange', this.handleFullscreenChange, true);
+    document.removeEventListener('webkitfullscreenchange', this.handleFullscreenChange, true);
+    document.removeEventListener('mozfullscreenchange', this.handleFullscreenChange, true);
+    document.removeEventListener('MSFullscreenChange', this.handleFullscreenChange, true);
+
+    if (!this.state.paused && this.video && this.props.deployPictureInPicture) {
+      this.props.deployPictureInPicture('video', {
+        src: this.props.src,
+        currentTime: this.video.currentTime,
+        muted: this.video.muted,
+        volume: this.video.volume,
+      });
+    }
+  }
+
+  UNSAFE_componentWillReceiveProps (nextProps) {
+    if (!is(nextProps.visible, this.props.visible) && nextProps.visible !== undefined) {
+      this.setState({ revealed: nextProps.visible });
+    }
+  }
+
+  componentDidUpdate (prevProps) {
+    if (this.video && this.state.revealed && this.props.preventPlayback && !prevProps.preventPlayback) {
+      this.video.pause();
+    }
+  }
+
+  handleScroll = throttle(() => {
+    if (!this.video) {
+      return;
+    }
+
+    const { top, height } = this.video.getBoundingClientRect();
+    const inView = (top <= (window.innerHeight || document.documentElement.clientHeight)) && (top + height >= 0);
+
+    if (!this.state.paused && !inView) {
+      this.video.pause();
+
+      if (this.props.deployPictureInPicture) {
+        this.props.deployPictureInPicture('video', {
+          src: this.props.src,
+          currentTime: this.video.currentTime,
+          muted: this.video.muted,
+          volume: this.video.volume,
+        });
+      }
+
+      this.setState({ paused: true });
+    }
+  }, 150, { trailing: true });
+
+  handleFullscreenChange = () => {
+    this.setState({ fullscreen: isFullscreen() });
+  };
+
+  handleMouseEnter = () => {
+    this.setState({ hovered: true });
+  };
+
+  handleMouseLeave = () => {
+    this.setState({ hovered: false });
+  };
+
+  toggleMute = () => {
+    const muted = !(this.video.muted || this.state.volume === 0);
+
+    this.setState((state) => ({ muted, volume: Math.max(state.volume || 0.5, 0.05) }), () => {
+      this._syncVideoToVolumeState();
+      this._saveVolumeState();
+    });
+  };
+
+  _syncVideoToVolumeState = (volume = null, muted = null) => {
+    if (!this.video) {
+      return;
+    }
+
+    this.video.volume = volume ?? this.state.volume;
+    this.video.muted = muted ?? this.state.muted;
+  };
+
+  _saveVolumeState = (volume = null, muted = null) => {
+    playerSettings.set('volume', volume ?? this.state.volume);
+    playerSettings.set('muted', muted ?? this.state.muted);
+  };
+
+  _syncVideoFromLocalStorage = () => {
+    this.setState({ volume: playerSettings.get('volume') ?? 0.5, muted: playerSettings.get('muted') ?? false }, () => {
+      this._syncVideoToVolumeState();
+    });
+  };
+
+  toggleReveal = () => {
+    if (this.state.revealed) {
+      this.setState({ paused: true });
+    }
+
+    if (this.props.onToggleVisibility) {
+      this.props.onToggleVisibility();
+    } else {
+      this.setState({ revealed: !this.state.revealed });
+    }
+  };
+
+  handleLoadedData = () => {
+    const { currentTime, volume, muted, autoPlay } = this.props;
+
+    if (currentTime) {
+      this.video.currentTime = currentTime;
+    }
+
+    if (volume !== undefined) {
+      this.video.volume = volume;
+    }
+
+    if (muted !== undefined) {
+      this.video.muted = muted;
+    }
+
+    if (autoPlay) {
+      this.video.play();
+    }
+  };
+
+  handleProgress = () => {
+    const lastTimeRange = this.video.buffered.length - 1;
+
+    if (lastTimeRange > -1) {
+      this.setState({ buffer: Math.ceil(this.video.buffered.end(lastTimeRange) / this.video.duration * 100) });
+    }
+  };
+
+  handleVolumeChange = () => {
+    this.setState({ volume: this.video.volume, muted: this.video.muted });
+    this._saveVolumeState(this.video.volume, this.video.muted);
+  };
+
+  handleOpenVideo = () => {
+    this.video.pause();
+
+    this.props.onOpenVideo(this.props.lang, {
+      startTime: this.video.currentTime,
+      autoPlay: !this.state.paused,
+      defaultVolume: this.state.volume,
+      componentIndex: this.props.componentIndex,
+    });
+  };
+
+  handleCloseVideo = () => {
+    this.video.pause();
+    this.props.onCloseVideo();
+  };
+
+  getFrameRate () {
+    if (this.props.frameRate && isNaN(this.props.frameRate)) {
+      // The frame rate is returned as a fraction string so we
+      // need to convert it to a number
+
+      return this.props.frameRate.split('/').reduce((p, c) => p / c);
+    }
+
+    return this.props.frameRate;
+  }
+
+  render () {
+    const { preview, src, inline, onOpenVideo, onCloseVideo, intl, alt, lang, letterbox, fullwidth, detailed, sensitive, editable, blurhash, autoFocus } = this.props;
+    const { currentTime, duration, volume, buffer, dragging, paused, fullscreen, hovered, revealed } = this.state;
+    const progress = Math.min((currentTime / duration) * 100, 100);
+    const muted = this.state.muted || volume === 0;
+
+    const playerStyle = {};
+
+    if (inline) {
+      playerStyle.aspectRatio = '16 / 9';
+    }
+
+    let preload;
+
+    if (this.props.currentTime || fullscreen || dragging) {
+      preload = 'auto';
+    } else if (detailed) {
+      preload = 'metadata';
+    } else {
+      preload = 'none';
+    }
+
+    let warning;
+
+    if (sensitive) {
+      warning = <FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' />;
+    } else {
+      warning = <FormattedMessage id='status.media_hidden' defaultMessage='Media hidden' />;
+    }
+
+    return (
+      <div
+        role='menuitem'
+        className={classNames('video-player', { inactive: !revealed, detailed, inline: inline && !fullscreen, fullscreen, editable, letterbox, 'full-width': fullwidth })}
+        style={playerStyle}
+        ref={this.setPlayerRef}
+        onMouseEnter={this.handleMouseEnter}
+        onMouseLeave={this.handleMouseLeave}
+        onClick={this.handleClickRoot}
+        onKeyDown={this.handleKeyDown}
+        tabIndex={0}
+      >
+        <Blurhash
+          hash={blurhash}
+          className={classNames('media-gallery__preview', {
+            'media-gallery__preview--hidden': revealed,
+          })}
+          dummy={!useBlurhash}
+        />
+
+        {(revealed || editable) && <video
+          ref={this.setVideoRef}
+          src={src}
+          poster={preview}
+          preload={preload}
+          role='button'
+          tabIndex={0}
+          aria-label={alt}
+          title={alt}
+          lang={lang}
+          onClick={this.togglePlay}
+          onKeyDown={this.handleVideoKeyDown}
+          onPlay={this.handlePlay}
+          onPause={this.handlePause}
+          onLoadedData={this.handleLoadedData}
+          onProgress={this.handleProgress}
+          onVolumeChange={this.handleVolumeChange}
+          style={{ ...playerStyle, width: '100%' }}
+        />}
+
+        <div className={classNames('spoiler-button', { 'spoiler-button--hidden': revealed || editable })}>
+          <button type='button' className='spoiler-button__overlay' onClick={this.toggleReveal}>
+            <span className='spoiler-button__overlay__label'>
+              {warning}
+              <span className='spoiler-button__overlay__action'><FormattedMessage id='status.media.show' defaultMessage='Click to show' /></span>
+            </span>
+          </button>
+        </div>
+
+        <div className={classNames('video-player__controls', { active: paused || hovered })}>
+          <div className='video-player__seek' onMouseDown={this.handleMouseDown} ref={this.setSeekRef}>
+            <div className='video-player__seek__buffer' style={{ width: `${buffer}%` }} />
+            <div className='video-player__seek__progress' style={{ width: `${progress}%` }} />
+
+            <span
+              className={classNames('video-player__seek__handle', { active: dragging })}
+              tabIndex={0}
+              style={{ left: `${progress}%` }}
+              onKeyDown={this.handleVideoKeyDown}
+            />
+          </div>
+
+          <div className='video-player__buttons-bar'>
+            <div className='video-player__buttons left'>
+              <button type='button' title={intl.formatMessage(paused ? messages.play : messages.pause)} aria-label={intl.formatMessage(paused ? messages.play : messages.pause)} className='player-button' onClick={this.togglePlay} autoFocus={autoFocus}><Icon id={paused ? 'play' : 'pause'} fixedWidth /></button>
+              <button type='button' title={intl.formatMessage(muted ? messages.unmute : messages.mute)} aria-label={intl.formatMessage(muted ? messages.unmute : messages.mute)} className='player-button' onClick={this.toggleMute}><Icon id={muted ? 'volume-off' : 'volume-up'} fixedWidth /></button>
+
+              <div className={classNames('video-player__volume', { active: this.state.hovered })} onMouseDown={this.handleVolumeMouseDown} ref={this.setVolumeRef}>
+                <div className='video-player__volume__current' style={{ width: `${muted ? 0 : volume * 100}%` }} />
+
+                <span
+                  className={classNames('video-player__volume__handle')}
+                  tabIndex={0}
+                  style={{ left: `${muted ? 0 : volume * 100}%` }}
+                />
+              </div>
+
+              {(detailed || fullscreen) && (
+                <span className='video-player__time'>
+                  <span className='video-player__time-current'>{formatTime(Math.floor(currentTime))}</span>
+                  <span className='video-player__time-sep'>/</span>
+                  <span className='video-player__time-total'>{formatTime(Math.floor(duration))}</span>
+                </span>
+              )}
+            </div>
+
+            <div className='video-player__buttons right'>
+              {(!onCloseVideo && !editable && !fullscreen && !this.props.alwaysVisible) && <button type='button' title={intl.formatMessage(messages.hide)} aria-label={intl.formatMessage(messages.hide)} className='player-button' onClick={this.toggleReveal}><Icon id='eye-slash' fixedWidth /></button>}
+              {(!fullscreen && onOpenVideo) && <button type='button' title={intl.formatMessage(messages.expand)} aria-label={intl.formatMessage(messages.expand)} className='player-button' onClick={this.handleOpenVideo}><Icon id='expand' fixedWidth /></button>}
+              {onCloseVideo && <button type='button' title={intl.formatMessage(messages.close)} aria-label={intl.formatMessage(messages.close)} className='player-button' onClick={this.handleCloseVideo}><Icon id='compress' fixedWidth /></button>}
+              <button type='button' title={intl.formatMessage(fullscreen ? messages.exit_fullscreen : messages.fullscreen)} aria-label={intl.formatMessage(fullscreen ? messages.exit_fullscreen : messages.fullscreen)} className='player-button' onClick={this.toggleFullscreen}><Icon id={fullscreen ? 'compress' : 'arrows-alt'} fixedWidth /></button>
+            </div>
+          </div>
+        </div>
+      </div>
+    );
+  }
+
+}
+
+export default injectIntl(Video);
diff --git a/app/javascript/flavours/blobfox/hooks/useHovering.ts b/app/javascript/flavours/blobfox/hooks/useHovering.ts
new file mode 100644
index 00000000000000..2062e70d26ac98
--- /dev/null
+++ b/app/javascript/flavours/blobfox/hooks/useHovering.ts
@@ -0,0 +1,17 @@
+import { useCallback, useState } from 'react';
+
+export const useHovering = (animate?: boolean) => {
+  const [hovering, setHovering] = useState<boolean>(animate ?? false);
+
+  const handleMouseEnter = useCallback(() => {
+    if (animate) return;
+    setHovering(true);
+  }, [animate]);
+
+  const handleMouseLeave = useCallback(() => {
+    if (animate) return;
+    setHovering(false);
+  }, [animate]);
+
+  return { hovering, handleMouseEnter, handleMouseLeave };
+};
diff --git a/app/javascript/flavours/blobfox/images/elephant_ui_disappointed.svg b/app/javascript/flavours/blobfox/images/elephant_ui_disappointed.svg
new file mode 100644
index 00000000000000..580c15a138838e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/images/elephant_ui_disappointed.svg
@@ -0,0 +1 @@
+<svg xmlns="http://www.w3.org/2000/svg" height="134.11569" width="134.61565" viewBox="0 0 134.61565 134.11569"><path d="M82.69963 103.86569c6.8 1.5 11 2.4 11.3-6.200005.3-8.6-1.8-17.3-1.8-17.3l-13.6 1.1 4.1 22.400005z" class="st32" fill="#3a434e" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M65.39963 112.96569c-.2 10.3-.6 17.5 6.5 17.4 7.1-.1 12.6 1.1 13.6-5.3 1.1-6.3 1.9-20.6.7-28.000005" class="st32" fill="#3a434e" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M86.39963 97.66569c-1.4-7.5-4.1-23.2-4.1-23.2s13.2-1.5 10.4-13c-2.7-11.4-7.5-22.6-11-31.1s-14.5-16.9-28.6-15.7c-19.2 1.6-25.6 7-31.6 23.1-5.4 14.4-10.4 47.2-8.9 63.3.8 8.7 5 13.7 14.4 13.5 9.4-.2 39.8-.8 49.8-2.8.3-.1.6-.1.9-.2" class="st33" fill="#56606b"/><path d="M85.89963 97.76569l-4.1-23.2c0-.3.1-.5.4-.6 2.6-.4 5.3-1.4 7.3-3.1 1-.8 1.9-1.9 2.4-3.1.5-1.2.7-2.5.6-3.9 0-1.3-.4-2.6-.7-4-.3-1.3-.7-2.7-1.1-4-.8-2.7-1.7-5.3-2.6-7.9-1.9-5.2-4-10.4-6.1-15.5-.5-1.3-1-2.6-1.7-3.8-.6-1.2-1.4-2.3-2.3-3.4-1.7-2.1-3.8-4-6-5.5-4.6-3-10-4.7-15.4-4.9-2.7-.1-5.5.3-8.2.6-2.7.4-5.5.9-8.1 1.7-2.6.8-5.1 1.9-7.3 3.5s-4.1 3.6-5.6 5.8c-1.5 2.3-2.8 4.7-3.9 7.3-.6 1.3-1.1 2.5-1.6 3.8-.4 1.3-.9 2.6-1.3 3.9-1.6 5.3-2.8 10.7-3.9 16.1-1 5.4-1.9 10.9-2.6 16.4-.7 5.5-1.2 11-1.3 16.6-.1 2.8-.1 5.5.1 8.3.1 2.8.5 5.5 1.6 8 1 2.5 2.9 4.6 5.4 5.7 2.4 1.1 5.2 1.3 8 1.3 5.6-.1 11.1-.2 16.7-.4 11.1-.4 22.2-.8 33.2-2.3.1 0 .2.1.2.2s0 .2-.1.2c-2.7.9-5.5 1.2-8.3 1.4-2.8.2-5.6.5-8.3.6-5.6.3-11.1.6-16.7.7-5.6.2-11.1.3-16.7.4-2.8.1-5.7-.1-8.4-1.3s-4.7-3.5-5.8-6.2c-1.1-2.6-1.5-5.5-1.6-8.3-.2-2.8-.2-5.6-.1-8.4.2-5.6.7-11.1 1.3-16.7.7-5.5 1.5-11 2.6-16.5s2.3-10.9 3.9-16.3c.4-1.3.9-2.7 1.3-4 .5-1.3 1-2.6 1.6-3.9 1.1-2.6 2.4-5.1 4-7.4 1.6-2.3 3.6-4.4 5.9-6.1 2.3-1.7 4.9-2.8 7.6-3.7 2.7-.8 5.5-1.4 8.2-1.7 2.8-.3 5.5-.7 8.4-.6 5.6.2 11.2 1.9 15.9 5 2.4 1.6 4.5 3.5 6.3 5.7.9 1.1 1.7 2.3 2.4 3.5.7 1.3 1.2 2.6 1.7 3.9 2.1 5.1 4.2 10.3 6.1 15.5.9 2.6 1.8 5.3 2.6 7.9.4 1.3.8 2.7 1.1 4 .3 1.3.7 2.7.8 4.2.1 1.4-.1 2.9-.7 4.3s-1.5 2.5-2.6 3.5c-2.3 1.9-5 2.8-7.9 3.3l.4-.6 4.1 23.2c0 .3-.1.5-.4.6-.3.1-.7.5-.7.2z"/><path d="M26.49963 114.06569c-4.7 0-7.4-2.1-10-4.4-2.3-2-3.2-4.6-3.4-8.6-.1-2.700005-.6-10.000005.4-18.800005 3.8.9 9.7 3.8 13.4 7.6 5.6 5.7 17.7 6.3 22.7 6.3h1.8l.1-.4s.5-2.6 1.8-5.2l.3-.6-.7-.1c-.4-.1-10.9-1.9-9.7-10.8.7-4.9 13.3-7.9 33.9-7.9 2.2 0 3.8 0 4.2.1l3.5 2.2c-1.5.5-2.6.6-2.6.6l-.5.1.1.5c0 .2 2.8 16.4 4.1 24 0 0-7.9 13.100005-8 13.000005-.1-.1-.3-.1-.3-.1-.3 0-.7.1-.9.1-9.9 1.7-39.6 2.4-49.3 2.6l-.9-.2z" class="st34" fill="#3a434e"/><path d="M45.89963 51.36569c-.7 0-1.4-.6-1.4-1.4v-5.1c0-.7.6-1.4 1.4-1.4.7 0 1.4.6 1.4 1.4v5.1c-.1.8-.7 1.4-1.4 1.4z"/><path d="M72.89963 30.365685c-3.5.4-2.7 2.9-1.2 3.5 1.5.6 3.7.1 4.3-1.6.4-1.6-1.3-2.1-3.1-1.9z" class="st35" fill="#4f5862"/><path d="M44.29963 53.965685c-.4.7-1.5.2-2.7-.6-1.2-.8-2.1-1.5-1.6-2.2.4-.7 1.6-.4 2.8.4 1.2.8 2 1.7 1.5 2.4z" class="st34" fill="#4f5862"/><path d="M27.29963 36.165685c0-5.6-3.7-9.4-7.9-9.8-4.2-.4-9-.3-14.0000002 11.3-5.00000001 11.6-6.7 15.7-2.6 17.9 4.1 2.2 9.5000002 1.5 11.3000002-1.4 0 0 5.3 3.8 9.7-3.8" class="st36" fill="#56606b" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M11.19963 40.565685c-2.7000002 5.1-2.7000002 7.7-.5 8.5 2.2.8 4.1.7 6.4-3 0 0 2 .7 4.9-4.1.9-1.5-.7-2.6-.7-2.6s-4.8 1.3-7.1-5l-3 6.2z" class="st34" fill="#3a434e" fill-opacity="0"/><path d="M9.7996298 43.365685l4.4000002-9s1.8 6.3 7.8 4.9" class="st7" fill="none" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M27.89963 67.365685c-4.9.8-9.7 4.5-9.3 15.7.4 11.2.5 18.700005 6.1 20.000005 5.5 1.3 13.8.3 14.1-7.100005.3-7.4.3-16.1.3-16.1" class="st36" fill="#53606c" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M28.69963 102.96569c-1.4 0-2.8-.2-4.1-.5-1.2-.3-2.2-.9-3-2 .5.2 1.1.3 1.7.3 1.2 0 5.2-.5 5.8-7.200005.7-7.4 2.8-10.9 6.6-10.9.8 0 1.6.1 2.6.4 0 3.4-.1 8.3-.2 12.7-.2 6.700005-7.2 7.200005-9.4 7.200005z" class="st34" fill="#3a434e"/><path d="M50.69963 18.965685c-5.2 2.9-14.6 4.7-18.1-1.5-3-5.4 2.1-9.6999996 7.8-9.9999996 5.7-.3 7.6 1.2 7.6 1.2s1.9-5.9 9.3-7.69999998c3.9-1 6-.1 6.2 1.19999998 0 0 3.6-.9 4 3.5 0 0 3.9-.4 3.1 5.1999996-.8 5.6-10.6 10.1-17.7 6.4 0 0-1.1 1.2-2.2 1.7z" class="st33" fill="#56606b"/><path d="M40.79963 21.665685c-2.7 0-4.8-.9-6.3-2.3-.7-1.1-.8-2.9-.3-4.3.8-1.9 2.6-3.3 4.6-3.7 1.2-.2 2.6-.4 3.9-.4 3.3 0 6.2.8 7.3 1.9l.6.6.3-.7s.7-2 2.2-2c.2 0 .5.1.8.2 2.2.9 3.5 1.2 4.6 1.2.5 0 .9-.1 1.3-.2.1-.1.4-.1.6-.1.6 0 1.5.3 1.8.8.2.3.2.6.1 1l-.2.6h.7c.4 0 1.4.2 1.8.9.2.4.2 1-.2 1.7-1.8.8-3.8 1.2-5.7 1.2-2 0-4 0-5.6-.8 0 0-1.2 1.3-2.2 1.8-3.1 1.6-7 2.6-10.1 2.6z" class="st34" fill="#3a434e" fill-opacity=".94117647"/><path d="M61.79963 18.66569c-3.1.5-6.3.1-8.9-1.5.7.2 1.5.4 2.2.5.7.2 1.4.2 2.2.3.7.1 1.4 0 2.2 0 .7-.1 1.4-.1 2.2-.2h.1c.3 0 .5.1.6.4-.1.2-.3.5-.6.5z"/><path d="M37.59963 21.26569c-2.4-.4-4.8-2.1-5.7-4.5-.5-1.2-.7-2.6-.3-3.9.3-1.3 1.1-2.4 2.1-3.3 2-1.7 4.6-2.5 7.1-2.6 1.3-.1 2.5 0 3.8.1.6.1 1.3.2 1.9.4.6.2 1.2.4 1.9.8l-.8.2c.6-1.6 1.6-3 2.8-4.2 1.2-1.2 2.6-2.2 4.1-2.9 1.5-.7 3.2-1.1 4.8-1.3.8-.1 1.7-.1 2.6.1.4.1.9.3 1.3.6s.7.8.8 1.3l-.6-.4c.6-.1 1.2-.1 1.7 0 .6.1 1.1.4 1.6.8.4.4.7.9.9 1.5.1.3.2.5.2.8l.1.8-.5-.4c1 0 1.9.3 2.6.9.7.7 1 1.6 1.1 2.5.1.9 0 1.7-.1 2.5-.2.9-.5 1.7-1 2.4-.9 1.4-2.2 2.5-3.7 3.4-1.4.9-3 1.4-4.6 1.8-.3.1-.5-.1-.6-.4-.1-.3.1-.5.4-.6 1.5-.3 3-.9 4.3-1.7 1.3-.8 2.5-1.8 3.3-3.1.4-.6.7-1.3.8-2 .1-.7.2-1.5.1-2.2-.1-.7-.3-1.4-.8-1.9-.5-.4-1.2-.7-1.8-.7-.3 0-.5-.2-.5-.4l-.1-.7c-.1-.2-.1-.4-.2-.7-.2-.4-.4-.8-.7-1.1-.3-.3-.7-.5-1.1-.6-.4-.1-.9-.1-1.3 0-.3.1-.5-.1-.6-.4-.1-.5-.7-.9-1.4-1.1-.7-.2-1.5-.2-2.2-.1-1.5.2-3.1.6-4.5 1.2-1.4.7-2.7 1.6-3.8 2.7-1.1 1.1-2 2.5-2.5 3.8-.1.3-.4.4-.6.3h-.1c-.4-.2-1-.5-1.5-.6-.6-.1-1.2-.2-1.7-.3-1.2-.1-2.4-.2-3.6-.1-2.4.1-4.7.8-6.5 2.3-.9.7-1.6 1.7-1.9 2.8-.3 1.1-.2 2.3.2 3.4s1.1 2.1 1.9 2.9c.6.9 1.7 1.5 2.9 1.9z"/><path d="M63.49963 2.1656854c0 3.5-2.6 5.5-4.3 6.1m8.3-2.6c.2 3.4-3.3 5.1999996-3.3 5.1999996" class="st7" fill="none" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M90.29963 84.765685c2.6 2.3 3 4.3-2.4 4.8-5.3.5-25.7 2.4-28.2 2.6-2.4.3-3.4 1.7-3.4 2.8 0 1.1.5 3.2 4 3.1 3.4-.1 23.8-1.5 30.4-2.4 6.6-.8 14.4-2.4 13.4-9s-5.4-8.7-5.4-8.7l-8.4 6.8z" class="st37" fill="#737039" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M90.29963 84.765685c2.6 2.3 3 4.3-2.4 4.8-5.3.5-25.7 2.4-28.2 2.6-2.4.3-3.4 1.7-3.4 2.8 0 1.1.5 3.2 4 3.1 3.4-.1 23.8-1.5 30.4-2.4 6.6-.8 13.8-2.3 13.4-9-.3-5.5-3.1-7-4.4-8.1-.5-.1-1-.1-1.6-.2l-7.8 6.4z" fill="#625d28" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M102.69963 64.665685c5.4-.1 10.3-1.9 12.2-6.5 1.9-4.6 8.7-10.1 14.2-2.1 5.4 8.1 6.6 17.3 2.8 23.7-3.8 6.5-12.1 3.5-14.9-.5-2.7-4-8.6-2.9-14.5-2.7-5.9.2.2-11.9.2-11.9z" class="st37" fill="#737039" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M65.89963 54.865685s10.2 21.3 13.5 26.8c3.2 5.5 12.9 6.2 17.4 3.5 4.5-2.7 7.3-7.3 8-15.1.7-7.9-2.4-14.9-10-15.2-7.6-.3-11.9 7.6-12.1 13.7" class="st36" fill="#53606c" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M65.89963 54.865685s10.2 21.3 13.5 26.8c3.2 5.5 12.9 6.2 17.4 3.5 4.5-2.7 7.3-7.3 8-15.1.7-7.9-2.4-14.9-10-15.2-7.6-.3-11.9 7.6-12.1 13.7" class="st36" fill="#56606b" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M90.19963 86.165685c-3.7 0-8.3-1.3-10.4-4.8-.9-1.5-2.4-4.3-4.4-8.4l5.9-.1c4 7.4 5.9 9.8 8 9.8 3.9 0 6-3.4 6.9-9.5.2-1.2.3-2.3.4-3.4.5-4.6.9-7.2 3.4-7.5.3 0 .6-.1.9-.1 2.1 0 2.5 1.2 3.1 2.8.1.2.2.5.2.7.1 1.3.1 2.7 0 4.2-.7 7.3-3.1 11.9-7.7 14.7-1.6 1-3.9 1.6-6.3 1.6z" class="st34" fill="#3a434e"/><path d="M89.19963 63.86569l-.3 6.6c-.1 1.1-.2 2.2-.4 3.3-.1.6-.3 1.1-.5 1.7-.3.5-.6 1.1-1.2 1.5-.2.1-.5.1-.7-.2-.1-.2-.1-.5.2-.7.7-.4 1.1-1.5 1.3-2.5.2-1 .4-2.1.5-3.2.2-2.2.3-4.4.5-6.6 0-.2.2-.3.3-.3.2.1.3.3.3.4z"/><path d="M52.29963 68.665685c-6.3.6-11.1 3.9-10 10.7 1.1 6.8 7.6 8.1 16 7.7 8.4-.4 26.4-1.3 26.4-1.3s-3.3-1.7-4.8-3.3c-.5-.6-1-1.4-1.6-2.5-1.6.1-15.5.8-22.7 1-3.4.1-3.8-1.2-3.9-1.8-.3-1.2.5-2.7 2.8-2.8 3.1-.2 10.8-.7 21.4-.7h11.5s.9-.3 1-9.1c0-.1-29.8 1.5-36.1 2.1z" class="st37" fill="#737039" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M56.19963 86.665685c-8.8 0-12.6-2.3-13.5-7.4 0-.1 0-.2-.1-.4 1.2 1.5 3.5 2.7 5.6 2.7 1.4 0 2.6-.5 3.6-1.5.5.7 1.4 1.4 3.7 1.4h.4c6.9-.2 19.5-.8 22.1-.9.6 1.1 1.2 1.9 1.6 2.4.9 1 2.4 1.9 3.6 2.5-5 .3-18.1.9-24.7 1.2h-2.3z" class="st39" fill="#625d28"/><path d="M44.09963 57.865685c-2.2-.6-5.8-8.3-8.7-8.7-2.9-.3-6.6 1.6-3.2 8.5 3.4 6.9 8 10 14.3 8.2 6.3-1.8 12.7-5.1 14.5-8.3 1.8-3.2-.6-6.2-4.8-4.3-4.1 1.7-9.9 5.2-12.1 4.6z" fill="#b3bfcd" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M43.09963 65.865685c-4.3 0-7.7-2.7-10.4-8.4-1.4-2.8-1.7-5.1-.8-6.4.8-1.3 2.3-1.4 2.9-1.4h.6c.4 0 .8.2 1.2.6-.7.1-1.3.5-1.6 1.2-.6 1.2-.3 2.9.9 4.7l.4.6c2.1 3.1 4.1 6 7.8 6 .9 0 1.9-.2 2.9-.6 5.6-2 9.4-3.6 11.1-5.4 1.2-1.3 1.9-2.6 1.7-3.6.5.2.9.5 1.1.9.5.8.4 1.9-.2 3-1.6 2.8-7.4 6.2-14.2 8.1-1.2.5-2.3.7-3.4.7z" class="st35" fill="#93a1b5"/><path d="M13.89963 107.66569c-.1 5.1 1.3 10.2 2.3 14.8 1.3 5.5 1.3 10.1 5.2 10.7 3.9.6 10.1.9 14.4 0 4.3-.9 4.1-5.2 4.5-8.2.4-3.1 0-10.7 0-10.7s-1.1-1.4-3-1.9" class="st34" fill="#3a434e"/><path d="M14.39963 107.66569c-.1 5.2 1.3 10.3 2.5 15.4l.8 3.9c.3 1.3.6 2.5 1.1 3.6.3.5.6 1 1.1 1.4.5.3 1 .5 1.6.6 1.3.2 2.6.3 3.9.4 2.6.2 5.2.2 7.8 0 1.3-.1 2.6-.3 3.7-.7 1.1-.4 1.9-1.4 2.3-2.6.4-1.2.5-2.4.7-3.7.1-.7.2-1.3.2-1.9.1-.6.1-1.3.1-1.9 0-2.6 0-5.2-.2-7.8l.1.3c-.1-.2-.3-.4-.5-.6l-.6-.6c-.4-.4-.9-.7-1.5-.9-.1 0-.1-.1-.1-.2s.1-.1.2-.1c.6.1 1.2.3 1.7.6.3.1.5.3.8.5.3.2.5.4.8.6.1.1.1.2.1.2.1 2.6.2 5.3.2 7.9 0 .7 0 1.3-.1 2s-.2 1.3-.2 2c-.1 1.3-.3 2.7-.7 4-.5 1.3-1.5 2.6-2.8 3.1-1.4.6-2.7.7-4 .8-2.7.2-5.3.2-8 0-1.3-.1-2.6-.2-4-.4-.7-.1-1.4-.4-2-.8-.6-.5-1-1.1-1.4-1.7-.6-1.3-.9-2.6-1.2-3.9l-.8-3.9c-1.1-5.1-2.6-10.3-2.5-15.6 0-.3.2-.5.5-.5.2 0 .4.2.4.5z"/><path d="M68.19963 86.665685l.4 4.6s.3 1.5 2.4 1.5c2.1-.1 2.2-2 2.2-2l-.1-4.5-4.9.4z" class="st37" fill="#737039" stroke="#000" stroke-linecap="round" stroke-linejoin="round" stroke-miterlimit="10"/><path d="M110.49963 71.465685c-.5 1.8.5 2.9 3.8 4.6 3.3 1.8 4 5.1 8.2 6 4.3.9 8.2-4.5 3.8-10.1-4.5-5.6-14.1-6.5-15.8-.5z" class="st39" fill="#625d28"/><circle r="1.7" cy="57.765686" cx="126.09963" fill="#99988c"/><path d="M17.39963 115.26569s.8 3.9 1.1 6.3c.3 2.4.9 3.8 5.9 3.2 5-.6 4.9-1.5 5.1-6.4.2-4.9-.1-7.4-3.7-7.6-3.6-.2-9.1.7-8.4 4.5z" class="st33" fill="#56606b"/><path fill="#3a434e" class="st34" d="M11.19963 40.565685c-2.7000002 5.1-2.7000002 7.7-.5 8.5 2.2.8 4.1.7 6.4-3 0 0 2 .7 4.9-4.1.9-1.5-.7-2.6-.7-2.6s-4.8 1.3-7.1-5l-3 6.2z"/></svg>
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/images/elephant_ui_working.svg b/app/javascript/flavours/blobfox/images/elephant_ui_working.svg
new file mode 100644
index 00000000000000..8ba475db0a07c4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/images/elephant_ui_working.svg
@@ -0,0 +1 @@
+<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 124.12477 127.91685" width="124.12476" height="127.91685"><path d="M72.584191 46.815676c-2.3-2.2-4.2-2.5-6.6-.6-2.4 1.9-2.1 4.8.9 7.6 3.1 2.9 4.7 4.1 6.7 5 2.1.9 5.4 2.5 10.5-2s10.2-11.1 9.4-14.7c-.8-3.6-4.1-1.8-6.8 1.2s-3.7 4-5.4 5.2c-1.5 1.3-3.8 3-8.7-1.7z" class="st0" style="fill:#93a1b5;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M116.384191 75.015676c0 6.3-3.9 9.8-9.1 9.8-5.3 0-9.9-3.5-9.9-9.8 0-6.3 4.3-10.3 9.5-10.3s9.5 4 9.5 10.3z" style="fill:#3a434e;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M54.184191 16.615676c-23 1.2-30.5 14.1-32.8 27.8-3 18.2-8.2 44.2-9.2 53.2s-1 16 6 22 11 5 23 7 19 0 20-8l16.8-1.1s14.5 5.5 18.8 6.9c4.3 1.4 10.6.5 12.1-7.1s.2-12.5-6.6-14.4c-6.8-1.9-10.6-2.9-10.6-2.9l4.4-30.1s17.4 1.6 22.6-20c0 0 3.9 1.1 4.8-2.8.9-3.9-2.6-6.2-5.6-4.8l-2.5-1s-.2-3.8-3.5-4.2c-2.1-.2-6 3.4-3 7.4 0 0-3.4 8.9-12 7.8-8.6-1.1-12.5-11.2-15-18.2s-10.7-18.3-27.7-17.5z" class="st2" style="fill:#56606b;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M95.484191 69.915676c-.6 0-1.2 0-1.8-.1-4.2-.2-10.9-2.4-17-7.8-.7-1-.4-2-.2-2.5.5-1.1 1.7-1.8 3-1.8.8 0 1.5.3 2.2.8 3 2.2 7.8 5.1 13.8 5.1.9 0 1.8-.1 2.7-.2 7.2-1 12.1-5.8 14.3-9.9.6-1.2 1.3-2.5 1.8-3.5.2-.5.3-.7.6-.9.3-.2 1 .2 1 .2l2.1.8c-4.5 17.8-17.4 19.2-21.2 19.2h-1.3v.6z" class="st3" style="fill:#3a434e;opacity:.98;fill-opacity:1"/><path d="M48.884191 126.915676c-2.2 0-4.7-.2-7.6-.7-2.8-.5-5.1-.8-7.1-1-6.9-.9-10.3-1.3-15.6-5.9-7-6-6.8-13-5.8-21.6.3-2.3.8-5.9 1.7-11.2 3.1 1.4 6.1 2.2 8.7 2.2 3.1 0 5.4-1.2 6.6-3.4 1.6 1.9 6.9 7.3 13.3 7.3 1 0 1.9-.1 2.8-.4 3.5-1 19.8-2.1 46.9-3.4l-1.7 11.7.4.1s3.8 1 10.6 2.9c6.1 1.7 7.9 5.6 6.3 13.8-1.3 6.6-6.2 7.3-8.2 7.3-1.1 0-2.2-.2-3.3-.5-4.2-1.4-18.6-6.8-18.7-6.9h-.1l-17.3 1.2-.1.4c-.7 5.4-4.5 8.1-11.8 8.1z" class="st3" style="fill:#3a434e;fill-opacity:1"/><path d="M41.184191 103.415676c-3.8-1.4-6-1.4-7.7-1.4-1.8 0-4.6 3.3 1.4 5.4 6 2.1 10.3 3.4 10.3 3.4s1.8-2.1 3.5-2.9c1.6-.8 2.3-.9 2.3-.9l-9.8-3.6z" style="fill:#56606b;fill-opacity:1"/><path d="M27.584191 38.615676c1.2-5-2.1-8.2-5.7-9.2-3.5-1-8.4-1.7-13.9 6.9s-9.5 16.5-6.4 20.6c3.1 4.1 9.3 3.4 11.8-.8 0 0 5.7 3.8 9.5-4.2" class="st2" style="fill:#56606b;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M10.884191 45.115676c-1.6 2.5-.8 5 2 5.9 2.7 1 5-1.5 6.5-3.8 1.6-2.3 3.6-5.9 3.6-5.9s-3.7 1.2-5.6-.2c-2-1.4-1.5-3.8-1.5-3.8l-5 7.8z" class="st3" style="fill:#3a434e;fill-opacity:1"/><path d="M22.684191 41.415676c-2.6 1.1-6.8.6-6.9-4.1 0 0-5.1 7.6-5.9 9.6" class="st5" style="fill:none;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10"/><path d="M67.584191 5.215676c0-3.4-3.9-2.6-3.9-2.6 0-1.2-2-3.5-8.5-.6-6.4 2.9-7.3 6-7.3 6-3.8-1.7-9.6-2.6-13.5.8-3.9 3.4-4.3 10 2.3 13.5 0 0 2.9.9 7.7-.4 4.8-1.3 7.7-3.3 7.7-3.3s3.7 2.3 9 .6c5.3-1.7 9.9-4.5 10.3-10.1.5-5.7-3.8-3.9-3.8-3.9z" style="fill:#56606b;fill-opacity:1"/><path d="M67.084191 16.315676c-1-5.5-7-3-7-3 .5-2.1-3-4.1-5.5-2.7-2.5 1.4-6.6-.1-6.6-.1-6.4-4.4-14.3-2.1-16.1 2-1.1 3.3.2 7.3 4.8 9.7 0 0 2.9.9 7.7-.4 4.8-1.3 7.7-3.3 7.7-3.3s3.7 2.3 9 .6c2.3-.6 4.3-1.5 6-2.8 0 .1 0 0 0 0z" style="fill:#3a434e;fill-opacity:1"/><path d="M36.684191 22.715676c-.1 0-.2 0-.2-.1-3.1-1.6-5-4.1-5.4-7-.3-2.7.8-5.4 3-7.3 3.9-3.3 9.5-2.8 13.6-1.1.5-1.1 2.2-3.5 7.3-5.8 4.5-2 6.9-1.5 8-.8.6.4.9.9 1.1 1.3.7-.1 2-.1 3 .7.5.4.9 1 1 1.8.7-.1 1.8-.2 2.7.4 1 .7 1.4 2.1 1.2 4.1-.4 5-3.9 8.5-10.7 10.6-5.5 1.7-9.3-.6-9.4-.7-.2-.1-.3-.5-.2-.7.1-.2.5-.3.7-.2 0 0 3.6 2.2 8.6.6 6.3-2 9.6-5.1 10-9.7.1-1.6-.2-2.8-.8-3.3-.9-.7-2.3-.1-2.3-.1-.2.1-.3 0-.5 0-.1-.1-.2-.2-.2-.4 0-.8-.2-1.3-.6-1.7-.9-.8-2.6-.4-2.7-.4-.1 0-.3 0-.4-.1-.1-.1-.2-.2-.2-.4 0-.3-.2-.7-.7-1-.6-.4-2.6-1.1-7.1.8-6.1 2.7-7 5.6-7 5.7 0 .1-.1.3-.3.3-.1.1-.3.1-.4 0-3.9-1.7-9.3-2.4-13 .8-1.9 1.7-2.9 4.1-2.6 6.4.3 2.5 2 4.7 4.8 6.2.2.1.3.4.2.7-.1.3-.3.4-.5.4z"/><path d="M40.584191 84.115676s6.3 16.8 7.1 19.3c.8 2.5 1.8 3.4 7.3 3 5.5-.4 6.7-21.5 6.7-21.5l-21.1-.8z" style="fill:#191b22;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M51.084191 103.415676c-1.7-2.1-1.9-4.2-1.9-4.2l2-10.9 3-1.4-3.1 16.5zm33.9-35.3l-23.9 1.9 1.2 8 25.2 1.1 4.6-9c-2.3-.3-4.7-.9-7.1-2z" class="st9" style="fill:#191b22;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M28.484191 82.915676c7.2 9.4 12.7 11.4 21.8 7.7 8.5-3.4 15.4-9 15.1-15-.3-6-2.1-10.3-9.1-9.8-2.3.2-6.8 2.8-9.6 4.4-1.8 1-4.2 2.2-6 .4-1.8-1.8-4.3-4.4-4.3-4.4" class="st2" style="fill:#56606b;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M49.284191 80.515676c-.3-.2-.5-.4-.7-.6-1.2.7-2.3 1.3-2.8 1.6-2 1.2-3.8.5-4.7-.3-.9-.9-2.3-2.4-5.5-1.5-3.7 1-4.5 5.7-2.5 8.4 6.5 6.1 12.8 4.1 15.2 3.3 2.4-.8 6.3-2.7 6.3-2.7l.6-6c-2-.8-4.1-.9-5.9-2.2z" class="st3" style="fill:#3a434e;fill-opacity:1"/><path d="M28.484191 82.915676c7.2 9.4 12.7 11.4 21.8 7.7 8.5-3.4 15.4-9 15.1-15-.3-6-2.1-10.3-9.1-9.8-2.3.2-6.8 2.8-9.6 4.4-1.8 1-4.2 2.2-6 .4-1.8-1.8-4.3-4.4-4.3-4.4m35.4-8.6c6.5 8.3 15.5 12.5 21.8 12.7" class="st5" style="fill:none;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10"/><path d="M53.184191 104.415676c-1.6.1-2.7-1.1-2.4-2.7l4.9-25.9c.3-1.6 1.9-3 3.6-3.1l26.9-1.7c1.6-.1 3.6-1.4 4.4-2.9l.7-1.3c.7-1.5 2.7-2.8 4.4-2.9l4.5-.3c1.6-.1 3.1 1.1 3.3 2.8v.2c.2 1.6.5 3.7.7 4.6.2.9.3 3 .1 4.6l-2.2 21.3c-.2 1.6-1.7 3.1-3.3 3.3l-45.6 4z" style="fill:#191b22;fill-opacity:1"/><path d="M53.184191 104.415676c-1.6.1-2.7-1.1-2.4-2.7l4.9-25.9c.3-1.6 1.9-3 3.6-3.1l26.9-1.7c1.6-.1 3.6-1.4 4.4-2.9l.7-1.3c.7-1.5 2.7-2.8 4.4-2.9l4.5-.3c1.6-.1 3.1 1.1 3.3 2.8v.2c.2 1.6.5 3.7.7 4.6.2.9.3 3 .1 4.6l-2.2 21.3c-.2 1.6-1.7 3.1-3.3 3.3l-45.6 4z" class="st5" style="fill:none;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10"/><path d="M55.684191 105.915676c-1.6.1-2.3-.4-2-2l4.4-25.6c.3-1.6 1.9-3 3.6-3.1l26.9-1.7c1.6-.1 3.6-1.4 4.4-2.9l.7-1.3c.7-1.5 2.7-2.8 4.4-2.9l4.5-.3c1.6-.1 3.1 1.1 3.3 2.8v.2c.2 1.6 1.3 2.9 2.5 2.8 1.2-.1 1.9 1.2 1.6 2.8l-5.2 24.9c-.3 1.6-2 3.1-3.6 3.2l-45.5 3.1z" style="fill:#191b22;fill-opacity:1"/><path d="M53.184191 104.315676l4.9-26c.3-1.6 1.9-3 3.6-3.1l26.9-1.7c1.6-.1 3.6-1.4 4.4-2.9l.7-1.3c.7-1.5 2.7-2.8 4.4-2.9l4.5-.3c1.6-.1 3.1 1.1 3.3 2.8v.2c.2 1.6 1.3 2.9 2.5 2.8 1.2-.1 1.9 1.2 1.6 2.8l-5.2 24.9c-.3 1.6-2 3.1-3.6 3.2l-46.7 3.7m9.2-103.9c-.3 2.9-2.9 4.9-4.1 5.8m8-3.2c-.7 3.5-4 6-4 6" class="st5" style="fill:none;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10"/><path d="M39.884191 53.615676c-2.3-2.2-4.2-2.5-6.6-.6-2.4 1.9-2.1 4.8.9 7.6 3.1 2.9 4.7 4.1 6.7 5 2 .9 5.4 2.5 10.5-2s10.2-11.1 9.4-14.7c-.8-3.6-4.1-1.8-6.8 1.2s-3.7 4-5.4 5.2c-1.7 1.2-3.8 3-8.7-1.7z" class="st0" style="fill:#93a1b5;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10;fill-opacity:1"/><path d="M44.384191 61.315676c-2.3 0-4.7-1.2-6.6-3.1-.9-.9-1.1-2-.7-2.9.3-.8 1.1-1.3 1.8-1.3.2 0 .5 0 .7.1 2.3 2.1 4.2 3.2 5.9 3.2 1.6 0 2.7-.8 3.5-1.4 1.7-1.3 2.8-2.3 5.5-5.3.9-1.1 1.9-1.9 2.7-2.4.3.2.6.4.7.8.2.8-.2 2-.7 2.7-.9 1.3-4.9 5.4-9 8.4-1.1.7-2.4 1.2-3.8 1.2z" style="fill:#b3bfcd;fill-opacity:1"/><path d="M45.784191 50.115676c-.7 0-1.4-.6-1.4-1.4v-6.1c0-.7.6-1.4 1.4-1.4.7 0 1.4.6 1.4 1.4v6.1c0 .8-.6 1.4-1.4 1.4z"/><path d="M61.184191 118.215676c.7-7.1-3.5-10.6-9.2-11.1-5.7-.5-10.2 6.8-9.1 13.1 1.1 6.3 6.7 7.2 6.7 7.2" class="st5" style="fill:none;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10"/><path d="M52.084191 107.515676c-2.2-.7-4.3-1.4-6.5-2.2-2.1-.8-4.3-1.5-6.4-2.3-.1 0-.2-.2-.1-.3 0-.1.2-.2.3-.2 2.2.6 4.4 1.3 6.5 1.9 2.2.7 4.3 1.3 6.5 2 .3.1.4.4.3.6-.1.4-.4.6-.6.5zm25.4 10.1c-.2-1.4-.2-2.9.1-4.2.2-1.4.6-2.7 1.1-4-.3 1.3-.5 2.7-.6 4.1 0 1.4.1 2.7.4 4 .1.3-.1.5-.3.6-.2.1-.6-.1-.7-.5 0 .1 0 .1 0 0z"/><path d="M104.284191 103.615676c-3.6-.7-8.5 2.1-9.5 9.7s2.1 10.7 5.3 11.6m15.1-83.5l-.39999 1.4m-1.90001-1.2c2.4 1.7 6.4 3.4 6.4 3.4m-1.6-2.6l-.60001 1.59999" class="st5" style="fill:none;stroke:#000;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10"/><path d="M47.484191 79.215676c3.2 2.7 7 3.3 7 3.3" style="fill:none;stroke:#000;stroke-miterlimit:10"/><path d="M69.284191 24.315676c-3.5.4-2.7 2.9-1.2 3.5 1.5.6 3.7.1 4.3-1.6.4-1.6-1.3-2.1-3.1-1.9z" style="fill:#4f5862;fill-opacity:1"/></svg>
\ No newline at end of file
diff --git a/app/javascript/flavours/blobfox/images/glitch-preview.jpg b/app/javascript/flavours/blobfox/images/glitch-preview.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..fc5c4204359659e32ec8d758855891e297f5093c
GIT binary patch
literal 197277
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%0AZf$5*<oC
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
KIoqB2DEv269CTIy

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/images/logo_warn_glitch.svg b/app/javascript/flavours/blobfox/images/logo_warn_glitch.svg
new file mode 100644
index 00000000000000..0fea63d2908969
--- /dev/null
+++ b/app/javascript/flavours/blobfox/images/logo_warn_glitch.svg
@@ -0,0 +1,49 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   viewBox="0 0 216.41507 232.00976"
+   version="1.1"
+   id="svg6"
+   sodipodi:docname="logo_warn_blobfox.svg"
+   inkscape:version="1.1.2 (0a00cf5339, 2022-02-04)"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:svg="http://www.w3.org/2000/svg">
+  <defs
+     id="defs10" />
+  <sodipodi:namedview
+     id="namedview8"
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1.0"
+     inkscape:pageshadow="2"
+     inkscape:pageopacity="0.0"
+     inkscape:pagecheckerboard="0"
+     showgrid="false"
+     inkscape:zoom="1.7951831"
+     inkscape:cx="-30.916067"
+     inkscape:cy="90.241493"
+     inkscape:window-width="1920"
+     inkscape:window-height="1011"
+     inkscape:window-x="0"
+     inkscape:window-y="32"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg6" />
+  <g
+     id="g2025">
+    <path
+       d="M211.80683 139.0875c-3.1825 16.36625-28.4925 34.2775-57.5625 37.74875-15.16 1.80875-30.0825 3.47125-45.99875 2.74125-26.0275-1.1925-46.565-6.2125-46.565-6.2125 0 2.53375.15625 4.94625.46875 7.2025 3.38375 25.68625 25.47 27.225 46.3925 27.9425 21.115.7225 39.91625-5.20625 39.91625-5.20625l.86875 19.09s-14.77 7.93125-41.08125 9.39c-14.50875.7975-32.52375-.365-53.50625-5.91875C9.23183 213.82 1.40558 165.31125.20808 116.09125c-.36375-14.61375-.14-28.39375-.14-39.91875 0-50.33 32.97625-65.0825 32.97625-65.0825C49.67058 3.45375 78.20308.2425 107.86433 0h.72875c29.66125.2425 58.21125 3.45375 74.8375 11.09 0 0 32.97625 14.7525 32.97625 65.0825 0 0 .4125 37.13375-4.6 62.915"
+       fill="#3088d4"
+       id="path2" />
+    <path
+       d="m 124.52893,137.75645 c 0,9.01375 -7.30875,16.32125 -16.3225,16.32125 -9.01375,0 -16.32125,-7.3075 -16.32125,-16.32125 0,-9.01375 7.3075,-16.3225 16.32125,-16.3225 9.01375,0 16.3225,7.30875 16.3225,16.3225"
+       fill="#ffffff"
+       id="path4"
+       sodipodi:nodetypes="csssc" />
+    <path
+       id="path1121"
+       d="m 108.20703,25.453125 c -9.013749,0 -16.322264,7.308516 -16.322264,16.322266 0,5.31808 2.555126,37.386806 6.492187,67.763669 4.100497,4.20028 15.890147,3.77063 19.660157,-0.01 3.9367,-30.375272 6.49219,-62.4364 6.49219,-67.753909 0,-9.01375 -7.30852,-16.322266 -16.32227,-16.322266 z"
+       style="fill:#ffffff"
+       sodipodi:nodetypes="ssccsss" />
+  </g>
+</svg>
diff --git a/app/javascript/flavours/blobfox/images/mbstobon-ui-0.png b/app/javascript/flavours/blobfox/images/mbstobon-ui-0.png
new file mode 100644
index 0000000000000000000000000000000000000000..25e1707c9932ea9091d2d794f3ca78b7f6b45978
GIT binary patch
literal 39646
zcmV)EK)}C=P)<h;3K|Lk000e1NJLTq00Bw>008R<1^@s64<6t800009a7bBm00001
z0000108b^v)&KwiBzja>bVG7wVRUbD000P?b4<xkN>y-7D@iR<a7{}~O)e=006}^N
z&T4uU#Q*@BU`a$lRCwC#op+dAS9$+G=iGbePM_J{w5wfp$&xHv?nT(f7+k;xLkO5(
z=%f)!Na!8om=Fj!p*e&Ap@dK!zy^#fu3*`+Y|FA_wN>5K_BQR_dwzeMGdr`Zm9(p3
zwRxXsp0j6X=FZH$=YG%k?ShI_q#{RHY@peOwi{s42jHAmsI#F>z?nYa19@No*bZzn
z<6dA8=rezJ13kc~dCz_%0p!d}lzCL-NQ$VOf{Ik+NJ|aSuo&jw3M-e`obAWCFgZp7
z57Z<mD^7hNFpoSiY=E>I*kizSw|Q(gkD)SjoFvr-UxWR>q#_ldtVl(UA)R~|@BzU^
z@@x{dXE7U#bB51mWnu-(S*BquJ5r*FLK!4erF*gnpn(}ZAPe*ZJAmhbO=Y0D%dpSP
z!D(fSA+Gr{e6C0ZC@WHt7gm}<egu}B7G+7kou$={v@?r3=5QXfNwXX{^8g2Eg2E7}
z!X!SZCJf^!=K*5|M7w}a1EyVujdo8DKr1ZtK)YW>DpHXnE2|)Q1?)-AVKmw73=GOd
zf2ww58EwIK=J{#b4}w9v5Aa+SWJgO-rVI;JQ>gJ{KrgTp*lZZ+7BlWLw9+eKa9`P{
z{1@;P@QlIQO{X*ituV@pROHA=1h@tGGYP7hHn&xr7G8XAi1SvSL7nv?+Rj?VnJXkL
z18UJ$$e>KQ=Fmu73^Mt$!>EZzD<WGSjfA_<_RcN9R`d5Z!#?xoeUp$d0PY9w1D-UX
zxxrwpR{_e3RHP!)meYXmgEO~^4!@qWA`5v%=&V=+XSc85!p1a-7GMrA8>XXsGYk~U
zG~pQlX$mqK{WhT>Fs!qPrNyyebnPw!oNb1AJ`X%?9-V00Xc>480KYYl^@e$lozzUT
z0+bc0$dQ$|LgP)4jGfOpvBa_~vK=i8!{<s^xyWMqQeZx?L}-)X1nZXo<N<C`SI=vJ
z5x|q698eK+9HIH;KF8PxgU1H7{Zj-J;jIXu@iJKU44Cc#HX2?0Vc>Ce2$QGZHw4@U
zJYZh(<Vgjd6`-t0MUI>#fv+Ir*U6S@v9J|++F}t|elE1v6{u^2^Uj3yK58un+{*u+
zfh3BYPfT*x2q08;n@hkcJAQHQ12!qbBN<|NG{zx<BC5G5DWxfHJy0J&a~{$IhJ~&M
z?l$1rThhgM1Mfrw=u-ySGbb!?R)Df16*;o9lxp!w>QfC>3*8{GQpovd6Ip&?lw}FQ
zStyoG0Iq}Gl9T6h67cOus)QzVUO9MLQygpR&$^X<o&?q+>DBtUGFAX#UZ+ZXgQY-v
zU=2WY8$_Ol#AA?r4xS$Z+n{6KfaRb8(jkMZ$G>;I0+bc0$dM5Vd;n(uBe<bign}g(
z2q&CCp4p*7FFFTSt*l2bpZu00t8>d9Q6OwI;Iw&Eo7YtXRp!=>n*YVk>#ga1(`K}p
zv1tF?hvbbK1~>rp6VS)~ba5vGJi!248DS$?<uisN2jh<d;1wN5dqJ*;;34S#EwC2E
zSV^PL8`Zq{S;seER)Df16*+uq0sFgf!8taK%qHTUBZ*wQDDF0#*361!RixXKP}dCS
zbwIkt=*!2MR2l3lP{Ok*WB~XL@Jrx1jeu&3pdrI!hqKVT3%0C*U3bE^yJ6Qh7%?h(
z#%SzMp~FB66`-t0MNU!?z;)2_bqFPDS>~s?t#Eb?;VTnF7S0Ejg0&1d3s_dV@eaY2
zGl7pf=(wcmyK<%}-!$0uwC09vG$e38CdMJ)r@&okm-*xS2TvdE&)<T!YyQTJeeegv
zjL)e6<%y$36>UM~(G-<Tg^ElmVGuXMyDkPqlxt*l>p8F_i3o!_1M)8dR_%*oN5GXc
zVbxBSP}%>*q`9R(bJ&7}Z6GRU6#b$yUjtMJ(0D&I-v#wgLi1KguRpN1yj$-a1i%Ie
zGMX!jecv%wfGxeS^=Y`~KHyJGjC`*E<?$u9St5;QiKNW3vCEd11b%Ne#2lweux&-A
zBdMj3{w2=xHZi+i#LsCE&MW338)n0bS|aI21W_aj<6|yu!3O}lUPoiqgy*Mn<xIva
zYl<6O9^~dPpYJFE=|0C*Mcw-js9g`%lNtrBKwC{?253{jqkzLOu$eqP6xc?NP4rU;
zMgheTif@$v3=bXGc`xu?bBK`&P#zCtfGdnnlrj26N@%4fT)yn4*gHRFl#UA1JcY%A
zg?|C@e+8ngIF{95ODCB^oFn0-VOW?1YG}5afJJ((iCQc`*m@fd0j_|s)L;`aV3QVQ
zt{fMtMVTu{g5y9u0%04hBcyvzbmeGzw;Nq?W5Cda4BDKT1#<LI7<!Dt&(`DJw;R^?
z!V^QtLY5#1*v=lnLD33y5bz~@^5pgVRWUKLERS6oCh|P+4dB*e%}Odj2|%P4{^E2v
z?Of>I1Jy~eLuNxAHddM)=qfFJ22CfS<EPD2l}aq9kTd}Q2KH5wi?{$Q)QIau;|i)G
z#L3bkph^jng&55w=)k9jl;M?ipk`@w6CI%1v_>P2(49OvY*I{`Tjp;?39aKs-!AIj
z)ke{30vh%}eJ3<;f%J1|Gh?@bOp6IIo4W=4MU3HmJ-zEJwhYZ<r?-MF!8L4SaGWA$
z%w6*&sLVt(R7MTI=y*`wDnO}|$=(cH1!v8ISGzFM3vr<@yj@<YWd!Y_+m6oNdR$*)
zc2dfdHK5r$1=AG3QnR3*G5c{vPH=fSa0|55H)APD>N5)kvwokn-$_V%NMVBo5dqTH
zN_S8NmZcF%MaxV*DNQlQ%|0$zX%KNx76<_%3bJY6yI1W;hnEM?z4xb!3t-QO!Rmvc
z*Ko;Uy~Hg**#;Obs1f=*%Q_J45MKb%Hi$PsDh^NC<`(LP!dA%Lpp~$mGG)ws_d5+H
zpQ97Y+YGCG9q>d2D36_lfE$310JA|_a5=C7fi)zk77|N>9R#{Ui<(qypr{^fG1vG`
zU~S2s@;)YpI?P80uq_4_Qu1;BJr!H$2`f<$Ux2pP3$WjX6{VFst_g{`=Qy$@DTx6(
zV+ISFin_6-!O)T*(h!On-8%%Xgjkc-!Q*ovZ9zH>)%8aAjzX0U!ZKl$mSKO@(+;CN
zxb8g{Mg$B<qk9Y3V?%!o25Mor1;#ocvk<ZiAXfzi8}b%-R*6?wYr+39?Aig`4$|B{
zo}nE%7|X(zQP}FiHHzoVuCW^JbN??xN`eYd9y>V$ot)Z&-m6J~N??mBE*eTuor)4h
zat0v;f)EmT2>M3#HI_l?92U^Iuy>)OB%WY`0s4LBW#?#?!1+eMnH&)E0i%pm<OGqW
zz<)z{K~mOGbs<kKZ=|+9B5E3-dTs({H%n@(1dT~av{{hou%IRmiQ;vyGWu}JtcRlS
zIaxNw%~*6e#!HNI{BfXLy6O;q+Dp(Sj%tnG-Hfj2(PqBOfj%#bj`;Zr3~yw(7dCI%
zO3%(gwrne~eg_0Y`qvEQ;2V8pScAnd9v;sD{|fwQqLx+x%3~p~1HJ>;4mzx@!=POa
zF_OFviv%vM2<-jZ#=iu6Mj>J<vv@`Y|0VE4;90#Bezpbnz5)`b!Pz-Db0fqbfb_Fd
zQ^|igp4P`9iaObsngsp^_#`w(J8&M25xXR4QQIAvYK<?g%B5~^buH!-W+_R|poRv@
zTshPp)vwnyN%0ZpVN>#(7b7`Bl(KqKW&{5cQ~S-QmLA|SJl3<8-TaVV)-Xhl47Fr=
z1)F&VFi+P40sbAf6?DYCV+@>s0)D`~+)0d{<1mc!lxCDy0KWjjAqZB1r8B#(V6kk#
zkt<7VZyI5gUx7UX;3&Pq3g+T}8Q#AZ^6!Twm*}1UN(nU%SOs)pzq^gvzOnQP@8-nv
zq^nbKnTb~D_z$SNSTdVcq9ba(%f)3m>YAl^=Pdk+ljLuUzdokV(i1T_BVo>19bjBJ
zbQrGIc)-tWDAPjyGR!Uq!6NushWXVPrO7o%?=X+F`Q*ewK(X|{JxDqcY{B9=Y_?$2
z&9jX0eJt)L#FOUL4gy7t0qTv8)PnZPTn7)1LAyccEMq*(7QW0@p5~hIT*eAe9%oXo
z^#d+IAm&vQoLz-InpnnygoP9R3oKhe$SSSJ`_V?|-2N>mUxDod5H?J6vwqj#pC^cq
z%;&nCO>^#fY7=?v7NbWgvoSwxRF20rt^63efWWB?md7(YO<xP~uR}83oUHv1tE%~J
z>E&}G)OTPvbRg%>Ma+(1)ixj+!H$BR(2B5e<uJ*WgK_0-j8Qw^#WRX_fQ3RJh+@%%
zWnLLGN2xae*=*Qk@qbO`QB0R~*njk7(H*Y6*B?fEqbIp?z<jcH<qRPBDmI^|jse<%
z1xVV6(7-5h^IA0w-J4-+58SYo9-ha>BZuHIhPZ`q@Hjg1=(si3pXxyQavZ+N#nf^d
z>a-lRJ4>xu$(NI|BUs@$7E&ACr;^VaOXL*y^3AZr)AvCRJ~7-*VOt%|?p&5T9V~7V
z)U-<^H^F0p-rl7_#e0G8p^e!E-KXkA2FOEQX2&&4jlhC`g1WZ~;#S+d#t88xSZXCW
zmm*w=OtTHiY{DRtVAX(Vg4m)Ktg}W*)%wU1uk%4!4gw276BZJ|92_j-q^Y5v7%nNI
zv|=+0i)Nq|i&h{+^|<;h4sgC4^9UXA$IWG;^<0#}axFu&@HIO43=0`$0Z`Qg_G4hJ
z2Wz)RSAPiBX5hyj?AQ*sTMVmqMwri|+`?BW@FzHa8Ds@0_mwaa-pwoc5OYb&9)d+}
zajV5{+7V&itVP5T)Z@@rw(Cv-od0b0#9&I$L}*(mA#S>71+^_zBo+v+K@uHTeZBzq
zX0(14LdS0Y%rMMOeWVi^G{tcS#D^_Q!1*rBSrU><MBS<j!&n{QHX+pc<7q9DDuft8
z#)1Lis<dS(;z<G-*pfPjs#zhNm?YKVAlf2?8)L+%CP|nW36eAuXBJ6XuxTbu6LBnJ
zq;YX@5&LoFSQInCkC-cG543fzBe~@kc<%eKWdt!r7-bfCSinzWu(4jBhpn?296X0R
z3Rm+L#(0baW{y{Y@|fbGR!ORiwp%PwrNZF77rYO#v=DP&n($sR?1y4aAsRvw$Fj?I
z;M2;0GHOoW6fqf}H+qYLuc9L?V!$%NU8gPP#W9DiiDzk_%4hRAc(9<?CBQo1Yv`K$
z-ti5Ts{AjhPqgq1YbchKJsCh!0B?fy7oj!W%*wDsGvSu6<)s`BrP2Xa1hm>PU&2xW
zXIRisq-8cx<$$#uYSIurM~C*L>UG?q9fni`#A?8e>C~LK)M+=x%4$theWj3j@}RDq
zX9Ua&K}y2MLhz9=ge_=J!LD&vj?ukEFVOQd7`_Se&ugWpER3>vV6P6=w!;q>z?wG5
zH9&0~USY#46wK?!Wh;4ZI6*DaCm$e&%u<;P%tJRQ!7QS@F=`jgm;+oJGB+VZf5-V_
z&V}=%R&ABwv@}RtnY=KVabDO*HMD&WZFZkKm)|=LT)t`n8>;5Bz$)rCAKrIg{VY7K
zw8?5O@NM9`@QFbUxyL0$+J=0rGLJLSDH|ULo)U-bnL6&&?<(NSAYUy&v=fBu1hltl
z_lRrh?1iNQ7RpIq(*lHqSV9Mpr|Ln@hV&AMS$ZoCMM1jI)S;i7<AN2@U|A3Ku8B{#
z!8iM}v<y~NgG3++5hA+Xg4Ri{oX$pQD|6-KV&G2c%9)D}50yVc&<$b;><luNMbv`T
z0(J-33*p%X@ck;d--ahG3RsLf@O$O+iGj6j;Kz8FFv=4-(^s54;8?G<A?Bg|hedGA
zXp<5d2@xAIwwc=y37P)~kg0{pe<CwyFIX9ItOj^R800d8m7&A<dLM(EfcTm4VwW}b
zXS1TZ+8nq1hWZo}!YIEC4-9K|nFW3Td>#J2_vn~))L|_PrBCWtfLE8%?JV$j!2b#D
z6F8LP7K002P&ptc#<!MZvV1QBzJ&%N6^BS747D{Hkmdm$LW9v<AR-N$EII?NIK(ZG
zjV0(E52Fkq>_DuF>ZHJrg%HUwLA)82Y(m&=Sm|1<cmz9k4lx&MT&Q<>K*B}M=$Q5z
zEb4J+BBEV6RqO@ortix6IK1SiP`evyEoiEN=6Nuu5856c52G9azAE7JkR349cqKd$
z$?~z$hk1yf5U|mFz*b{{FPxYNW1qPA-hke}SD>$45}nQ;D~=LI>mYU(jGhUxb{I|T
zEq&B6uk#?5h0$)kt&h&1ABs7bEP_ZJWQ)ltI~@4}+~Pwx2GtSn9$Us`R*22QV$MFV
z{e<CQMkRkYTy&$({IN<D*gHz#c*sYv42%3fbMir<j9v9Lz-j?%rkG{`@-@I)jESrh
z#IJ$dl+G2&9@_5{hjXud1f4zL%!6<(kZguV2TqH?N&%OqKvn^*=wz&==+uS=y`eYZ
zFp7nbLXJCFKw%pdIfX?SL5MI3;#5&XmKeSFr0Bi0mAG-`+(|3dv~o2eTH3<6ScJ62
za$nuMP6%*pa3q8d7)DtQUy|^n6!2mY%M6+i19viHY(bxa+^GJPJn#%2H?x-8f*-=@
z8l$Y$0<Q-XMtMTZ3CbwL=+NDjzykBXNRd4zA#*WAu7u1=h_pc_4iO77%CJYJ<HX13
zm@(ke2$A|sy6*IdJ6o0lFq$X@SIkHW|F~AcS)ifr>z6}SHTTCXR$T_W!qB|W@jf*5
zJ$|E(wUm0Te&p*gJbLtjQcU|WO6O*N2R-`ep!;ti_>nk7E@AAQKLM^Zw{>9L2X~?K
zKMO(!`oB<s*#dkU!9{ils-q6?wAnZpbr9}wp=mjszto%ojXlF-x`?eP`#rWK=C~B3
zLpv5W4w48lVmMSI2@`Bf&@sE2S<E6w3vrqoqBI3I7K*fQ;~)^ysPy(Q$OOn*2wjF{
zHQ*#6GUG7H2J=T6I32_iOuI;suui}&RuA8||0y?O7ZlG@;QKto{oKft+yMmI^lvEV
zNg^j0P{s^6o{bLG_0r(J6mr*rdnV)>b;4{;7y_oi^&yvaa7UBm3Q63#7P;LK+$NhG
zKJIfqxs1eJ;gdt)l3bGeqHVc<FJH?)If543Via%}z*~%Ayw8Pd$qdqKKWe`IRc67~
zOoPwggpS!P0p&Kd3VX8#kHdBD964u+QrPH3_G)8W-66`qy*Q+A=b(6zHeRs8?1mD-
zZr}~T1ELgpeF(pE#$W0@bm&%d0+^j5I4h3RoFMeII);avQT|*|8jJ*U35XETfM+<?
zFwo>c=roAdKyVHrS<R)?^B&^7jIe>Kni_~1bd`Z_&OL7HoDfDi3sD~>ByFP29ne${
zHT6(m4NW0vk|vB&9wdzNX44-mt>cDy1I75upPK!CtKOjkt+M^%aR%=GSH8*}e4hL`
zC`xmcMmeeE1W~;sMu)j53azUkcq52aAgUp-O1<5M=MZGlh+GpvV?Ba8g24b02@%xf
z5jX_+h$e@Cf}StTA+QL@Ay}{|STC*+e@_H}D-o=i!j)4!qpyFn+0kp!K45E_wDiNk
z2lbz~z(Zzt*bA&FS(c8b(P#Q>QfS`%)Dhc`^>KZ)=l|R2iX|<;!zgY+utyxW?LbZU
z3sxES`q$_zZ8frh!qc^irxWC9Aj4?{ti<C|kwLQ8;L1jH!od!pU&5yo&9o&)iCR8X
z38X`wF$(Ne|H-I$8LrT}PnCeYt4%<n#(-kVsPwapF`^Y{Euca!xo|UitCf)GAUfVO
zLQcE}2MnXk8>*>*m+(&{_@UXA2aT~gfPf*^^EGbZGvPH;f?;s-f#eCmC<QtZ|8k&X
z0l04hu0^}Di{WERfL9CoMo2a4Rh<Leu-V4l8IJQ~K!c?}Iz}nitUOEXP~~6EZI=LE
zYW9HX($WvYP4G~lNykq$(|T9_NPvYZaRE<2?ocV<5npL)!k^(X8%u5+D8=D?v1xy~
zc67AY1)xHJhfRUG<XKEV7qOKNo}-D4oG+e7r>E{FO_os<J_@kniqBZ^-fElhIB3Zj
zIN!jIyc8=|OS&4YFgOVaH$%%TXl{YFW~gn_KzdHPXbY+}i?r1&&=pv?I98m96W8V4
zW@s?_Fz{nwmsy%|qXZ2D&#%Mh7JR;c^X-i6dL^fb9EXfDg?22S9ftH-;QbBQmuod!
zNKIVqAdQ!6!;a0yBaH1t@FF>ETs(5vt{E909gkeKjhpt!g>BrZN6t3Sdp54;kyAGF
zLvmvG2hVW0y+*MVW$J}s=B2IPdw6HZWbq8cyME1KR?kqVgt28@W~}zzz<&aNJR%@@
zxZAMW=WtmjJ8{GT-!60^M?z)~oD|PF?Jw8>yd8bMRs-8ev57XGV<j7CW*eVngkb`F
zq4tN%Avo88H(L%NwG2cRIOkl5UEPMNzX3ZA=`g4UaBCnEgQNf@VUr0PD46AUCs0il
zSJOs|I1e{eODL*viBSr;WX21$SRg%$jv8(UB0q!dr(wgUQ5gAfU~26C2d9u6XN*#!
zoivvLb0v67!ToE<T@P+`(e^0Gd9!hY4szi-!vkGha^!Fwmt1WQx7{T-D~CJFB`3`P
z{j$+^$$2?k+aZ_F;#w}bOcuAnAtyr2pX#d}OSyTXar{g0`w`wX2VGR-FFAxP2c=1L
zfDYmL8^fC|ZHnUaz-Ps=jD9;pVU)%*zXa4WbYH<|1j0XrZ)Ab0MiNCaf?|*fT?`KL
z#iY?BlEh*Se6P^XrEN<{VI>I1<_?CV|ChpTMRY)DHq^S%7=f6k_~XilVq&tQq>2a~
zWW;=ygcfpHw3-GEE&jLIpvV0@%>E&?bVF=}vNpDB$?#2n3InD^&aZ)2RT$+9CJEqD
zbP=jh60B<=|L1x)54Hi@LIlBFQ0)YE6`+iRtst~AD(42!3bHxS>0{9VvLM6dV}J~M
zkhPHEQS<*n$O>e940=Pz`r|N`;e%y*Q2_6OsvC86^@!Q{Hy^@L3&!4h9*RDS5$%Ej
zV+R!+OP~}p1kM8+xnBq3LSsR@Q^5~~t|2-T@qxYj-;%^Z)d759h|y7(TUkSB+ZwD(
z8LoXVW7(I9FNAU4p@YziAv|GYk8DPV2iKZkV@i=V0U`QvsG?a#c{aD4yFH(@6T*ow
zZMeMEEQS>P$@(Z~uly;5cbhlX8qIzn4{L5Gg7&OEt~DB;Q$UU*pmfcC7C9H(8)5V!
zh`BI|LoArZsG3jAu3;1*V#u0Ng2gd{1*3>~j373O$i@i5W{ggnher|FFhO_#krf2t
zortVu{_hfmvwEx_MzG#6+v8Az@{4y%!Q)-Ez{`yNY^IDYZB$YV9l#{f6{UXy+%Ar{
zTwF2Z-&d4TbR^$}2&7S2u2ecEvQx~cM3X4D>lSrf2hC8&4(Nps8M~sUh3gQc3N~{*
z3-3SVhSryfK}equ_L5Tnnw0stUFcFiMX$Hky~$k-i56&Z`)Lj}G0XiDE@uLBkhGVn
z;0NC^TL98zFI23|qzo7@M29Gk0gr~zp{-voD)hx@$qG=uFwzKI*8=n20ruM<Xap$)
zs0L(yDTJ$$=F93ZhoDwM69iS#xZ|Y3k*P+HfH9z-fI+|}APcwzCMvgx2*HpASpoGP
zhwr&LN-8+Ojqt${&NH!<%MTHE`cx^A@-%JxWFv~sagaQbbGO7eVA|-cp@k?GqYEak
zG`rz5Gv8l?WtjM%0aqKYw-dNo!M*!6SLDmA${Ds(;7;0k`y&c<s{=ieu6%92KZLg$
zaw|1RCMns%qh%&XqkF5{VMIdQht1Xq;kGfH%kCl8u$?eZV{si|ArE4!{M!<kUkK0{
znxx-HflGb3^l2D|4PU2^0&$MR9^5AzP)guMQ2%jw#re<?(!A0O7DLd1v|}IwWLO%&
zlQ9SuN#qg;nk7;}U`wQDp0_8DJ^`ahdz64-q-_)6BW;TSVP0n;Z81J?hq8WRs!X#G
z#}#Pa_)(tYga7a>aLArVRmxi1AmF)^ae*7ww-TM<<Dsk1-;83t5HnNe&9OcMywmVK
zX~@{W1D`}kU`*$jXE4gGK=Zj<Xr6N!+i!Re@G?Shi^1T31}-?@L6SgAGc1chwH<-B
zERlLP$wz{@Zq;+(T#1yU`jrP;x?j>~r!!?%#8vx}QIBDM!2P2na5$Uq^3dVBCQlqW
zmUn&ti4<Jf3~i1kFa=dlfm#Z79Dz@Ojlg#T>}3M~RRQ*k1%7jYEd{;^u+>zfkF7HJ
zVmG$x#uua53iG_g7nm_XD#!~Tlq4qtHY=%fZs|fhfrpKTQ81wCb3ACIABBb3`31nQ
z4ufz7LT9$9BckdapQKU=ctpS((ann94Qv+0L`yZZW1)lJe*|PvcqjylSE9HSMMmkU
zgoDWw1_i(WXn`HSD&Q~wrnY&&CST_jd<R`Ls%yUo83LXyg!pYp@P}DhI%n}fejan(
z1`uw5uzw5kd5}A%r+c?TThegk=v1s~CoiL*-Rfc9feueOCFEE!$`IJs!bM+z)>+`;
z;;H2z+H^q_1YrW9V3tL2fr2KBz}5b~00qJRMxTH_2!;soATVlmAPG<efxyFuK<Mx`
zGpaz4MPUVkIVf~wxG=^U3FBcGv9i-<9b2Y(BZ3be2AmzpRdW*7l@b$`Un5~=SQ^nn
z@l}IhvyIAq8}OI_AH~2l&s_msZ}iIHN2t!+3Sa1<>m9ZEh9$J1WB1QP2is1TVzC#!
zZx0SbaunPRT(r=IAOubvh<P9$0?`7p^}y@&AE!WJCNs)iJgimiBu~-Bp@fm0;8vVt
zz+}bi;q;qz(USy80<{8UOHARM4Y?)YrkcqG9<H>fIa=g`L0maVE*Qg=f?N>bN=YtI
zSW=J)0t!+x6a)lPk_iHADJhuIPX$;=t{|XTuwVkdBaGHnOgk<HZO9jvq<;|LJ={p}
zH1PZ(so?752F87U<9=8FYy+5A18+wM%<lhKs#M;wSafG}`oZs$<a}bZ^Kblsm(@Z3
z-(dD@!K|yx4T3oaG;7fw9V5j|owKyIUI+M5SR$Y`!ZR&!TEr@Tsij7rZaVN9H2~Q;
z)pOZ^{xANFUvhuO{Qj@NsiA%!14h{byyP6H{sgFI2uL%k&IH*g17r&Xvlc)i2fozy
zLT`T~VCTT=1-no86t?r=h0OC7*dBprGi>L{c_D^v58>GqY!BPBQI?0w+2DAPwZZcc
zH8#Gf?II`KRRsfzMO9o-N)VDIi^5R{%Xe$H!3W3x(R)H|lgEk*?6Wb%1vD#da=gI+
zQsSTt+<;<%sq#nF4A1MsjPf-9$LYLi@2`mVw=%SvCHa+vYT*uaQ_uZK(N~|=SU)z0
zC$sZtf3%8`J%X<1A^Tz$th8uE@~Y{r%j8sSG64Pu6@G->e-8W)j(OGBDysKUkuY%9
zT!_B|a{ZuICkfP<1hOvX!#NNv4Zs3vnb=1=0B@?%122QLyUOMk(#FhvJkMstR*ZTu
zVk@KvzNN5oAW$#@(pQK<5JH(i8HK7T!kyaW>#~AnpqZsq>dZBN$Bp_HGN&{+%4T4F
z$A3;6vE~B5p&~Qg7%qU(MX<5}@!eo=gv3sLLDfX{_GoWB!`x4nU-JbPefb^atCy@B
z`#jWLa6r^9lL3gN=*&idB|H`Q#5RWrUId#n?AjAYH6H#Z?vpRT<99J;7WzZLh6+#~
z70ISy+fC5&Dv*|oQp-@X4j;k+e%gbO0{0=T>;;ozASaIwL3po;g&am?BLv|LA{)gE
zkCDwr=nrSeWFz?D3}gcW$3+Z!pj^Zrmw*gpBKUR&QHT)O8AK*R@Mwmvb0dUcTL@O_
zY(_;0P*@0*S#CM=$eOLVI2PkM)KLP;2B3?=&5ZFuY;@Pgip=Z;egxF%@Iw!F`_S13
zpLhTrL;6P;Gi<Sr2{niJXV*PEc0Rxco@anpZ9X3{vV@WEfIAHJD?m0dp)Jm2W6<UT
zNCt`0$fJCNpR$~ns}5dz|15T=q9ciFHfCG++De^3({S)Lxeu$9zzy^}7Qxkz<oGrL
zy0sn^pgiJRd3^|8@(!qbIrx^uXCcTM1YVs5mjH7@x~7ym2a5~=F_<t$fF1%yfk6V3
z!W$3;vIImZkY#`fW90CNpvWP;2x4>;HO9D)Miz2lM|9C_6=6I{Oy)pD81X>}US-RY
zD+i;P8PYo5LR8|TlUiI#&Yz+hSKP&qqz&^Um=bFdbOz0q3ii!NYSE=~A|X;E1$Oz0
zGa6y_t6}Bkz<t1f0jqVAzc|_^ivai?-Mlv6&%Bn^$h9jG=Rx+R+7dZIqOd7rI>DNS
z`z-KtM)(aGdU$|M?4&)YrsIjLSleKoF~7+QBc}tB`~7;qWXOL6HX+af{P&zn3&wf@
zKg-vJ?y6n}v82KxicO;07S`o9ew%|YLll&Qr+UfBMxru@B8-$d6d|M>MNvS?K@<*B
z=25sv>1n<v^JH)tm3i{G1S*dja#6i`M1Md~2%-D}N`z3#M#(%%xO!A&qsTMRWRV5}
z1-{ZsSRnKY3Cb*%vTUyK5cuYqoLNuBB2_t~%8iw&=jsp*TL4$`z{jq<Qw3Kpfrkdc
zUv~mEwL8|b7i|E%+$UExi(XMfckpNIPJWrQTE9iC0Ub>K8hwv^atnMit9x}EHR%Dq
z%VwON_i=IWB2=!GTq`)cp>Gz%YhddKVaU_Fu#fh*{~q{XBHTxYJ`@npLqFeRF@1xz
zT(`^JGtwoy2kXdgim<44Ix8)eD^FJvz#^)~!l<9Fx4IP$x_0?36-IgFqzU*YWL^dX
zqMAOnjGTy3kBZp@t02<|1c44;M9~NooE=Ejj4$#?<=~4PQhE4d2&p8#D1frb2%k|U
z84w;Q8((->zD<zvkOMaUu!n_>?@KI8gPsNWY7|RF@O|^V^01U1+m#Jn9$2<EtA;>W
zkhe{vZlV22jyZwlMTCNJwcP9tU;uN>u*_n?KHJzfi^Xlq7o{}?btxV7M)(8JM}ZFl
zdvpx+WO+Z+>XvZoNf@2ld_@$p4KVg*sQ+{Bj!SmXK{s$~0HHB(?*o3vcvm-7Qh1Mi
z9ndAf*V#nF;$cSI@38qd!z}vM3I=r>37#{|@piOry7K^ErJjIqaW}KutFY^8>3T_P
z{(Q+@=5}mDXTwb`om241KLQ)}URofiSr6Y_eyS}R`vo}uJAlg|B_#7z2Vv>pQ;?&m
zGiwOyE%3niCN9Om!NfpH2u2AQfMD3fN`i+V8|Yv*KhRvx4wMd`9#8~B0_o}aNKhzC
zK;WZ97<@AZ3MIspBf|nEjjApUrUBVF_~xvIEyxJn_cJI{xJ$gQ2%1I1R=(_S$1qIS
z%qK(c0gK8upbOeY%Pawt<e$y>S#;pMsz9W<guwm4Wf^dnhwx(_5gEW-wgRt4@tck)
z*LP*KEBrKcGtGxfhsc`+aYQTBJ_mf0Jm)^!!eX9cKIc*3rEEewPJX9N0gOz;&(sY(
zgvadeKe0JzWHBwQ4<w5u!yIxQi!Kz@2gkZr%CV3aB0%KD!1v(%2AC__u+##CkisHZ
zoWwspO^~orfg~`ht^(#0kgJ!t!5BH&ha33hWPlq;a#9C|@09^Vfy9#mRv>jI&!EH`
z2(ZQ^z6`K@iEjn?fg~gY^1dWw1z`Sf?PqNFCDh1$EawPgdJLKJ{9(;DtE7G^B`1cU
zTl+oj#j5O1ne|grDu!v9=Zi6#V*GM_9Z3)D=EL{|>_%sLtTy8YV~U?*CBKgowhi8q
zm&NGxm-h2pan5&9AG?w&-gG(Xm6sShFjT{~eti)(p;Hy_C?RB;8P(I6KDnqN!aH~|
z4ZNR+ariyw(M^r1@icuo->AohjwPh7cnP~}8Wz4-%C~c{wxXRd!J!iXs}($ZcRtUf
zH$D#or&WOR$VfH1Q1Yb^7jZ6QIn|`<EWC?=Bq$*WR2aM@K2Z=h3TSW3lk+mT_Aoh5
z;o3QJUI@p|)9-~Cwet*nAuKD8HyXkj%2OB&VUx%6L)caxPld2WX`c5(gv!S8*G(~Y
zRGCnpsWe}w3@~jFi4v7`&7Z~EzY$9Rn^^8jOmT0hGJ<aqc)Zhmcp}^UpyWo_u9KLg
zZy043odx$NV`zVz{WbzW<<;^Ca5FJpmuN!PzMa93y`I(<8|UK26cO=7bnC=3idU->
zosse#bX$sn(xD~=fgg!`atVSE@U%`YJ48tU-(oHc7dL==4QsEG+#pG#+e6;K{v11<
zlj0}?1#9nswQI<0Q_pW^YYRX{^*$2P2Md1&D=z>yoI$e95zILq5@Ct*1R@Y96%hCX
zSZWXg5K6I!9F3k!!Or1&VG6dw@nF<e6g<e-O2?Ud3cDM8UtxRT+qz7QkHYrNT<N*y
zSEaG||Eb2UgmiHXHOWu|rHCLXb(&6UqLy`Zd)rJ<#wKMc7_L|$D{547od78S&n#KL
z3`C2}FE&y4H<#?AIwu2wFTwL;x{c&~bYZMF80G9abV07W(e)~G6K$ubtYm<9GsYdm
zEBN=}1d0E2*uAjfy>+u#8eXK{B4n9*tF%$CP3Ss+e~pd;{}-s*a}-()D%Kqly>=LQ
z4PCtRS;adXoKz&ZVeO0M(&YzWlg|LBwN0{9N?=6*ah?f1#5WqvrqRxniH;RUc_=vC
z1g*b?^il|kY9cf)iV-<ILWARA7fdNXU!qK{iJ(|_;tZ(DLJ$U-g+drcv(Og?FRNRM
zv0F#;1Q|+wwnD~)RGb$_A7mKvSqO#oOYFiFuPcr<I<ak9XRp{3Sui1tQk3c5nt%p`
zklh~CM;P}L?Xz21=D-5mm?4Wr=8RJRMPr3Jfc!H&o!97Un={&DnC35xnr<lNyH0h`
ztm9(7#YLRo@kxSZEcCw^THolR%e_;Zhu(5v$>R{(YNFC_gPXntpWk+z_hwzf0{)Au
z5wECwf3jKD@2Ms9BE9>2=27e#W$VX~q>~|Z<XH!(B-}Z~?#I@1C;yWJFF@yfo)Td~
zM>*Fn{2GWa1dnJ{L^WO$!#OXCXG=zuKuUolz!5rdC}cs{f$$j+HXy08kO{-yEbMi`
z>j%0(J_pvQzTIU=ciNN=$buQYNn@Fr5g_F4=XD;)nrWVx;=b>jJ$6JZCUk39Qdv-@
z2DA#G5J<+LUI0FV3<3{99)XW8^;Rmp(S~9c+Hd1BA)oy35B}BO4@*A^?bXm@ipjR1
zLpp9n=jdLhvHBKtx3g1JlJv5hJ6S<~#~%r1+nXYH@gsjc!n?=XXB#U(dqiK9SHfT2
z4F7OOiSZwYvW(|Rv4=WBqmqpQbW98-E2iR_oqq^Q{=pi(&hPc$F-TAkq>9}rUVf@9
z6YHohL8?9n@iq{&`f(L;d?DfmtauVnN&*Wa4%m*sl0pMj2;`t{(dm0I5Q1J0#vD+i
z8hpes1X6dES2if6TS$%1ry2wEw`WFqfY*=d81(c^e{peh&Ktf~45Q2&H7={$GWix1
zf>NN+s514=a1m4+^OB2?g+h);Sr)5;3DETtijPN)X<&*l%Cn(y6;!Q)_(QNe&<9_j
zZO6>l+s>tX;mQE+g~Kq4osg14(Z?v=TpDzd4lN6arnh;Q^Y*Z36bwEobhXJ(m4M|V
zkoY2u<Z;{J;yEz51|I1>F1^9s1oW_y9CMk!wD&A(&b1~i7|DdB2seOS#<;0mz2G_u
z5Ci#qgiU0!1-2Buj{i{s%EOY{i=giBAQ5s1(Eu{!gb;2R3O*`ok#{9ID?r&kmR&$P
z1+WSf?114>o1TQW_Z2L#3J_$$>IXjn%hT$&6=;yLd^6wQC^WcQ9t1X6SsjA3-`9^p
zDstG?rc&-iz67FTSdYGfz>nY%#`o*U2eo8;mn?3#r(`UTprFtNibd&=#U(!O2VZ6Q
zVL5o-zpp9sBB)sf$wy$TuPM-C;6>;?VJEy~2!3-a#u4?ijb{2;(Z4npT>YY(<6dE@
zoE<^f<<_ZB8qmB+Uw{um@)pEkmY^Aa)B;aD$Ao;x<4Am3+4kP^{i^oDUfb(h5L2yU
zJj$X7l0nHz<GKvyqQ(w2qye)jifsH7*1BJZcIp?#fd4kSRK-R)L-`WAtDk5FstAcN
z9jK&4oDRfn1W{DnLDYtDEM2B7FkvwY-7wiEi5CY)K_P331C4;GiJ^lB<)rx$8ppjP
zRY&107uTlIbVMW#b_y&B(nd&Y!uC}q=!!r?R375a_62k6mcif6@RX;2U0@9OKSaAR
zC%JDwd@9@nOVOU}`nQEhE_^r6JC_j+Uk^mjFm&D7_gzs)W!O2S0pkJSquQXi9^Dl4
z{^RBMy%hM04Wlmyd%=|~aTa_Cnre(bQ)ZVmVA)wzjfOUA`=m+!-oqAt*V~cf8%YJv
zya~SXqf=!kJsLo1p<A84S%Pe4#rin*InL=8Zj((!Kuidf6(W=r*y$M3iK5H?m=%c7
zv`(OXcYS4>T`{~+md%gWm^w<QTtq>p^e7B3Oi2Bi4hYjzJf_$jmdr^>u+7Wl;aHYG
zy02z^qqf$Ejd9>l=)T_f!p=e|(&(fonKtkkaN4X(;l~$Y*M7n#C;pRu`4?0fILV>o
zt%~!Z6nt$6egOWMK1plfH*bYEubp{=$D=B50KN+ZSI_&HmHLYmcE}?@5Mbe9kCOx&
z4~dVYm&{O-ZlcsPMuZ_UB<U(d`NOTdGivZ(E%3evytSfwADSf4J)&D9KoxuD63!Eu
zlOi5=h}tdde3h;fGLDn4LS$nITn&_#qi@QBf*}R{N*nMzOtB18p`uun=V*`akv=sJ
zPzNG`P*H@k5P?M40#N{AY?)PK(C9&;Kvlc|w$#DxqNsPPGV{_*wz|vqRm)dF)f#v{
zr+28iT0^HEk^wB*rZw|f=pJS#ZLoTP4v1Ve(uZig7P-O?@LPXDLHr7r*I*MF&*Cp~
zFg^sc{tZ+H`HVhD9n4z-|8P6Z9n+i4aS^YY1l|OwMgIND?K)kuBmqK>x7L#eHg(Le
zL=nT!c?b%oaipt**Q^4&y>KqS^<o=VTaf=o1t<?q>dZx{Nz6q?&&Cz=^F%@p!d7eZ
zBD*v!h7lr%ki!U7fb1Az#D@W2gJYlR<dRd`xab2?KoA)1K$&@&Y2`?c=_3h}#XOy`
zdBn{qOHZ4SsDux73SvHlayl$ZmdBhIb0mu<r9;&R?qY)D=Uix71?fM*7Ek~B(+vx0
z1!o;J4eEx4o;GEkoU)C-;Wg^@B=-C@>g8=nTR@&3T;^kQ2EH*h7lG2R0@1I-(nmm;
zc8;w;8=QW(4rn_rRPZzoZ{f8BvtB_ye)<_9C8|twN{J*eEK={nYDkjLX@2P#8>Np6
zTcU)%R>0mm6xxOB!&4y`{yO`#*El&<jw30}z|VmNRY3I?o{4#!M{+jGslpG7<*`^m
zE*cP$1%j|7ut$;7A;2-vVIi;xjaS+)eod?FF}ZL=vG!xBKk3BR94o7aU8bmowK-vu
z+MLuyI(8&sRv7Jhw~9Z*EyGxG1PvuOmi=9-lFs{J`DbCH(#*1;2`Bdh+o3AZmp9Bw
zn$`18zDk2T5AmAokZv^$=ONL*9JlHX<U++cxK-%+CP;h^hQ0<v!{Y{u0?-Kq1KM<P
z%%v6hJq0pX|Bz7Xt^Y6X)pj5(8%ue}z=MJfIhVlo@Ewm42~i<&M<uzPNyb`|L}TFA
z!`5q)JobO_JnAO}*!8kg!H9jdY?K(<^c`Yc?zEzsq@9xKR)`7gs7dXU0xJw5TVhFJ
zpwPm`de%ZNcE-EK7xSLw6!Y@vNP7z4w2uK;7Efsmoyf5t9p1dGmWf&3vZyFko7XrT
z{mv3MGiAK=7a*4@RqN=7+}B~#7vV>J`^}C#@u@=xU*c}wMr%hNzB3TBob-1{gs#WU
zLv8?u(~t<E=i6ZTJCOKyaI+d3KMy<sK0MqHuY419{rWigW%a;X-txDhSJiby!qN_B
z@vs9QyI>))k$41-ParKk-=-@E^&=38L&k;f9Mlg(<Y$T<19xyYzbodb{T(NHn2<b5
z$NX#HMMM$vBwONmM6Dg93qFY<F=E*kSP~If;5j%!7>h81Bo48~k|f$><}lGCR}9^9
z%f@J#GUQHr&U%3XQkc(<nMY_JkaV<W-_@#Mo`M>si@Jvt$Ot;Vyvn@rg#gJStGpS#
zgBf`mSOl&pDd{nA&xR#!@K?auC+8jE@m6l7Sq`DD2P<rilHgy*ojGGxG0N4<t0D1j
zaNh`HQYYJQ(KZSN=MTb1yEO|xrcw>81>9GUrmDga_5yGMb7z^g7MP~degLDnfO-Pl
zctD1LvEn=t(6dZ&4{pvZ_**#zVkyN@yfkN0$9GB6Rt5PMo0_oeP!~COg9i&_xE;&3
zG&htG1UgekL#>4jNmD;ujH}Y}1;P2l0m*~`NOwvW88Qakka@@=@w5$#j2uPF@pR=w
zjO&n{Mvv?Q9)wH|(+vZ5!O$1r(Ocoy*^{bZa16Ma7T(AsjL<YM4qpw$tmgdx;Er5F
zE)JOxMDpN{pyvY+c^|kxE|q8g1@L9?N60An+lS%5Ud6GEu}tztXCsXt4PRG?sR(vT
zIGzsDEt-h}Pdicv1;jlHF1q;Yh|v*UbQj$_Bt8%DYi`Tt^lJSrCwZ36(aiNu0Jq?9
zt!zUyTWpF3tc=<qqc)*X9GrQC!|fmv*p7>EB(?<G0vVkMLB}gZPW*OJ%@@bntsMXU
zaUCzxSY2j%9N+6x<{#EE2FXd%`5DFwiuI#=;y!C)5YUx~e*k<8GToYk{`h3FUw)A#
zyi?wecy9>%&2ex;kjt+kGjbo1@i}CSdg_KC_Yh=01i44y>%jLI_W}JoJn??rGWM8+
z4uyanK=g{gY*e-B_r>Li0M{aLe0-^Eu?l5j1wOK%pi_`*8DlIaiH?H16LKwrv6v*h
z5$^u6$n^851wUVV@~Rj`i_FoJ^MM8y@pOwMo03!uht*0r_$p4uN+3i#Sy7E0v9Wd7
z<VG1~zL;cPEO+uwuJd>x=kN3|$2-K+Aj(c*l|ege!n2OSLAPWG*g0P3qR1e{q#t&?
zfaGRi1-fLc(#m6%{<Q_K6m442KRnW{f6z(e=$VDhBHqfH_Ap!<McN_A<zwW>{*g!=
zYy*<{F-r3+<aa@2v+gyqNgpdfC$j#pF64AfKvJSRbv}3T-<%<86R!`;f*^1$JjchD
z3I{=;EWE&nf<g`oXz_@+AtIxORklFH1$PfT{&Pt`4}0+QJ-~Gp)%(z;1zmjM5?O=a
z+)A|w1C3$*l$Zr}8q135qK?YKGAW7D1Xz!Em=_F5NJ?2!S<9KGoCKUEZhM;#v*l4c
zT}3F58j#fU;w$24SI&e+4^Elx?H+&NH~}a@XRhVY1=@ZCg^W(!c}SC}kO6^hz^B0f
z0qhENlFGv;!@cp(z<Z*nA>a37tQ%(OcJW^R3hc~pu_MJf*ctYj%OAkqpMZ}J;SK2H
zJp)76!k=x^n(7NGP3VHo@+G`l*1h%Iu&faTLKnN%6$OL|5-QEfPQ5?_V4VkRzV%R^
z%uo8v@lR?wnu=brQOubtN7Yz}(Wt6lZ)N8+iP7bVd@EKaj+Kw%<|MWcVNc(E5y6O~
z2uYy<Q)(yqxbfLGVU&`RzuRVF5D}0TM+_XxA4P5zGkh@)I0&O=yV~ZkHwv*Rx-t@0
zndvfq{DUNh0+BN(wGNmImV)>mXwjEKnhFrrz$zW}nR)_Ref{H3_{NZ%f!7y$iS)Ff
zt~>|nSP=L<NbfSF(?VeBvVg_PQ{{7OA+(Kw|0r^`jsULH0XC7}!tI3@IP~zvz-vJG
zr*XC8E{R(r1xgB(P)MZ{9#kMufkIjWA<=cAeE}l^LPERte1RAd41@$bH*EJATjvA+
z!bzUKb%a&#^9*#&0iprAU~C<2yw|IusT*c54_G4uR@8c$ITrhDNylXKNsxlLFm!Uf
zrsL%BNxLkaujKF1GNUlcauqs8smyM8+WSZmLX(WQv#kpk?FU8Od-{hvK4fx9))?9&
zZ$r|9(-|ej!}eqRs-OsJ(9xNv0iWiih)RF!Civ8^S={pfHogvVmklq^8*5|4T-*K5
z3();0c;*LO2W$lUc5SM7C0hUf@1ys<8-Rb9tSnxRzTFCu>xeMR+&4jhEQGYC<7%oX
zP4ZQ~Si#?Z2x9*;0WoA0-TQIxIN2)x`cqByo@w(S0bGD~WHhpm$5#TaL=jws;*}B>
z#s$eXpv7W~109apz{+*!@kbFTB@=rqOcgFU8Bne1sL)A+1jP@QhjaoXK(#Q@o-vTM
z?Av7V1Ov*>$>!Yc=<*qZ7=>XqIc}qiJoiIBDAgBv3SE+8!-;uo%mIE5(dF<3=JQ6r
zibv|S0(`?k#%-`YUHsW|z?(d`?!^A>_1J$ojJH>xkcWW}L10wzjlg?3XbI+-%FEGd
zr^QsW@|!UDPu{?iWDA+ArA&$-fWlqi=<Hw-FkZY`nch!1(nx+dkrV{l{-XY%*T*BJ
z{>)dLtbl2k+zv%HWtm&PsMKyk2RsyuElvd$=Ngk_j46Yw1Cs3IGMadU?Qmff8sk8t
zfQZdXtH^7MwL^<oRRqBzFqZ4sOajny!=*eJqzUsI<JcrNdoX8W0Q>X-vgoi$3WLez
z;hmuqj73gmr<|;RbZ8}xKLJlhf%9>R5#fJu$>uWplqj>9{_Dj3_1l35VQe{klTYwU
zE{R5wuLGH+6xHh)II<_t#Y?a+zaH=JzD@9H9eJ}F7%1IHw;y?sbkLRa%0;28udorQ
zn-f-}=OW6?QdjQNRbBzY*I&}&fvKZTY>f<mELB@=JegUgt?Rc+19T20=LCjn6)!+>
zSNa?}vd$}eZm4XYpG={efj8iBF1xYiN*+xMMjBdJ<V$^qBj_q)<5xsX0zSnd#t2Q-
z#4BM8g}OtSr;%*+OxdH+R(fryi3qe!CthZQbUAsY9=O=-5ck;-ET<Ni%lTh6(a*Y4
zCCm?^1CsCLgsZ4g0DcAhH9W<?@EcmjUPbJGC0t>&PjAv1+;{$Fc>T}enL$E7g4dxN
zb3drh&zb06Je^1GJ1;f{<$c!vA`n}VY_E?LdnN0*<Urgpk#maDkI{E=XzClGl#N?t
zlfg;6;t{Y6SnR+VDR65HUF$W<Rk4dq3hl&AoAa3kdf>CYFq+Xd#*4)sW*g8=0dhX@
zk3~ve4VxRGu~o7pn!sTW$XSMn9?>k6$?UP1r4yypMrK+_yU&rrX8R~cO4_3AC{apT
zH6f0Bht0GY<#gIdWr58B+}_1#{%$t&C{L8)E&G5^0AD>ZcZ5FzvqR_@@+;!!z~_I8
zb#c7RI_FU;&9~3||LJOEzE)R%{6@IB2S2BQ<v%qNAE`C}CSXaK?a42)jIV+9R`9*6
zQBJB(*!+csw0(ji5PF&=24$!(ZD8!zAb&gYW6ZkV3_SECdF~=>Zu*lp@7so^cfs7V
zAUp@G6s10?dGo`&fvx7T9vxY{T|fVk(}0h|&F8~Vu!7M(n~-c|_vtQ6&F(hAUo;=P
zjCmFD)H)gfSqxY#ni8T@e$ZrBPO(SWLAB=cH&!3%-*F7p>nZG%Q%Nyu=rJg$Fcxvy
zPpIji$zSnxu!rEm6Y+j{74SVE^b!z%N4K2A?|`@KXJZD|4g!*WK77W4J8gWt8~X8o
zaSg%9qddUT%FRhb`(w~CI|-v#LrMT~K#srcqfH5|2rUDy1cWdqgF7a4N|uzFybAEV
zQ{W+u1vzxRy`k~LJ%z4Bbqa_LV;_ex(?v#aKSEQ);whVpj=B#Fc$$nh)}X5=Uo80G
z*a{d)@!3RzJ@Z1u(1Epibcj`GB7RkyfBR^!ckFOLrhGvZS8KAQ9G6N>>{o>@daF4Y
z5uA*}MSEn7Q%+o9o*tbCp`co*;<bE?C+imQ*j4X`be}fEolsFDw;TLl^dzW0U=F`1
z0p+MMB)rss<z$n+nOCFpZyxkuj7!t-;(x%o?RC5fk3Sw2aJdRDYywNoCy)ZG%#{;B
z`FvK{T)`S~<s{=|`5F&M@e}rY5#_0~-fh9&kj-ci!S`d>$^zTfMYa=`W*mK49U3MA
zeSgeG_w;=WI-GH2IjBa6*nDK3hQtpW(pl6&O+0Fy`E!mzCN&*^6#JRWGPYVI!I850
zM6b?NDs+lc*X(KefQnxCa-WxdYAqNhZ8Gip3Of&2<nj$fIiEM+CYpKhqQ3;WUEBFi
z7_iJ5!{mY<5RU<8@+6WWiuU^8X^@WrmoN@PvTO|(=NIXhShwT)Sl7P<Zo7~s?&H<K
zBaEkp9vPYaW?11!a=F$?6*%^B)U=o|JR>W(WB<391<&hN@S=<;?f=$1nSt_LR9;Y3
z(}0UzW`l$A2AtL97QJsmXSXg+=mNrq^eEKjf#_Lm^FqwFTW^B!M}XZ1ii!pF50xzz
zlcCGYs+AixCnr1ZdmR;}x;{xtz7}h;?wy=u);!r|=rAB<Ey%l}Ie-^|`ygb-^l82v
zV&8>Te-Gg&;5JSmhv@$Re;K@*_(LBgc;kX@8Q~AuTs?t6+LH+7gn1t%-UC<sLKpKK
z;kp=H@;7kreMkQ5iWP9>nP4*;a*--8&fCkJGG>zsm|`wqjdkS#jvWwNbCc%TTK~&(
zii+Kbs+mikIRv;!jSU9~qkNn8)B)N$p9O2DfWEUKoMvxJ4YSiBi1S$_A2*1THwR50
z!dHxzfvmkBnzw%vSj4<Abh*$F!}OmizSA~19A{z$@roQ+!st5aF38)VLBSOeS&uek
z0~dk&132e>Fm^lK!3p9RT?4G;Q7+ihkJ?3nx!eU@U1E$(Vvv*P)xe*@#dpH_qgZ#6
z@ZsFc;NC^0({rTc^!Gx09Av&7VysZ+{xyM+#T$7)^A&$M&^7T`n;NISg;ROC+w~$x
zo1MjtNvxugI!q~ya*NqV^82pyzcEO3>q;1DhHRL#>=3KFVC7sO#iz0OIHi`Ci^{Bl
z(_xdvnz5eIbLF&^^r9l}LVp;lB7z3199zTYt0bl*qZOI1q4G4nR^+cCnb$s73D!D&
z(H6koufX$ff#f}U9gLm`zw8I>;oSa*k@uX1HP?Ee^o&1A)7anqQgE+=%YUp{WsX-R
z;VXX$Bkw(OUo)`+E_)f+q7gE6l3M1JmUB?1d+%?)=HdeZ<p``T*5)O5N(1HCMK2<2
zEhK0(y}cr3>Sw$RuZm~?%jnc4uY^eN7}i+$)fn{USs4yNKRoWinbpLx=`nZJPP7Za
zo(4!3o5>Wzj?_WHv6xqrD4Gv-VT~$Tj-EwubSG);6*<HeWcEK7xC7#Yy8mkw-(Jn8
zg5v&G=>HCM+yobY8~&KpEntspyXCJa@G0)5=7GN?cvED%wC!Q=64TVZjh6CiIR7_r
z&M?*=I7h(+*TL`pi6atdE+oAA15lF&ym=615d<|sNh<^b4m@A+hx~pGjsbl<>3ZeH
zmQ#0`_xD<iEfA>rOfMh#-DQ7&0iDIxsqHd<69!u0i}kQ4!I_R^?GW!@Ud`vwaS@}G
zLNce5*Ox{VEv4o{mS4^gCC5y$?nD4YM45A@=zwu3_4b;67-dDKOa426KL`F85Pl+{
zgLT;4<2=epL9wU>F1Z<M>!9hg&=%;h#N*a(xf58&S}xw!iRxb1LmQ8xQ>JGimqYS$
zSoL%G0|c<N5q@_yM<iqEf-7Okm5`HlkZXg4w+QgnL>_XG`I?LKlWv4^w`L!O5}6)7
zm6y1cUN-xaix<r%os=_YR~MkuN}tp$os9nX`CV{-6atrqq-3qnMGY?Z8KfFEw#vu>
zn554C6I25>X#`;;ar3UUf!c&2ztN(#z!W%<XfG3SN|~9(f<G2Vwo%s!E`zxv2$oTF
zZws{>XZL+1879R-Yde?ap(YAk457Ku@+5Q&=&ZWqmZ&t6;galJ#Nx99xmYeF2v*@K
zokgar@^j$1(DQ~n;n@P#E=GX=O`&^iABN1Th1Yx)+&SPCR$)7_7RwUn`2?1Slr}<I
zDCH9b2c9=SC}`}IG<I6%A!+OsG<FK=JJ<6^e`6-U9f{7_|L&=JpU7Zz2VOQua5$RK
zyMd3H8>L(CJ@4s({%+{<SUv)~M)(Li*~dfI%6{7D@WW-yLYc>?LEc?N*iD)Y)nur`
zqndy?0U?u95E8nSnIlY?hKmkEYD9PLDk5q{#eQL=ItLeTflFN+aT%?=f|XPi4IK{f
zTClzc^H&vHD;=l1)^CAK@LPD;M_D|s30V+Ku-F`Qen~*Xg|HBSa}}hQYuoK1+a-Z(
z{uZiN0JK5A3PSQMO$3FSs4N|99$XwU4`<TDBDiRXLdk`*_moz>z5cyqN56&im+3-}
z6KZe<?As=`SWu!^)~D!jvEY5?^nXUeE_k;ixup*Z^)wFyv-s$=W3!93F<hacHCnDp
zGs$)uHZR_P7-dCXNQusZ<x4>%7`jKYQzy8J0{A(7rOpNUeQ5d+T=_j6NO9b?kysDx
z=2_-F+l9)tZ4;%<yj%`4PblOyaMADJXUvir&P~E^uLu5V=3l86c=ZJ^?;SuW1Y@TY
z#KbHEy@ldE%cOhw0P{b{{oa8&o+6(ckAc~GN(1HDKFmqw!QE`mb{`Yt6^p>=Q}RHU
z*BLJdOCuixZj`VGqEYA{1UIFlDW_uwEyA%S^y{N3lQ(3QeXS4^0%OOw$rBzNKTav^
zpdDCgSVZ*+^5Y^v2Py_#XTW+FhVRmWw)4Q+4m_t@STzB+Ky)Rn{1<5234h=?s*auo
zo}!m|dme=yZ9AC7lPFa)5f)YbgcYrnzwi7Ci*qi7(;viogxLzXX68Uy3%n%>u`e5)
zDSr`&2+HPS#uFb@K}~|~1JCmp%kg_|qGZUZVbs^hf%3^y8Yrc55czo!JOy&HGp^X1
z+C~=v(m_8H`)Kb+J__6fZwtXU>!DA;SW+{?T0UnK=N4c&C67zQ0D?VE3}aDH)h3(3
zDnnA3s6(6-@?k@bbLKuB1s*qNVwPe1MVI~Y`NIRVrS1Wqbs@V9nlo_Py(PG>1LpzA
zDo8@`SCBgkdOrfm2jJQOe#f}M``F5(6u5TuZp50`jxviJzF$9K-TA(C@Od-DPlvV#
z;Sm-q!$_LY85z@%7T|^?)ZYSQrvqv+<QpLEEfo6J9Ot`Y!aZ`ZxvzrpiiYJR1-cmy
zf+EQ;p!11NU7#G!51?{uKo-*gi#K7vu^$eb)3_ZS)?r~Pn4?`OO_aLZq$kq459@vN
zZ617e801{o9Rr&WjA#E$W|BpCb_AqtuuRy6jBt{x(w{(AtC?@cd(dgejg(5+J&$%T
zS5xX8KAF)U`;#<;u2q-^ntEaBR){1ug1V!cC&wUsAHi6Lf(y@{50C!{sy+kfd{tZY
z9y90Qn1&HO02G2Jkk94ExP~nIMr#E7S;Z6Q@@25_+rZeoqM`8;D65FOUs;SU#yjhy
zkbgBuSp`FjAt7SGMRF2nnSA{ZJh%K{Ka6q<`5_>SFgy14QyD1bt_#!T=A{i|SzuV>
zGT<_!AzX*9V4p=hSc<B6%BY1+=26R(iO+GZ&Hqgv2D`xX_~Ra68=qs8pW*Q<JQILc
zbmfc)=$EitLQj<4*|4S`1+y#zPR8PQMcYEu+()l4o7Ns<?l5Hp8ADgW%NZytc0&!!
zYmV*NDWI+Xwhv1>VZ{Rw88pl5RlpfqJE$e#Wn`mg@U$iPqYu$KIQO5S>VKhmsjfWn
z7N(Alc){dZV1ylrjk{3LH-$-3PH%o5oOLe5nI|D-p?}~3Bm`Uw{CPFh{~J2L!;M2`
zg=zfA%h)%6V}>Tmke4Uq<#D`1*&~nT-Ot$Q_&(Lc)K$Ar9Y)zN;U{7q^GPC!K9$5>
zR(|Ui>sCa#-c(=uAvS-+B@X-%hZY2Lkklb?kqCsz&6WUvhYnKjg}==Jan7L*5Fa@B
zw~GOB<6S!G!+C^Z@%?7QsWHg%19YHjy8+2TO1bY)YL^=@u^;6z5(PCr%-RakN6l9*
z)}N{}*0pAkw+AHDbIj)n(!@d!vT8I>y$kkrz*!La2AuviXdcp<%$>)AU5)@f3{bo4
zG4Nk8fSLp})1D__)*M)RPrdaB5f28>-2cbUH6Ym(h1}<X*8*-7hR%aTJvf}LUlvRX
z@0uLFa<DmtylGxlT!sSV5YWS>Aw~>hpVDknW8pv(1(NKf3LAG>)^Nw7l}wqfWW=ob
zQC^3_BF8X-JQh1}cm|ubSUim&2aF=e3-zJHhl;+SPWE4jp*)N-UZ%;P5+3Z216jUI
z4_{?H-$FfjJ03fD{iZ&A0=g;n_L*gvmZFq1S2TGXTSn;ub-fUOQuk$)?V3%Er|XA-
zr2*W~1&NJ>siuWXxQfQuQtYA<*=A12PoZuFybK*mRC6qT(ryA&-xEqbNrql@R_}T<
z`jcL~-?;$iET|=I%qKK4snXAQIuCCF+UJ7!9*EZiZUQndhC~wK&<LEPkEtdxh?&l%
z6GB=x(sJ}e+DJN)KxTNHD?N9r1Ep>6+RN;06&qqP`X<?5icKMk?Wo%*CbnE>nnV1^
zT<;NM##m%-+N`;0BPFKuK6Fz{8*beOb{OjGVQyGgQjXiY9cm5W6mtNwlu|0KDKz_&
z3_QPxjm>Tfq$#lL!W0RfB|YNsW*YfEwTuv_l@K}e)6+~Wh8b|oUH>^s752_Un;R$l
z{j305yWr1T37+nt@JKEFN+OV)56)wdEojWW1h@_Ce}P5cfqxj%5cNY|psb$2*gnWT
zqZng|j!BHnn!#M21G5qmh9zLfz*~ea!htTjaV}tmz`Y*bMq?pJk%Hk>5KVxSi%T7P
zn$jpN_Nlcxvw2{LrhxW48Bw5y$9LY}p7KCB+zaczA1Ayxq#Fj->f3DC*a}C`{fc*m
zV2#irJF{H~hae<hOSAnWY*`P|CM~<M$RMRD=_<2j=w|CKn!r1#$Yi+%Nb_l0`7l*@
zJkE3M;7sbt3_G|Ca|7G>DYK|0P4%`29kU~VMF_jhS1FJ6?jg|~Q|EJlg3JO$2SHsF
z|6GS%4U%oYfk;-r^=b%r0*`ChZ3b@E2J_e_VC5tF;Ol^&ybyrd50Kq1$OK~~{E45J
z@h1X}m_#Wj%-f;;4B&oCX?oCbCS0~g?<Pqgm<xqZ8Uu9{@S0%wOh{CL>!r0*h}mVQ
zdy?`n)46U|HTmcEFc1Likw*V}<dg@>1QvF_i}W@@bY2>3*yKtJa4pj}L&1GM1XA0M
zLk`3}SYl$%EjtUM(Xh*yz7b?GRneN#z<8@sKx}l8Q>TKJnuFiu5_5;R*gczjTF2OX
zM*5=qn8d9DTMDx5d1Qdlc^31{IdjI-0wi)=={{y~83I}lZjTK}kq7ofC0iCls2=96
zhRACDTjm3`;NA;^dA+DE1I_@x4apnfydmv$TyHd+qZX{*2cWtYLyu($S9MRwQOVTK
z+Q&w_-?;;`S|Luz8X@Y#X>)Y>rEnVLry+Pfh?KdiV=E!ojHnBPt^j*IIs;>z>{3>E
z^3djaF*%NM6AF0Dqn{Dt=m3U+QywUV?K5jmn((?L;hwVKhoY^pN~77OjT+hROu&;N
zh}%D9Em0b%VHP2(DG+7^@QGt1X*4jX%2tcHeZLIvunn#*W@3lj8V+7u=p+;i82Jng
zIiJAFe~e`Qc`hAw`Q4K>>~cf2G+X!N8eVnjNa7dg#Y2mbaxHK+Hf%{EJ5DTXg#MO$
zfUOEbIau5SsW8|t0qz4guLGm5h14eC2_5)QZ7kkb0iS``Phq8}>lHnZc6<#V=`X7f
zm=O$*Rb$Ps--jH}WL`TDr0zQ^v;|kR0`CX@Qm=fGuZQ9DAQ^+ChtM?p8gp-qO*)`F
zy!pg9$^wF)X`jRZVH2JH`wE2tyLLztekw{bY>HwOdutCG)8mNQe0Bnl!AmUoL&pyp
z3AYf9&1O$?6InY%%8C$9TXaZ^tQ4kRy1+v0tAD+_a^fe;-`)&gtcHQPFjVWZ^ua#-
zZD(;sV-LTwTj?3*(vHpG%!bCQ$DeYe&N3Byia@q^2>V>5h#_zYj0f|02F$=LuiXk_
zdFa>-$!AQ%>}5u^Tn{{;_s=CjJ%k^Cff4<$1?Ucj@6lEZ=hrZANSib_nH~6u^;_r%
zeW<aqTKwchRofbywBcC?p2s_&4M#qWP+(mE*?YikK!*xvPY2ZowPCaqRm=fa8FPkj
z+9H?Z@L|p;#!>D9vbz7-9&BTA{-4U-33gv{4skn)SEJ3#Sd_ZVl_!Aw#$377Pq{dp
zhE#;RmWQarW6Y{1OBEdmB!Vn10*TAM6`;rE%jmfE5b)wtlSNKu9+m|?VMxY4O79gP
zA$t2p;$2a$X{h11D#TdGrYUpXy?K6iiL{zXI0<5o--UX-A+YBog@p*JkT|A7$njeV
zWJp0x2I{s#{3)<vAeT{c+@+0m@)scUIt2d}knYxmv$vpE=}O=~Ks^I{z5zRa0<(s6
zulg&E#<&^XisFz#GY=?~ATwM`c2d7~F)4vahy%|<Z9oFVu(TATtuT~@)RwcMryVX7
z$dGN+bS?q0K#uoh(aYrUoHvu_A+Nk1<$8QzO+X*_lK{3EV{ip1g&f80Fd!MGRM2*^
zTxG`F1gx%yL^Cg5-N0SV7FDyOI5iRaT*wP(O-e#iudrfw;eBk76;`>AHu~-%_`(qQ
zHf*hOS$Nd|`CrV#-qFiTTU)s&12uzW(kVrY975nfXFGDCOw}TS0Hi`%W0dYJ#qpQ`
zAE09`-R)pKYqGA+WMX&Y5W2?m+kv&3UD`@x=WC#*3%DQp3j|Tfp8-$)8~E>o)4l`F
ztq>Z}@LCU?g^rot2>hHG#!hR>B|{Vj>b1kVOu{q$@%5nG%~-=jMA;yhb55ZR+AP#;
zA3{Fp=&vi3&f&;S$M|^|^YJ*!G2k~81PqN5HU@`ZaWdw19Lnu!)kPVK!AR-=AXzTg
z0`@2jcS2!1?6fJY4~VYIQMax@Rla~2gE1eJU#dG)k%P!T>YKvtg^}l>S$&+G=V4#B
z5$SJ$o*cyUbjP@|r|J4%%kTh#K-s>Juq4Rmkg^kkJ_tsb*y7~)TI*6lv=6LJAk*lW
z!211Y6)E6+bWz~6+1=6E&#!^1mk_uiI!9oS4YTG${ing+U~cl)1GV}cjljj|7_}RL
zztu;e{s0?^02~)WppxWr4GaWKcodz+Md$WG=P7Vz@5LcU6M<}^x(Y}sND>EiCGctp
z8uiC3G4To>>3J!Rax2nO^h;njwxQtPuH2Yr&H24g6HTRvMyjS)bevbg6+h!1UP|<r
z4(a*~p-~TSEXL@33t!sUIY>yb!r%v{h$)q-X9vF;@Xl`dQVO<rK`0(3mF&lU?R>1C
zb#v)xD}R^;bMsvNoCtomEIH6YEP-S%5Y+%F_kujju%==Iq{(^Uf>H&j7yhb9ADxBS
zV$6xv>@PT?gp3`)ThUS8ztRe^_6O#8El*eV;&2a(2Qs(`kX7h%<d>ntTOI?x5ByrE
zog|HpKOgN){H>vGSLyROevTyYe^FQ~P$^<V^`s|d!pzi{A{+?GES3OEX}dB2mnv8a
zqG<9mjIU%<JUVk7N0~#1TETrjT|7XQ;qmJ$H|6*eY{!izk)i3o=}r_4+(q-t>!|zX
zd_uKn;@T2xFUYKi?2yQiMC7#HP0ET5a$52s;4|QOFuVyCE&Bw6xernAY=l<{xT6|s
zU7oJ8d!4q|U9my@@SbL;y8aB|r^3R=CW(|GN-2+anb9LO2>_0*J(gxl)#Rp|p6a4=
z8rlqC-T~ytP~=E4k0sQncbj;Nr=c%js@!s!w)oM7E=_bR@GK|~x+Dw*&|p%U$^#$$
zFbvddTVplptVLpIQru=ZG{YZ1fF*JS4v<WdKUZSUrIgZMn4uf}aOajZrpP=)fHy*%
z0eo%yt;je~wpe*$y`vzKOj*)rlH3C%xt=gTA;d0xJffKpsm3%kHX))BLNY*(04aH3
zrcjY7!QgL);Y+8(jxlhP5mIsxn`KyR<>Ec3ai2p?Vh+SmdzCYgdWJCVfRGIjC5c?%
zP`g=@pJNd>%+fZ>pE7?RV}36wt+hj08&P!l;k*F04iWsiouTYv_7b8N$y{J5)MpKG
z{R8aE>k!~3;4h3K-vi!e*!3(}8(`71kmv-bm?h!DNFxki14yOjQ<aMn?wR;iva=^y
zH>`uq<rZ*!q$}{})RCDN5Di;85<DooDke$%;PcP~lj9`9c+h!X<bm%YFdff!Vw*pI
zQMnn*fHLRrf@B!tvtb(b$0a(PtN!*^QxgSQ^&`Sg7j7X2!wuluNT(g@SLhmPO70Uz
zS&;+C`+?7bvkitFXsP))1NKK~ur|Y$FM{8W;xCLb@(idm!+$nP_CKp-&0Bqv2pR^n
zM35v+(^9ogpq9u9Z1Om`5Jzg`bRk%n#GhY9=e8vL(xPRg#G%7RyT3t~#H0uKy{6kq
z2pV9h4z{18jh~G^NKYFE0#8HRUSPBi5VlxKO$w~q!y|A?Zl^1v7!wvdtrT;WHOY#s
zvj(R>PfuxuKJeTlAO~3ZYBiD(0^kvker_jXY!M@so3RWiL$V#*M&o*%o{ZfFI~yV2
z2JH>6MTLGuKD+~I?F2cSkh}n_0*DY(g;7>y8mw{vzHmD1=!H;ok}8IfJcW(TWoypm
zF2^Fg?p27k9|fGH*c5@NGq9^xW`LoeMp$xv2x|#CTjx(w*Y7#Wb}PMY_X+50mQ<e&
z$xe`s#wuy-iN4YBn}Ih&n-2^TFvwOOBY1w4j_Lp>3|5z+;Cb*fg05jfK`6qr!<<P?
z8vIFG*FGtN9C&WAi(UbNv=GDyd&P8WH3?OxJ6}4(b5<;Iql?y9zP3&F0qiA0fid8#
zm78!9P;PD;AeyKpIn5GQFS;G>OTfhuupP*HP)G+vs=`>ru;d&FZ3KBC6&KDS%0B=%
z>h2svQ8=yo1N7S;qru(@FI@(|*+Gz;&2SuILGbdvHF?5wJ+iXrtpjPvQs<)UJu82&
zw4#@UQNNOiqEdI8&~Xd5h2Ywg!6H&XvNHgy7ut4#(+u)VbYAQ*u)|m-vn9|3NRlDW
zPSVuPdy3ikSgeFO5rlyw1?=Dv6kDkwA<m$oj(y@NWv&E2<;b|dd0zDC?;u=4`6`?t
z0f|!Gr()pc^yg1cnsb1;526dmn)xO|-WUS7ANI11B=8jQMCB%&1e8sgL1OsRr+pHF
zdU&J>()G|T{+lHc7a6Vvn(;XsAq8@(BEA)w5vzRXAbepqZ0`lp@)?qRh!EScg895;
zc$7PX80irjRZWH<$P!j_2xU%Vr!_<isY`6-VN3fO<hb&Et+|BPo-C&+r*x^I#^+4*
zXuF}0#h~@Hv1bmVs~Y{X_|9$wk^xR*j0h<#auZb4Ar!lfQo4|sT?<w06GvIa!5C$L
zj<hKN4t;E=u-!t8L`ZI~MxH)VHx|rvJ}}MK`@r6^*AM8EQBd?qDC9|SF9qOhm78x0
zpe#frSwhq9jtFq+DBQChy7J$}@2JC3=aa}w8pH}5VMFNnPoq2T<sX4hfx8t(A~3HO
z_E-@*?7eVRHT=>i2)8k)z!d^EUWOa7a2m2yi#EJSl(;JhkjMahdEB^ij+g}K&N{VS
zFuMyvO(4pH*30dLMbKPsRPw!9yxw*O9;m|}3egm3ACWZ<ExXByAu_@Oxr*5_sBdHj
zA*luP?W}P%7ho&B1pPM3ixDP_h>D3eL~CYq+e%LaX^Sc53K-V8a#sEL?~_!coMG;U
z9(I#Lu^#w!<>s3LD0|j75RTU&Q`0QE|1#h=aOtH#fKUy0Wf!y8t0NX`rq;HQmSh$>
zI;{ehhba-wD!(`fcJ{!EmXDL+<3!j1L6Xq$)8t|SRlW7ZB3q~#yP9wx#8v<Wfp38b
zmkQ1ZQ;kg)djX#W7blW~=tgkDAm+1AL(qLhX~r3}EF{wmeXRobA)i#vhf6JF1Ohx@
zGecIM++8*B+*P!tn3O3qh3Rj?e87AegIrz!cytx^;%7jrDAA}@O6^er2&<$rPjlW^
zv{CNYq^b0mRwE!Zyk;4|Q-es+jAeg?oJWFRQ!uv4iW6rlploUFClwEHXL8pBz#jxu
zcktr57I!UgWseBarJ%`&GX;3EQabIh<^7sf#x}#KfK63kxu0gK^*Q9Vqg;BM#hvLI
zl;02jSqvox$U3!*?0~s#I_N+YFM=86NmMOf0`W0zf0VN*1+GtT>C+Q{>I-NX23e#1
zkNVsU;j@*2Ai(cCjgY^Dg~c57q0%iN%XIHx0)rbOAeg}L^z;4n^$zM%XCg)<UP{Io
z$2Bnt=MFMoV{DWY*mF5ricz$`UxTC)Lk#gd0)}zOF=jEuP9hX|nrSsVJ4JzVq!1!8
z1}ZUwcU=YCu3%3N`~-pX2;o$iaAS(raEh&}ij}64O@&b&KCAr8A^6f_*t-MfEr4!5
zKx6Ul8-%>82~B^4u8+O420Rb!xDKfnw#kMm-nb`=?yW#((9x0e&1Nu74^<xMQ;_I}
zY6+os(>&J!*o<O;-Z2Z6eI;?=^tnKT=7`<|xSc!?@NfXfbUz8m00japI_clmhm}`2
zB2Fl6de{g+sEI1}H0LX<fl0q?bdq6Wm|{9>jtsp#MT%zt#V9b~6X9o!0XJ3H<N<(k
zXip7|t|<o<ogrAVo!|a4i_?7tsaO@w^--2dpbe~Lrs!`4DG#4j-U#pMgfE;2d&ZzG
zs&9cv7*q~&)vz}Xt+g;O48sW>&+Yj-@j^P2*v83pO(N)CTrp-8hBl})JDjy!5{?6P
z%5>Aqf>=i$kHNjOQKPS>&4QYR`zlp(pLsB@n-{yX_ahlGK<Ti9?%i300-H2diE?^x
zqE3`7Q<)DwuMVXr{jy?W!HA*1)y6U)*vs?S^b#+$kjrQ3mOTZYCm_R}z|Sf-+W~-5
zwr#<#t%q>!4Bz^C?%?}Wzr3B(e>se`E5_noQLI*w=bCLcR5`&%MB<tlxX;3%rTh2!
z7BpT7{GYxr&KDpDA)AGk5pWmkyy3~Y#wQg>ijwL((;)Ud(6Ael3ar_Njand8CjmR@
z^97mSYmi<QXE+dAJOv(=1GDdgH3~T7fZaX}So&oF5cmq;+R5OaoGw9VRqN7nqG)r}
zzoBG<oZh@IF*%O%eglww<|aIk0*Yb!=%*W@u7r^t^zeTg(tpif;4_t*>>xl{Gq{LU
zReHvqKz|3M3Dd#T)CL38x(Q_LbmZ*0u*rsVtx6c>k&^cSpM`hsfiEwD&OH!o(Isk?
zf|@*R_uxgnrVQn|Ao7$NHJ;2}EDtdlv@rwLZV1v^rS|PW2MCELP~_Mh)Z=x%k+_13
zs?cTNrq;F1TpKTa+hMP#lgcG#qsoHHgYRr%pwGv$l7#IPR=o860|Je^Gn}s(etHsm
z_ZpDY%u?_u8Ai$Q6Y%q}yMUd|l7)iZLm{8#+voxq6*-{lJ){VHbA!!reYSEv_D^r4
z(cOVl9fqb5oGC%Xz-~UdJO{^GJ_39S-n|QM?SRY}h$vJkUHWzD9vJC>s)a`DN>WPm
zu3UnB#OX1NZ~+9-B1P*2HZy`n-{{Lwy-S!Kz{2?&Ob;xh1jHC*J6(V(pi4Xsw`vRx
z6eG?CMt6cDMK~HOYQ+ZQ)Woo+Y0g*V2g<Fk2C#J<(gK1EUG(rB@JC@UWc#5L3PA>;
z?&nG1-zuwN8bImD-QdK~b@^t7mbsrowB}mYoCU7+QxZ#`Az^PJDdX5gP0;xYbd4;B
zN(rGOV2|sAcT^d|P~OCAmAQO5bOGE5r$|5aLzDtK5j?F;M9R+re*!s5PteLp_gj!d
zaES%cb0P2SYojc?6A19h1AD0hAHsdWpf-`<0l^Li{9TOpgRtufhh!;4D=1UwW;m}7
zr5Mk;#3e`VxW#iAmw^8zUZ{b53cuiyiFM<Fpo>8ge1tJw3!@?j0m@XJg6h1v!=~@e
zQNMsaRgj+#7u7<d@p>W_Aa6s5BSg@Z`iuDv6_7k^c|Y(^=<Wt_V1qfeb?8c2SCrBH
zO2wuFi>dlkv=Ium5!gx|i#@@sQJqVv58#S4a2fc4uF^XmK0aZdp%5tgEEw~{TKnzM
zpk^5hR3UhrzF>fSpM_5&;i%Nn2oyhm0uD}duINiFK)d!@z$g+57{zChEeyzuA(ut5
zl}y|3=!BfikW+(vR;9RyJ1R?IT0psTK#~dyq+~DJPnV=r*meJMg@E7AhGq#PEf5)h
z`Bd3)zEKI1l{0?0@*&_$My<2ZrJM!~aUV4EicDkW7k7M2MffLyJbf%+Z~xm7du+}t
z!0dStoK}3{1gcQREfu5qnwpJeHBgS*=fh_2dXT<;A8#m$MGMi;%<BE@JI@1I)(qcm
zpj&^ROV9y9eLRm>xR-5NpeIk#@1_?r_6SO?;m3sx|GMAc`HJi#a(3t<;si#I>ql2d
z-D;lRu>Vm(D7S%)5+3{02JTx8&p*F`d^h;{=S~0Mt>_M78x2UVHt#>)#uH551AG|x
z0Pt79+tB@GDj<1?@*Ik8JVI>cjTCmalIpj()&l2ht!ODV(JWP(=g1RgmujYv4MTre
zrwPTw5H6<O41#)wz8+wt9*0JfiF6qm389$o+ystO00vl*-^q%67c27JEX{YbcW4KD
zhwh~?@O|joLEDZU%pU5dYm^ah0IRTx2fb~4m@X>ax@WYnRvr4AFWm(X=3w`w;HJQd
z>P<8P)FTkM#!gw0iX5jRdVtj^a`bX0U75GRW;jQ{X-(*U+UV$?aou-(UI3|erNIKY
zBhWjkeQk{dc$}!elg`~1WI9OT5{geWk&N@z1I`1JY14-y3n;2XMYx7O@{IWF@zsMc
zq7Z}7=)yw|djdf<d+1~jZ={B9vQ)y$X9ASH+rX`cboC5fIoAO{g9~Tr6Mj!rx1isF
zAVxqKiG{;rbndrBC8e<<$D=$1bW<S37Cu0}D@M(3cvS$Q7o+`<{{G(Ukwcp!{j4UA
zNxAQ9;C=>Gp5DC?D9j;hq3f+rTLKNr8j>lGPuiQ%vmQLMJcG#nNk`N7LeztY7SL6I
zw9K(v_3$=z7Q5*M{#d#4GXhFAAN*9Q71fL+jBY<P2$v>+)8N4l=(Qzb%Vr@q1#?-$
zm2+K@<5HdiU{`QGUAe2^Sy)woj#}V#l=?02W1`FvP-2s+7szX^d!qCL53<fujBKk%
z2KBl#Kza%+oO131W4gO$c_BFC`F>&;Mj0dES@74wUJv?oLE+{AR3}w-o=k8rH+nwz
zm6<auG83S*6cm)+r(B~pPrJqJRp`j`-DvY>2rew9LN>Z4HKdeK6|bD{iX3+`N`}Sk
zP#?wH1udQMY60%0=#p$jT$})8J}S^T+}(i67@`+}!k&pocN(7qI(b$I=<FbjOF9n0
zq*7ghnuwBtDzeByV2D7e2?&=n_iSB0LltWfh4s*z8xNxl1Z=R$k|nQx#kW+3Z&j}K
zVE|>a8@eM<TLU{ST}i5k8H`t$@pkBW7HYRDdgD+Vi7;SGRxE`l+M(Ts$8B9#bt}iu
zxat&N0BxuAum2dGRMW)=7|PXQKLsz#!{TNL{dvh&<f0?6isF(djn2Lq%Y)ND-g3O`
z4-Bw9NR}~n^tiCMjU<wEH8^D|x=qmu?*n;kGT4l1CW*pEja&Ky^QtT`Odk>ZUhukL
zL_tCKjm!YgL0-9d+{-P*`4|z+cq?|vLj+0!qZ=1Eyxulf0Z90rfL;q3j^Hc<r(mvZ
zMfa}AaV4Jy1jD?X&0J4m8_e4cuLeRdMYqMBjCq>N0gzFeP5NW#s5-CA95pWez<q2(
zvG=KJ7;K_mg3~}*B0%v5%o!9LgZ=2o#d<)!<jJrTrS6B=sQza)_|@8xy$NzCJVy92
zi}?h_c9fL{4~GGimI28=bCHWxSPscq9U!g{D8dLFBE|LKql3E=l@nf(<3VmVz_g0z
zxskzPh&=~y$U(~-bVJ+8>)*#vvVc6$?L#()My7JLJX`*zO{nl@)+6vAYs2+p)W(6@
z0D_`wZAzg9MMhyzIhY8LexOzD0Y>py2l>@75@=c*Xw`i$uw5UtKri3pPyFNf7apoy
z^}_?o1^v*u0NVPYv)_yh%Ek@T7#Bj@bI`dA+IAy$T#D1W8nt^RT(lb2t%N13;Sn>g
zJ6dBaEa@&Ar}4h?PVp6R%H;rj1XzMjWBN4Phk$KxVIN%Sg7s(S>GBYb5Kx~P2d%*n
z^w$}3kuwBi0+aNB&jbI;2rkc#zMf#?V$zmRvSt)&hQZA;?j^E-L>YrD^NM>w#`5V;
z>>|g*_^MZ<V#W|qzv%Ubo!~2<99?{$Kl0ZVr_GTBN&p@O769iOb)h()j!240_}X3L
zU;(ZK-{l3)TufE2XhjYnAEub<SKJ{p{m}9>ydwv(Wxx$g5yzGp|7!&8yDVf4o18Lq
zXgAu+sn|oB?&spj-MgtFhtuk5qvrwjGJRN&fFVX0Vm$%Rpi~w{b?6TnecS1lVHzX@
zpHa5+eb(^)3P>JtpsX^WxWSB7W#fiP&rKemN0*k~3cLh(8YRw#U^QW}l5BW2h?VFJ
zm=|K4EHiz5MNVE8S&eQ9V*#%OB77N0u7Us83v-jeUzg$NxIHqMFxv%;6o5eqxv4sh
z{RDU~FoxoEo>ym*`%@OjhuW|}woGaG9A}R{0T}{D*~%EJ3Ft?ufN}MCpkxmnLstSC
z29#o$t=z~{{A&dykGx3DAj?6rIPX6QNNzA7`5Zcb?oS9-B3O+jR}!{YgZ)B+<Z82&
z%O3}OeNgtO$jL1h@IAEWrH8K=mU#B_8aACP&`A_>3Wdw8kiEcwG6*%izt6cucL+x^
z!e*ZA>p?vZ(E!p_=x`BAp+3c$=)*u40XtE2(Z?{`@fkz;N`K`&=9ctICjaLkI|PBs
z(aT0Y#kvYe9z~#>3M9+t2Lj9IfQ7)bz+4o6Lb8&OTusPb3Dye<k}H9y(RxjBJlN}>
zpw!x_0+uJY_-LQD!>!a%0L~`EyG9_^4!n;k+K`sQK!z}WldQ2#7WVf!HTb*^h%n43
zdp2)F?DDCT5N=iAD9YV9{RSYr2pA!tKrc#d0B?u(7VTA<#%4<P$bMiw<O@3G`zbc@
zS~hXZ{xgUwa@eZ(nAzk^Gds%Xwd|#bswrY9r5PQ1Xr+!3_R_>Gc2Q7pI<Peg?E^1t
zOk~<<CdKi<ulHHS3JxzOsv^g?x?O7?UjQxu=I|wSi=7t{;qPuTi{+h0Ihw*InL$^z
z8w9QX7VU%kY^<AsHv>=Ok>XA+=jW;#XDysJ1}zTcn;|P;*f^qn0s<4zqu2)i2o!QK
zW>{oKe@u~7c7m7DMeQ@d&l%z)tYs%XRI+W3zKwDLJE)?XerB<Ov9kI8+2lgQCKn<2
zBZ^hntR^m3G9*@mwGtvPG@D#a?1<PkD{|t?=L~W<=&BYj(8?Dn6)HdH^BE8kxEy%R
z<o!}teasscIat(|hnWztZ6WauU|O`|a}Dq;8Eb^c{2i#lH1R)5ECpRD$kpm5V<pvl
z0Mus99&=e3@^ye_f$<2Tf*~i}y0NCGU=Kbw)5SMfI|0)xax{T5YE<k#1BAtSlnwN-
zfLa#N!-h%YbJVhcUN%t63WoS2D>#om!D{Bnm23^IraiQht*oY<7jB#kf(Ls2stQ=1
z#4_Fk7Wgq=;`5~0#EmS%0WSFxgmb_J8TfES#}1cCfM74MrJOg88J#!Kn5qmTEGhO5
z@N%Bu2Yl1-Bu6KWyXVrhJ;sOtnn8qptv@RTejm!)0;4-%Bm?<8_@%~SS|!IH)xkhx
zz;A&MlVuazfpryY(F?13k3e7BUaCjVB|5T!v5_kJ+V;Z8xzx68pr^Bn+BWFvq->P*
zbPDR)Z1#3m(>S}C=eOlpI)6S}2lvu`K{Z>~t)+cQhOO)FpnXZg%&XbDZZGXi=CgHO
zj`k&Iuyq}@FIjwmv7h!OD-SS2`x4l?4qEEX_G`xC|EC)7Uk4YRH<3eB<YbUfpe=a|
z4V!!kump+A&w!WT2>lD8>ctTL^i=OJAn}P99-1kjZG@u}t(|g`CDSJd{2vS0$s8}l
z`v0Cs|9{0u$$;oJ0nG6s7=zJWsNp@3-35NmBzYIc8Kk0^o(P#D9{cR1mml*Kn;5RB
z*~bJZ`+(|o5N(68CD1ogHs?UO-XZo9r&@;TQ974<V*=)94#mD`9b}y5andvU+y{UQ
z$63UofMrmD+2al*Bj^go3(#)t7~m2BFaHe4CKGLYt_Gy=KG##n-Z((0pyMpdL2@eb
z0jzD{|CA_-OZ)5Da$f^EFGGJr4%h9*+SQH9_dsC??dL9~DF&0Qqy1Vb_XCBmatyPD
z+t|wO<SHO}>}-^AbY+B2Gsa8fSf>U|{hWYK32g$=X+c{R?v5U2w@UWz4QRXwvYUX@
zj9Rc{CZjmWIL+fsmIJ)L<vf^m*br9%$m2skW#FbAxE@IHCEh?4F4e%tZiHo3(E0^*
z#Lra_uHN@3IdW_Vx|K0Z?%BV38_4qKa`gDW;_Ez5=UoMy&kizs>`7t^fcaS%&1tk1
z1#}}aZ`?vqOv4`39c}ZzVo1Sl?B*t(V<&^;E8ut>0A=+EL_1-u4f^6xKA)&+D>HYg
zYfG@VGfBg2$;NH4d>(9F3+-p`U)A0J7|^~%9$=io@fJ*CjR$!BgO9+fb1IlOJ-LZe
zI*Fvzxy+j4<Vk}@Rfd<<nca7$eRCbd_o~o2ghiWU`7v%pyP}JqbuMrfx6(|2iw(Tk
zJkw+W|BDWq>@jR|0$65{=2uDqK{2Z=f8cF00I;@D<^P1U$<TQQ<d$_a@~3VhI~`;_
zkg~^>>Hv^N%^ov&JgB?ddVyj@1>DULU#E-Rj2`IIEAoN@<^DjjFAj`IYTE=oodLCN
zHa(qTYTN4A+qsw8Hc9U`Xq^q4w!-|yu=OcuUp7;)RBT-*4lo`H_)c%U|BtZhEOY+_
z6|mgDeE29I|DPFm^^S`$5EnBS#xdNBGF>=nz_AVZb99VPa>7x51$-RX>BH<-8f-TJ
zne8irWo8`s!bJcqmgGMGRO#(v2>kD1v4)kvJyp<iH4HojRha^zJwUuL?g3Ip=`IIJ
zPvQA69QZuMFt^ajQ|u*QDYAP!sov(I%`&#fD1tF0vk>(31w`YP+0m=bD`E&K_5SWc
zx(%8FC^SREfDX)!0o%<QTZ69SyWRYM2pytPpjZo0(P;MOsm2k*5cfA8vc!OMP8D$c
zgkfD_gA^fi>0<^BQf6+M6XF`nn?uQiyuTCW7A<<?_CWnwIZc4fm=)+7`(oNW7d4-|
zjZzFu@jJ(jjpwh-XJ0b@dAFE7Z|;ro;u^TD23r0JLT54Fq83VESsq_A?8Ejh@CFY4
z&8?v5^?nTa4&F|9>?6>(5o8$BXF@1d(#``)<se5Dj6k8_<N2eEs-4_R7hfjJ<K)Pq
zD*o)_4=CFN#3hUrZID1*z=#EHA?Qp&TM9bU(B^1+<!o@bL9zoPPonD!%-Akj90RN+
zHmvie9KAz6?m@;saoClE)C6X6Jd0<xs<;87A^;Zi@CMBPYy)(b`Bf2fi-#z+jVlLD
zVL+~kJUL3mN97di@=MSm32Agvb=0WlyUcqEjMohte-we*M!OQ{piAq#16_i=Y5Z5-
zVvbV%WtgMjatEqj1|es{%e3^BgQY$b+a+Y3a-DiVAZdOjvI~N50KN3r!$X&X+6d_^
zq<X+lLP2P7%xE60a*BK*U?fm%VVEDWhab{IFFEq~1QiB(yaHw1)}YvFLz|`BY>v3l
zSr2VJ&{hYXorpGx=mgOQp>0sp279-ITmZpSX3ZWFXeq=2$D;s}#qp8D150tjT;|Uj
zR%sb9%apM`2W_z{f>DxEGmt`=y|8God$Hm2iziXEqO_VZRA$$7&EqHL<VypXwPpH7
zR~c(`%b0J>=sOn}WV_6KM#*3O?*`Xqq3cJd6kH`tv)5l3Q|2}%W#2>$SZ2te_y1vK
z4ttObt1|_&Abt(}1`l}BfjuvR!fmL`F4pbsVxHfEEQAR>a6HI+K6?uAdldi802|pw
zKfPqfo#2HEARWi5cO&%eh9!;A*$weo(ANV?lF;cuT(28tmTnmB@rw!QmFP%}S>{5|
zHam41oxqa?{$SSn2J`<u^Eyx8YXLrT&i+RRT@7mc8!IwR`P3o5^Xrt#LkJ28o(HyI
zais!V2pl90#>~4N+i<>O9q8CNEGG@5#SMoWF0%~A%78LWi1{Qqmjq`KA_^?t#{lpZ
zvy4(2%E!)t=&GAxjt6Q!@KS^8^NN`5?qd=dLkE@*7Pathm|o8GGvNO>x<)QMT!7rY
zD7q*NhhVsgO)^Xr#h~r8Hh`abS)O1oz4Vi%7mxfDrrC-duRs|yDpkzv;MN4j^#9vC
z_vpxu`p$o<?(K)(BdI0LQ=?&+wk_M3Ok;LT%)|+pYzT+I!X_tdHoIq&Ko$~V%pAl{
zfMkK4c!*-dERdKy7M5MugaAHXAb7(GU`%YV<+n9hevBl0G!JPs8qG)}_2b_Cqw1<}
zxB78gQupXl-*b9u)X%zgtG@O9{pwf0@<`G<*P|1aZBK^vXsBCR1sp6rgO?~NRT``g
z9DRpf-48;VSfczs^*&#u=SOsbW|8dxLH5#}DfsLc2s<f~3t;KIW?=v)1oKgtmN0BX
z>x}x*1M^Vt>h9MIY=^nGL;bcppdkuwn^uo+(E_<d{>KrIB<E4-ffj<a)i^3;+r!Lj
z!{qrhID8kCg0hu+=-@#o#W)L`O<v$UlO&lT&1Dw%a2u>EqFicD`U%oAyjlfF33UQw
zNGr+9d2d_>BtqAsL`kXApd5~A)ur^*uVd@9=FPeTH0g2c8NA-an8~?M(nqLi{cb@W
zA)@;I^}4TE1Y64ag5syS1$e}v?mH3qVIzzSNULura~gW))zbFsY)I6B-JnI{e^l>r
z_aCL+a<xVNHhp35q-=s*&pW%gse6v{4oYT#{|S7J3~&<oUJ{4<fl<cE@b9?9PkDjl
z;vqLF{a64K@Uu>dQsj}N*Hygeh;~3oXm#PDw;<Z6JL4*FmE%fm*CAb`%y!&Tr9sxb
z_7tC7^z%9F;S_J6o>$V0wRo==bUCe!-%zdxkC1*neHTPSjxv3PM98q3s;m&Rz^qV9
zGHKnRGB&s^Dk_3jkd3N3Z<4T8cb#5EteJNB&IEigtWKnsQ?Sp0*Eu0>a2t3fEwmCQ
zY76RH1ueTJ7we(rx53^s0G%m>>#4@qqLN(f3PzH6BzX$o?we91ISYJSPd#Za0p9~=
z#)0pUVu4Qs4{#o}y~c9`+rHm2Sf51cH)JicZubtrl@Jr^3?pf4i>y=6ysISWTt|R#
zi<Vi*TC6n4;*azG4gvMtMi<Ap4vQv|gche4@8HdOAFVd}o~!qs|A6{Cgyi+8z=*5K
z@CuUD5nkj1Np^8r>h6Lr!<u47^m8$w4W5m{+fILm*~v6*xa_nc7KTP42uEy0T|Mr0
zf%6(0nx~2Wg234_haEvlVfRK&jPFGGz5^a$*Q7*%^XM65xh;wcDc~Q0{{l>tLS;RD
z=n@X!#$||yQJH~ry&2hEkp}CTD7We^uvOpFo!VpGBH+A3cJZ-In-LAGYxQPaLCA8*
zWMsk8*2=M1X;27Tv~K%(b@T2J^lbqmcGgsonk{UfbrAB+nB>h#QjPpM)Vkala4Rl$
zbb(L=c7+6W^)@FX5+RipQiz<qXvtY%_A^{1=dmJ}`Hi{^Gel{!Td<lz_C%2hiM;kD
zXq{L6Q}e!J^Q(a0({;TmTjx^KXT1`*MD+gwSu9hzmE(Fy0pADiLap{Fcjo;LIK#j<
zxPVG}dB*g;4N8=a0_vm+oeWE#ThJrOEm}dgRi_(RLV`$R3A=EK4HPdoH$JMFu3b~W
zR_(cqp$?|j2L`9LGp#{8D?K8d%R1^pVYPZ<+sb^l!%!d6%Q+IFA+SP<U~EgRrZftN
z^eniuFct%n8O4yc04tnjE=`&bdSR5UsBp+SEYShc(?snEQClZpY}1*@90wx4FiP15
zau0|`;6G`y+**Y`N&dq3lfVGp!PwK<<o?+t4qqX|dHx=CnwY^xCCWv0;gxlIy6w^m
zvO`awRa!d{X~1%em&!mdT5fK90(C}p2k;+&{ir~0FCa(Xw2$bwS@4zxXz&irT+lo_
zPj&^S$)_soeI%7s6Q}!&Dp5-E8W4M6XIQO$FJh-w`St=Lq?s~eAtDxh6LlV)&|KcG
zf8(T{ggyJe&gTcJLxrFHD^2phq{%kA!lu)4OV;W9MJxZ`4)8PJPf@w1R8##F_~&_C
zehmDGr*!Mi89Ld>M7b3eAF&m+g<z+)i@YV#KzNMonKcP*n0bkDZ`S3jP}^a)15e<c
zStH#94o?>^BW21>pLqAfx}4{;B#F{OGr8g9yMWh!7-y#nTv!$s6UxoQ{Bs~jq4k&5
z!cE2odlv{Bcsc(Q-h`HfCJQnG8ZDTWs-Jmbl=VKhYBTD*h8uxbYx3T+3QgtxPB=S<
zWQts8R!M$L>qi#-lJ|n+jso{GL5e>@g|i!MXrgQ&8}A#U%HfG=O_rzi@V$&9oUDnq
zjMA)mXYmJ%DHqfh#5#3roi>pBKsJ<``}|3Pyv#Y#+=sN4!gK2(9fGMQh*(NxHB4l6
ztW}I=hckI}B-C$)Z5@*&b$F)l9@iiWY>BCDr6Kba@bN{G%mB{-Uk2ym1n_sv@&tDQ
z-!(mNGZ1CX;Aq2TMB985z(u6JkXw^VO9LM!feh`~G=Y0Ph_IeZJhwA%C(z39l0^~W
zQ(Kx;GXmzesCzL(HV(5@f4)ACe~an1EKNz@FME2p`Yf5rvAq`{b3bZr=KjUnjiOfV
zr!OJ+HkbJpUjXhlea;YN71ZQuT{bKZ5b35!yI{L4tCX8K?))=+9E&De*>|>uKkV%K
zRGfcPXVKaOJr*KtsWla`8{Av8`5I7yyZMCae?mt?IE}fSH>ZhmPLpPstjpUXNJdI0
z@BiLI=25tpv||<QiA-J}bsj)%4u5#@*)u5j>hL@c50K)Ee3>+VLe5cR28JkCMNM8p
zdF?wjQF`$l8C*y8`(>4zawp@b0C+#`e1RKj<IVOv*uk|QYM}#rF~^V4iy>}1D#F{_
zrCI^<Rb>vrTqK*bQSWff=(o*lyJ1$}A}^|}k?clUS=O?4N+?iSKm+@;5>(JJEFd!j
z`~Y==<711zGKtEo@ni~z2T1d0e2p0n8<I3cSuw^?`D9MA^lX|q>J05&4DuoiY@vry
zrs=^_pOY>pGtgtPymHgrs!yVp6TFzMHv?DmZVm!XZF97d_N?kwl%MZ4WTP2GdG071
z!tWN~tBg-hY9F`Tpx<gKip!8)V4oqIAIMXjmXE*7lV>sWy{JqquRJrO!2baLhHP?R
z6*2h_z#}Ofp2X$LJi>WCf=caTV2E-x;myk<q=|5XC{ucXTm~E=0WKQR<#P8>R{Gd9
z@M3%-sMxJRIPylY&qG@SY*~P<7U_3c$X0=TCD}Z^)q-1qG(9;PQV%x_uHFlo(l4jU
zCU*07^?2DYr8d>QBrH~*m|gxY&o|*cqJ10{{_~G`ak**TWB&no1YDlS;jegtr??HX
z!flAMIwto8;CUP@l5{dj1M^z>%mEA7*@fEWmYW72&i(W73-D>6{TcYp*K+<<?XI&e
zBJ75G@Ft+;n%NZ~M~S-nwlC<17pJ8C0d7=njrH@Zw{5@+=bR$fry22mh4h*3`E#YH
z*n|i80PvH=ztY+g`WEn*gTo0N{+fT`r@Rjp9b~WxA(wWj3|S|7KpfY-dK9&TXBvn*
z5KY1Be*n?*IPqP~oNL5w5ZEuOUY$G+Ny6}g{+{KQCGmjPuEomri_ZaHx{Zk+zWp+L
z=6~YuNuLl_%M92_uu|$2t13biBUsn>@b)@i)Ly-FsDu%-ni(Cn3&(qFO83B|etA~E
zeM-A_PHI=r2-$SMXIVT4o20K7yG0(^p52v0lCJ*VuLF;P!}DbLG(Y0|{AWXwMp5SV
z@RxPhTj?N6X(pS*p_$8cbHRCu#%CnfS+LrHIRS39t^b=h+)evyMN((mpz{XkxCNXp
z=4FHX7*=`#WEx}>Y6-<&w!vD?XBnu6<eTS$tkcr(I~6|ZsQs#rDEt*EvKxzLeRcI_
zeak0gy|_*{M}@u;m+>59WFB}9xEB~s<8p!&pXVNa%m+~Ug$xW)dTR2#?k+Q|Y#((7
zwe{-+YRgX}V{VlB%YvShf^!R6B3zQV3)OZsKCmH(^mDpZZ5DU(HS9&L*F-_KqT4hx
z1on`Y_Sd6UcIteFYxPWV5#;mvtmiKm$T^Nj*x(AXNA`rWjOY~Ub>0;1uTyTlnwRwL
zX;dQq`@p%F0v;#H7x^X+@<BtA26;p|stgVys9iJ5B``-T(osDw_A<jI&ZZ^NDM)(f
zk^rPah^pu4+Zc-SeW*mY>YxO866Rj_DH<*|L3l4D-ve^5+C8JVW=aqLU8wzl?PQls
zY*xAwR!D6w+Xg8_o`9S2$vvN=NJ#DH^j3WQxe7$FpV)%%C4}^T@)Rok=O^I0F7PC?
z`~`o@<9q}Y0%VXS$_1_b9@1W(1ljys%fTpWFSm=sHs<Kyw0j9=Av^(go9@(V?;6sn
z_9n!Qp(sI}I|#sk2cY9b4A>7%ZzOZwR;;Ebh+6)WVH|xwMp1!UF;vnFZ#~)e8KleS
zwyMD`EE!$Tg!fOfev<U1Sa+>$xZ;BPzyDB3ot5ZIz;ST6gu}y(^0(Z@GknRAq(L50
zK8?zHI*3DpG=ro_;4nxwu|)zEKJgOD)qMlXm$^@Sb>hDADd6AYaE>H5F~lI3Grz|6
z<AW?*ek)?@AWXd#tU)Xe5_0l?r_z#=<*@!=v2xvDKwFI7G2Cy`&3ynaHex;5273t+
zV;dIRk@|FWxtP1$V&m-l+yU<S!ge=#mUW3a;3hbG31XZi#sB4g{+@oOb+oyGL1EK7
z>!rLzz7Oab)Rrz6wNkiG6Q#_X(pj9+GuY}Fk_?0|K*JcM>vFdodU+74sTSiK<xU+!
zg#v|8S=Al~egWl9u7}wLc+yI7jU2<3W4?qu`;fG2)4MrOO|Ca=VKvTEcmte1>#gw&
zGs73SoA2`}ToXucP=+YIO@?Q6nQQf2h1vr*iaLSrak431{NJDmGfj$dPBG!0q;<|g
zj%aJA1w=IW$K{H~>&<z}aw*Cw$L{rXw{FY(`-FZpC=;9(I39uHb<i%!#KCIfIifu0
zJ6}_x_A_nQrnu*`46(S0dre02cQ215^UU`f;IyMI@Eq_h?%+`#=bL0bp#}!UiSiuq
zk_&bQS~DQr1>ip55merky}+$xHzpRQAQfW1ohf$~=6A5=q0`LHfV_&_RZaA90VNZ}
z{8L_83WZJc_@nyeV&z7_tiDZgE%<FXC$I)>*e-FmgV=%>yzL-)5y=Tnir4B_w)^aQ
zk1+G2`D;a>yYO!%Tn+4p@l-ZV@k78@i1R}(F=J@Jpe#|QNN}1#E|Oru8Kmb-f`+L<
zq|*)jA3ckEhj=geNHV{?3V4>Y#2MqqxS3mNdU=rOg#`1BgAhx=#2^`=NKB9*<PIY3
zpngaXmH9D;_a`=vqAZ2Ws5P`}G|$yTvK~fk#E68r7W5s96zF2}CkVD6X+ec?HEE{Z
zqAi9FvYQH9^t{?eHUn<7#7q}#&y2z=QJV%=KLp}+aOPsJ59urH=Uz_h!*2`>f)M3N
z201{25qFTD=>#LAgY;b3$B26mJqSjar-vpUr=AI#_z4T_AkG-W#EA32V)-mF;=If-
z*Ja|k&;N+{$zNooZIEm06O4ohx!FlDA_nOt!3cv~WhWSsgY@XXi<eV5i8gw;b@6pd
zlm|(`qoqxAZrM0DvmS^Y$L&0hXl+wsG1?4v6s#!1TC&}98Upc!clLtNgy|Xb{`ky@
zPlbv+Me27%-=ih)@=5XM;q*+_uzp#8>DN#@97=B~+le`o&R|)l_aMx{%O~JxgAf^k
z&U%#shSvf0yhMcOS<uO+N7+K0>Ba6ZeVV}rw-f8!)nvwJvmNtN5m6CnpLU{viLDUG
z|G0ni92GH#+TJ$?Dva_$*t-MTx}l{_9U^B(j$qlRu_{_bTN~nQKb%cv3Gy89V^nUm
z7bv$wq6c%Xw!sRBa)AV6VVDZRd<d=)(3}E$4z;9uimmjrlTprTf(+qbGj$ewE~F7J
zPm=uEStea*YzJoxg|}3D$D@{Ydlfgs<78916~Y+6ql0vB5=FTLX?S5&J%}r-Iox$P
zz0<~yS$%8?K}4v?1AmsDwRpV7K{%CCt54pM<YQ!$n-)Kk^J?9V4GdO9lrKC1|N84N
zB%oKq(9c0{hSbmvJl;FZiJ@K|=^du)2XT5%4|D#$II;LJlVfq{9)|m_g6?PGzBqKB
zfT3I2_DfF_y_{g%4TFpfCE(UU7)rn|=amO^*;}}LDz7Z~K2$kEf}Xtc;DIt`_eMhm
z<!bCixfdbpM$}P3N&4&;wY@4Jy-!DIb!w9IHu%W`gjp*BeLb9>R{GBi4WB^;=zbgc
zA-$Yr+8EFVH6+R&o@WF>4=y9@riUaW?5B-ko+VC_VXPP8gpUuC%EXDa4>P$VPVCGu
zlOj%x=a?i;ha6^ttBCO+$E{mPNKIO-y!0!|oYx;M{ki*ZNpO%&M3xI=H{y8MhuZGl
zib^l8xBVq8_&l6zhExlLJ7Kl~T4Qjj89EwazEzza7LI_^qE2v>Et+?uWWACUF&Kgp
z>>J?pc}S<Shvj%*aY<jz1HeDAo2N-F=D{#9s2x!%lH9`xb@XrpBfLl#D%!+59;IP7
zSbHIQ3esmG_B>41DLZHqhZr#7K#VApvYuEHhQ=LkXoC@VY1Z>f%ZquHzTSI%Ka31P
zPj8XW&#2>}4mNM%JN%sgk3)(myMZpa*viCI9ow*(#%2mQi$EedsXt($EcTWtq$IR8
zsnFLc3)=MIjB^qi_3}>AQhN$Q0x}j@Ua7y&y$@2CAeGLZ=OBGu&C%-dtoA!Th6?2R
z8P_wNlkLL5pr)ob*OKB@oaRMdz-Ce_hVTOH`Vm;qfc*nV?*M)drcXhv+j9U4pK`FU
zX(PrYSMPukSHccQ<?UTA9@PHF2eVI&6npOwDkk1nb|2V4;XkW_BzLFz%ZV3x4P%@^
z@E2T76pQV2(S=1b&D7!2h(#-sW}t3Ld6@;037<K9*5}fk&vS9+r>F3UFpc`IDg7AN
zw{#elZt4^;&viIvkBq_EBFf%I7_tfVf;9wgFE~Rm-v!CX;c^l>!!X?dv6NOLV?eir
zF&DQR7!$C&f%Iqt=el9yVbu1zA-ROSDlJEPbCqHxpC5Y&Mh?J1bDx1e0Nw+?5Aew}
zgZw#vNHhP97Ve=7hb9CuvSE~M2pvY*Kr?%=*+UB*L}?&Oh$t3zP7a?eQHsS=A5ut?
zs0El=O|WCyq@DsC_Tw^zYgVNV)+|w;7>e_g-eKfW9H)1fT|;rwJBR7|ew;~nn9gLJ
z$);gqkvNm}!^F~8G1>hrv9UOl-NST@C79?QCpLNu)7^t4#uA+F9;D}y1S1C?rf29z
zMtTS78A>d_ypR`Lc~kJuAABeQ2R8F9z8(L9@F$f%m%yKMfv+)<^X9@yfjXEW!V?Id
zpqcFAP#xKGdAz(E{xC{Eb$a-=_B<Lh+F(5q<ypX8fb<x2J`Kr}uq_IcQHWiyOB*KB
z+9x^g66;p`C}M<}lrdtQW)kSI;kiz@{u=E&M6Sp;*~3yrslGSh=bPdFO|K~PqGs-5
zo-=f!_6bH%OF*SDVFrr9=7w~Eb_+T~u+4&y^m&qIVbQHWUfAd5Ixph|O4g;JZc-Un
zs}{R4lvh@Q8Pu*`Uy%Cd!!a%W;3)iWKEPJ);9pRAfEUaVFi<)C@*f5{in42Hc(Chx
zai-j1q#K8;hanS(R}8~g9CqunJHLD$F?JAkKL?{XV|5SGI-cO%!9jQ-f$SZ`$%%P<
z7=~^v8Z%k7a<j!!F2RTJ)>={^`7rlt_B4rUt42EcIJA!Hy58EkHf?K%#^VX;4nsN)
zyM`edhkeD!aTvy~hVIjdvA5uM55h|~BDx2u8%xlzdysRFCg3JTQnJ>hSg72zKFT|k
zwAZHhC)IYpy}<t}W*-<-i*yFo7+DT%pf;{e+uAXg!O3mf54jIw<DUP~U&_V5J8+m_
zh8Pn};;`3I(&W{^NFDT~w13cr7lBvc`3?R4N@vBfl|f78pWKY9Qk?;Qgxi1Qkl79H
zcljt^WZmp|E;g9gsyaIqfGpQa{SO5s_qEpMhf0uTb?a%fhXi}#N@LhwK5&02P-(E#
z-!2j*(b@=!4qAZ)5^9yRJ)zGJO@ILD7CWNhMwAz`{80|8TW&TS!6DrU_9^()|3~We
z$+b%~>s6gqs?yd9VGdOI-b3r4jjLBrn+Ro21;~#Wae#9mCrP<#&u~hFprH*KT65o8
zGT}vSb`Fo_D9g=;!+Zp;`ZKt=@L|3Kd<Kr)tx~1eqt<f%bd4&9QiEim!q2XSI#=@A
zp_<mWuRxuSt-1OJZR6_H)1K^x*cb#W1(Um7Vxun8-EiHQVCuRk*G$D2xv7C3FF@QY
zpMjBU)!#4n!YlG$uSL1Z!-I}~gz;3G3E)?>_XM>H-VXfx4W`zv{51q4owacdeamj+
zD~~>1=VLW$SDEXJELI9#x^%fpha5mM9d6|nv2Bc44TXJ#f|~gvi<O&|B=3gvN8m&%
z%~9aZS&{^mcy|ny(R?MOR11O}ST_T=uSgsARef?;NnKatw+3q3#(lou?5j~-%h<m{
z|C=?xGYBqI$0R#QB|G8eaG+zEWS;RJ94m{+9GijrAKH9G`5tIJ4144qykA85DT|Kg
zPof;k4HVD)bL9!DkAa+l(O1!+sD0g!mHF+y8V>3pb<I&?{YxKfvCWT_n27qS+^*ua
ziTAqM{o7tCV9bSX5MyNbb9dWVV=c6Gx6(RpArq*8^lK@<|N5ES_c=o#1Mk~*R<>_^
zlP1UbOZex>kFiT8dGKa<_p4#y7#~HQd-GeUV;z>`7pgu<7N2gN3e2rTHLSB9fc=I1
zA4{*}SVaQhP<86tUxGA_79*)E*XDjDx!BIju4iR+6|a5!dtF6>ERy3=(zKzw8OEAn
zPXsXq?tv^Bg@>TM7g|b@)iuCzR*)oPnPgcehH}Fa=%=-@y(I7VA8~}#6A?bKGlh8f
zd*K`9J6SEsvX~;1fvk&TSETNR39@eq@yI~dQq`}aKY)wiXud7z1Z?vvws-$P;Wj=R
z%ywN#yGmtVyDVMZ7O1Vj?I3pn9S9<j0e2Rd0FhB^r=3BxiGKy$t%8@J7#6oNZZh1a
zt!&%h_8$ULL<sF}LA3rB@*BZ?X?NC$AO{MXgMIxH`klgRtrR)c(hNUZrXKyWK>g*F
zxrU>_(K3G2{`@-23hrZ-`bV)g_WDYhI`*x=#iSHv4wKm^`fF6Bun?|~+bm_dS?$my
zRO0N&a<1ED1;<1?l@n+4CR3CEcL8VgEj<X_*at^j_zdrNFS6y1_rYN+(9BzH#V}A%
z@$>~GrSgib2KDYQH6Ue~YpAud94J%Y0(CA^-|DL1f#7X?l(MRQIdv7UeS5#F*e{|9
zGKBwR#ezChK+IKC3aCZYc2pk2nqe7aS%@e%P4@G{z$HFQ2o4`18MWT^yGg#v2YsC>
zu)pU}b;zcasofv&>yFlnEC<S~mH;ZJ;DA0_O#&JyUgyB-(Fxe*hpN)nmAA38;&zp=
zO&iaiHk6A(&&?yquu$PbJ3W#Vnp?M4mn5B{GoCqLDmX3yHfi2PUBu$oTlgx!n>R4}
z%b6;xZC<T}BqarpKr5+62Ue-ht4)%5@9nRh8D2R_*3xA#unHtuxPMn#SE-0m7Ih(d
z1i1y+<tpK5dnR}*upJd?Qbov;WN8o_5gfWRlOg`9;S@fT;7X8XDY9|$d{5=>d(eSf
znO~}sI_C~xxy-&P11jj%ssHjwvQ~8sG7&AKZI@YBS!yXKDpVlJ4i~z#QNE}mci?-n
zig1J8LZdE=<A}CAx~5V`Th50pzHgNZqwFVu4-1yR$TJXtY${VT!4=tZ25<%B+P{jE
z_o}MnazXH`L+pL4a^XRg<Z|jPm((?=vXrim7zN7R*d$=5HnBUjLB9B64E16|jmCHZ
zN2xKzE5#HLBE|W+NSf*2sOPIob}#0bvxNz#ui)f#=ZXew!NJuZpl&rG%hCi{SV^vi
z{j}O-xf<r{O2~3x_4>!MNis+uD_#fL0<;L|5U?Ay^>2r(c+mC}WSb_)SP3OkBeFE6
zc%?{kC*mpiO*!w#?=?*1o82WysjMmP1RMwhD<sL4nS{Z~GFX=HC4+Z0m1M=Pgq4tF
zb?Uo3vK$CVZnDaCU0N5}C7{cNHdmb!CH>?S(Ztw|TJMQm!LcqX)sIG9`Y}hSPJXOQ
zzA43RG8A;OtHf_vd8VAc66XGDn}p@bGMIVYw`ykVYF|Tz_oRQd>%P(jlQQm@RjsdF
z(&x(cX(?U(F~9BIqSfPWO_ZTEHtGZUEz9pH*j(5WiX<E6I>RjR+bio-r8`!zv{&Vo
zjbyb=!s0|13`K*otA<Ke!UqBoPjyIgwaBtoq+YqQR7c#mq84WMpi+`=*QS}ZM#p?$
zzhyA(xv}xb+zpTOiS42lxzIRY&g`zfLJLNct2G!ZUCj@uBKxaBG%IcUfh$OoL0l-o
zCfAy)_5giavR#v;7e*=8$@mZKx7_UN@*>lOB64<j_@d;P{MF_5xz#WQ%aNq4*?<UA
z_o4(@Gm=~hS(bWVZB=kz1rupu-){}-TSf)TMU#6yLR5PDHzQ%DNg<?Uhx5h=Uu9dZ
z25cG9>96JB4emlHnO>~Fnq*#`wqM;9?$spA0IrRiwR3s0+&Ig`flcvENSJp;LLoXX
zx3_blV6NUO57_c6+ch2#1?pbXcX>rrWbpd0Es`vLPx~sVYrniQZCj*IZycg5V2TGP
z%ngq7Ei~HERP^F%U2^b+lF}r3CDd63WYd48Rb+tr1!?yyM)JXZfLFe*loS9Piz!6m
zTeW$Ya{epF;7O9(hzahKzfv2*=__Vx=a9}C>3aQz$Wi)yV{3ef6d2Tf{(IEFMy{bh
zfJ>%-t@aC9`uxRpZ;Feh>T+|1q+eu?P}pW$-Bw!;5UZ9t)43v#RyIi1gb7QbJeSTI
z))A#`rM@-S?mnra`$*rK?IU?@yI5am1M^0K6iAzIDJM-DAp~199nGAjt_CJne>qco
zEtBSYf-KYSa&5M8|9ZF&OY3WIc&-pENfNwy2-AT(kRe(YT02Nqw=d=?hjm3rg&Z{N
zg+TKan_PnmzE~TxMi$A+*LUL+Wxl!;>t9P$gi9gWXg4N9JFcWzxE@+su5pulrHXA0
zwQ>JCYvcaS=CUi4k$3~+v+y)&TeOSFIXbU|St!@nfLM3+UFR-?^*I$8!rW+>g|LlX
z7Yo~;m#t=q8W<R&TnS-rP^3YE<+L?%{EB9NGB7YiX|Pt|9>o!eOk=FJ$G#aD7@{;-
zSGKopi%eZ7KQMRNz`zitL1EilSi(+++D%g!7#N~7C=5p+!L=GqQy3T+qBJOisFb#>
z-9c~$28JjN)*)PB+Zn>9Aq)&EG{<si(h%i_MoI`vxNG)E4+8_EC=J$vqO_%L?y-Tv
z3SpZj*vPokMv`3m{O!O#;CIYiX<!gQLz+>tff0drDe^aJ-VW?FBxzt^OmBn58@$91
zeJAh&a5v`6a{~iIl$#zYm_rmw{ULy~X&?gwLzD(Y$yg%)9v=caf%lo)Z(tA*n|?88
ztjP1-8{paw!*(pTlao+xfFpeH)J{Hk;jO&y8WSOAU|@)HS?Jy<7gE5wJ_6%AaA)3%
zeT|_=0|R4vue!=ycW{7EzRJ9P`$_h;+{1gGcFgTJFffWT5N3)z-@E~??MR^~??fKH
z-OBX*jm0~D2c#iO0|P^p%Q3y1H_DZGF_XWyXgnM}%=y;cpPG8dJGjPhq=7*xb1Z9k
zo1s$3^q$#BSCi|g>_0;5`Il(>`rmQfF=3Pj27zFRazi6Tp%c^`$XFRNrhN<y3{e`>
zyMUrR@-fDDv~%HY@8UXRdK(x7p+>Xpxxr2EdGfa@9{xDnPoAUY-me(b+rS_wHvM8w
z_XfVUBe>%`VCL5#CuZ^w0F$HAz`zjY%CK=vZ!1GOPf`NAfp=mYF9rt7FvJZ!Y!K5s
q1ohyGJedijz}GR!gA5D;;{OB7l>er~Hq0{s0000<MNUMnLSTYM^WhNy

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/images/mbstobon-ui-1.png b/app/javascript/flavours/blobfox/images/mbstobon-ui-1.png
new file mode 100644
index 0000000000000000000000000000000000000000..64cf3cbf322e9a636e9fcbefa1fbe0786e27172e
GIT binary patch
literal 43609
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
BhnD~V

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/images/mbstobon-ui-2.png b/app/javascript/flavours/blobfox/images/mbstobon-ui-2.png
new file mode 100644
index 0000000000000000000000000000000000000000..b767a9122a05a7154fb0c6d1cd3bf9dc81627e23
GIT binary patch
literal 40376
zcmb4pRaYEL)9noI?(XjH8r&hcySw|~5(rLkcY?bQ?(XgooZuEX&-ea=bJ1P<s=HS0
z-My;2Dn?aV1{r|>0RR9X%gKII2LOQb|KSTbnEz%}lHIre47`i1uKRx$_`jh-J+$&4
zpz`~!<00v0Y3kwZ1^~cix+{*W$ZNVNGRiTlyC^9rD6!DP0L1-ZX6*d1hXH_G0l9DD
z-+eYNbG>~GhL^uYIla5I^fV_Mw~j*l%kaZ&%e>*JZ)p;ben}+rMkI#gN+7~V6Ua(P
zU_j8PhGWRXal`lEhBdG+k7!<}GwU1bId*XNiCWcqE&p(=|C@KZZe6n7<L!Nu`#M9v
z>yC0nMaw@fr|g!TlJoxqFLMr#Sj3gx8o7sIv48>>{JC9B`~_40EoAl$a0)pKw1#W~
z$s^`Dg3gcs@vp00JN(}kH59L6?xJ<8fcp^3rv?HYR0lv7h-*(zY?)ulL$C?+|GaR(
z<_`d^R$msgzQXCib=!|(gXOE^{m8N)XLF=8Q*tin1V$Il_r?N25+>O7fHem-C&UsM
zA4DI>7qm4Sajs10Jbu75(DFauwiRs2m;Ryf5yR<V+1G<Q&@!y&o#5v>Jw@DG1G_Ez
zJ?Q70`Z>!3Pe2^B{Oi3`P#?gTD(LY`&*MT(0%fPLQC@SR6v7a{f|ZR`1*U~%-Nfz4
zMi`$(yNa8Qgl(jfu@_nQWXOq}Ba4d(Ij|_(JQ1?&eKQmZjPljz=k*r(YN&bFvc#$$
zS9L?{li*Xw(TWVa3S%HDrimU@^HK8xk?IK*t-fJ)0N{lPAVT%3!!rjb>x?(U@41rE
zSmwuEzg4sjp?G9nxh%iD0zKD)BXkOR19`y&qVd*?=Y&n=A0JO2XWCOMzzaz~C8z9r
zr5JBPP4pvtV8p{ckU;d#LVAxPr*~MFp(6uwnej>gU{3C_c03a>KX+#S=8x;M!qEDI
zS-E`u08bNz&OJFLS2_qa2U0dE;l-5m14Yz6bCBriggTD))||`;(oQ6LxZoGCTCA?E
z(@a0TCA*c;-G%JDN8zz3kOk^Quf+v<F6nWnhD-(Gt92*j^+CC6_J1riDX_=+L9Sdw
zposwIKX)&G2^;6@T5%6$s$?a^G(qXaNHUA_6xmY7yrY-kLe1S14@E)rVCa-G$1j-W
zpa9XyY-&D%Ewjy1^-5ar24>SwzFOQ4BqTl}Fx@8XK0DR~RPJK`>B1p12jTTuTo(=~
zBU(MwBr~5NcMgoXd}>QdXhge&RHO@_o>PN7N8aO}BQ@zxu7(<F+EZ2NgxTu)s8ecf
zgbE-SKq!Kbc8UybLDNM^Dph?T#|nZ=t2VKKjr-2#hW3jbJ^(VskZae>`GXTW=YL{v
zwRy#YApaJ#x31)Z<@@vs*n@o`tf(VMbz?*A3rw)KSLjSPbZG)D7s8BJ!x@R+lZX0o
zRfilo>~Z7%(?aOkia544H}F5AxO!-q>baR;L)P*C=w&(-h2bN};o@^l)J9CAR=q_Z
zvKlP%23rRf%Pn(*Zc44siQkk@|2w$&>52d+*r$SMxX1AR@BKd5c)R9-xquJvi@-OB
zckw#YJCAhzI7RQKKL0TS|66;lK*RlkGC$d6$ZT#qVpUe8L>}|GHl^$rHP}SeL`8`z
zbV<`J*crsVnMl3t7^)`lj=4z7#{HE~LFWHtu5$o+4R)F88lwl7f=FXdjNz0yS&0yZ
zdWZw=xibKZli==k)q4W2Uf8ozFoo)0hnnVlwryn2PIvJ|Eh$5IX3~L$j(R`zC9XjN
z%a0bFlx`GE@w(BWe7FB&FhJT8iAJyillu#jD5a;&f?nu5Z2s=g;E3|H8A{1H#kz<G
zLYT>it_-|4ia3eNH%%?!#|9M4Kf#07$T#|F910?U0q_x#>NTHy3)!K`EH>WQxXH;B
zHP~}q<r?M#eeCoeJQ$s!+2F;ci`b3Q9S=Y45P+L51U>FskZwkyEL?NCgA@O3Fc5lb
zrt^iv<WGpAt1^P04LNbO>cBAalEDxn6z8t5yI3&4v#J+I*%@Z=X|F6Kg^o}g_J-J>
zgoD)iyBPD|Z9YGECqJI_gJq;e05=8Up&=?o#Y{(nv?x<XEA*J5KzYy&7pG0EafGgJ
zTBTp9sb;!o&@o;xGu14v&@d#_h$W;AzU+64YE-3ZJ(lEe#~JbA8Rk-#q4!$}Xuz?T
z=v8dV6y(#jd@+~5OBPbiX#X5JH-5eeSY&q_^N-pvtW@y_)Y1I9(Cd`U$3<}qQ-;Rq
zmIY>QZ1LR0pAsPOkV_FF2uRg!tcy6+fc7p_#wUwH*Ct92{6xiDo^DwnX3}iyTf|`#
zZLa<GmJ`$Ku18tkFZu~@d<(=nk(89<6;sS8gjHtvk+L#*2oUE-%JCqp=-jt(=9(~B
z`BGl-|Im^dgDrYOZ&fH>V~LN&iZcLG{xTE#rNBJW0*R7<w$}%T19YaLzCJCk7?Snb
z4Jk7u>mjYs6)Qd;q*5`L@9KlvijaG7*fzR(Q-({0(1>~Suy?b@9Q(dPx-QqTUXa<G
zy(bYz7~oEt-_#ZOW#*MSW~2<03W)2%*2+PKg9^VZ<O0;TB_oiLH=?on3%0`1{VQOK
zSNG$R4_+bE&|RE|<eN=%GW?T^xT?<Xi9lid-rbz>XKg%JiT9UW=nU-9kJ2OFC3MaB
zBf)I=3jz7c;y4=Vq>_P<q>WM81b1Bq@fR>XEIaV7T*3$&D~<|(^0dPHA6>o~u%(&$
zSIRlxI-ar7FU0#EPgYRn)k-7Q3+F9v>mb!#?vf0>!v0Xt-q1gCF8D<rR+F+aJBXDK
zBO4BPT@)vKmD>lvJ+UH#_R~lQtBb*kP({=I-f<32r@3=u*WjzlK%ER^bOYL(Ak-58
zon22i4ryCx&hh=x#ofl^9!c5GuMFhh641F~youBuKZs$1oF$+LV@`>J%tr<Ir34!2
zcZKdWBA~Qw0(l1IOsjpD#OF$gc#4U$$&ba%kNrb8BKI0iLrg(0heRjwH;a=r-yik2
zu2iy&teg_jT281It_SR&_GSFVz|fP;3<}CkV%2L0VfQDf85mCrA#MT)aXu}JG7ntW
zv)^+R_F_FhD<N)wc5I`lnzk%bDCW4BllrTWk2Li$js)L-SDxG=r}m-wFexcF`)&m3
zO^V{L(`NdD`H#hOg{ro}-7c5dzDA#J{fGW|n87Se0+7008UNApc<EQX8K=3<C>4*o
z6y$5}cy!Im$4xyaFMM-iNrblUU{7j&B3FI#@!WwVt5UxmodL?cW};2&_eT^L-jE<d
z(~i?Rfccf)yU3=kTv4u6F%rNS$O1WH$#W~TH;Na<xgC(+xw$JJ;sp7K^5?xja0w5G
zUC>hmilh;f8zKOoj#5K?u6p)dscB;t&g4ATQZH@bnspGi3n&p6*;jWR;nb@+JUk@4
zGdz@~t%v4N-#b7h9C${8_r~mjtAA>aPjkmLaiESJiKr)bbN1t3%m7-{_0Z6O;AO`w
zpjQ@AifQ=!o{m%D7Gg-N{<hbSKa<@gWoq75=0;GtutitNuS+H|eqv0^>XPbw12Pd>
zv~ufp?v()^Ocn3k@Evyn`0G)yW1`;}psW@1jl7Fqd6>#}4w<frRIVCc=pQ~{boQ+o
zqg?z^bnlK0f`K)Zd(~O#>ediZgw(WRmyo&pj}uHsIr>{;+*AEB$|Muq&$jF@s`ube
zHgG+wZi3uq014(M?O{uiDO=R2Zsh$8c?qJ)yu}N1*{!RafA72CkFy@eGNjmgoah3)
ze!Kz;=A#a{QE|FMnR(`sJ*#4F)Z}$>5-6IycHCQgx_=zEjC*}9J_I}YKtxuM=3;>e
z{O}L)Pt2{a1^jL-uTWSv@zP4KyQ|Bk;wvq|NAb%^7hO29qr#a^FS@bCQ22A^7SuyQ
zF2MS?GGfP8=)-CBy68yQaBMA1BFisQ+vhL&cM@wtqsU#LwiU3bnQWW6pkv~*`Pipq
zGEGcm`ek+Z+dGHPJKCbap>DN_({p1JpmddENd{e>Bus-1C{Llvgk_rL8eNpwY%h*r
zqbO+(+stjJeo=GKhGvV=@Eym*NdSOZ`ObVx%6F$y@5g)Z$_N^;1y_<0fPJSa4%|<O
ze*Uww1f#Mc=NoPHw~W^W8ubVRPLTMH;9Q`@c07)ddk&ITZT<!-&g{Y<Cf7VDN{sWQ
z2#Y=FYP2KaIdTuOKe5kziV|t~DjZuW(x~g7RIY<%5LQ|(Z+Gs#Y1f1cz^1G0JwkV}
ziv$X*Wm4@Rv2Iz+JxOwy<`NI)t;vDh!dqfc8R#S~pOKd;)B}963h<%rIO-AENTlGn
z=xU-NnI$ypupZ%cTIEs<%1mkznF+XR$s^k{)FJ$UROR3tswue)_#g-Hn#e}OjRMFB
z;+h%!ufBQtB?*Cjs{I-1=XJdaeQQ6z=+)VIA4*;~-p^#X1y~W#O+0HroGJTMYt<se
z51j2qlRyizJrcYV)3G1aD{Z_RS!3o+pDa5g8~cFdst9ae@WQ64xPmtUbra-m65D>U
z0KAe~bOM)W1det{#&8)u?EYqT{17r5uPsA=pSfcTac$I)H5MJE1<@Kw@d`hF&SnvX
zn%^Ceo>-6uxjzH)5syNWy`mU|hX7lMZ4-|{vr1OVrG&6zH+&G!GCj#Xr7>3lT!p3<
zw|7~bTN`0Xz}kl?I9cM7f184E+Z5YG>&k|~8Hgri7uI%|2WiDYesvHd(GBxJ?W%Zo
z;z;h_-=_4K<f57HoP=m{2ZTt&tS(NX5+M^=oF2&{{-nJ6n0mM19-^Z0IN%;}Y?fBa
zJ%#rnN&b~&hRzre&*W(m0;xc~^j6SAlmm~JflxH@fs~1=)<=I&SQo)QY0H0Zp@$mL
zwy}qUbXA^4+&(^I4WJ8K5#_{}ydyE^Zo2${i3z+Lg<#n>tP0JHvO5;#I!24r&+Zq;
zgtpXik`J3Bh>+OC`>BqgZI_j5W~km&AU@wJHb__t58*4GUwIWL_~5EeAy?y1P&E!)
z&D}l!nkPMHdOuu7E!KlHZcp~N?3JZuMZ$UVl}_~EnIFg4dkCC_+AF1w`=V;Pb}v74
zTf?IpN;YVeJp_Byt{{btT|tPzQd_&CP%YU3go;tW)WUa$cW#mU!Rfgrl~O@{<{WB6
z*xQJb)3B!q#g84JQP_<~^I}p3sks0S?1lIWP<<A=V#rDt9Y~?I|BQx@aSp@j;wL>O
z3uzb#Js`m=_wr%ua)Z4UM81=xW_|3D*bxEJo{7$plA&GYtaMI>YRZDsIu~oGYSdCs
z!4>K{+?YGe0AXAkv;q@~dfXr^+qqba@Aa-6Q^d5U#i803g{IJzRu%CkYiIAkHi*@1
zj*qV&4O&Y)FjNiPC#q>Zy@YE;*TF@kb06Qu)FH}45Fn$;ay}0WiAZMd40~ZO`yz}_
zfrmAP=;!Btg=#-R8-VN!@BOPITc&;4l?tYuRI6ATs`hm0vs8lwx-L}RUL0xuXm-mN
z+AF1IJ50;hIyM~6{PGQ%hMs*CvJ`N2!F9wd;y#JVvYT1L%Zotem0;{Y{AIcb=ut>!
zJ}sx1uMt=S+WjcbftTkuyNSQ?lT6CU1*!9NSALoj#c2|y;tIHhT($OzoY>316+4)8
zs%-oahODO7WEtOcZn3JEK+2?%>Oghd2(iUIyOUD#mhsMZ^&5y`V%0xUp-@L(xC5t>
zE-h+^*B?+^t>h|@qGi9=)@^<ce8b-WkwPSpi!i(W(ej4>u1h)w1p?RJ{*|~t+GEuS
z$*tant|izg_bycXgwpCodZf3>_-5#jjhM`CmpZAZr&CU-Q+-~Mc2SAzNc9h60K4LV
zUeygfHC`$`3=i6k392ec#$hTq73+7QBEkf{85Sr0pa+ebBKGvrz42aV-tOcAU5a<I
zpC^msv)Ap`fLrYz5)SJZ#aU$0tdYR57fLF6R|2us>3M-WsZdcSD<To3MXYi<L|c?e
z-yy2Xn+%DhG`zYll6(#N+*$c|+xYY<L{;g9EUefTc~1YrqBC73vh9BIlwV1~I(SO_
z@~`u@QUvjC*b5O>HNKp{DiJ%tqD$Do(~oY19LT@ax7jr&q0W|H`Bq#s&hE+AetWs#
zV~KhzM6PxT`?}V|Gs0V2;4wc~2}BCStOJ#5BRyjIuDs>gq1WO5wMo4(Iq3gS6RiV8
z(D%bPI!B+GX>lb+L@Hr%^Vj!DsS8I5QP=`yYL@gzI=Jk;=wBt8;k3-<J`TjxuhgmI
zVRWbR(tokcU!gUbERm8SVw0xoarA6ijW4Y|LV6xODmLfu^J#rgg-&K8^1Y#aiVX^9
zY(T7LcS~cyV_Nc8u%e>h-dMj~0>&ZnF7Dc>9sf?zy*x@uO-%cI-owzDhTz1LDC0f4
zUM944Kq?GU8jNFLs@?)Lo#xUDHFj23>Oqaa7^`<BQfcO2e)AnVv93ffp!Plk<ba;!
zy_4pYZslG-RWl99hd>XDs}DNf7=%pNDpcR38M08$Eehpy>7`2WLgSK;Ht4nyHE#g>
zA*%RLgYN_EXKiq75EYMKzqb{&^|mEP*o*aRH{**`M66#!RmhJTj>3~aw)-K;thx$w
z!(d*}S7(wLsRW@%JeY>|so_$5Be5<Q9UGWrPSvYMHVMXu-<OZpl+Z;Q5s2(!hyNw_
zUy5#Z=Zib8?*BuN`8Ev*s{>-f18)TrkoDh~{7a*9cHS#K-GB2<9)*)EaGdkF&q;8*
zuuWVHDP|MT46?Qay1<oLE+Spb|J(>`)Uf3;Q~+2;F(4AC6SNWMuw^1JTi9kmG+%3p
z)7cA46LJQtKx>f#?CpqoN~xIbv0h&Z==-ssF8CTbfwBO9-X(Q4z-%D?1`eQZKkNhR
zh!l_(909)@`HR;{-`q-eqa=Q1(Q26*72y;3@nc__hr4uxj;d5w%(HBc2q7oNq%`6e
z^BP?wRK_A(3lPeNqG``sxo>p}t1j1ZJBgH1h|Iw@V)zV8VYnrLDc_8|UcR3Oo*Ci*
zq8YUW^*itn2U>AIKI#nKZ9F|04~yLT3-@=dDU&#&XeXBORI0o4^7v5~j|=vYe9Yg1
z_GG(3eErilEaZ<}JR+SKwYF@8(a8t{j)AMQ6ToAhznLmP<>uN$qvEiLjDk^HEpsBd
zRfGVepC(a;L-hWQk6zLsQ{d=dI~5u6QY>O;@pgm<8&2Kn_2vkLW%K?sWwPaFtF&GU
z=Jgt}247OyKH9Z<U4pwPb-1%VcP?A#g}*<wxTb^CRD;+@aj_cR<bm$vQL6(%aAo8!
zST5*9Z<r|@t<+-ep3TgT%8wYgo=9hPBLwmGZ9~%Y(tU5lw;CMSX%RdIS%KkJ*R&2_
zim<siO0&AB-s`R0r1m~?84D%cU|6;~WFwX0Q^W@fB|l22tib3e5PTlrjJ<4rs8CF7
zC=*c4No-yeC&0YWgex^ZVt$xy=kK2rV+$e`&OB6l6V#)`w<(`>&aQRphh)R2q)+B}
zy*(JLwppc}tpUMbVCPLFF`6dLnB0h84a6BcQebo~{<z07GZH)8<H84veZMk~P{r~m
z8i*HqYKaD6$#5k7v1++v2_q;30{u!^a0LXNu+Dzl4K~jYMS8+LIenLukpQig0A<Ev
z5<AC$jM~U!=Gfs`iAqpU0#G~j`-X*&y?La+MosG&$Tlx`#$KK6#&FM5?@}Y4?S|=P
z^<F=yrcJq7|M?IBOW#X*)Xw!3LS`fFAA4EV6s!bN+3188`Jgt?J46Hzje|6=11)e#
z4{6_$I$#TKwtU{1>^m~mZd4@HZ{<M@phEL{K^<<8e5j1%9@*_Lkn_>%7~_}Ve3Gg<
z%&{+<9!jKBP`IXMH5>LDW0Y%}AriZ3@#E|1WAjEHEKB4DwTq;Cb!RSV@K)%j{4UB0
zk!ldc#3k3B`wG(^w?QvpFEA%|d>g^4CbT(1Agl_Ak&13n)IujMPG({hw+>;4ewJ&v
zUo?+kP;*<#%{*>};i}Y&b%`EQfEuu1R#h*eV3~qb`i&7<FH28?;KCWIM(I0Iq3s2T
zpB0o69+;pyMs~t&8KNb=ao*QpI59oGO+|%Ka_FkX)y#HNXk}yX5b}oBego!|X?kQ3
z^H&NJz>^Mr<tfg-Ai?>T<Qd2fDIcxkppgMBgw%<JW*XD0(@!u6p->ZgVx^N3IZU=Q
zc$61p!qcL>sto8?_zXZ1db8mB!$Sw(FN;7|Zj0w}&vVga45O5?_jl3Mu_JTs@Jyok
z(zwxmyh-F$1)O^uz-%ywq_A1Q%)J{S+MbclpTC<Ka;L%3g9tig_sE}_pdnWzJ0VhJ
zkR%uGsyHS8t09h;?TLpcS3l<Lbv(b)e~c}-?4=$(Gl<+_FiC|F;f5t=j0iuNJej)x
zypq<~(LRx*c1Y$j2Q^+lo)f1c%g~JT0xE!%gIAXgZ~ZLi;+x1>v0imnGjybV^EVos
zC<+jHlMo>euAbuExEpQ#NfW#_fM^WNi8=A;Y9>Jch$VMRY#qL45NZ+$bC0*+WhdH-
z@{7PJChLxVb>Yj6$LsV$bHR*NS}q&OgpG9$!Hee1!ZQ6NiTwBDlsx#3B>ZXA>{A6~
zjQW@HxF1pl+#|EFRG&I(pKpa-et>qdgkGk-<}8H&LRy{j>=)B`ggespG0N={h022?
zIaoWVmLJ}c;6wcUvp%Td>x64Ud>qan*2|jFizoRhd>3{G(1xczVDZpWMqY^K6s->Z
z4$ApWPYf(3##7;h7DKb`SAAl4wrtD4-1D?`TQm*VZk#YeE^YKkR(4#t9^`=^)!Dgd
z;*!5N(hkW6`PPXIgakh8$BW(;#N;4K3MM<j{G<GFWu#`ZjA$P=8g~040~W1pyU}r~
zTXYGfCjas945a;ogeZoHsryNk*~+vTf%L<v7G4MQSayyTm_XSWGaHt74Ybe3F%KxO
zKAU2z1lM=Lrbp+a_d@ItAG6&#?@C<;FQ<Upu9$Od4X=H}-lb3bAz23viRai*w6xcB
z_kEAEO&o-wc=}l!75FqhLn%jfR=-MD@@CVsL4tdo%Pwt^%Tjxr+a5j@apD?+0Urpr
z4Dl<zkEDu~)kLZ&O45+l>sOPaTPawO#)IEyiE}N3yAl_|4}xfT2ZCH<N82wk=Hg1*
z%+!k(h546`D-8U*lj-ztmr6KJ);*+QsikObikd**Cy`VkY+%zOJrH(uA(a7MjKERY
z1(NbdLh!#_V_(a9L=lWe`4Cb}{J^I=v78b^Lu;T45~Sw`n9H%b(1cg;5(Rh~;79eh
z2Cg=X9c-V07Y(t49qL|#(~f*hqG<T1_5c>)u6<bUF84-R%HYw-+jHg~d*sS0s6bey
z0Jh@aAC+4(^5*H#J36fQI0g6u$xH%fW`r!Pp=`0*sH$bQ(w*Z|N)oHf-?aO7iZy?#
z_sfoNwUHTrh+o3Ib`6Aw#+^1ozSQHNe8R{k)e}B0{TCXWhQMJb75$o`@lP=K-c0C9
zK_T-{{3$Cb=_0s~G>$RM_GbYe+|wK<%-~jX;D<wTV422=u%54`;hi`%ve*WT5{WEA
zZl$Lajun!&E?P4XzLe~k;FmAhlm38+s#1AW)0(G5SjMF}u-{$?z3sY!XKwn|hti{4
zx*EDYidhE*TTK3{Ggl!Jm^hR;(g)*!kstCqxQ4#DcVJ%E+mAE!W^H$)9a>AJa(gnF
z&5<*JuL7$_h>vc3g0_4jyZENpqv<3OrTjwZLnyLp@wv0}jH>;W7>*d|uQdI#)pSOJ
zKcf#>bB@7NiA88=QyVprV6sBc&H}A6ydR6j#0cyuhaQuS)c?iox_50Y-t!GY!_w{7
ze{tDpdr?wEvswe`!Uv3QkhNyiE}*GkRG~KL6Yhr@(4-Awjr!c@2AwhIIS>Hw{9)(5
zcMJM#z|+RZ)WV0wQh4CSqo$^qez$W;yM#zJnQ}H|)auN2c7l0qVzE#LC5d#t&|!M<
zKHoM{BHud!vFI;&vw4sJe?R7B^k<i?6jSnfL%0r_)THGJPyc#{5H<!#>fuS=tB0@3
zddZ4>$l2632X#Uyie2eJ1+%qHC%|KqF6{E>rEPhPDU9}%q57V)8}II5in{b_%5u1U
z{4%-bWHWcJ5FiTYtlu0MLf9>r+a9_@snCZ!<0YF1RL-mV;3+ahxHbmUZhUB024Bz@
zW!zoBUj7#2A2az$ItVQiE<)gn?4>P&)V{TW2P|IVqO)kS3Sc-AHPGND@Ipn9pVL(p
zLX%yw3^5EAnaFb~uy{U2rt`-a_`?^<m4#ISk%|i*KPhOUcvmH|4bfOvuPeft?x9P1
zr7<F$k9-vlOf6&!4^_1gwfgKYFp}cjx@5cWS;)Z3*Ji?DylEkZrK0pgJaZSJV$-zR
zWKXkV1U+t$>DrC;X<))QEtW}VdHbNf#LjIc6kwy+oDoyy14)3_@{;eySeM?@_ThP2
z511#ljkUjPG5%pt<%hpPvzb!RObCOoa6Vwi`{<I;ki{Lbna0ANI@lxzdQIt;Ab0(D
zKa{nn9lXr`BF8y4u`%T`&+G{eUq09+kISr;kQc_-$&erwKZW!Zhq`fnl55WI%q3(p
zH$JrOSeI+*a>dLfcyd_9H$M8>%Ix!Ad<&D-Q_ufl%4R9$#4QUi?*;0tSoHAjoR()c
zoqsS2NOip(<>a-_r<>y5ox$f(wgPkH2HJWNC2mdY&GPCaUr+b)Kt)vAdWq6d4;SMi
z54LQ|6`$u-y{1M^OPn5)2ifO8y_s*@2PPOEvF~-SZe|qrD(mfRZ+2p4xH$-)o~<f}
zls|*AybJF%E22#fs(NA}t_KxV(Zp0GO2hxi>bQ_a`z_x4gZ)XP5_r(+M`rrm#N>hn
zw;J>jl7V5(=hCu2d9;jwjGQ9SP~M}2ZOg!y@nwzGZy<TxxaIVGSZXV2j3dDg!<c@f
z2rCvq)M;T}tP_|`;^I`Koi7@a$YRSQPWr21Wy69n?q@$>U)|xAJqu1<$P=0PmhC%-
z)GYQ5c~x@(QKzvOSVC+6yZnUUy`AtsQoV2b)D~k$1&Kcs;vuz7swm<c;d}^b{8E;h
zspH={<14M^X)!B_U+f7lUzbP!?rxq61vP2UPSAsw%J;%_Z-bD-jlzqHuXyIFsr<M6
zMMosod$0%;rpm4+KD~Dy2+j%PYnp%>qc-K(>>z<x%dtL^UhWAtjS{OL6YO19kEg4Y
zR@8WXZC0~z-^co2$Y7I==m;mI%~C<B-$un-5M@K%s3qVxHcI>DlIEF+9T1{ZBfr~L
zCJz9KUC@G%#eozEGjM1?DqBqSo5;v4ue~G`pRRtERu&9}$D3jY$Q(_!8&kIheoIn7
z@7&d^!td7glm4I0UAjX%{Q{9=l~WU91R*hHD$S?2$l8<%5X`5h0!)weuVS9K(51+H
z1vpwnD3t#_A2i6l_<E?CJZE~^O{IJP%ItkL?(=o&qy96wdK2w8Cv@X<N_^He;ehCM
z0hAVY@khoejv$JoqI0E)37clEMR8%)bkH}3ao0-4^Kclh8g{m_&g2P`lrW0bUN>TK
zMcx%tQ6)7mN~_}q6%#^Why09QcS-|9HL~bcWGX6`>s^w;uWhfR@?|5nBglg=s3!fC
zSL0be{DEby7S^e~keE_A9ML&KJWc<+=U<_QcdpFOdZuq`z-a}%NcOGImen2;vL?cC
zlFPjyMS$O4lPG)uaZt&=cNZiqB>Dl)T-vhGJtyo3TpbfI<KaSAi@IXK!us2rdav{2
zRC=Fz)l*U{&qPSZo76vff<=oecHUjTY=NDBwOhn9pNMh0`HE7|y3zE5!Z^1HLsYAZ
zmxwGq??y7(umN8i#rE9QD_lc|bMpWv!N{An7MFyYt`-D(CgoPMjzo)&duh+o>`BGq
zwE$%FtRy`ODrtWtIXJ|o$9MLmOco$%9iLi;75hYxdFEphJ)&a*WDVEh?|=x)<BVQD
z;{(WxhKX^LBkK&JFQ|GbgnahzpYT`0yKfA)lgb(ageAJyDBY?s)P%w{3n^Xf!}~BJ
z<Np<ein8(mpFKQ9lA1$jWRXK$VxL?6XfG=`0ZPMGpUZD5c}Nf{`)&xVy3bC^t*z2$
zk8ga<XeghYmdEAp2q=ehd@$JZY~RM_1FO=EHcTiZ;jW)h(rcHAc)f=;Ec2ZpHXQQ@
zNrv&Vnu5ZAH*9x62^ViE)BYCF<&mK!RS$hgBfq06gb=5KLYKj4K9VS(eB}uKQ9-i^
zPoDAEi=~Qsz~3GKU4q)Qsi3qdpFWA}b7Nr>_{upaf0UBEeWlJ?mh#U{4Z}$%&K5fW
zH+Ooyd;Pu-OuTursW?0;BLs?ZbHg30dx{!ltkK@HDhNx>mA#hkKX)CyO*mre1q~h<
z79NtN<fHL)J;Xm)@S>2_;JKwjVEXon{w1kcccB@S(;AcO%XR<+oc8Z5*9TW5;zhCP
z#Z95^=Qy>D0T1g|NacDo;rv$V3>BrtC1Hnn!>&uA8_|SY<T`7SvQwLw5L-&Q^P3>}
z+!Rk4b1zA%+1|K@ESi*Pa9^O{M%jBq=pv6VMk8ydqrM4raHG(alV={At%VN~eJV2B
z(nyYyJAMe%k2Kr5r2((~>}hL;UnVd+@YgkzTa?GeR-Ei_?wsg#>W8Y5pC5u=RM?g`
zTa6&W+j~RMG*x*~q{H_wZ)Ix9dhiD)*?T9eE`lWHoVFxwMiaxZ{c`^QU3>shLjG*|
z5lhlf_sE3+3e$K;%XG<P!4uv=(yBE-XiYEqGv@O^y0~^&WdcVY7n_HOVa(BY_*u{(
z#J~A=EPQ}Xp*1w__G1SXA2OOfL@)x6J!FuR_l3c;lf37fH9=wSFj>j5Xe^5uQ^`Lz
zOHgwcr@p<c`iK=(b_CnJ3lR}=Qae?0q>CQfSPCEdO%=aKK^&_4G?Y1h@(2{1XuXN<
zu=e)guX!8>TvW%G3*tznEhUDUr`k)`=?>k@q(Qa-LeV{p|EfBUf8am4K1QPUt%!e+
z&#gWP4cj0+x>f@%ZOA|J+2UBP@?(5H(fFt|l1BsPIN?ad-EsfAUNKdK%P5Bii@<QN
z?;auZogrs)(@AT1oM$NZhz%On5-j{eG;8GdiC^lZU9IuW$)-Iio7aMsR|Ae3AZal6
zvs+?4d6wMBM*X|T?jpaPC!D2r)a;b*RL5*YXPqo_0`iC#K@v2-@=8U9(RrZeT<V_}
z#!Edd=?2Kb{}h*PCP*>7r=ZkoS!(RS6o4)u9u2B3*^WrXso>GLqea8uW8rzE3Af_P
zmiCQL5^dCYgfkJM`e^TSlMkGGgZiim_O~sW*VHj-ne=e-%CZwC1pQS6fh7|^2t#>6
z9iy*tDLK(nsWIRP82?7<DV!yWKS}~@SM1M}3AaIggThdIkPutIt%w4WXZ*DfI3aJ`
zt%aT7+Sh3?zL@widdY-k`i@=DuBu_`Emy#UUb1#agX82n75}(;4+2}n;3QxlG7#Sd
z-dxJ%9A8|xn2Y9qY;s_^#Bi%-d+HqMWP<QOW8)YpusX$;8&NJv3(s6)={_>`)Sdtb
z3v1i>S<;Z+Qm%U**AdSoX&;lil7lngM}9?^8kSr};j7Az#TTpd{SYe(T^zbM0c>nt
zxB2PSoEDu*Z{Ra)-1KOjZZ<L2BW2w}lIAI_ZFauYo6b4dbrIqF@o$AX;P^9E;_rT>
z_p$$TJSQ|s-ZlGV>Uv}c5y@xud|j(jQ>YMJcQ<NqM$%CMDvqje7sZ#;++$=y1Gp23
zq`+S5TfW;JQeLG{&n?=EC*m01plsM8!imrm-~FjbmhcTuGO@_)xJ!te0FA=r<Qo>J
zxUnJ1uuLR+11rKb6@4c|N)_s2X96J{l|hAb$jS7cm&FdtH&V?e*i=;8Cvb!EflgT@
zd5oY9KMridn|UaY(O+14;C$Q&L}@zQS<+z%{T2=Ng8j@NI3kR1vK$gTXjp_v^H`sa
z^k`X7XR^4{QjjeKB=(SbEfZG%<tG5ULeD}?(%rI@(_f7z&k+75%1sXs1KOrqX?wWb
z!QD(Rx6gIx&LAIn#2?FexjoE>uIyjJPk1?jxLixlKW<tQTzQ~z`tdA@ilj2^_P&~H
zLE<u(We`yqA40Evg84VFT<lPiHp1uAU8_F%2}??m*6gz{a_prCy&2hZ`cSD@VK@0M
z6X<Np7NZUSAbtuIWVw!hrwm>8lJ*K-T5L+Jkzt+8nN(ufIOonLFcesrD1#|OACR)A
zy&&o#%?Da1=$s|7m`C1NYV_tZ?G{sVFcamV+_%Zxj*IxyevNa^M@Ud`Hf@m{@nxD*
zB1p@M`#@EgiT;i;c>#PXBlFgpK2dCGlJ0AU+y{wOYm$-<mqaZi1$6axaL&w|DGvFg
zc9q96f=G#B?lNuB_!2AwvUh>cwB){&wn0j<et9r6O?H0isZL7B8dge$%<J<Emyx;%
z9^nPIKg5pg`@-pgZ#T(nv?eshnvX4rpYH2?J5#!}`arrsuSEgOZ{xW0IiqwCIS6qu
z{j+QFjQoj=x9-$5#f<*HXCcOz5;Y(;O6A^S3A4%Kt(sice+wsap=xk%;ls#w*7p`*
zjuCXX@#8yaSmK*pwy<dFpP1i;FLP;XMc`!0`KIcE<nV_6&?R$BgcoE02)kwHX%q*$
zaOQwIEfWPMT5Ie|D!(cltBwRSC@SyI=DW@$yng0+2g(9>59mLCK_+qD{)lRq#%h4A
zz}#cEdExxnw{7onS2@q2Q(#-h`-O)v=(UAm53}kn5H%2ALyH#wb5NA)USC)8C=DxX
zGIKg|>D>xCZ)#={(Lvuf)u^(khk@+FqtN7;QZgEylP%svW@|o*=1e%N5j@?hq>55Q
z-(LnJ*FDd(w15yI?Qc0CxK84fa0rW-{%AuM+qBa=c<(Jxb|ul9a$I>P)ty|4ML=36
zWAuC{T9U#~s+%?wMltryF!}<Kk|tbHzBtvUK&`(U#wH)Rx24_`hNit|hd|7zX|+~b
zH%hZo3a2(WoE&mTC4nl&{Y?75k7R|shFWi77^4IKxIHZ{BC5@damkN#88%@UT}XrQ
zO5}OX&3CX-JMW+0kM)7$PGD0Pj-czqgMZ`!k)VY!RmGc*ykTKjpG(udT_dTmluMkv
z6!O^k87_#=CX<H*@^qVpsdsM#_Cl1hn$SiXBx8ZKRofh+n+(xf_*VMl+M#8Hc<CQ`
zYFE8-nd_OARobLB4a66WpcY-*Z(@34k{fIGX8@X&g;|PJ$4nl6bxR6J4o18ZO{fw{
zb=*eTqkV5|^K`**+DsmF0ObN(HS~_VX%i`mJ*ZJmwU5zZ4LO|@n%H$N9*^6HnWopC
zK~8Xcs>8g7<1J%eO}JzI=dPJ`j6-Y4J@jF)j^|3g&f}<c(oDDa_>cIL3!%YPejHFl
z@qT15xVUe04$XNmgL8yGC4Zg2t#5%0@xOkrna5I}n=eC+Wt=zhigxP=&?k5Pzf1FZ
zMV|(fZ^>QgE7n<%<NJ`MeSptXTfCM&!)!fVat&BX_aJWjiZT%h%z&OGCYC7(fkovI
z6j+`xJDi~wwHyNjg@3N-IBcBfspZ_UOiIa4DQnXNnm8b<w|K0&@zm>S7M+3`nm<&9
zHjbj&MQ8_F#h1VB&H$Zo$5`ZPVhW-X0x;3UgJ|-6D2z1y;hH!r=VU!l8;fAN+Z)7q
zbiAhZOYXzp$$4NOnk>Y^9aK_#S?2TR*%V!dmeqU%Q-&Pt$5#G?+_+Cyj!15_-rWSB
ze-pL`q78KB9r2e~XWMZn;|oGBSF-J_pz1hf-RusNP0N2&@ZCtF>%!CFk$t&%RebiT
zXN|lbx)Z6L7kc;CG%wDW25(9z*Z+0LJ>HKb@tqTgqoRZlxRwRjjD<?|nsfaoNoT`z
zQAD(X*apOum~kf4R1144E7UapIiXNcxHK9a@>hO72q+|M*VA<rQ`gs2x1cXUDUc?J
z-d;+mO8y0%Oa_htu}BD@tI|-^CfN{*<4$0$+{Q=2Kh_tbcp!7VnJkCw{R~yluUsH0
zx;cPA2d^SH;V&tMzi{jkC$HEtGL`j)!H0P_VLLY-w!OxcOGdJ74fkAmCc5E#{P{f`
zcZ{x>8BH&Y5({0(7k>9TCs*9I8%<tjzH$D0CojmnrG-S#<VZ$Q-hHnBHzU#Ydh&4P
zN+0y*m(s!1XIYRk407rj{<o9Cq#O988*APM++l%T>7c$C^U33i=7{E*Dr~XIx;f%y
zDo@yWG0s>jjD%eB=Bc7=>r(}LzujfwL=S)Q<I6)wvhp%~m&$u=e9@v`H8%0uxg6z}
z8YxU%H$Mbhi}b8zw@^7Eo7H^CsW;w)(qOQ;jVTQv&38l6F>KZmS1#>!Di*0yIc2k&
zLcK__TKg+-ba2BX7T!GKpZ<)d?tLXM(>V`8(Ka5n{_YFjT9){7kcBPNJlCcp8(SR1
zhb8O%hOp>??jOT4`B6viM##r2R)EiIn<2CqQU_U=aiFx0k`t-&&Px*#I6it@lpj7K
zyOv_Sc5}j3B4NlX`IdPwM7dQ(?uljaAy$hrmmUMShF7$lFw!7D@(vd(ihz7Wf=UZV
zJ3^#OBzwhIRZ6zo(BzKq%)(l>sS^Jlru=lz-t$Td%ea2hY8_g>UM~b)l0@<Iwxw{`
zs4Sr5_(wUzo2b2!e%Vn;YO8D{Wukf(j5!AY>k#c5Tue4hX1FX-D000BH1c7g3DITp
z=tcl(+Kg&k=ZYfp#D864Jw{;I={*)s%-&5)y{H%oDz=ckQMRD>pbGkYW_dNNqurVP
zkK(Vf{k#KN+*(sKzI?tY_b!oi<87c4u2+A5ycdOVRONe<?_S5-xE$Y@lW|vgwp0P<
zp-H%F^G`x9ouFnUqv<6`xeUnV6xG1A7>{EMKC#Lw5c+`-I^P`)T_6@YtHeD_*^v?V
zL6zOn@v>^JZ^gM?^K-BQ_rR~I@4ud&oyfC;YP^)nFMmv=!?=mSWO9)G@s78pN`C5S
z_BASH!c&liKXg$iV<e(_e|#A`00sqg|M334riCgcD*=Zstx&!Qc|mRG*S_qa36eKW
zMlB{sLr2<P+l|UD%5Hvc2pnRsBy#p19e2-k7A3doE)=rccKzqV);DX-G<o_`2X_8~
zGxeezL4~^yyJS1;eY)$GCX&SN%ww|Cf+wsfhu8RaIFi@4uK_*r7sDzTQy^gPlz{>M
z5`I2~oqsn_vBJqPYt=ONV_t%>k1AT7bRh8E=Zf5ogJ^NHuxCWYVwhgZ?}m;dZosso
zXUoxr{n8tHssq95iKkT}8>;Wu-2pHCLA2tPOmvvt;BqbeQOSntP`{(p8C0#}fvn}v
zdvlweq8SqBLPcm7Pw>N*vXt64dN0WeHDDQ1tBhXXh7(ry0DQP2&aBu44NT)uH!T($
zbR<I46MRPm?`fRVm$AW=2gfV5Cz$s){c8jCRJmWLr1LT7NMFFYk<Y!6Xz<4IB+<*Z
zyxt5ifn}wyFM4F8o@5*YOB0N5d0T1N9nB&BtFa>X_^t`U^%~haonKA**Z3F$^rCZ^
zbvsU=TxtuD_T)s(#Ux*b>v4-(2Dqy+=PEegh2(Ee@QueLG_-?3Vo*FLW{%)z0J9sY
z`^Vf-9|FXey`gCzED6OJtIvI{9ZE&*$<?{t(*;kfrZ(k<;iltUXwI%jzH2zdI^0s`
zSlcojwyP|)i<(UhdMCcji#VKy<d!ImIGPNUSY(KhL8#3!D7_5ROn#4IT{Fed$ufEm
zuRfuoWzTOR^$pO)%`osMS%MBF3@!tShQ94;x|2oUw;io+=mbOYe8ehl!i6xBwL)}~
zrX6*r9j|sHFc@KW1GhcXjLbsdE(m+Ew!7RXkcc_njhJT4ojLb7b<l$xC*TeP^12Ht
zG$~L8?@aJ`DIN<J<>q3VyT>LqujyR8;IGtPM|twC3Bn?aTM)`qB>Qf<tN2mYP<#>D
z8!L?xe~Op|ds+7`rbWucsp|9;OYx*)&<s(5B%S`dd?8UH)jX9eH{S$oBZkbY3%AXO
zEK%y=Ui-kefe2ePI+v@zWf~RyI|XUQ^`qV^)nHd8rFTOZ=r{~X@)*5ECN7&B8*f^C
z5oug3i8a5Gj~QC`rl71x@5eea6vP+uh}Qc(y$f&s1#GImL2D-CZnzgo0+cSCjK9F^
z`!wI;ueTSuP>q>E4*|Tj?R}L-9_AN-#+%c=ml*Cu*j}KE1W69ZUp<}HXPq(<Tob!s
zmXa!w50>Op{vZCBq*(nIW&9h)?k;WAgiSZn*Fp%I-#_ZNvbhRhG*)8Y3BLECEZc_m
zsH@aQDpUb25w;K<7j+Dp3=u^!EVmL^+A}0DYm+0lGNg*R9Px5~A7V$h3yaK>I3Mn>
zyXM{%+9b3Fs97%=B3lqS119K8IR$<~vX9n3^uP^(tk<R0kzMFDSlP2jITTqQS~7To
zf(MU)rU)Q^x0&)9W=CY(`iTW3FX~1z`PT~l&809(5|7@oFucg=UAD8uKn3|ktTL40
zVeqWjCfz8WNGT{PG#n&DJd-Q<^Dpem_ekYGf}_Q#N{t#?B+$Y9v2d(#?bZoIOOE8K
zZ(OwLTyDNDOFf@b2S|G|153%G$(h|?qB72u^_4|;M_lH6?kaID!i_ldz_^M2q=M!~
z9h*PLdL8ZRcqVFis^1C7q1DHOemYBGN--`Pak>XM*BrcK_G!Z2<JAjw&Gs^nWTBfz
zuaC7GKS4>o((iOv2b*e+k^-w1c{>DYBKIFv1L0Dh4+of^m~rcg=S*9_k-(DdTH^aA
zy}!Pat~9N-quVr<3S+Y6OxP(h)@JrN=cj4G@~&R6dd<G`{{GBfR}yhWUYs073Fr!#
zW*sflK85vlqt9XMF>dNYUDts_g$)@>0=y3hJ3};$pO0mvkOPkSew9o2%xa{Os*)e&
z)v^6G#)Q`di1e>ATz<O<8}$?(vr{F;!;&O#ruW{rsMKR|vX({!-GzB5opGFPAPD~>
zapq6f?JQiv>#fI9t$d%O5C)~<yuz#oQ<0Ig$~#WYjwPDEUj|GRtYv>}{rvSQGAZQ(
z&)K;3NNfJ`i`)d7rtzM&z2T~9ugxH@=xT<u-GZ3SiN<(OBB=jsFDhNAotM-I8BEzj
zqTf)aam+K)y>6x4Nb=KfPCMj<D+`KNA$Z2uc7(5=@h-mr4mU=+F}QALN`z}t73t_W
z@@uN@R)Wu8ujwGLo+~K_X+;L!I4(FP;QA$4oWH?zB5DdTX(h*_tfV9d^Bc9w4qBv(
z3bO+a&5ej}&pKFhAv7jCH?u~J-oa<|wxfyRqk<_;qjU3bt#>zl3Jkt+X*OEGWvH^f
z*E`-HG~H6U(nraYP1L(s6jX?FrhMTIq)(m|0cx_aO>%jwan{Dq-t#vnYO@*3v~6;|
z%VD_r*K|QIn|ej_0Uoh_5%;y{uU>G`7cL+u{3J0eDd$eXUOM60)xpc4tDYRu9*U4X
zXosIh`f0rFq<!ZodOKi0O@jwpqPRP;nP{=81GGw2RE7&?0{fJwp<Z`>Unm5LCYPn_
zDW+ZLAdSnwe}oF|33jucv)a$jc15|JS_7Z@oGPP@vL(JNR7Cbvoi+K){XVxlX<shM
zReXxOM(W~&v%Qcm!zdtv5^1h^wwbtt)1g)pC_!w85LQ35gSUk>u0p;lQ=5QZ1zt%a
z_8oHzbl<>j1yV0|c^FQitRYb}iQ^q>;AAfiov=0Ym33O!ypN2txcRnyv7)erQQerS
z7d#I%2lwEo;8b>bF>IMaica5v^^GHDgY60SZ+ewqe_s$B;qq@djneAjUO)DbY^`|`
zEUno?a~UT2zNT;uYhsb)d@>r{b6(Z&p+2a^%#Q`WWrb<tQ2C>tgL#+s|InhI7H(*#
zWOp=)!1@nDOx%OAbA<J81or}a4cuej7qZ&4KU3CD${#*j?f9EkqrGR@w{73)?Qy$!
z?hc9;S(@q9Ulc=jP+*f1$J)c>5oKiw937qG2gEhj4no!G@)fGx$l)g76MrID42g%c
z?_2GYeFFswKF`x-&TNM*kP(6?GEsXu=v>K&*!KPez&N?=@g{Z?Rh$Z6MBR4wKB^}h
zT~>%TfeHA2lri=D-kaLh>Pv)I>P&UPr1bE?RhkfAxUyq!>T$Nd($Zi~;eH-KxR`Ze
z5glS!LKxNDnZ7V`-)g<IELmO0nZIc8;|r+nJb3p${mDG9U4+<rAPe}%q4xe9yy&XN
z{dcpv%i{3i)ukG}apRzb%K>|p{h}5h<Ep?QM!DlpiS#h{St>=Lm|T=alf*0a0xVzh
zx6Gn#=PEf_AcG3VZPyFYuL2Lag9Er2%6{tbxZy@3p0GKxCYw|aq;F)RFr~`Uxc`2S
zE!Q7)R!8&XDqy@|(qxyT`rplZZJ8DwTI=T*Y1g@an+|%pHUg58hHVSsn$by?o27<|
zoJrb;EK?G;SFNOaOeIG;_9O+sV=~vYi0MKn^v{>8yO;?liMcwrXV}iJc|Qok4`ohA
z@7*^g!*JY874Fo7?)(FUU&{AZML#z4ut$U3SB4Y%Ue8yRUnA;nAG^?GGzW9+GS)Z$
z3{2&328C2@;J7T6*+?%$G)vI`N>xc=0IbrYEyzHK%E7At48x%DtjCT~WDKwgV8IeV
zFa{4>Bfb-Cs{+ueBE*o;5f9f4`d!1=t1YGXQ&$*Jb#5%9d;gr|1NjriMU@7JBUw`t
z$j4Vqi}r)b8OuzMB)rHF=5xWh-|)uaxB5}n%-_(*mf7x_XL9cx6tQ;=VY4XH#4L-)
zTN5+h;<YoBEVFT^+fmyS0IM#SpT+z^L-)6@<5A!>bHCm8>c;*CZO1#MJ@^#0zvQ#6
z-NS*digwa6|0O5D)c23dP;WTbyTQqoFPY_!^LJL}=-EJ_&*j6uVuq7|xSUFppBxQ0
z{1(!I+})avzwP(wnp>runRnT)MqOysUXq|n2hfv+GHnTxOsyF%c0KxmGPVJ7#km+5
zD{(Vzvba!0Biy<=Kag$ovZ-XD8?dn8ZtdX$^`OR7bXa=mF9eLi*F0DLn76`;1iR*V
zA{or(_l)l+#I$3WPUTO3f?W@uzlOb4_>vV~7zs1Ph8%v(3P^_pReaeV@RKv(n^Ovx
zL&^}uw|t7`hwD=$Vri3n-u!Wr>1ld4x*T=;2~@;i%q#Qy9qi~f54=49dNqio-)_h)
zoq*jk#9mQO0f#y-KJz>0aYxjMteKon*J;%&ibB5svL031dza?-L}Fqv45X|9*1^?=
zv37^ht^$Kw2-M^SV&o3T2CH+RaIm-~!J-fRvO1<Ji__`f&}M`JILv4na-tId(wMY#
zE|FTfC&Y9dL|%JQ>KHm3h2kwamqvweAE&$<Kk<G6m3|LH1&``|4*;alIl6j|#zc>@
z`&y`gF{8S{B5jw`?c1!B(;-lN9!d>zqk+FZ1zP?1bQ@L*VuptwSG0nO5eV8mp@hau
z&}Zk;eS){wp3CbrLI(3w>Rg&}b<S(mt$<dnCj7%9IMd&|=?-q`=oLM&(!zR7U+EY_
zs?5{xmUk2K4fxNfi~LriSIS@^>rw+6n&r0o{F7Pv+4QsVQ))E*DDenNWVVy=@~JY+
zdTc~%T@+)B>|GXyy<}6%Bx?AL_RiUo15)inIbvAKqAvGo_2{t<I|-+^ZBv1_5d~xd
zbe*r(TD7f*Evp;U!7QA=X*yWT3J$q5$_^~RZc%pJ4f~opK_-Z_Mn0l$C-$jC!EnGK
zrGeDgc4si=@7WbLXnW?J03@Yn2mKm#z&_<3II|`C<T0)BsFd!_MZ>s8<%}_cosf-u
z{D)P-W&O$j17<*(zgaQ|eM1IHB7BKwxv%wm72{hkT0v`Bo*UA>V-8(h7%L8rE)1AD
z=)Iy#rY?(P`zmH`x;7}KiqN_ZAvz2)+IO4s89p4sKok?A=?R3r?|)utAAiPV$~sI+
z>v+rupC3YD@cSStci_10Ruu29U3@d9)s~S`Oke7s151Iw!?)h;r^x8`C@yeAE8OtV
z3MRdA?f9#XTK0!R|D?}phLUk$XRpY|)<72LVFE78FzIO<w%%8sXxGE}fbSlV4<h(!
zg<zk&$<$&He&o>4tL#SiMt9TcSq#DC1ae&&hXRym-@N;McOLyAzppG@cc%-R(th9G
z;l0=0O4!|xkbDIn)%g1Cz?jVZDYNd8-<r`lXeJ_*i<4?GiQg$p-|+{)b#0!v=;e4&
zgv>tZHEpy3ZDD<(8lb^~W1+=N2SPz82thzV(CIz>aA%hrJh1fSJPJrtfN^HL4Yz+_
zUidECW5>e&=Jl#Z&cEWM6^AY4f7tW5u_KuJfREoyAJR#p%^p8UuR%}-;wd`2$8RlJ
z>4795_W}MYGzH=j>L$)+PYz}rhYjE->eXK<t?Q%)F=M@H)HII-50k_&@N*zJBz^w}
zcf8FX&lddkP7eOg+3NL2?o=n-bGC2AwLj&!>NA*EeGD_JUt&)6@0eNrJLXhZGqbw7
z+nDS$!O5!fL&`xB0XKlsK3pMjob0-jrXq%rg&A|W%kf@%<4Y)Q7)Oo($+1X(MmZG$
zaAHtZXtYR2Kz;xUtp*v~iOuby$ff7mls8yzmtQmRZ{S}*>kinP=LmM;yrce(v+DsD
z_-`h!2(BFZepv{@fY3$|YcLJ%*KWP9c>pFl58(eMa0;i<IKm;e{}G6TVDx}rx~VS{
zJY@`p`7JDd2wtGmS9xsC1RuiAJL8Y+JMSJi;s3!?e}k`G35)&#Cst2pPW3L#td23K
zx`~<9A2X*q(K}VBTG?_o;7Vyy6t#o$2Ow2~>pYLvhKDY-K`ogwTBFe5s1xjtDRHH+
z$4iu7Knh&{u?QD@I;)tjEmGtQL!c1G6+mGOVm96r$Z3#67I_mWc#ofOFdf7(trDK4
zlXxAoYizskCthJ4l2sEI6Zz@k6d!rg9Lc%2!x2M2S*($@KpMfFQ1HjC8`LQTPN{(Q
zyC62sx*W%y_+O^@PJ)9Q7!=>Vub_^(S4oylC;5V)m48x_o9V*<%6Xj4DS<nAe9GCJ
zGW|~8ns7EJ$~&3E+05il=1{{-zQY`T!%TK%4sS7&%b3Hpne3eEYUZdm&=Ln_viOyY
zbW<pu#^Dq+^=N7Ob~%u#J1M}GEjTIyTH_#ag+wc5BIR1(cU@fZJUR#zfG7mQ$Kv>m
zX<breu(Oy<Ts#MoLzim-E@!VRI-tWONq-#^?fVVzv55nWu{!#Z7)s`L8R%J$={Pig
z+5aPc!Ub$RZtqt|ewV{Ji84$CeDKm#L?w-^6>u^%J!s7Dt>x`*25L`&@MwrNu`i=3
zqy4#)ObSgC12f=g>My7hgE<#5@v*DXka{jR&4=luEWMLJ=xoNVypwV3&!%qOoyf%5
zpzgHhz?~Rg@@*LQ$le$x$^T+%fmSm|RpU~Pq^&ccqX48vs}7Jclp28}Wha1n9lRAb
zS}sAlO{P+ic1#s>snEi~4S<LqO|TH6ekky3q9TRFkHn-h<Yeo}byr!E-MNg$R2=#o
zCiCSAkjqTFjC{K-{SFf!*)DyUPXk~0;*0ptYoYwW2S^iK=QD-i&j-GYm%B5d&es4<
z1{cR;bkYYiS)f}gsscVb4J!OFdGlU*GE}-d(~lj(8EOW2ePN=CmpKQxC6Sxq!vxCM
zSI(a28+|A1i_fMZc_*#!olRonowyBWgFX|}{&Y9s6-+aN1sFR!hRNapSO3&5m~pmy
zp#y9*$vO^5w;mz8^#)f#8-WqH+Hb3!n;zQQ5fG5butG#MPM#v>^HX5?nGWCx$Mo^e
z$yOCpMg0mczIit|cM33W4aADT`8Hg8Kky-3;F?Ux?ONUgo_cldZzr|hxaZ=3%s(<1
zJHQL~4NV#WkuRWyb(rcJPvAA;7?O;)kJF54WLCq$)>k-(2@q*#<J8e4wv6ceAB<bs
zXd<a(r?YZ>ZiWvND32e(jPkJ%Jq6MuU`Yh_jYG$LAU+G+7$$wIk*>a9F9a=Y|LYQ5
z0tg(_!&|?GvIWbPv~_yz{{|ip#|3=a<c4b;Tmd31ASQ955OE}hjv|bS32J4d-DC*k
zDhHO^K(6#6;>-9UqtGn1JLUVv*1plcZ@cAKUXE$J7Ff^RlIwndUT9tg9Qn|&&$l?*
zJffpjY`lgHb$EzUgyF0m3Pg9q<lT?83u_bb`SD?h&xBFJu9@L!sAShe(vVDc@DT&`
z$jC|mKv@Ly$`l;Z4h0DawnKXemQRO+B2m)LG)#WS7>r;4|6vk3{sNa(15LQJf?Nqy
zfR2IkAvql-C6w19o51O?xau`b^ryj>CLIw_;D@k-5I!Q>A&Nsjj_tr`9vX%4xkwW(
zH&T-0MQ%&E;fHXErcoQ-;DwkvY<~w{FwV8xv);X371Fp0c$%?X%x~`v{OR<2RXO{O
zJQ7EmBHu$40WEAmz|VlkLEJX<lQdv_yxOyt9{^5bJWV(R_yO}mKA1j~BGJ7DZzS*)
zP+x*aFNApw39Fmjis@aPlVSNN&$)AkyyaFBzk~EGx03i6{D<Glgx^8%xLff(4piR?
z)e5TDLiL$YT?f@MW>^0U+^<<JvIHnbI}SpC)}}LIkr+w)B#0{z%|@eiERE6^FwU>2
z17TsU^6tpz&z=C%o3{W=9F(W~la>oG-E;yNGV%WQOO~*sg_th+gI(7D#|-O&YzB_A
zzRO22j%({8^V&>oH))S=jb6nMe(_}<ICC^({CC$yIq+n!n)A?bGLzIW!}_8A$7&3l
z27i_jm>yp-Kz7(`IC9)7_7Fd0#kYOTaE7Qw_8sX4Efc(cS`#1E0QJOyeXLr0Ru1-t
zF0$$tSanX2BW{7fIYBPG8(w%j$mGZ2AGZbB>mZo(y&yBsm&~c2!p!O_=2UBDRxiS-
z4x$|g?G}PtXo!w(#u`%^oK_gW5q4=ZDF&qyN;Om_V0;Udv_iyoTa=2fG(8D^7yPrk
zK73X&t%3d<EPdkgV+)1}`^8$VrS=-8OK7ciZS5Fb#OpDnbl-0=rG5o@Ry8WbM8toI
zp)zO59b~GWeY&y+>b{lW<6V})^Y_=M-AfON42Nf}y(w+xbr6$QxX<9vFwS}cyaXY}
zVMwVZs}*aGd!5maGVM`@B31OQC<r!iuvEnU56J6Gm0OX6ykTmTHQYlL-z0lHSALVX
z#^1y7Ro~?H*gbr?^_#r3_8tyToyZ)1%}hozhYy*_`OKk+w6gZJqd;f8Rx$vsWgy-%
zilNtsKw;`=w^{8ldN^Gjj}|0_Ht$>5&AYNk7Ls-?%=c-@g_tPt;h1c*XDmpzVSMOO
z3}e8xnq`zsfnk4^j=X(poY#69W(-FsSx0+kxOgvPin7T1+@3O8=gr%T|1zC~@2tRo
z?n86my5K*z`ZoTCycB&@<QWWLihMD~q&4y<3|I5AzK_@o!xA_alOftEwHQ+Wo~nT0
zTPLz%Y#XkMG0Df!!Ln!6`vu@01aM{x+#M$rH@o+bF+r~Ta?m$q(>pkiTYz(dc%yR*
zRGbqeb_?Xc7o_bRX#7=>)FGAdjD*WlaBVwO7eKW@RlCMLrL9(I3*h=vFv5k8jsT}0
z&2P_Uv(XAKi!lwtXM^Gz_HgOxLO(DISUz<VT#)I_Q(!H#`!GSqO1s1Y81k>pfj6ZB
z?f+gsAGKWrW`Q4)m56wV`#F@!+yrQvfsIyWIs;P~{4uzFJ^WB_ADMFc=J{`Ic<ZL)
z4!e{xLcGILeB=-3LPwj6Xr;m1+ja;3fN6P~#ljETIdCX_6p_}ztH6Y43+m_%Y>tP>
z&yNk$KI&qEE4xt}>4-t%`@kOo6OH{ohGV)e2Y*9Y;Wu1+hpsv2F{*AkmEOj2x17pz
z=R8KmA5P`bFFwXs4?mUn{{D9kym1qA{;-^x)jwfQ^+=M{0@-1eLYpt>^Fw(t<d;JG
z0KL7!fM~V5wk4wrpYd600SiReX|{)lFSajEe`d%h<k7R^-qs&gW`FP<R)3z5rBfGT
zXd)x*k0>Tn8Ogsr&5s3`-q!&9@){_a0^xk-9k4GIe+i>>9`A4-zapJob*MM_L^aGY
zm;$_|R>gNf`bY3jN&~0!)^{L!CZ>7B;AN@xseBd2%onWbCz7};@I`yCN1P7w68k-t
zSW}|PP7nm})x#iM4ijb#e~;|yhS~&NoCIADjH!q8gA0IrbAUH|)BBvYRNWGw<{Z^?
zJS#xW|0#~$H$crJigT_EaQn-Old4O2F*1rHE?LeT<}>r>M>8ktCpjM5+XmQdmPG={
zZ!*x(WnP8Ia4Lnti+1~*<Iy8j#zC}uR<gDNRIKZq`t(ik>#b$80aG{OFl%;y6BBT(
zwV30#mS32zl|G<rIpa@JtNaD93ox0fjIm0x1J2Kk;~@WM%qaXfTn>vOu<vDXW7?!U
z{*yi{>)ej1G`{JR_yrsBiqu|Df<1|G0&L_m8i{iOU*QaE5D&p<2-_R953HWA5xwAK
zp>@BM0VBf@It|l*Vg#^`v4j{)8K0xft7D`s0*|m;=b-O;Q$Hs7>x<#P+C$8AF{V_|
zlR3N_K2TP1Ej8>$6^+zzCRMyh4aYH#7kOl#<CqkEkv}vX$7$^^GTS|lZ}1{>I1!(I
zj@S4(h^u+Sw~G9tFH#i<<98%J361Y$KBS$V*};MRt>&meSjJBXqdiJn%tA|0%J?d?
zC#BN`6q9T}q3_>eqTUYhU+Ybckzr-r4ot!XUK7@bIT{n-30u&FPrHQ2O_!Q@eU(+s
zJR9`_&u4j!HST`M6sshTvfJH-_J_R%+=by@W@Ur?GV4>eDV->R>1i&mdWSzw`!Sz8
z9TkB^3HZY(bG^SkG}Eh;N{jKg0Y3-*Z}6YnwQJ>(7AV7jh-1pChk)->NO@(DWup&9
z9T%}7oh^_)vvCwGXxg7wcp?BB#@lQ0Jf;<8PKNEH<V*~uD}w2%bS3aQk|Wq9`Z}R$
zN6>V^>$sIike>58tT+PdUq}D`XsCW2(fV70tyfdD;ywbSj>IXhK*#_~j<Kt&kD^pl
zQ%2!ZJ{#?`aSf?W+Xk<cnv}Muw-*ujb=y&60QJ^<-5$gvWB>SdNFNNrWiWH1SsG{C
zi}DXlSHLlL9i=Tu`q>_sq2~?ox);6Zqz?i7g#9e;Eytw!O||Cl9rT(Ja?2+In)4rI
z?YR%4nSsX95#D+`YhxWeSeDe8<i}2zG)#h&4?=09L4M9|dfp}{i;bDiw}4ozku^a-
zV`nPU&u)vQk5&*`LfeTB9DhFCc(J4rjS05pWQ&0EmIvUB5ZuxRXIuccya#6-1h>3R
z<cx!;y5()Ya>hZF{_Sm6+<Xv|zxg)pmmkF8KYg1y=Nv%CZU3g#8AYOW3?qvRDRcyW
zQtW5vQ$tzrADqfEFs<d?X@uHN-(4Qs0;aTvMopH<v0YWOxeN$h1i`?;jGGGkhg*pj
zL$UYe@3V$hfK0=yY--<*!?n!rl6}RmJ9ES*03X;N`M;Qe$E{Y&>?7Cs7EA@<t$rW%
zBT)AU=)LODzLP}WTu#DtWc=2UPPD0u6^=GO$I>!5e7%{m(>?#P1MF#iE=Sq_mo}AA
z==&ayic+;TZgEu~nxz2t<2Ck8-N8Lg$Tp*1Je`BOE%i~NX-bz<!%CXc?bP6>DNT=2
zOFxrG!hc5l*{u#{FZUx8l8{q@Q=};>Y@*a@#pz}?Zmo{UqKmGT-afD9C2(y3g2`VJ
zOq@?;`=Zxo%ni=U3ztzmI)JQjApc7%2|h3mxjTkidB6?_kiE&3cD(pz;3w>XNt=sW
zZ|A`n4&SQ2&)`)MKA-Fqye*WofJ=ek<_1cSJ4gsNhoNMPbh_o%w)rkN>&=FD0@kGY
zAFD-<vAW($z|R<VEjgXauPeyU+tznF2|R#qB-yTM1txFJ3%c`al%G2o@z`PTx%Uu{
zjfT&yLOiD6bN())_?(~ezbH0;&QIbN#pav*R6eC}CgxG#tDu-gII6|&s=jUfwUb*H
zg|}2LH>S{42Z65)5-&lbJxEb})2#PX2NKNNz^L}|l)Jo{U(5l4Jdn{jrsA;w9S{=@
zmdL2pG<RIkyus5h&t;nRQI55m<ssI;_n*EsOE4APJozD8NH1yu1U_WrD~SDlu%;xy
z=)re{=xxW&vtO#*(!xTp<TMy}Huxvo{d^73%<h~(DO4N@rxY7CbgNSFdnFUCg?Via
zEe{jbrgcRQB)0}oHmA;{fTc92j<e%5JMP+JbVn{FT(K+5)lw#I*p-rkr4-2|0ks4u
zupPKvz}<#lIaAJmLN;iAigF!LAYq>{l;*?ilcD{QIEmU)8Y@;2Vixg0>P5HRe?*an
zh))p-6yo>_@Dr}V2~;6lkgDA%CNhxagxE2)%ZqOYF5=KEYq|{+?Yb6t5mUSGefmvP
z`ooq2=6`^=PjTBb8+^6Sp=XU8&9^a(gkj=jA>}l(0Y=>fr4`V6zbO}8Fv)rjYYEW8
zsZ{NDFpEnhyUdc?W>4=K(0LrxqtrZ4(bFbkH67D1?q@lu+d`m}Pb)au&tS&VWBsgp
zPT}6`M}7Bwiq2exe_cI#<V4$v3ATM<1h1&U4C6Q|-43qeRUof~$m@_d?GlbIP4e`q
z2eUy&sJMSTtc1XLj)-sk2L$u+tr_`2V4U8A(jeNuXO$BZNF71R0`NJeMaB+Z?&Aia
z`j=U?$bQJ~XkId!7{{6$$P2)O7`|qrf580CFkLQ@Hl_Ap!-;or=-pE{@?}PB$?u9a
z<k3_aD`(v~Yal!eiVws%oBh_c&>R@YnBym~>;Cm1OShqY5CfW;th>U5)wF~+zXSXV
zpS`B{=U{Ftxc<KPlOO5$=^F^YQcnEPfyOrpzw$8~4-PQr*B4VZxs`G6jG*YijYu&9
zLth)}xE51uVL67ATf$IC*Sn0|(o1mAR=JSB1oR~^c^s5TC_KPWpC0#v`!^U{0M0HD
z2tdJjUaE|6cxfK^9VjScgA|ktndP*Qo+ed>iI^X5pY(d!{~eR{Bkkz3@2jxaFpsX?
zf<9ey{t`f6L00IPXE0%6KPG?tWlRypcl!s;VaW_3x&O9FM8A14QNEdNdtYk5RBs?G
z$uJR$Fu~(|Du7zEkG0!LyaAX?jEx+^mnM|3>q&b+Fw|Eml<~l7dv9j|Ct)Z>!_X>w
zF~8&Q!`~mE9up;8j0rBOffbOy<`A(jp#1Xxeqa0NwvcyKK5ZYjkas{nZEtoTYsfob
zE^Ti%k$1pZw7of*W&0P>{8S^3TgYCIG$KwagkYoX)6j?^-}W~yw!7e3yU0t2<=fm9
zixWeZBe#WON#)NBt!a7@up9XD;h~s;tx5j?{{$F279#mP84hxSZxob7StNo?k=QjU
zFG9+s@mmtH3(J$gVPYitc5!)~$^|B`(=*${^vod_S+9Tcz<IqzM8Gu0%@`jq*j+MU
zcg?pk9K_eQmWd)@p9z8o&YMB>+;1?F9}MQGA6pmehrON8%Mb7>pnO<xGNlI`%%b8j
z)20fh276rMa*P>NWwpTI7T<^73d5{$;i=yO4YXUN{|%Ycu`KDZ-=iakgagf=si%E+
zZenbly&niueomZr8749-PW#+2k$t<5t+dZwLS*0hw9k!@-1lJKd-GtL{0@iwDS&Uk
zB%(Y3fh3te>jRDR>{l6nM#?l8W|;%95zDP6*V;u?cri>WfxI9TOgDJ<XaL4bsC*Qn
zd-7VSl3C>wSP_Nsfv}m<UcS4OWcla3bjV}u;__#K<~aLzp7C!IwAXfm1<L~adLV0w
zfBM0qE3%iT0nNE~xAfbIvCdA8Ihcs~maOJ!z_kLxdr3;ZdIX&Rc^Dj6dVo0-=-P3{
z0aS=rDB)0!hvP=U{xJvzGHJMd+sv(2o1BV?bIGj+mM-wmMpBKxq60(TUD5v;J|e^a
z%}K95%4nsU!9ziwIax8TImiLWDU$C7iA?Wpj1ZYVmgKu_M5Zso_s(-1ShbK3uHJ`N
zC+yB{g<;BM64G8Op0Ag&&{`_qIE?xK)vo`(_qwUmvOUOU76qyU;O+vx5KJqCh=fxB
z`6xVH3cLH6Q8Afk0<dQ~%4Dx}oanJ$@~j3v9TUXN{-p{-w)f1o3j23ChN)U$N55S@
zUc+?tse|Q}*`WDr=(va(Ht_QjhSHM4@DP)jDDtZosTK^wG%p7B)Pl%+$5S3V8Tg*j
zr3NpP@v<U20jUIn2Q#DCVfm4JGGU}|u#OJbzN6e)0b7;>U)8Nut@Az`PJMu)zr&jY
zgJcpDSIbGiK-t`yn76hx%t&9Hsi`nJ87CyeB-}W`UdISQxhDxXh~Kx24SB2CShk+Z
zmggvlk3)+IAP=ImpKK2;a)X^9-WVG4m;T!PuK&Nkon6%HuLS>3p=ucff)Fdk^dWcP
z>C>R}ad4*6G~s<pn8^!yunuYbX0lg<^*l^Hg=4HH;<?zum~cs%)lx>1sbJ&*sm{_S
zecOe<vNzD|4E+3*e444XFTG}eSkUeiyJNVCZ(8%bz8lKuUQoxwu8)Kucubh$6M_S0
zA7$hsJ5IN{TLvf#CF~yr=P2Hp1`UIE9NC`D1np1UmrTh$-;nQ5m*{s2Ry{`izvd)2
z41DyvVC%-^q#vM6PFFLECrEUqB2_PuTH!bn$L(({#Bq1S?@Xk?`G}IzWwftuqrK%r
zgj@x|{jEX}u?stUEbIoB!6AR?`P~xYob9a2`6a;BL0HiMff-P=34Ei?5;)GPMN{U&
zM*$w3{vO|@3giJmgf|Fq5R&C|X)fv2(R#-&rdf7jeF+oP@vx!M0%w!Ol-?*t+b2h}
z+KyLW*V#lIGFs-tR$UHRGx|`A4~sETz_pk($p%bRMq0$lx4w(7+KW^c^g&=qG|Hsi
zkYZUIlwCZ6hXR8CIjn{mjOp?60OHs+5VT%~p#6zkc7OCH63=z8(c1K_10RtqDKsdc
zoE%KetK7>^<N6duQ~ZcEDa05*tQ=@eBgVJ{@7YXrg`hPG9ib=z_Ch2=)}*%G#JY^#
zHOgf}`^K}nrFXP%4~2VsO0WZ(ieQg*&@dA!QdzD=1@mJ(7Rcw+k0Q()>*tMqTR14v
zPS9V-JQ-j)4h<QMyAoq4uECUXTSewqYrB}!_L@4F2zk<KmwLPLOAbbUo(-O!mU$85
zXy+L+5kIev{&-B}JYg?DnvAo1ESV_SN_-&BU{&C3$UAOG&|@|BI<iYfTY^T1XlGtc
zK8%SDH53^n)Bf4x9j~=tdkCfWjBm0T2=N;KZF-VxhQ=&z&7t66fO1!<X2cbYk&1*1
zK|6*98ed&SV*LtgTI$(Y86ccDnxOa!ZZd5*{*+yu-NzQIReIylG|GcsuzwuT)vnsE
z?8pQp_JzVQ)Q&YqTN>~^Vzdrb4h1kH8iL4F@JnX2MIau9m;(XH3@N(*w%=IgXa%Ni
zOQ#9`S7`fR11XblhrPZ9P5*+TGcx{3`8KfMjy5@~vjOu*%otV&jgV2U<)7j6`<O;I
zr*(bD-!LhMF<_zHVG?EwrWw?3Fzr-W8R_(P#9+j?p*X~hjw0je7+N(TugLQ_Pg_3S
ztb^#_H#8f(sflAEh#EeI17Cux2md|J@9wGSu)q6gOnz-nJfL(IC+GP$OPtL~oPov<
zCC-aUoGF`Fc;q}bjc{peaww0TPN=B_6&ppeX(_}cG^GYNJ{U4d7pttkP}u*5`jQ=w
zdU-2dfy1`$1^wj}W|i$>*Jw6#>B-%VEjS`;4NtN9UWiO{h)%lsMHnJp1mp8}^1#2*
zT&RD#n;BVEg2O&k$ay%&L!PrY74g4NG=2t=yqgFKjgW@|uOg8cuHs=kB4A$ndxnjZ
zy8<`QhJ$DKy0PuiX|QJ%DK=h;i@@iA2rZ;=SNH(JadQv9YCXTh%D5TpgSGiJl+LSS
z@1w_Y4D$##l4+DOunJIP-A(Bj-1djzUfCvSq5;Tb0UvTN-^oE@A3(X#NzX{T;B(US
zIW~ds<un`fVfm41ie|0gm{ypqEAY#GNa+Ld%O#{V_+@%<<G<6S(y(A-khp}}@lfhB
z`>ae2{KcQz(qDeG8IIq|7xbcjUfW}^C%MAtHPr_|VF<!w4Ut50J^lwTfV_qLXGDyV
z!FLf6m-3F0IAST0X{S*hU`C;ah%06U;+^`*3~8+I8Aicmhf62_#TFj_x_$Jvu{F^L
z-Tcx7zR_i{U;{DE;BB%pjfNv-z-AaV9%^sMn^77Z&F+#X4u_LNoprPZXCr01n{F5u
zMt%oUnwsm49=Q&tKW?1f-^SEk%t<y-x=spQL4RXQP*8JB3fCt%;NuP^-c*lNpbp*z
zHF+!|<wrRwL`n{Nocs9{+9zqaCx8G(c9_lE)ux%_ww9CsWB)j5`@1REUuo>W=|J!z
zs4TH-=rDNVj2nqKS5QVPz6Tv5W6DVAdSu>#?DkP3zCal|0B#i|JT(<le*++33nR<i
z+#hgbyXqH;O3GN2A2vJHi)>M5X=$Aa(~pfwsxQtbFue%53%hb)_Qs{z_UWtVQxi)g
zfWQ45I##fWzY+sJ#^lH6q&rX|)7E|1_n4o8y=0n{f>o6+b5sc5D#7lrq?xoJL10&4
zO3E(Al)23!g>ZWr*Q5{)uuoA6-2ihsjE~X<T-&Lg)VBZO58Ltd?eL~#r%+T!ps)#?
z-C^cRIAJRIEDP}o7sfOgi!B5!90~p)1V_M_6ojG>wK23nR`GQC>C-580J(t6aez%k
z`9JoWRKtrarxO@I{4;trE%VcgE9FM^D2+WeVZZS4Gj$k=Yv1A6-ECXmq^3@d^cybJ
z-An$z%&NPETy&=op!6pY2_QJ|=ubdtVUiT3)aAiDi)&a}A7;W!`!n|aB*B<~NlBP2
zkdY*$1=`njOg1EGDiXx(m<n}uNV>Uu(nZ46&~hp4x&qc73X|LDj2yUOXy|}YnOR$b
zatMZ@w7|^zd=0JwCv55F!qUxi8s#9wHiWzU3J{-X(q6En2BwYQ(({hORe&s^I9PoA
z^f$w8>`fky0KdqhQEsa{^lh>OIC?S7|7j<T$<xfWvF!^ewUb7;FxWBRxKsp25}qii
z4L8w{dW|`69m9w9Ns1OjWirX+F+ePdNPrG^9h-`h#2PZAKj`G{LO*gba8*96YKHwk
zfOZEW`y0h0d>SMlgi-~00!B1KC=U6}z%=8+kRn6rckKGhf#hIxdI5Za>r<;BXtJwT
z<k@uDEs_p24nA2*psZ-)JAOt?TMJIekFUbTFVoxb{(PJ6A+h!UBHUpj-Ph%`$^JA-
zCuu2;gCB!#5oIz&q$5mgyn$7zyGg`gW$aLzrzDx^fDlQ90^Myinq+45`vbYVFo5L!
z6xx0Yi`IcN4W_RJIo6QZpU|Kzjk0MYboiiU7EDPRO;UEFQRYB$aA0Qm3hPPfYYCM?
z(ME`VWow+<1zCU$6jHjFQl<&kxDY}ys<s4Ksle}UgFE#ddf^Q$c_z+S;Mpv1R8D#k
ziIc`rlA(<OR7(ZCKZA~q0>vtPdoRX+);}44&<Y$!qn!?<E74A}%h+^Kl2~bG^asVK
z`q(kNWgxDD+63gUf%Qqc)Ui#pgS!qkIj~_GjA?;%%CHA?mKK=<$-&5FR`)iHa;;(%
zPlLKuu)jxX-@;BLLKYIti?U=i8wA?8n}V7RrAbn<fUV#E{Kvh2>7ODy43qxq+<hNN
zqx2<kH4JI2gSLdrin>Y2Nw45c8PBMrK4#C2?-TL05Kw<0;VeU{N#Mjy@W4qT6%;#3
zf(jfv`U5%Bdoc2IOkii-Ca5Zcb{8V!Z0~ddwMRph15-MnWgNu9kP_g_nc72?Rt(d1
zGhF^7l+((tfhenXgH{D`i(z3GLSDJbp=u|-QH1mSjC`L04l&OK8+;UoTx3uI0dTff
zFz09Vk;}d8>K2+XeU;zO-S>fj(w{=6z#YQaz82O+Q#8jMN{=q#fR6nc8(u=#tsvoV
z!F4y_`Zcawi|cR3b;0!~dL08fALXDWTn!CBgTked&WFkc#*(8P4qFcLB<K(@RzOJX
zNQsNvi(F*2$?pSGxIR^aQv!hvu<L0?OLwiOC<(MHDE4Z-3`&Xt2Zz<<<yG?W$Vhl5
zen8VG69h$qfRhHwaJE`^oprrExtt&2v}v9u({Cdu{cUc_N#UkpC}TrCh(L<Mu`qpn
zib>=loI-GCp&cK%`=K2PZYkPHfD79CIJ=KNM}Df09Z4<*oNJ&aAC|Sjiz!fLP}Tr7
zQ(;mQO!h(0vE|xwC^<w)V9Kej1WqLBLQx59jKHZ0NOl9n6i_!L?~M!OA%sID8e1`{
zz%M;JTOe=<;AkI$4z>>e<D0!d{Zo9GDwb%t9aEt_Cj*<_XQEsf?C3gY((ZucKa=7%
zs1Rpb)A~%b0-W!ll>lc~v<iarUHkkudmVkbSk90=Xutib2~!G8)@KujoZez}mYfV#
zUIm`8s^NEmZ^31t94M}b$d93EB~<j%cEnK(g7VmcDO6(Hf$MJ)-58SjF(8Z}1h{BX
zhtLj)erc4-`Wx%)dAkE@5>ssB*}lHaf~#AoS0>qMS?=Bs5-7DFRLay#9L$IhoI=wm
zP_4qGH>M30#qBam+;kqU<LmZ(*U|6G-Grga|K%>hR+KtSdCLT*iutzyxK@EP9vUx(
zAN-H0>r>Io(J7!+4pd=n4yKra10yE^AvP4F@(a<45F!Y;0<pnXw~x{n*HvMBsGBj#
zl)4v?{18(Sq1S-We13>()!dHBGs?+efl^tZ8tkY85Xpy*7_4o9i3^}2Z2(jQ_yihl
z#|ZeOeV$%Nk#kYz<WrTGF_p^;F<p;<YNMx(`#k)34SZ1>-*2zJD`zleD4)lxYdIn@
z8SOw(5y66}0ZFihouM;0EqefD(%KG<^d5Fk0q+5C^aYY{@*@f_(cBIEGk4#Is!=*v
zLiylE9FrQCngDGMELjiHSHPKSTIHl22)Kf@<0IgP?EmyR`f@<ok3Y+6-{37B(QVAh
zmWx+S?8|r^Ie(b|Q30o{hjmv&T=boN?F?AbsKqL8#_UeGBXRPfun-l3&J+6-_)9GW
zb}_qM3>R#()p~pLUA6%q*x%ACcj+<CW6C9pH?;}gwB_#m5P=fE&9NTB7*OCJ3Bf6#
zqVU=pm^2rvW`o`fNJ_K}f$Kxtn!SCEPR{i1ra8D)d+5)Z?-lU2{Uvg;1=5PC>vWy@
z_i9K;C@+WFGPu03pAr=UT$yQd(z`6c6k`{_BosmN3L!twRN(Ux0<@+%KS_Zo1*MeD
z{Z3<wwl|Y$>(O1}z+7u`_u5M62`*vcrD^7C!<WqUefT6yZ{;($JQjp;6(FZV&1fj_
z!McSoezqp?5U9Ke>SGB<Tb;=1eUu`16EddLKmgg@6w_>tmZv+foR2B^S>1UJf87qU
z6v`W*<a-cl!4#nEO`Fsvi6-FC51PJ+3B3g|#nB{1<-CyRa3qjW5R_(J6(n7k0+mDu
z+Ca5~Z0`pk!4%PW7SjczkM~@|l&e~)Rq%UE+l-tH%k=h}dpw*m&w=(bD4zmyS6EXn
z2s-d#osW{%Fy-r^P@AIAuJyrmADx|kje8p8a7rfG=vodahap}*_y1s;wgFc_yMWSf
zL;Ym<_6Vb+^;Lu46^2(CAkzo4t8`#GBq^X2iYmc@lC}86C^SKWI0|$cf1(37(SlC5
z<CLXP5#7&SO=5yW&kbZnzyIr6TH4@d;H}&}-vXfYf+@opeL#T&p%CbDi0!6vraIJ5
zi&8Q%z?zRBC`^rgWsmDVik)_ido+(E!>iq$li|zd7Q~JQe%1Nx^$8GD;Oi@)>b!xg
zmrD(0{ZfM#15522s))H!WIoh7>d=aai;pytHWV#%h$J0;mvj)yJ2c@w4TV;WKl06i
z=1r1wnNSBekqh2zv4}&ybir0iEEOctD1x+$3WDSj3BialFd|^QECuu&KW+JR8k22_
z7TIN~T3u!f{FJesaoV4U@k@RVnl6EP#zMYI!=#_V^a~7p4(J4lL*6Ec_K8>51J}Xk
zAbjo1<geO;=y;dd=i}@bY_?!+0(QphtclsXSxEQnNEcWQ>QzYBL7g^>$n9lnts}wf
zq%YPyNN**Bax!#{a%+JkXaYDuK4HQjA(RI(`w1Q>b{{`&srghua%*K?_}J2F*i}2-
zl>afN(;>j2)YQYGcj4^>un!#fP56lgss7TqQySJc!D`jlo!kuk8&EWVSWNYreA28X
zZ(M>|!Bn5;j%VqO8Ox&u_z=|l=(ae?I?d)dH0y4k?LvA5skXOCeW1AqxG)Ej+Z9lH
zU>O1t11(J*#5@S(+-Kd)?X3lpevEf4cY%kdNtR*S7COK|G`|gtHbS}pc0C`yTr{}U
z?e-Q^&8hD@`5LeYv}XD2a%$ZU2zM}Y4ETF&Zn}@T;sVP-y$Pv$bd#dJ9yX?Xp_?m6
zFT<^Whr}Yyy_kgcoNPZpDeNsXJ?R7?v;g6UXmHzi|Jbo4J$R7xXp|yrEeTA+8{spl
zQ8|ztB1o2F`d-F?lL7d6ks;vZodW-Rn(13RkR+2(+hVr7-lYw=4Tz)b^ZDSdDqIDj
z5-1xX`T<7|AT=gq=sj?kqub-8>okoCsCRo|N_K#|40ppq+G-3)UX;7v+oeY7V0x;W
z90ei4AA(?xBiY?x>;Gk)1(0?@=q}rlgCu|yeiH=qk)u%#8YD|GnKuElJ0-1X0d|3-
zeh#OKfiI`h(9i(&{nn&>8k4pNElVcx{+eR6grad!<m*d&^mfP1pcaC=6w<A@jf%EL
z*sLI-d)a1_q>_ta<1$)SrMX7KZ*%v1y8}v3oAhDa&kks#=j6=pp<CrL1!*^fl^Mwp
z%v~IW2I+!dk)u%#+T1R;{~wrn=`v_9g0UCEd7}rDiU=(Yu(qSWEpI(=BhU)EkhSwB
zv7yaD_@Q)k_cf6Xkd5Fjf@BT4y_s}df>fKPSwW-jgH72;NBSMahK)3=*8IXXT*;hl
zUqI=Bq=N~ahE0ZnJ!?3eGrN0=>}h&iR8+R<rQ1VOcj%sh<0+m>n=C1j1Ik|H3Ijfj
z#uU1k0DM04@|AF{fP$0YJ7=2WY6D3}99Fju)L7;pAU^=*pqr<#=(S=x6!?l@<Opv~
zc(A8HH9~qZBv#@!wUTb{pe>c6O&bHgSAcYZ^(5Qop_jMwVG~@S;g#Gy-!6etXPMkF
zKqBl@zI+J#a~HR#1n6|qdo)Vdm{ht8m^xW@we0r}Xr0@TddlS(?#x(w9S;IVWnR7x
ze%S#W4u@O|XE_5GDpv|NHW=-$Z&?Z4iKARxM$oc)7v5eeQ4;dUKvCG+6&1LfA-xpc
zzKUd18;N9+CKozLk<$G%gNTz(uRyoGPsc*VBPmmPZ*A_LZ<j#nElocr1uWp@tK#?|
zzrf~Y<)oW9KnUCvI_ctS(=SDL7P)Y;9)r6LZm!isH@Og_n@`2~B2Th^XXLjnkeCXG
z><gEc4$kCm+6?Pb195F01Xke^fb<laUyZW1MIrqz<>Or<v_o<=I=-HC^J-G@1Z}RO
zRi#Yws?KbH^rl7Dkxsrtx}llcC7SCs{5f~Ow|k8;fN_u;8(U~o=)xC+SSY9E^eL``
zn|4TR&|NA64P(3VnIL#hvOzLzHOZ;Kne>3cOQ3bMiG5F>1(ODG)_4o7YBwnYeakA4
z_Ze-GO49YCnEOne_EZPHidsg5YH`<YB;C<OJk>!nX%e=vn4%wmR6qhZy#&>=jP?}?
zo^I3pROgEBZm&Qox`AW~rdohB-b!KGqsC%6ZL)9s%@iwEuD2-U7Tr1NfM4Vo#fA-|
zF@DG|V*-oc==ruw;i^&?b%nWp2OT`{nUe`v*I+6^_G%*tPZN9?M=io72{h3D&OBaU
z0ZPXyh;`t!wz7^S9VAGJbWa~jZvY9nYe}a*BE6xJ<?A(9q~Rf*TS?n1P-X{i{8p1R
zAnCm%0uU&I{2X_(Ptg`6w}@^o%w6+2rl80t1Cl>8E`!c5AhTQvbaNF5J4cG`hB^Y1
zhc(pV9Rd59-g$bf!+9x2&-30h3lp#z-7835JpxK6!^H^*90vOiw6AHX-wf+@-yo?q
zz)Fz!;;MG@Lj497KDdFkaX&f_f)##Ry<`Kgwx#X^CvEh`ZNz06>H1YPuSs)To8}sA
z$`Ivb`vb~sa;y&%xGA!u%s{9EP979TGs`I_y;$9P3qjH`#Ym*jK-0h*l9&G0feE+`
zm!y8pHoh`f(z7^-wBI*se`x?i!iG%TIcCAJ%4(3i(yM>>Qn<PVM*JMAe2_c?4j#mA
z-m(c+_vM$gAgOSHzv8I5=H6;(n+;FTha@RNqoF82Lnacs4{!u<!EHwC8j|sKG;e6<
zv5lJR6s*+)S^hcsq=8a&vk83m5wn^>pvzlbZTj966hVHD{M^4rxrJIJexs`m9_n07
z8-m$Z>pBvcMpwVxBX;x4j<N^su>rDDg8KN3pnabQYC#5y)^`QSHjJX=V*HC<do*t|
zUPdMZ+HXIj0N4}5k@PglKIBrkx)e%Am>h23esHoFc$0}Y!ix6HCwCXAJ$#!D{JC=j
z)NA3xJ77t|{Dh)WrYmNj+9ReQHf?BDk#1W@^QJh@)g|~<O7oHKF99egpCnL<-c^@Q
z9Q7t<H)gX9;7A92g(iTL!^uGa-cZGFc#?7u6qeql1QP{b0X&4MVK~-mVO7R#q}~FK
zzZ=kn?P?C#=fYVgu&4F4+eZnRJoYfXx2-tIn&JSJK%V`7jXfrb=|kPg6wLx^KZZCy
z+0Mrw0W;{c{LwFfCb+s1O1}mr6M&OpZ{NVTp`#5}HTUh8%wSv-h?_ufGEb&p(|_RM
zg|?36I4F+wqK$awwNdyrYe;u&q-8@JuQ#Q*K9OQ>8j?9*<aPy={$7+!Pdf<HSBU^G
z%I)MqehebH0M5WT>Ci_@lWt(@WJX73h(p7&)!84DEi%vkpTM*;T|$Ph8P5VEuh5s1
zrFjPI`x3wjn4rlX^iHf9pLL8^@-}4uz86zVQ2-+`*)kCe*6XsgMh~n*_A|x;hhlu=
z$M^1o%$Dm61*l~-jPk=-2g3L=IL<e)Q$Cf3+ImyPY(S9@Bh}*>%Xu}t@lU96!7YMO
z<GQU2t9H5yWCJNyk#5^WTSJ1E>QnqKk>VW}I#ez`w><*o<YBCM8`=p#UMUm?asiw{
z17$D^EW7j#Q{cD=$7o>^*{z70z_cNK88`!zd2%1{I;J@6w|iU37HrP4kG@d(w5$zs
zFpwOxtG<9tZEs<Jn3MH?p7k5f!lVvH*ye@ZehxRLp=~!9bu#!*g#D*LNuMD35%3-0
z4O|8Fn_%rAk;=&~v6hDVXW;2|rs{d#7$^<f=419vx&!Ujl2rAyw$xEmpX7n2B=;pX
zo70fY0pt$lqa36K8Gt}36o>jQm7kMd92$<P?(PD@qM+8`Ljn>XDQg}FfazJjIsqJJ
z{fvLMuU!xPz1yeUnT5jsEYsPGB8TlcLu8s5xYj%m5Av5ZZJ!tTa=0!GZYfOJ6;3!E
zPAG)2zWz$5O#}V|oKN$nJ6V|+T)@qt>-*B{;PuC0lR`TqSWrBINx>x2?Lg6h->oB=
zYNWQgo#$HO+}*5Ll7@yfbmUCz&k|5}qfrLRpe#RUat{YAJ(88wN#GF_D>2-^N!AZn
zY5je#V!G??0{l1YEMM#A6VCo5e~xB*FeWYsH0{@oClWpjN@Fnj%ka5CXWm$r?Rz68
z>#wU1q`3}O#s?QU=?;=4Ve@nF@^Z9FlPYP3qQxvrYa+=unp#r4-<IUz)(++;l59>I
z^E#bd4WBijw1FF(j3zlEAEGj6at~gz3L)TN@^ad&X159BC-k~7dK%y)OaNx@Pxq@|
zlZ7_fu^h+#<@93B7m=Ys<}*+r+~l8_hq9ZoB1#7_8VRx(K@kFt<N&J@o2qF^xW15f
z6(lnLU^x9mh>eHS{oS?5e*)jdRG;iB@j9q!H`(EXk?tC2dJW8d6lJRl)wb93=!SY`
zyOMQD$;PCQhBPG8Nz$p@LfA3gW`}&~4n2?-DKu!G3ETvVpe&LD$xjz>YwUs_kMRS#
zn67*eVUqK{YF<|XXx|GGdqFhRE4Qd>5Iu8l%-T`A;$+w7JoVBelZ7#%m}vhv+0HMR
zOPc?pco`GbNe3jve7FTs#s>uZ3qJ>m&CHO3v9uWs{AB^8rlX6-BLZKA!>ZuyJmYmA
zth3K;g{7^7btXJ!!FEIDsWzy+6?Nmyu*iJ(FJX#NCS{g3qS^8eNjce`G|DXW6P8A)
z{e}V*&WF-KhWwF}PeU{&iv6-R;3rsZw9cCB-v>@L?^F<f3sUbuq!4@ufT*Htu=+n1
z)x4m$lQk<W`d!c;jWSGzN*Q5jG|FG#66e3aZRdiNqK*)fLM8(T5n(UCWKv`WMIuhf
z2a%=#F(S0Y^NH!*@fG|K_ID6t4}mX9^XsDwEh4gf&>;B|riH|YE~;hxBY62Au*l%x
zO_&}Kom-l<%yY8C1EuH%C#|ZW{UGuUK;{RbNN&|8J2@Gi5rBy#FTzA@H(4!ms?~Om
z>;kC^qOdRx1#4mVr=X$$A{7{h%_w+g6%^!w8-#GcRDV(u{C@MC_mR@0S$d|mZPOZV
zg-9#-M?fSA0RfKR@IE`gt_PZ*!?1#aKn46Z!lHONrzO_Wra}}4l8i3cNU3y*1~mbx
z5NkBLL!j`3zl2bDA}U&dZ?_-7_osm<Hw36*a(Gq4pdkGtYv;e+RU6#|tM5epyB=*{
zyeBt5Kche?dZSSW!HI%D4EX_L+~u^$?M0+@{GEzPPpk*d#ZZR)J>DAH2h=9mumGBU
z5KTboawuJG!bWR@usIH42>~C70)skX2mE;;1*H5CPMUmvd7RZs_W)UH@@m?^Ee9tF
z9lo(N1Nq8_BZDTCFGENIaPs{yYLQW3N0dXPjHzXLuwgY{57ru$tlkH11Y8Gn$a)sW
zfd3FEE`kZiL3AX1H(*Tckqnbl6Tq;ry81L4F;#va&C)qBx$ZA^J-!24UV?u=x7~68
za<VmnQtp!Z$KyFi?8CFFww`!26wfxx<??h{rY^nLe(*H2IaY5uPQt~d@UN*bXEv;u
z4c|ViSKV_8(DY@q+<o@yueHXzBOnrh_^Ad&9Y5qpGZgd)nJ9g`oeM_-RRyqz3uE)~
z7)6eyQHDl2Cf(oXKp<+qK=42a><9lUfMu3OIRX=rj##H;UDh1*@_)Rvhf3gJ>mfK6
zDDr12RCR*n%AsBEP6p^gi-Yr!80x`;!0o^V@Nqj_Kv%{=#+;r{uB`Y@1C+a(l#NeF
zlE7PbijKC+AfH_6S<n*F5@Z_WY)ej`iisQxQwZbz{b1n%aM(ZLoMYgLOY(XD5m-wj
z-||lM$1KFLXt|3QL=TgpQie^Z%y1Y^j)Tt!fiRRM2L7)3bq3R;WQw3jdv|@-H1`Y0
zmoQ1gfx^V?i|A5kd_cXqFHJFzo;g8JjO!gR9ZYfKZYR_Bv3SjHTzJc^tpD&g!2Q{8
zSpZzpD@f*KCv<py@7g;)A-<~uHe!k}6l1u7`Bq;D@0?dZiQ$&lS`}wBs1kTjgD(ja
zHbP(#h>0K{23c$t{<A<68~9G*elDN|xD>^=Nt2<Q2dskX!+@6G!$$`~+WhVF8PM@t
z&90a&kIZ@{HL`Pjwpmw@)!hJ4z!Sa9=1$&Y8;ZMuUsFe#Zvan$;SW5}1rlEduErD;
z$w^K=(b!BHlV6-BQz0g4A3Fh1X4iMHI!!tOb3#yG51}}WOhWz=!z@SzVBt7eeG60^
z19yhJ<i9qq<s(SEK?k*TFq7=2SzqrBys|zwq9I`mLlo`E*K?%-Hn;ol?9JiRVRo1J
z&ydeG!EI2k;mO|L{VMQN;En$7q`46$41PlnMsh&86;YxYQxG9&fzs1A+@0b=Ys))T
z7PAD9$cMm@(0F7<1N1Z?2dE^)MIW?t><4j5sY9_5sTRZ9jdV0cVe}~w{25SUoQ_YC
zX*Sqs?cxcbb^{qRrgp*)#a=NeFcj}W+aJBCkAo|Wxz(l9)b0hrdQ;rrnF2ST+Y70D
zrDx~6z%!T}`nP+Y#LFzb3_(3{;4kEQ*5rV4>uZy5VG6F!plf;Goudvwc40D4@-S7l
z_wDDpEnzlGSc2tYqji9~x?)=1`YENp(+Q0YK>{|_!&?<l`4u?oaA@;Ez=2W1)ZWPx
z5J*8d4doqB7=b`+XdJ)p8k?gas4@lGI;(hm0>F9l5Z`}>RsVVbF5sw*xZDcdjG<fg
z1W9)uyz>{-AL~ig!`(nN19h&=Ne(CnFD@qLx`0gSw(e?4PWpz|QJ9!;6OcrSAe$$R
z<m2Vzk;`00G(mbIH17(fXTixcV45HNPFH15+7j7m4XqjqT46*x#76W1ioI-MJ+?HR
zsYWS)P|8Ho!`{r=A#Ux*xcm&0b<=zLwYwOW{T}tN`K0SjF62*hAejTot!#2PV3HLy
zCU-gd=_s{3q=Yb)ymz<PuMR2SU`%uwwZ~q<nnzEhrSbnU_A8^QItES%!DPpPmD2~v
zDT}|&8d{a9LDbis>kBaZ0?szT(a<I!l(re`o`$*I-sW#(O4aqf$FKiA>b^J73t%pW
zOjMtnn>#g7dJzXto7obTKv`YmISg}eS8z)pEwB~EOjX)`3Si}=yVhNXU_bI?KKm6c
zrDMWfw7mBgCw_1y`+oie0>lR8GL*J=k1Gdv7l&#;NNsqS9o}u$5KWr0a?+bt+l96n
zqrXxZ?h@E|1Ny!fQS;zEOz(-6x%s%$1Eq^8%b#R$9gq8yB&MIpG7P=A3fusx_ap>C
zmVlElAm5*}z6ULi1<M3ciW!~1k;N=!N&O!<RL=$RB~0b<z@Wg-Q3IRX!`&h~t#b>?
z^%gLN0KYP^B)|8eVk_*+Ly|Fep7$YXU@5HsHR_%hQE!+K_zy5O?{czp1EsQ)=LL+H
zX&l4PoJnEwW9L{?dpA&F5>1k2O9j4sO&BPQJ46El5zgI+!Ak%rhRq2=?^l3RM)Q6f
z(Bd@&t1-oWPhr4z34Ahq1J&(ssSOT!%_pR5Na|jP9>*&~B)fs=Ruez{mV!(}dM<4E
zBkG=4ao;e0qjND%_MGH^ayjt0ox(?N!S=4K7WtB$@l$|ebXwBB3RY_fMh%BAkqqJ_
zA(8{i!AnBIq6kgz23ful#6Avt98-xPiV6DMiD4KV1ssFvGvgohog1!WKn-DbbO&qK
zdVAtl>r{Z_8qHH+Jhh!$w&<pT4jM#549JDjke&z2evSU;TvBsP&iDUe$X7Ypxq-3+
z<9>O~`fFxlT>QiJzAwkjuO?tWpd6IIZHM(sKnEdIXl8tpFiBV1Dy%u_6;)cS0#pxd
zg48Nlu_?gPXfch))w1h%Br`{Ng)*XcTn+rjZhW(>hB=0<og?D`q}mEUv-{sJCPELH
zs5(|7b+9J9+pRaK%mUNSzl$)W(gCCm^y{c4x4=K%CHbC7A-f<4k~y(J`6#Ajm%wlw
z4`=w?j-{9ahmDv-<w<}*rzLIg!`c?`?XHRVK_Y=Xbgp4Y-vE+;jt?QR0-9@~S-}P$
z)E7$f&xvvH5gnWnYowq7-y9(0(^d_U>3!XTDdu~Zb*GHQH2B&ds322%qAlz1p=pZS
zngOf&a2KR3U?#|LbUZU2Ji>k{g3X-_0Ixv5$ogHKzj!UgA4GqA_y2G2Ok*TTuKWIr
zh|H|3>*{lQj-J`sUGDD8P+C)5$rJ@kprub11q%?&4+3PFA7n}(WZ8mEjX^Hp2Bb9&
z+l2hVYT1yX53(r2CS)2iNf|O%2SqL@a?Rx~XU98x?;PDT)7{fYb$4ZEWcouybyZha
zb=TR~<O2%XFQc+LGBV<y@BQP&3;4nkm22={Ftgi92v8z5pnMsVH02rl$@TbC-@}CO
zu2^^bNz*ko=-cqt4cN#FE{fQ8P_wT@oaAn6D}jX9CSd+9Eao8c&HC*8D12-}@dM!~
zh0q68u{)L3;2>u~vuMZqf_2CL3X@!I8nda^j|123vzoA<V$VP*2Mna$geKKM6w(9@
zF802!`|Ggu+eH88t3+Rld1n<qhuN_$0m(!ff$|Yd@~W9$hchPO)c?lBOg@U4dCi&L
zT!DKp!n=`480%+h>vBw59N{Jpp)F9Pp!7a?@EBxb2p2(q2h#6B>EDC?NssWO<EZff
zk@6eBvi2gXCCV#eD3*0#(RvUsV?wun-h$<Id(B1brjPIW((EW8eFzq1>;FB4Nx0HN
z_V!rr-uyJ<pZ_+&*WoW?_>}F;i6qhtl>e~KU3DEsxsI8%dBGBCKgf1?N(Au+RQ?KH
zy#w(D7|F#Z4Hp6^fRv~tjIyn`m|bzs+jZji$KaJKa6b)ya0mYOf0KOpJ1L5PFOU19
zONeq@&qBTfR%&_HMl1jZ80K%Og-v$DernH2+2^V-GwBYHJJf}z=6_Gna=Bq=U^@>e
zCWwD$jn$XUn(co>5=fp%d!VeFba4dHycsX@?0yr9?_wqk-mzB82Qk*ELB3eS=1cI>
zO;|3&<dh)o#Ryr0fGkp&tlEj}Ew$%RR`B>Fq_<$H1Yf-m*WOe7_#YIx{Bnhnk8YtJ
z6^Kk+Uw2JA(9BrMWiR;+%;Zqny6Z(uPVrIu_hf^sU$W12jQZZ;gK`(nVP^EJgZQy|
zE+V+i;$!e9>+m9G;$|!9X^HFsl=W9{JP-F?Fngt(It{928088kko?>B1zf`9pFUOp
zN(kX=aN{Mon}v}X81Z71ti{O{02g5sye86_C}7Nm%`r$-VRH&@jY<A^#pA|{qx_wF
zcbR-9B=Zr09FGmHq4si~_i`D2*#gtTKDC;nJ{eS^o{6%yfrz5M<5**6d)07PHTi<|
zQmVsfzT1F9maV5FUYLhJ#3Y_vO?p-$9Rp?j9xOf&uU5@&eUowX7^`3wCcJ=&n5b<u
zaEW?;NdvgiJMiAO;6@DD^H9zUl#K1HqeE;b3{*N<l+&%GB$RScO@XRH&Vh}R;w5i{
z`wup_@<%KD(2Fs}4}<y<P@@*7T3VRx<kMY%)xuG4GE>1KQ04ol;SVF@3f)b5QzD%M
zWfiy%3opWV*1^95m!AT$4WWVg0$;c8c4*zJE7l!L@A$#^J-Gd6aBUsp%TOMLl!z_p
z3fp7TAi&F#AY~73c+rIiV^F<g)@$b^IC;fVYLqYMw|MnVm8X9%=E~O~_Y9<d0_3E9
zrl}@>K5hk|w@*S-&0B*vW`&zG@IQQ$`R5HDmfU`c3=SxRufbby!fO$1J_Q$l6fQcT
zWCcU&f5QUg7G^Tw3Wkd;?*6s6;Pzj@->kvLMVOg}tPqIpM20RpE|~=t5;ECjXUCo%
znuM_;+}Q%PY``uhB~Cg;${puUYMwVnL%wwvuKaiSz&A`fHRl;{F98#nWuq>{%k{sF
zH~#VC-aIz}!88%eEW8iCylR4lU!nh<E)scI4V3qf|4KYx7$bUrmh(TT`CF-F9NtE7
z!@iWaEbla9L2{z?FTDoy&%@uWz{*89a{)%v0zrc{DJCE`*DDWZ9))bO5o1rHf%z(|
z+<|!AT7@x$L_{nS=tU}8Q{9A94`BQOTn^zv3G$~P^%STN!V?RZ$vg?pIH<9YhboVQ
z<QV0X1j_~afTTcx&$~~+*H+BxhifDgHHq{Kl*Ol~;@po(9-9_CZfrg~47CNwGr&{U
z-EJ|98JMp4_BY|%%TSqtGmpVot~rNI79q|O$1V`z6(twX*i37Q^dwsmENsB~Z4f>t
zct17|CFCBO2gFg#MhKO_?9o$Lg;T3=LBo>;h=1WSPXJ^5GG(_)_5wJkL5+dR0uCY4
za@hxzy}S$8+JNIZV(yu}MrPoD`DT*lR^nnOGB7W0>Stqq!0{R1KBX|>AZIaQ8710=
zd2B-PO?cyba4m$&6EJ=ON*;JL20uFnXk+_3LXs^+NSUByKa!F#ngYK7{vwFD*5WP>
z<2Y2&a)W{gBOX*U;1}Vo0A8=c3l$SZ`~Lu_;stR2229;B({m|@1xFI7D|jd>@-$_k
zSf{{*wbD{3JnAa~<&c1M9z%}D+jj05A{N;I-nk54Uo@LjzJi%`OeB%P0_9IQKZC@#
zrlF#~R(PLC@LN#*b9nhCyq<#alHnvrU2t}QqyR=ijBKy#7B1wnGT|rtbq9d7f<h4%
zD-dtiU2yaKCQ#g>mR3wj6R?w45P4>EGb4tV^w|+^{)+c3)1t)sr80NaG|^On`(l!)
zILfW4fRf{&6h*QGlo-W9iV{m?7$pm&*+6cmwD&E@xqwD&?;@*gM~H1F4D9)OqP2@e
zED}+rf|;#;<0<&R%O(c$Ivr)bNF<RyfKs-al-WyG%zOL(OYp5#xLW}CGF%ulk#jpi
z63pJ+qlj2;pLL6nbD=3UK_b09K=K|`#$a<EMA+zpk64dj5gNH-VzdO80QjX=KRd^-
z^TFS$@<QP*zf*$rG@Q=E)P%w1Oc6%PP*#%!nKYS_<gQaf6kJF-+{<K$Q&~`w3MnMY
zLZd*SkWwNM2<+Cm#0b7f#2O(1o0uToCE)hc@b(hC<{MDFW3Ac5txn|N##zVTv|&UW
z--Fi{;7tLm7a(^Y&gUCf1vi3Z8Z*VJ>REyu$d_u)d;*e#uqsmu$~g!|AY5s5!7<i6
zRZKRa+Vc@8Vhf-xlGy$x&wQrJeIGu1&xAeHcG+$I*G$|cmx4zWJW_xO7jg>Rw4pd1
z2~G;60-?-K5K@|;hfqSG*mh?sUxK^q@NQt1do594RGZ|OO5}(GWdOVZ!He+L9e6#2
zr2>e@;MBMo8R?c68rKif^?@tEDM2ZfH2fe%0V4|5$6&1r@m3?WxR|Ae8neE;29}$^
z7hC<@H$Qg=>BH|~(haqG18&Pbv$EwS`$^<P$y|$bpGbgmmj(G21mA(%cj5XL-1i`Q
z5;9Y8dc^QuUfZENZd^aX4sNnkZUwA~3qA-Jd@KdQIBea4c!!~wwqSXO`ih&^TebP~
ze3Hp|laKL#CawA~1JB$Z=Ra{w+IevGbhri-FsK(}ILKQNyal(H;MGl7cECRi&LeQP
zWWJu>)4mx8$|iKO!J2f-P;6;U$^<M2;DTo)OyuFV58-{9HpR(dR#M%?%zf+%FZ{23
zk`H{!EW!Fu3~9l20+PcY)WfUmQ3uPpp|!2^##hV|xQu0>)->GLa5IKA2R6@vnt_P|
zjJwSl3Y#ZxYN17CQ`*3li42rW7GPUqVgi<f0Lu{xV<osBfL?BPm}_8p6SGXQq41w^
z`BNsm<~fdZ(%N=|_$XUBb0@I39W}5VZX3%N;&q<)HD9U1SE`!7h}U_gpjmtjf*;XL
zTv_L-G0j;Qa%ek-v>ldVgRZP;YQyRT#017fbpkRIP;`wR%fQZA)>f1)V+@%U3zju~
zxv9LBzGBbydOUexS#%*kPMh5=)4-qFc2>YY0OtPsGd@1>`-X5ij({_F4C~zdy}4rs
zICICOjr*^sjDO6b`eE4mkom+Pc1V2)&OQN8mf*bWkfSk^kke#IIu;x!sZHM0Ty8UR
zJ)AMKQpe16+fejfhT}+sW!b2+kDluDc=E&r#y!YP(uRee$AoPO;3I#+Cz<e>`wC`r
zons6*bH}jGN9i7pTYbBI5%mF-PmQ=-D#7JZ&6A_>zP!u(6r4g^fUK>`sacV(1<9IT
z#N;Hj(aXZIF1N83MZuvo3V9hc04oTZt;(8H->t5w2g~<zJeiUL#z(*zWoKZI07FPm
zK=7~N*LeI>#t;8}m^-3tqEDigbJXtPIMM@i2X>D#jUoqR3CIIEl!daO0f>av;|AL@
z!UQwVSnJMAvx?ZYwq^(!C{DVJD-hf9dNtq;tySq;y|!s>x4LUVyVv8%l{_$729a;T
z$ruo6;4O4etXvI=fIkA>K6(RY?jae#hYehF$EdFT{?Fd7YjA*@1CK*?CM^sbOp>g1
zvrkbE5c0RbX*_J7%%llfQ9A&R?eB|@fUvqxdE<9_J)T@Cxsac#H&@J6M6AFcA@x<2
zjtoG4@#q26akI#V@xKrG8O*l$Jfy(xwEPoc^~NdF+FmPT)>dhbKW>PqRG=n3rgDv8
zL|&WkZ$Oj<=~)omu%o)BZ&Ke)uvK}@(d_4Va>WHEQcyhw)muQdO3WDg=0A;Jh=_r&
z(sq}w!<yM_*O3LCFVZ=Ty3g$oJk9s<>|x)>T-&-{ZE^a!*}dv~5N+N8$kp2U*}=7|
zQ|B5z>fB?aiyYaqJcf^K2A7@OBPu{m!c@tzMod%rE`5VoW)YDGvK0{4bhiyq_QOv;
z9LX@#z;qVwoPp|m1Z%fgdb=1#8Xx%b(E^-7faF2fy|cl4br)G28{p{(Fb@kP8@)<%
zes8cm>=%*PuKRPd@xa<v8eohYyzoXKtL+JW4wIKM)!>4=whbF}1!^p0tOR9T+r+!M
z$(?$|&Vsvka;LJZe5VE4T;2~q`S5Hio&%<{ze4tmj}E=B&EJ!6N3r-K@ZSDc|Ly?|
zf(}4(_9(lAoq}ZjzOK^RavT0VwExZBAi4AJtL)Y0b3^PO`?vAcru(?sbDwLrC^voZ
zGG=>3Ex?!H70j+w+5^7|p2v1lEZ;YK%sMtxO-=XAC1V+%(b%bD;+79-^Zk~(vkq$@
zZb1Avh}_c#3vU3Amm&Xslh5G>@HmiPhqbt_iz<wyVZFy=k%tH@x0t3x@QtfuE0Gw*
z;Y<9KsPdtBpSNzza95D**IDB|-ND0i3FrC%$)=aUzLw=6?_sV(dZ1NZ&GXpXvK(}K
z&-K{G4Q<<}RVjY>=l*4^q=%zM2(^cf1V(oWb%2gSlyr9L%E6T`K}}CPq^4ZP3IW+5
zB;z_r$0Z5_wt|rK9=deCPi5^q@7^jib7q-rzRboNyn72~&Ok2z-5uq*{5w`JGIM4H
z;bq9C(ri?>dOAMx&;aE>!Mt%?{@t(s_Jg|_RghKGN1laeh<UCn7v&%ZL6fUJyT9eN
z|2=%-h=Jtpm%;GwVQ!!Lwgbr)FRp#-doaL$wS)Th+pbc`8Q>J?Jdi?2>*8t~2PvJp
zE{rZ>h-><YLuwkP3ISt)<JPQ6bG_O}Cp7Vp*_{1<)-%u6!X%r@>$;%~T%{lkdOkip
zvS#U&k8?kzikUK>mA?wlG`N6!yRgG>ulfN=LG#`e``^Q$!gdA?l67!8n0x34BwM)Q
zL#ylD06?<S{=IiyrBogu2qS|~QW@TEhc1jRVxz}EC_=<I<j=dD9)V(DzVABb`lv%a
z(q_uXOZir}jHI12+FjnL3zXv$Y2EYj;gS?^fs}_R-{j?gobR?X4SEpl?`pRP$zBF-
zcf3^3_dF}Q1j&N}%eeu7WQ%8TsKDNNefw$Gn!QLA`j)h8(1l&F7#lqX92-U{CVa*w
z;hbl}%>!+&-<}fcQ1^WUeCbS0Uj~**XUx&xYA)ZYOF#!zB#}MR)$!qwPut^Zc(#nI
z;Pwcg&$XEJzuEv$`^)bR4{&DNk@2F>f!oUfp6e1M`<KCk<$jYl!vywWSdzWexp%PK
z4mgh;Z^!+|j<>yDI|gFU0}sR`q-S8N<S|V>SV*g#agv?tdea;|Z=<zd^&o0r4iLC0
zKoYoKgWmb^<jW#vhJT6AhG=Jm|D5}!R>4t%9W$}}cK)~ru<RTpTf5qW9Jq7+4cMW8
z<S?j&{U&cj-}N2bO!!=feWUj}_tLI%n|oAsE>gsXi;RtrY-Y449a86DvK(+xL8(hF
zvYqZ%4Y(<_(N@<IlxSVPlXrBw=P4fwsoHnqFE06!Sk5l+{4eJB>-?e%u<SHpML!@p
zEGuWOO+C8-$iZ2Xy-^9hgXGl#*0+DYa`+(GSzUM6MTQveQOP<t83WRBAq9FgBy}Dp
zrd=LM!)Oqiy&AjZBHQU(6`NDYskGG<;CQXe+w+fMM?B@jBM$uf^hO-;;I7-+?e5K0
z>7<i)xSFDGkn9I6d*?2j<M##)mc6^Rb3+E57WJJSQXd&;eZ`JG*GZiuzTqB~^=u;P
z!zj2jkbVTlOFkDoIBlJ`Ub)D2y1xNXM)EXJz<)A;d<MxAkXQH{%Rq?3B(R3R@j65c
zNF?wkI!GhY=paOL8y&o9kJ_I1+LVV#9;V5h(U(6{9g%^{pShptW=)dlg14NbeU|BN
z`AEU=UBW)aT9U()ZYFo;4iO~#=Kv25BzM-eQ_E87vIk=zra@c;_i>NXwAoqlqJon0
zcC6Cwm5XerZ)DA+jrpu{o^^6;0$b$JY-Fu9WOk%xBRhtcQxKg4E)uxcfCT~<V2Z%K
zWRKYM*Ma%A<tJbMxjFpCxr~pH-WI?1kMezVszF<51G6Igr4V`<yu+|0`vLs9Lj=ig
zIKb{Gnc>y96D!zSl;s(R(Mgyr1U#08$2^B~KrReWeQ=MR>e<|@z?%@hXMt-G6Z5wT
zgf^a1L6`<IWOE}Z3dB(vqB5?!1mP5}0>TAcb<G~J=PyBc%IfpG%1^$m04}rZkGM!S
zwtkN4h#2;O?bcEllodG$3ZYYqKpfyR>;)_j8g#m;Ylkn=ZHs-l_3gw8mghW|X=%8}
zG~f-0d+b#AQcnRFiS#*40^nM9q72~JcuGyAw}4H&)H<+Ate4Sp8Rc9f(hDdDB0Yt2
zF4-gYd}H}3TRyQ}e)1&*<aaB~2$`b#GpGCMb^`><mLSQ&42ao&>MRZzBnQ~`vj@@6
zgX<>;W3hJzmeR3&;h?$4PW5!gY)0n$WuO3SBe0f%wKS|L(it<@eSZ^JBb`|U7V!P|
zP`ruc5;~YdMxcWQB-iW_bG}wiWBEO+Pdps-<Zq@@FzKZx`47{@qcLC)79@M65DwFd
z9B|torWM?i)h-UaZTD(dIV|q6Q{D3nKdCnjM_KXKsjx|9bBhWZ&jWDKn}(zGyfI)3
zy=gc~xJKYE5K9Qu6mW^aH7<4FE^L<@y}A6}LxGc@TPjPOOf<gA=kwwy4A^12+5@Wl
z>|tAx1L6S>4dgrL0C!vMN2^`su(-!gb)^hP=~oR$sg&U;{i@+8O{k;7uNsb0sS%)n
zU$xufhL>>F6e0xWE<kt<S1mvYuA15|H+qZm6D|J&fTijgR|+;){yr<>FaT$lRdyHv
z+2Nq5Su=JS;!y+0?!Z#?v7GV{Kz;UTwD<1%3gkg?k?nNPF&yPq#c-6KV>rsKisdLB
z0L@m#a+HqYD7Pxc5{>jVl(Rsj!F8sH^d*!tMHFDq*UMF7xi;k|TTHefpED>&1RotZ
z)6n6p@+=*OVjQHU(^U7_#(KtF?K(;DLD!xAUEA5C(B>V!P>)PqTh%v4cF!$l0N3hn
z58ULs`Ny>iP=<UN|H(3B3&?N;(i!~oX;2Ee2Ivrfd=u8!kT(}$aRFbx2k*X#PF+Ik
zDRcz#{sQ{eHRSyT0x;*@*5x`f<tJMLKz<LVaf<xuSndG$o!R|d?V|@gH4wjgte&l4
z&%o(fal~FsdxPcCvfOvq*U`$`k>`7VeWlCqH|rC)_`p4&2y4saFd4Wt^^GS^VHU?#
z0IeMI%3(8QY^2zJ7J!RyUPBj3M7aV><taADPm|G_oWF@)Ucgy>fF3Ck<y)6Cdin$d
zWdxtWgHk3x&A-ixVFQlMsW?jui$@%B(1}~~YsUaEceMMn;)qyht=sr2Hiu;+)p0~$
zppEr)y5|<{w5iP-sobE;>n%Wr6M#?UjaR`<f$N$~>W%}-1J?n^1?8ZWLJ8NTE)oia
z#FCfX%9po+ueiLx=;{M>p+uM~uwsERt2J4Flj7n6qw71%PrCfpnj>%udg>f^#esA#
zTDYy^SO88PEX6T<%5@9qx!I#^88>Eg?_*Z{pk)>u*Kr{X40q@{?60ptXmVxXg6kB!
zynX^gpD4HoZjqahK80&t^Az>{AEiAiKxvIYBSWJ%?k=9YcV`zcwgb-9u_-ntPm>8W
zxzLoKY>N^qp(`CPRXuZR*M7l6AGpU)#Ib+gSC2s(Up<Cx>PYME2HMAU9VFi+;v#`A
z;JT|w+@|s(5dneTz;(0izP=M(5NHqA6`akN8~kz&aRjl3Fork)t<gF(7HJTI_8|<-
z-aSFUgUdg(uSHoaFDx%mUVT7dUG<gX$+sv&9O-$kKQ%@oi6qjaY;Wehn@glmF0wN{
zx3*kAXW!{HuClgyu2v4Aa)?sK<Bu_k!y@V&WhWM8b9sJc0me_XMfnP-tXpJpES+@r
zL=x#8D0gv*SiNSF3^$JQAUMi)<!dLMqkI+*WF|k!fAo^Ro=76S1LbZm5z9SdxyVkr
z$KLdV<S1K}pJa~mUssSgBAS@u)e}I!C6dVD0cD^wxGt7ER0R6{xNd}GiwKWEYg~5&
zNkoJu$Y@+lF2~5x1Y6&2xE=z{-g*^W&*}os=6{wQzIQG5r{-hpisdLH^J7|Dj#5LQ
zd2rb!qpRfnC5p?6(NzasP=q<fN;${oc#(`&<a{VDM~tq5E(pTh2rK206a8>20g<Cz
zJvm7!!O27t=>sTtbCG-NsrCBb640vH)%Dd9C=K!fM0v=qePIVq?&c_e8m%BywBaa0
z7tAiUD`nUmhm6JoBNiCjmos{DA8rr{90xs~=g*U_o=75n0A-zvG>854jV2>Zg~hI}
zFFb%S5341}YN)*N5rd>ICepm-Z@aa>D=1|%ushTjBv{}3R=zu6Y%V{^5|a@^i7-Es
zbo4|L=?f_9TqI50WNSUUzVip7kz!ZZ*8s8xm^UB&!af}3PouOs-<qRLEXut^xo(_I
zB^^DHM7lLI>LMEBC@q(G9ikZsvfyr*gq4lu8za^_+yc^&CEpo0Y4rjoL~_-EYz$}f
zA7h8_-IAlUdSJg30>057Tuv;?_To#WgifiX+b5DpUl!#yMp-WL9Za6_bR7$KlxHpQ
zZK+yWdPjNl@3qn;;3#uUILhI^7L{^!f|H3P(l1apbBWEeFZY<*r+ja{^2Wy!j&gV+
zln`;!%@awaPoQk(5}Rdz?$PUrn{1~i9OZB=N+DyJbn`?K=@Te}4BJ2zN#Jn>v4X$_
z1p#m4Zh?#;(w1w~AT>l<5L}n_>oGK9H~QJ1{8%F_TNsS20}fO*I-p@a<Ic|m37IX(
z1yEEls@QhbSJJSVH`|ouBot#9)u0808Cc1j{O2pB_LFX&NFsd>q{azR0i0zXVF676
zn)7T?pJUO$2K70WcZ*S975L#V?``mG{oOiz*8YFg@c$)dUMI5kmG&qw>}j^a`mFJv
zHS6`{PX>!IC4)-R%@awaPoUg=>Qq-hkR7S^Re^6jvv;<J4^)$f)U%`|;3uwkB8hZw
eWKa(giuwOOHG%!ZyJh_V0000<MNUMnLSTXcEgTpC

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/images/mbstobon-ui-3.png b/app/javascript/flavours/blobfox/images/mbstobon-ui-3.png
new file mode 100644
index 0000000000000000000000000000000000000000..a1fb642a003b2e2b652b1e6022a84f08dc76105f
GIT binary patch
literal 32449
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%0Arc|r&DC@j0pAQE7yV9T$8x=q-T1FG
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
Cr$1f*

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/images/wave-drawer-glitched.png b/app/javascript/flavours/blobfox/images/wave-drawer-glitched.png
new file mode 100644
index 0000000000000000000000000000000000000000..2290663db49ef6336a1bcdcd39d81811fcf4b5a8
GIT binary patch
literal 3544
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
O0lJz-8rACdvHt>8r?yZ4

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/images/wave-drawer.png b/app/javascript/flavours/blobfox/images/wave-drawer.png
new file mode 100644
index 0000000000000000000000000000000000000000..ca9f9e1d85d222e7bf80649c55783db928cc79d4
GIT binary patch
literal 3269
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
D7raE)

literal 0
HcmV?d00001

diff --git a/app/javascript/flavours/blobfox/initial_state.js b/app/javascript/flavours/blobfox/initial_state.js
new file mode 100644
index 00000000000000..52066aac420546
--- /dev/null
+++ b/app/javascript/flavours/blobfox/initial_state.js
@@ -0,0 +1,144 @@
+// @ts-check
+
+
+/**
+ * @typedef {[code: string, name: string, localName: string]} InitialStateLanguage
+ */
+
+/**
+ * @typedef InitialStateMeta
+ * @property {string} access_token
+ * @property {boolean=} advanced_layout
+ * @property {boolean} auto_play_gif
+ * @property {boolean} activity_api_enabled
+ * @property {string} admin
+ * @property {boolean=} boost_modal
+ * @property {boolean=} favourite_modal
+ * @property {boolean} crop_images
+ * @property {boolean=} delete_modal
+ * @property {boolean=} disable_swiping
+ * @property {string=} disabled_account_id
+ * @property {string} display_media
+ * @property {string} domain
+ * @property {boolean=} expand_spoilers
+ * @property {boolean} limited_federation_mode
+ * @property {string} locale
+ * @property {string | null} mascot
+ * @property {number} max_reactions
+ * @property {string=} me
+ * @property {string=} moved_to_account_id
+ * @property {string=} owner
+ * @property {boolean} profile_directory
+ * @property {boolean} registrations_open
+ * @property {boolean} reduce_motion
+ * @property {string} repository
+ * @property {boolean} search_enabled
+ * @property {boolean} trends_enabled
+ * @property {boolean} single_user_mode
+ * @property {string} source_url
+ * @property {string} streaming_api_base_url
+ * @property {boolean} timeline_preview
+ * @property {string} title
+ * @property {boolean} show_trends
+ * @property {boolean} trends_as_landing_page
+ * @property {boolean} unfollow_modal
+ * @property {boolean} use_blurhash
+ * @property {boolean=} use_pending_items
+ * @property {string} version
+ * @property {number} visible_reactions
+ * @property {string} sso_redirect
+ * @property {number} visible_reactions
+ * @property {boolean} translation_enabled
+ * @property {string} status_page_url
+ * @property {boolean} system_emoji_font
+ * @property {string} default_content_type
+ */
+
+/** @type {string} */
+const initialPath = document.querySelector("head meta[name=initialPath]")?.getAttribute("content") ?? '';
+/** @type {boolean} */
+export const hasMultiColumnPath = initialPath === '/'
+  || initialPath === '/getting-started'
+  || initialPath === '/home'
+  || initialPath.startsWith('/deck');
+
+/**
+ * @typedef InitialState
+ * @property {Record<string, import("./api_types/accounts").ApiAccountJSON>} accounts
+ * @property {InitialStateLanguage[]} languages
+ * @property {boolean=} critical_updates_pending
+ * @property {InitialStateMeta} meta
+ * @property {object} local_settings
+ * @property {number} max_toot_chars
+ * @property {number} poll_limits
+ * @property {number} max_reactions
+ */
+
+const element = document.getElementById('initial-state');
+/** @type {InitialState | undefined} */
+const initialState = element?.textContent && JSON.parse(element.textContent);
+
+// Glitch-soc-specific “local settings”
+if (initialState) {
+  try {
+    // @ts-expect-error
+    initialState.local_settings = JSON.parse(localStorage.getItem('mastodon-settings'));
+  } catch (e) {
+    initialState.local_settings = {};
+  }
+}
+
+/**
+ * @template {keyof InitialStateMeta} K
+ * @param {K} prop
+ * @returns {InitialStateMeta[K] | undefined}
+ */
+const getMeta = (prop) => initialState?.meta && initialState.meta[prop];
+
+export const activityApiEnabled = getMeta('activity_api_enabled');
+export const autoPlayGif = getMeta('auto_play_gif');
+export const boostModal = getMeta('boost_modal');
+export const cropImages = getMeta('crop_images');
+export const deleteModal = getMeta('delete_modal');
+export const disableSwiping = getMeta('disable_swiping');
+export const disabledAccountId = getMeta('disabled_account_id');
+export const displayMedia = getMeta('display_media');
+export const domain = getMeta('domain');
+export const expandSpoilers = getMeta('expand_spoilers');
+export const forceSingleColumn = !getMeta('advanced_layout');
+export const limitedFederationMode = getMeta('limited_federation_mode');
+export const mascot = getMeta('mascot');
+export const maxReactions = (initialState && initialState.max_reactions) || 1;
+export const me = getMeta('me');
+export const movedToAccountId = getMeta('moved_to_account_id');
+export const owner = getMeta('owner');
+export const profile_directory = getMeta('profile_directory');
+export const reduceMotion = getMeta('reduce_motion');
+export const registrationsOpen = getMeta('registrations_open');
+export const repository = getMeta('repository');
+export const searchEnabled = getMeta('search_enabled');
+export const trendsEnabled = getMeta('trends_enabled');
+export const showTrends = getMeta('show_trends');
+export const singleUserMode = getMeta('single_user_mode');
+export const source_url = getMeta('source_url');
+export const timelinePreview = getMeta('timeline_preview');
+export const title = getMeta('title');
+export const trendsAsLanding = getMeta('trends_as_landing_page');
+export const unfollowModal = getMeta('unfollow_modal');
+export const useBlurhash = getMeta('use_blurhash');
+export const usePendingItems = getMeta('use_pending_items');
+export const version = getMeta('version');
+export const visibleReactions = getMeta('visible_reactions');
+export const languages = initialState?.languages;
+export const criticalUpdatesPending = initialState?.critical_updates_pending;
+export const statusPageUrl = getMeta('status_page_url');
+export const sso_redirect = getMeta('sso_redirect');
+
+// Glitch-soc-specific settings
+export const maxChars = (initialState && initialState.max_toot_chars) || 500;
+export const favouriteModal = getMeta('favourite_modal');
+export const pollLimits = (initialState && initialState.poll_limits);
+export const defaultContentType = getMeta('default_content_type');
+export const useSystemEmojiFont = getMeta('system_emoji_font');
+
+export default initialState;
diff --git a/app/javascript/flavours/blobfox/is_mobile.ts b/app/javascript/flavours/blobfox/is_mobile.ts
new file mode 100644
index 00000000000000..7f339e287bfd61
--- /dev/null
+++ b/app/javascript/flavours/blobfox/is_mobile.ts
@@ -0,0 +1,34 @@
+import { supportsPassiveEvents } from 'detect-passive-events';
+
+import { forceSingleColumn, hasMultiColumnPath } from './initial_state';
+
+const LAYOUT_BREAKPOINT = 630;
+
+export const isMobile = (width: number) => width <= LAYOUT_BREAKPOINT;
+
+export const transientSingleColumn = !forceSingleColumn && !hasMultiColumnPath;
+
+export type LayoutType = 'mobile' | 'single-column' | 'multi-column';
+export const layoutFromWindow = (): LayoutType => {
+  if (isMobile(window.innerWidth)) {
+    return 'mobile';
+  } else if (!forceSingleColumn && !transientSingleColumn) {
+    return 'multi-column';
+  } else {
+    return 'single-column';
+  }
+};
+
+const listenerOptions = supportsPassiveEvents ? { passive: true } : false;
+
+let userTouching = false;
+
+const touchListener = () => {
+  userTouching = true;
+
+  window.removeEventListener('touchstart', touchListener);
+};
+
+window.addEventListener('touchstart', touchListener, listenerOptions);
+
+export const isUserTouching = () => userTouching;
diff --git a/app/javascript/flavours/blobfox/load_keyboard_extensions.js b/app/javascript/flavours/blobfox/load_keyboard_extensions.js
new file mode 100644
index 00000000000000..2dd0e45fa714c4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/load_keyboard_extensions.js
@@ -0,0 +1,16 @@
+// On KaiOS, we may not be able to use a mouse cursor or navigate using Tab-based focus, so we install
+// special left/right focus navigation keyboard listeners, at least on public pages (i.e. so folks
+// can at least log in using KaiOS devices).
+
+function importArrowKeyNavigation() {
+  return import(/* webpackChunkName: "arrow-key-navigation" */ 'arrow-key-navigation');
+}
+
+export default function loadKeyboardExtensions() {
+  if (/KAIOS/.test(navigator.userAgent)) {
+    return importArrowKeyNavigation().then(arrowKeyNav => {
+      arrowKeyNav.register();
+    });
+  }
+  return Promise.resolve();
+}
diff --git a/app/javascript/flavours/blobfox/main.jsx b/app/javascript/flavours/blobfox/main.jsx
new file mode 100644
index 00000000000000..c6ee5617532f84
--- /dev/null
+++ b/app/javascript/flavours/blobfox/main.jsx
@@ -0,0 +1,47 @@
+import { createRoot } from 'react-dom/client';
+
+import { setupBrowserNotifications } from 'flavours/blobfox/actions/notifications';
+import Mastodon from 'flavours/blobfox/containers/mastodon';
+import { me } from 'flavours/blobfox/initial_state';
+import * as perf from 'flavours/blobfox/performance';
+import ready from 'flavours/blobfox/ready';
+import { store } from 'flavours/blobfox/store';
+
+/**
+ * @returns {Promise<void>}
+ */
+function main() {
+  perf.start('main()');
+
+  return ready(async () => {
+    const mountNode = document.getElementById('mastodon');
+    const props = JSON.parse(mountNode.getAttribute('data-props'));
+
+    const root = createRoot(mountNode);
+    root.render(<Mastodon {...props} />);
+    store.dispatch(setupBrowserNotifications());
+
+    if (process.env.NODE_ENV === 'production' && me && 'serviceWorker' in navigator) {
+      const { Workbox } = await import('workbox-window');
+      const wb = new Workbox('/sw.js');
+      /** @type {ServiceWorkerRegistration} */
+      let registration;
+
+      try {
+        registration = await wb.register();
+      } catch (err) {
+        console.error(err);
+      }
+
+      if (registration && 'Notification' in window && Notification.permission === 'granted') {
+        const registerPushNotifications = await import('flavours/blobfox/actions/push_notifications');
+
+        store.dispatch(registerPushNotifications.register());
+      }
+    }
+
+    perf.stop('main()');
+  });
+}
+
+export default main;
diff --git a/app/javascript/flavours/blobfox/models/account.ts b/app/javascript/flavours/blobfox/models/account.ts
new file mode 100644
index 00000000000000..02541f1f93fab0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/models/account.ts
@@ -0,0 +1,149 @@
+import type { RecordOf } from 'immutable';
+import { List, Record as ImmutableRecord } from 'immutable';
+
+import escapeTextContentForBrowser from 'escape-html';
+
+import type {
+  ApiAccountFieldJSON,
+  ApiAccountRoleJSON,
+  ApiAccountJSON,
+} from 'flavours/blobfox/api_types/accounts';
+import type { ApiCustomEmojiJSON } from 'flavours/blobfox/api_types/custom_emoji';
+import emojify from 'flavours/blobfox/features/emoji/emoji';
+import { unescapeHTML } from 'flavours/blobfox/utils/html';
+
+import { CustomEmojiFactory } from './custom_emoji';
+import type { CustomEmoji } from './custom_emoji';
+
+// AccountField
+interface AccountFieldShape extends Required<ApiAccountFieldJSON> {
+  name_emojified: string;
+  value_emojified: string;
+  value_plain: string | null;
+}
+
+type AccountField = RecordOf<AccountFieldShape>;
+
+const AccountFieldFactory = ImmutableRecord<AccountFieldShape>({
+  name: '',
+  value: '',
+  verified_at: null,
+  name_emojified: '',
+  value_emojified: '',
+  value_plain: null,
+});
+
+// AccountRole
+export type AccountRoleShape = ApiAccountRoleJSON;
+export type AccountRole = RecordOf<AccountRoleShape>;
+
+const AccountRoleFactory = ImmutableRecord<AccountRoleShape>({
+  color: '',
+  id: '',
+  name: '',
+});
+
+// Account
+export interface AccountShape
+  extends Required<
+    Omit<ApiAccountJSON, 'emojis' | 'fields' | 'roles' | 'moved'>
+  > {
+  emojis: List<CustomEmoji>;
+  fields: List<AccountField>;
+  roles: List<AccountRole>;
+  display_name_html: string;
+  note_emojified: string;
+  note_plain: string | null;
+  hidden: boolean;
+  moved: string | null;
+}
+
+export type Account = RecordOf<AccountShape>;
+
+export const accountDefaultValues: AccountShape = {
+  acct: '',
+  avatar: '',
+  avatar_static: '',
+  bot: false,
+  created_at: '',
+  discoverable: false,
+  display_name: '',
+  display_name_html: '',
+  emojis: List<CustomEmoji>(),
+  fields: List<AccountField>(),
+  group: false,
+  header: '',
+  header_static: '',
+  id: '',
+  last_status_at: '',
+  locked: false,
+  noindex: false,
+  note: '',
+  note_emojified: '',
+  note_plain: 'string',
+  roles: List<AccountRole>(),
+  uri: '',
+  url: '',
+  username: '',
+  followers_count: 0,
+  following_count: 0,
+  statuses_count: 0,
+  hidden: false,
+  suspended: false,
+  memorial: false,
+  limited: false,
+  moved: null,
+};
+
+const AccountFactory = ImmutableRecord<AccountShape>(accountDefaultValues);
+
+type EmojiMap = Record<string, ApiCustomEmojiJSON>;
+
+function makeEmojiMap(emojis: ApiCustomEmojiJSON[]) {
+  return emojis.reduce<EmojiMap>((obj, emoji) => {
+    obj[`:${emoji.shortcode}:`] = emoji;
+    return obj;
+  }, {});
+}
+
+function createAccountField(
+  jsonField: ApiAccountFieldJSON,
+  emojiMap: EmojiMap,
+) {
+  return AccountFieldFactory({
+    ...jsonField,
+    name_emojified: emojify(
+      escapeTextContentForBrowser(jsonField.name),
+      emojiMap,
+    ),
+    value_emojified: emojify(jsonField.value, emojiMap),
+    value_plain: unescapeHTML(jsonField.value),
+  });
+}
+
+export function createAccountFromServerJSON(serverJSON: ApiAccountJSON) {
+  const { moved, ...accountJSON } = serverJSON;
+
+  const emojiMap = makeEmojiMap(accountJSON.emojis);
+
+  const displayName =
+    accountJSON.display_name.trim().length === 0
+      ? accountJSON.username
+      : accountJSON.display_name;
+
+  return AccountFactory({
+    ...accountJSON,
+    moved: moved?.id,
+    fields: List(
+      serverJSON.fields.map((field) => createAccountField(field, emojiMap)),
+    ),
+    emojis: List(serverJSON.emojis.map((emoji) => CustomEmojiFactory(emoji))),
+    roles: List(serverJSON.roles?.map((role) => AccountRoleFactory(role))),
+    display_name_html: emojify(
+      escapeTextContentForBrowser(displayName),
+      emojiMap,
+    ),
+    note_emojified: emojify(accountJSON.note, emojiMap),
+    note_plain: unescapeHTML(accountJSON.note),
+  });
+}
diff --git a/app/javascript/flavours/blobfox/models/custom_emoji.ts b/app/javascript/flavours/blobfox/models/custom_emoji.ts
new file mode 100644
index 00000000000000..736b6b620c31c9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/models/custom_emoji.ts
@@ -0,0 +1,15 @@
+import type { RecordOf } from 'immutable';
+import { Record } from 'immutable';
+
+import type { ApiCustomEmojiJSON } from 'flavours/blobfox/api_types/custom_emoji';
+
+type CustomEmojiShape = Required<ApiCustomEmojiJSON>; // no changes from server shape
+export type CustomEmoji = RecordOf<CustomEmojiShape>;
+
+export const CustomEmojiFactory = Record<CustomEmojiShape>({
+  shortcode: '',
+  static_url: '',
+  url: '',
+  category: '',
+  visible_in_picker: false,
+});
diff --git a/app/javascript/flavours/blobfox/models/relationship.ts b/app/javascript/flavours/blobfox/models/relationship.ts
new file mode 100644
index 00000000000000..a2391683ff64f3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/models/relationship.ts
@@ -0,0 +1,29 @@
+import type { RecordOf } from 'immutable';
+import { Record } from 'immutable';
+
+import type { ApiRelationshipJSON } from 'flavours/blobfox/api_types/relationships';
+
+type RelationshipShape = Required<ApiRelationshipJSON>; // no changes from server shape
+export type Relationship = RecordOf<RelationshipShape>;
+
+const RelationshipFactory = Record<RelationshipShape>({
+  blocked_by: false,
+  blocking: false,
+  domain_blocking: false,
+  endorsed: false,
+  followed_by: false,
+  following: false,
+  id: '',
+  languages: null,
+  muting_notifications: false,
+  muting: false,
+  note: '',
+  notifying: false,
+  requested_by: false,
+  requested: false,
+  showing_reblogs: false,
+});
+
+export function createRelationship(attributes: Partial<RelationshipShape>) {
+  return RelationshipFactory(attributes);
+}
diff --git a/app/javascript/flavours/blobfox/names.yml b/app/javascript/flavours/blobfox/names.yml
new file mode 100644
index 00000000000000..4f743d6248edee
--- /dev/null
+++ b/app/javascript/flavours/blobfox/names.yml
@@ -0,0 +1,40 @@
+en:
+  flavours:
+    blobfox:
+      description: The default flavour for blobfoxSoc instances.
+      name: blobfox Edition
+  skins:
+    blobfox:
+      default: Default
+cs:
+  flavours:
+    blobfox:
+      description: Výchozí rozhraní instancí blobfoxSoc.
+      name: blobfox
+  skins:
+    blobfox:
+      default: Výchozí
+pl:
+  flavours:
+    blobfox:
+      description: Domyślny motyw instancji blobfoxSoc.
+  skins:
+    blobfox:
+      default: Domyślny
+es:
+  flavours:
+    blobfox:
+      description: El diseño predeterminado para las instancias con blobfoxSoc.
+      name: blobfoxsoc
+  skins:
+    blobfox:
+      default: Predeterminado
+
+ja:
+  flavours:
+    blobfox:
+      description: blobfoxSocインスタンスのデフォルトフレーバーです。
+      name: blobfox Edition
+  skins:
+    blobfox:
+      default: デフォルト
diff --git a/app/javascript/flavours/blobfox/packs/admin.jsx b/app/javascript/flavours/blobfox/packs/admin.jsx
new file mode 100644
index 00000000000000..9a33fe31e14461
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/admin.jsx
@@ -0,0 +1,25 @@
+import 'packs/public-path';
+import { createRoot } from 'react-dom/client';
+
+import ready from 'flavours/blobfox/ready';
+
+ready(() => {
+  [].forEach.call(document.querySelectorAll('[data-admin-component]'), element => {
+    const componentName  = element.getAttribute('data-admin-component');
+    const { ...componentProps } = JSON.parse(element.getAttribute('data-props'));
+
+    import('flavours/blobfox/containers/admin_component').then(({ default: AdminComponent }) => {
+      return import('flavours/blobfox/components/admin/' + componentName).then(({ default: Component }) => {
+        const root = createRoot(element);
+
+        root.render (
+          <AdminComponent>
+            <Component {...componentProps} />
+          </AdminComponent>,
+        );
+      });
+    }).catch(error => {
+      console.error(error);
+    });
+  });
+});
diff --git a/app/javascript/flavours/blobfox/packs/common.js b/app/javascript/flavours/blobfox/packs/common.js
new file mode 100644
index 00000000000000..439ae9d27eedeb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/common.js
@@ -0,0 +1,8 @@
+import 'packs/public-path';
+import Rails from '@rails/ujs';
+import 'flavours/blobfox/styles/index.scss';
+
+Rails.start();
+
+//  This ensures that webpack compiles our images.
+require.context('../images', true);
diff --git a/app/javascript/flavours/blobfox/packs/error.js b/app/javascript/flavours/blobfox/packs/error.js
new file mode 100644
index 00000000000000..ab27f51435037a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/error.js
@@ -0,0 +1,14 @@
+import 'packs/public-path';
+import ready from 'flavours/blobfox/ready';
+
+ready(() => {
+  const image = document.querySelector('img');
+
+  image.addEventListener('mouseenter', () => {
+    image.src = '/oops.gif';
+  });
+
+  image.addEventListener('mouseleave', () => {
+    image.src = '/oops.png';
+  });
+});
diff --git a/app/javascript/flavours/blobfox/packs/home.js b/app/javascript/flavours/blobfox/packs/home.js
new file mode 100644
index 00000000000000..a6ce1876212a82
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/home.js
@@ -0,0 +1,11 @@
+import 'packs/public-path';
+import { loadLocale } from 'flavours/blobfox/locales';
+import main from "flavours/blobfox/main";
+import { loadPolyfills } from 'flavours/blobfox/polyfills';
+
+loadPolyfills()
+  .then(loadLocale)
+  .then(main)
+  .catch(e => {
+    console.error(e);
+  });
diff --git a/app/javascript/flavours/blobfox/packs/public.jsx b/app/javascript/flavours/blobfox/packs/public.jsx
new file mode 100644
index 00000000000000..9df968e084789f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/public.jsx
@@ -0,0 +1,242 @@
+import 'packs/public-path';
+import { createRoot }  from 'react-dom/client';
+
+import { IntlMessageFormat } from 'intl-messageformat';
+import { defineMessages } from 'react-intl';
+
+import Rails from '@rails/ujs';
+import axios from 'axios';
+import { createBrowserHistory }  from 'history';
+import { throttle } from 'lodash';
+
+import { timeAgoString }  from 'flavours/blobfox/components/relative_timestamp';
+import emojify  from 'flavours/blobfox/features/emoji/emoji';
+import loadKeyboardExtensions from 'flavours/blobfox/load_keyboard_extensions';
+import { loadLocale, getLocale } from 'flavours/blobfox/locales';
+import { loadPolyfills } from 'flavours/blobfox/polyfills';
+
+const messages = defineMessages({
+  usernameTaken: { id: 'username.taken', defaultMessage: 'That username is taken. Try another' },
+  passwordExceedsLength: { id: 'password_confirmation.exceeds_maxlength', defaultMessage: 'Password confirmation exceeds the maximum password length' },
+  passwordDoesNotMatch: { id: 'password_confirmation.mismatching', defaultMessage: 'Password confirmation does not match' },
+});
+
+function main() {
+  const { messages: localeData } = getLocale();
+
+  const scrollToDetailedStatus = () => {
+    const history = createBrowserHistory();
+    const detailedStatuses = document.querySelectorAll('.public-layout .detailed-status');
+    const location = history.location;
+
+    if (detailedStatuses.length === 1 && (!location.state || !location.state.scrolledToDetailedStatus)) {
+      detailedStatuses[0].scrollIntoView();
+      history.replace(location.pathname, { ...location.state, scrolledToDetailedStatus: true });
+    }
+  };
+
+  const getEmojiAnimationHandler = (swapTo) => {
+    return ({ target }) => {
+      target.src = target.getAttribute(swapTo);
+    };
+  };
+
+  const locale = document.documentElement.lang;
+
+  const dateTimeFormat = new Intl.DateTimeFormat(locale, {
+    year: 'numeric',
+    month: 'long',
+    day: 'numeric',
+    hour: 'numeric',
+    minute: 'numeric',
+  });
+
+  const dateFormat = new Intl.DateTimeFormat(locale, {
+    year: 'numeric',
+    month: 'short',
+    day: 'numeric',
+    timeFormat: false,
+  });
+
+  const timeFormat = new Intl.DateTimeFormat(locale, {
+    timeStyle: 'short',
+    hour12: false,
+  });
+
+  const formatMessage = ({ id, defaultMessage }, values) => {
+    const messageFormat = new IntlMessageFormat(localeData[id] || defaultMessage, locale);
+    return messageFormat.format(values);
+  };
+
+  [].forEach.call(document.querySelectorAll('.emojify'), (content) => {
+    content.innerHTML = emojify(content.innerHTML);
+  });
+
+  [].forEach.call(document.querySelectorAll('time.formatted'), (content) => {
+    const datetime = new Date(content.getAttribute('datetime'));
+    const formattedDate = dateTimeFormat.format(datetime);
+
+    content.title = formattedDate;
+    content.textContent = formattedDate;
+  });
+
+  const isToday = date => {
+    const today = new Date();
+
+    return date.getDate() === today.getDate() &&
+      date.getMonth() === today.getMonth() &&
+      date.getFullYear() === today.getFullYear();
+  };
+  const todayFormat = new IntlMessageFormat(localeData['relative_format.today'] || 'Today at {time}', locale);
+
+  [].forEach.call(document.querySelectorAll('time.relative-formatted'), (content) => {
+    const datetime = new Date(content.getAttribute('datetime'));
+
+    let formattedContent;
+
+    if (isToday(datetime)) {
+      const formattedTime = timeFormat.format(datetime);
+
+      formattedContent = todayFormat.format({ time: formattedTime });
+    } else {
+      formattedContent = dateFormat.format(datetime);
+    }
+
+    content.title = formattedContent;
+    content.textContent = formattedContent;
+  });
+
+  [].forEach.call(document.querySelectorAll('time.time-ago'), (content) => {
+    const datetime = new Date(content.getAttribute('datetime'));
+    const now      = new Date();
+
+    const timeGiven = content.getAttribute('datetime').includes('T');
+    content.title = timeGiven ? dateTimeFormat.format(datetime) : dateFormat.format(datetime);
+    content.textContent = timeAgoString({
+      formatMessage,
+      formatDate: (date, options) => (new Intl.DateTimeFormat(locale, options)).format(date),
+    }, datetime, now, now.getFullYear(), timeGiven);
+  });
+
+  const reactComponents = document.querySelectorAll('[data-component]');
+  if (reactComponents.length > 0) {
+    import(/* webpackChunkName: "containers/media_container" */ 'flavours/blobfox/containers/media_container')
+      .then(({ default: MediaContainer }) => {
+        [].forEach.call(reactComponents, (component) => {
+          [].forEach.call(component.children, (child) => {
+            component.removeChild(child);
+          });
+        });
+
+        const content = document.createElement('div');
+
+        const root = createRoot(content);
+        root.render(<MediaContainer locale={locale} components={reactComponents} />);
+        document.body.appendChild(content);
+        scrollToDetailedStatus();
+      })
+      .catch(error => {
+        console.error(error);
+        scrollToDetailedStatus();
+      });
+  } else {
+    scrollToDetailedStatus();
+  }
+
+  Rails.delegate(document, '#user_account_attributes_username', 'input', throttle(() => {
+    const username = document.getElementById('user_account_attributes_username');
+
+    if (username.value && username.value.length > 0) {
+      axios.get('/api/v1/accounts/lookup', { params: { acct: username.value } }).then(() => {
+        username.setCustomValidity(formatMessage(messages.usernameTaken));
+      }).catch(() => {
+        username.setCustomValidity('');
+      });
+    } else {
+      username.setCustomValidity('');
+    }
+  }, 500, { leading: false, trailing: true }));
+
+  Rails.delegate(document, '#user_password,#user_password_confirmation', 'input', () => {
+    const password = document.getElementById('user_password');
+    const confirmation = document.getElementById('user_password_confirmation');
+    if (!confirmation) return;
+
+    if (confirmation.value && confirmation.value.length > password.maxLength) {
+      confirmation.setCustomValidity(formatMessage(messages.passwordExceedsLength));
+    } else if (password.value && password.value !== confirmation.value) {
+      confirmation.setCustomValidity(formatMessage(messages.passwordDoesNotMatch));
+    } else {
+      confirmation.setCustomValidity('');
+    }
+  });
+
+  Rails.delegate(document, '.custom-emoji', 'mouseover', getEmojiAnimationHandler('data-original'));
+  Rails.delegate(document, '.custom-emoji', 'mouseout', getEmojiAnimationHandler('data-static'));
+
+  Rails.delegate(document, '.status__content__spoiler-link', 'click', function() {
+    const statusEl = this.parentNode.parentNode;
+
+    if (statusEl.dataset.spoiler === 'expanded') {
+      statusEl.dataset.spoiler = 'folded';
+      this.textContent = (new IntlMessageFormat(localeData['status.show_more'] || 'Show more', locale)).format();
+    } else {
+      statusEl.dataset.spoiler = 'expanded';
+      this.textContent = (new IntlMessageFormat(localeData['status.show_less'] || 'Show less', locale)).format();
+    }
+
+    return false;
+  });
+
+  [].forEach.call(document.querySelectorAll('.status__content__spoiler-link'), (spoilerLink) => {
+    const statusEl = spoilerLink.parentNode.parentNode;
+    const message = (statusEl.dataset.spoiler === 'expanded') ? (localeData['status.show_less'] || 'Show less') : (localeData['status.show_more'] || 'Show more');
+    spoilerLink.textContent = (new IntlMessageFormat(message, locale)).format();
+  });
+
+  const toggleSidebar = () => {
+    const sidebar = document.querySelector('.sidebar ul');
+    const toggleButton = document.querySelector('.sidebar__toggle__icon');
+
+    if (sidebar.classList.contains('visible')) {
+      document.body.style.overflow = null;
+      toggleButton.setAttribute('aria-expanded', 'false');
+    } else {
+      document.body.style.overflow = 'hidden';
+      toggleButton.setAttribute('aria-expanded', 'true');
+    }
+
+    toggleButton.classList.toggle('active');
+    sidebar.classList.toggle('visible');
+  };
+
+  Rails.delegate(document, '.sidebar__toggle__icon', 'click', () => {
+    toggleSidebar();
+  });
+
+  Rails.delegate(document, '.sidebar__toggle__icon', 'keydown', e => {
+    if (e.key === ' ' || e.key === 'Enter') {
+      e.preventDefault();
+      toggleSidebar();
+    }
+  });
+
+  // Empty the honeypot fields in JS in case something like an extension
+  // automatically filled them.
+  Rails.delegate(document, '#registration_new_user,#new_user', 'submit', () => {
+    ['user_website', 'user_confirm_password', 'registration_user_website', 'registration_user_confirm_password'].forEach(id => {
+      const field = document.getElementById(id);
+      if (field) {
+        field.value = '';
+      }
+    });
+  });
+}
+
+loadPolyfills()
+  .then(loadLocale)
+  .then(main)
+  .then(loadKeyboardExtensions)
+  .catch(error => {
+    console.error(error);
+  });
diff --git a/app/javascript/flavours/blobfox/packs/settings.js b/app/javascript/flavours/blobfox/packs/settings.js
new file mode 100644
index 00000000000000..049a3a7187632b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/settings.js
@@ -0,0 +1,42 @@
+import 'packs/public-path';
+import Rails from '@rails/ujs';
+
+import loadKeyboardExtensions from 'flavours/blobfox/load_keyboard_extensions';
+import { loadPolyfills } from 'flavours/blobfox/polyfills';
+import 'cocoon-js-vanilla';
+
+function main() {
+  const toggleSidebar = () => {
+    const sidebar = document.querySelector('.sidebar ul');
+    const toggleButton = document.querySelector('.sidebar__toggle__icon');
+
+    if (sidebar.classList.contains('visible')) {
+      document.body.style.overflow = null;
+      toggleButton.setAttribute('aria-expanded', 'false');
+    } else {
+      document.body.style.overflow = 'hidden';
+      toggleButton.setAttribute('aria-expanded', 'true');
+    }
+
+    toggleButton.classList.toggle('active');
+    sidebar.classList.toggle('visible');
+  };
+
+  Rails.delegate(document, '.sidebar__toggle__icon', 'click', () => {
+    toggleSidebar();
+  });
+
+  Rails.delegate(document, '.sidebar__toggle__icon', 'keydown', e => {
+    if (e.key === ' ' || e.key === 'Enter') {
+      e.preventDefault();
+      toggleSidebar();
+    }
+  });
+}
+
+loadPolyfills()
+  .then(main)
+  .then(loadKeyboardExtensions)
+  .catch(error => {
+    console.error(error);
+  });
diff --git a/app/javascript/flavours/blobfox/packs/share.jsx b/app/javascript/flavours/blobfox/packs/share.jsx
new file mode 100644
index 00000000000000..73d65cb9b58764
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/share.jsx
@@ -0,0 +1,27 @@
+import 'packs/public-path';
+import { createRoot } from 'react-dom/client';
+
+import ComposeContainer from 'flavours/blobfox/containers/compose_container';
+import { loadPolyfills } from 'flavours/blobfox/polyfills';
+import ready from 'flavours/blobfox/ready';
+
+function loaded() {
+  const mountNode = document.getElementById('mastodon-compose');
+
+  if (mountNode) {
+    const attr = mountNode.getAttribute('data-props');
+    if(!attr) return;
+
+    const props = JSON.parse(attr);
+    const root = createRoot(mountNode);
+    root.render(<ComposeContainer {...props} />);
+  }
+}
+
+function main() {
+  ready(loaded);
+}
+
+loadPolyfills().then(main).catch(error => {
+  console.error(error);
+});
diff --git a/app/javascript/flavours/blobfox/packs/sign_up.js b/app/javascript/flavours/blobfox/packs/sign_up.js
new file mode 100644
index 00000000000000..42c86aca9fc2b5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/packs/sign_up.js
@@ -0,0 +1,42 @@
+import 'packs/public-path';
+import axios from 'axios';
+
+import ready from 'flavours/blobfox/ready';
+
+ready(() => {
+  setInterval(() => {
+    axios.get('/api/v1/emails/check_confirmation').then((response) => {
+      if (response.data) {
+        window.location = '/start';
+      }
+    }).catch(error => {
+      console.error(error);
+    });
+  }, 5000);
+
+  document.querySelectorAll('.timer-button').forEach(button => {
+    let counter = 30;
+
+    const container = document.createElement('span');
+
+    const updateCounter = () => {
+      container.innerText = ` (${counter})`;
+    };
+
+    updateCounter();
+
+    const countdown = setInterval(() => {
+      counter--;
+
+      if (counter === 0) {
+        button.disabled = false;
+        button.removeChild(container);
+        clearInterval(countdown);
+      } else {
+        updateCounter();
+      }
+    }, 1000);
+
+    button.appendChild(container);
+  });
+});
diff --git a/app/javascript/flavours/blobfox/performance.js b/app/javascript/flavours/blobfox/performance.js
new file mode 100644
index 00000000000000..42849c82b10378
--- /dev/null
+++ b/app/javascript/flavours/blobfox/performance.js
@@ -0,0 +1,30 @@
+//
+// Tools for performance debugging, only enabled in development mode.
+// Open up Chrome Dev Tools, then Timeline, then User Timing to see output.
+// Also see config/webpack/loaders/mark.js for the webpack loader marks.
+
+import * as marky from 'marky';
+
+if (process.env.NODE_ENV === 'development') {
+  if (typeof performance !== 'undefined' && performance.setResourceTimingBufferSize) {
+    // Increase Firefox's performance entry limit; otherwise it's capped to 150.
+    // See: https://bugzilla.mozilla.org/show_bug.cgi?id=1331135
+    performance.setResourceTimingBufferSize(Infinity);
+  }
+
+  // allows us to easily do e.g. ReactPerf.printWasted() while debugging
+  //window.ReactPerf = require('react-addons-perf');
+  //window.ReactPerf.start();
+}
+
+export function start(name) {
+  if (process.env.NODE_ENV === 'development') {
+    marky.mark(name);
+  }
+}
+
+export function stop(name) {
+  if (process.env.NODE_ENV === 'development') {
+    marky.stop(name);
+  }
+}
diff --git a/app/javascript/flavours/blobfox/permissions.ts b/app/javascript/flavours/blobfox/permissions.ts
new file mode 100644
index 00000000000000..b583535c00e35f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/permissions.ts
@@ -0,0 +1,4 @@
+export const PERMISSION_INVITE_USERS = 0x0000000000010000;
+export const PERMISSION_MANAGE_USERS = 0x0000000000000400;
+export const PERMISSION_MANAGE_FEDERATION = 0x0000000000000020;
+export const PERMISSION_MANAGE_REPORTS = 0x0000000000000010;
diff --git a/app/javascript/flavours/blobfox/polyfills/extra_polyfills.ts b/app/javascript/flavours/blobfox/polyfills/extra_polyfills.ts
new file mode 100644
index 00000000000000..a8d5530c5fcb6a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/polyfills/extra_polyfills.ts
@@ -0,0 +1 @@
+import 'requestidlecallback';
diff --git a/app/javascript/flavours/blobfox/polyfills/index.ts b/app/javascript/flavours/blobfox/polyfills/index.ts
new file mode 100644
index 00000000000000..431c5b0f30f350
--- /dev/null
+++ b/app/javascript/flavours/blobfox/polyfills/index.ts
@@ -0,0 +1,21 @@
+// Convenience function to load polyfills and return a promise when it's done.
+// If there are no polyfills, then this is just Promise.resolve() which means
+// it will execute in the same tick of the event loop (i.e. near-instant).
+
+import { loadIntlPolyfills } from './intl';
+
+function importExtraPolyfills() {
+  return import(/* webpackChunkName: "extra_polyfills" */ './extra_polyfills');
+}
+
+export function loadPolyfills() {
+  // Safari does not have requestIdleCallback.
+  // This avoids shipping them all the polyfills.
+  const needsExtraPolyfills = !window.requestIdleCallback;
+
+  return Promise.all([
+    loadIntlPolyfills(),
+    // eslint-disable-next-line @typescript-eslint/no-unnecessary-condition -- those properties might not exist in old browsers, even if they are always here in types
+    needsExtraPolyfills && importExtraPolyfills(),
+  ]);
+}
diff --git a/app/javascript/flavours/blobfox/polyfills/intl.ts b/app/javascript/flavours/blobfox/polyfills/intl.ts
new file mode 100644
index 00000000000000..b825da66214f51
--- /dev/null
+++ b/app/javascript/flavours/blobfox/polyfills/intl.ts
@@ -0,0 +1,106 @@
+// import { shouldPolyfill as shouldPolyfillCanonicalLocales } from '@formatjs/intl-getcanonicallocales/should-polyfill';
+// import { shouldPolyfill as shouldPolyfillLocale } from '@formatjs/intl-locale/should-polyfill';
+import { shouldPolyfill as shoudPolyfillPluralRules } from '@formatjs/intl-pluralrules/should-polyfill';
+// import { shouldPolyfill as shouldPolyfillNumberFormat } from '@formatjs/intl-numberformat/should-polyfill';
+// import { shouldPolyfill as shouldPolyfillIntlDateTimeFormat } from '@formatjs/intl-datetimeformat/should-polyfill';
+// import { shouldPolyfill as shouldPolyfillIntlRelativeTimeFormat } from '@formatjs/intl-relativetimeformat/should-polyfill';
+
+// async function loadGetCanonicalLocalesPolyfill() {
+//   // This platform already supports Intl.getCanonicalLocales
+//   if (shouldPolyfillCanonicalLocales()) {
+//     await import('@formatjs/intl-getcanonicallocales/polyfill');
+//   }
+// }
+
+// async function loadLocalePolyfill() {
+//   // This platform already supports Intl.Locale
+//   if (shouldPolyfillLocale()) {
+//     await import('@formatjs/intl-locale/polyfill');
+//   }
+// }
+
+// async function loadIntlNumberFormatPolyfill(locale: string) {
+//   const unsupportedLocale = shouldPolyfillNumberFormat(locale);
+//   // This locale is supported
+//   if (!unsupportedLocale) {
+//     return;
+//   }
+//   // Load the polyfill 1st BEFORE loading data
+//   await import('@formatjs/intl-numberformat/polyfill-force');
+//   await import(`@formatjs/intl-numberformat/locale-data/${unsupportedLocale}`);
+// }
+
+// async function loadIntlDateTimeFormatPolyfill(locale: string) {
+//   const unsupportedLocale = shouldPolyfillIntlDateTimeFormat(locale);
+//   // This locale is supported
+//   if (!unsupportedLocale) {
+//     return;
+//   }
+//   // Load the polyfill 1st BEFORE loading data
+//   await import('@formatjs/intl-datetimeformat/polyfill-force');
+
+//   // Parallelize CLDR data loading
+//   const dataPolyfills = [
+//     import('@formatjs/intl-datetimeformat/add-all-tz'),
+//     import(`@formatjs/intl-datetimeformat/locale-data/${unsupportedLocale}`),
+//   ];
+//   await Promise.all(dataPolyfills);
+// }
+
+async function loadIntlPluralRulesPolyfills(locale: string) {
+  const unsupportedLocale = shoudPolyfillPluralRules(locale);
+  // This locale is supported
+  if (!unsupportedLocale) {
+    return;
+  }
+  // Load the polyfill 1st BEFORE loading data
+  await import(
+    /* webpackChunkName: "i18n-pluralrules-polyfill" */ '@formatjs/intl-pluralrules/polyfill-force'
+  );
+  await import(
+    /* webpackChunkName: "i18n-pluralrules-polyfill-[request]" */ `@formatjs/intl-pluralrules/locale-data/${unsupportedLocale}`
+  );
+}
+
+// async function loadIntlRelativeTimeFormatPolyfill(locale: string) {
+//   const unsupportedLocale = shouldPolyfillIntlRelativeTimeFormat(locale);
+//   // This locale is supported
+//   if (!unsupportedLocale) {
+//     return;
+//   }
+//   // Load the polyfill 1st BEFORE loading data
+//   await import(
+//     /* webpackChunkName: "i18n-relativetimeformat-polyfill" */
+//     '@formatjs/intl-relativetimeformat/polyfill-force'
+//   );
+//   await import(
+//     /* webpackChunkName: "i18n-relativetimeformat-polyfill-[request]" */
+//     `@formatjs/intl-relativetimeformat/locale-data/${unsupportedLocale}`
+//   );
+// }
+
+export async function loadIntlPolyfills() {
+  // eslint-disable-next-line @typescript-eslint/prefer-nullish-coalescing -- we want to match empty strings
+  const locale = document.querySelector('html')?.lang || 'en';
+
+  // order is important here
+
+  // Supported in IE11 and most other browsers, not useful
+  // await loadGetCanonicalLocalesPolyfill()
+
+  // Supported in IE11 and most other browsers, not useful
+  // await loadLocalePolyfill()
+
+  // Supported in IE11 and most other browsers, not useful
+  // await loadIntlNumberFormatPolyfill(locale)
+
+  // Supported in IE11 and most other browsers, not useful
+  // await loadIntlDateTimeFormatPolyfill(locale)
+
+  // Supported from Safari 13+, may still be useful
+  await loadIntlPluralRulesPolyfills(locale);
+
+  // This is not used yet in the codebase yet
+  // Supported from Safari 14+
+  // await loadIntlRelativeTimeFormatPolyfill(locale);
+}
diff --git a/app/javascript/flavours/blobfox/ready.js b/app/javascript/flavours/blobfox/ready.js
new file mode 100644
index 00000000000000..e769cc756079f0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/ready.js
@@ -0,0 +1,32 @@
+// @ts-check
+
+/**
+ * @param {(() => void) | (() => Promise<void>)} callback
+ * @returns {Promise<void>}
+ */
+export default function ready(callback) {
+  return new Promise((resolve, reject) => {
+    function loaded() {
+      let result;
+      try {
+        result = callback();
+      } catch (err) {
+        reject(err);
+
+        return;
+      }
+
+      if (typeof result?.then === 'function') {
+        result.then(resolve).catch(reject);
+      } else {
+        resolve();
+      }
+    }
+
+    if (['interactive', 'complete'].includes(document.readyState)) {
+      loaded();
+    } else {
+      document.addEventListener('DOMContentLoaded', loaded);
+    }
+  });
+}
diff --git a/app/javascript/flavours/blobfox/reducers/accounts.ts b/app/javascript/flavours/blobfox/reducers/accounts.ts
new file mode 100644
index 00000000000000..0fe91fd34d1ab7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/accounts.ts
@@ -0,0 +1,84 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import type { Reducer } from 'redux';
+
+import {
+  followAccountSuccess,
+  unfollowAccountSuccess,
+  importAccounts,
+  revealAccount,
+} from 'flavours/blobfox/actions/accounts_typed';
+import type { ApiAccountJSON } from 'flavours/blobfox/api_types/accounts';
+import { me } from 'flavours/blobfox/initial_state';
+import type { Account } from 'flavours/blobfox/models/account';
+import { createAccountFromServerJSON } from 'flavours/blobfox/models/account';
+
+const initialState = ImmutableMap<string, Account>();
+
+const normalizeAccount = (
+  state: typeof initialState,
+  account: ApiAccountJSON,
+) => {
+  return state.set(
+    account.id,
+    createAccountFromServerJSON(account).set(
+      'hidden',
+      state.get(account.id)?.hidden === false
+        ? false
+        : account.limited || false,
+    ),
+  );
+};
+
+const normalizeAccounts = (
+  state: typeof initialState,
+  accounts: ApiAccountJSON[],
+) => {
+  accounts.forEach((account) => {
+    state = normalizeAccount(state, account);
+  });
+
+  return state;
+};
+
+function getCurrentUser() {
+  if (!me)
+    throw new Error(
+      'No current user (me) defined when calling `accountsReducer`',
+    );
+
+  return me;
+}
+
+export const accountsReducer: Reducer<typeof initialState> = (
+  state = initialState,
+  action,
+) => {
+  if (revealAccount.match(action))
+    return state.setIn([action.payload.id, 'hidden'], false);
+  else if (importAccounts.match(action))
+    return normalizeAccounts(state, action.payload.accounts);
+  else if (followAccountSuccess.match(action)) {
+    return state
+      .update(
+        action.payload.relationship.id,
+        (account) => account?.update('followers_count', (n) => n + 1),
+      )
+      .update(
+        getCurrentUser(),
+        (account) => account?.update('following_count', (n) => n + 1),
+      );
+  } else if (unfollowAccountSuccess.match(action))
+    return state
+      .update(
+        action.payload.relationship.id,
+        (account) =>
+          account?.update('followers_count', (n) => Math.max(0, n - 1)),
+      )
+      .update(
+        getCurrentUser(),
+        (account) =>
+          account?.update('following_count', (n) => Math.max(0, n - 1)),
+      );
+  else return state;
+};
diff --git a/app/javascript/flavours/blobfox/reducers/accounts_map.js b/app/javascript/flavours/blobfox/reducers/accounts_map.js
new file mode 100644
index 00000000000000..9053dcc9c0528b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/accounts_map.js
@@ -0,0 +1,24 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import { ACCOUNT_LOOKUP_FAIL } from '../actions/accounts';
+import { importAccounts } from '../actions/accounts_typed';
+import { domain } from '../initial_state';
+
+export const normalizeForLookup = str => {
+  str = str.toLowerCase();
+  const trailingIndex = str.indexOf(`@${domain.toLowerCase()}`);
+  return (trailingIndex > 0) ? str.slice(0, trailingIndex) : str;
+};
+
+const initialState = ImmutableMap();
+
+export default function accountsMap(state = initialState, action) {
+  switch(action.type) {
+  case ACCOUNT_LOOKUP_FAIL:
+    return action.error?.response?.status === 404 ? state.set(normalizeForLookup(action.acct), null) : state;
+  case importAccounts.type:
+    return state.withMutations(map => action.payload.accounts.forEach(account => map.set(normalizeForLookup(account.acct), account.id)));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/alerts.js b/app/javascript/flavours/blobfox/reducers/alerts.js
new file mode 100644
index 00000000000000..1ca9b62a0210b7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/alerts.js
@@ -0,0 +1,30 @@
+import { List as ImmutableList } from 'immutable';
+
+import {
+  ALERT_SHOW,
+  ALERT_DISMISS,
+  ALERT_CLEAR,
+} from '../actions/alerts';
+
+const initialState = ImmutableList([]);
+
+let id = 0;
+
+const addAlert = (state, alert) =>
+  state.push({
+    key: id++,
+    ...alert,
+  });
+
+export default function alerts(state = initialState, action) {
+  switch(action.type) {
+  case ALERT_SHOW:
+    return addAlert(state, action.alert);
+  case ALERT_DISMISS:
+    return state.filterNot(item => item.key === action.alert.key);
+  case ALERT_CLEAR:
+    return state.clear();
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/announcements.js b/app/javascript/flavours/blobfox/reducers/announcements.js
new file mode 100644
index 00000000000000..2134b04c6df1dd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/announcements.js
@@ -0,0 +1,103 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import {
+  ANNOUNCEMENTS_FETCH_REQUEST,
+  ANNOUNCEMENTS_FETCH_SUCCESS,
+  ANNOUNCEMENTS_FETCH_FAIL,
+  ANNOUNCEMENTS_UPDATE,
+  ANNOUNCEMENTS_REACTION_UPDATE,
+  ANNOUNCEMENTS_REACTION_ADD_REQUEST,
+  ANNOUNCEMENTS_REACTION_ADD_FAIL,
+  ANNOUNCEMENTS_REACTION_REMOVE_REQUEST,
+  ANNOUNCEMENTS_REACTION_REMOVE_FAIL,
+  ANNOUNCEMENTS_TOGGLE_SHOW,
+  ANNOUNCEMENTS_DELETE,
+  ANNOUNCEMENTS_DISMISS_SUCCESS,
+} from '../actions/announcements';
+
+const initialState = ImmutableMap({
+  items: ImmutableList(),
+  isLoading: false,
+  show: false,
+});
+
+const updateReaction = (state, id, name, updater) => state.update('items', list => list.map(announcement => {
+  if (announcement.get('id') === id) {
+    return announcement.update('reactions', reactions => {
+      const idx = reactions.findIndex(reaction => reaction.get('name') === name);
+
+      if (idx > -1) {
+        return reactions.update(idx, reaction => updater(reaction));
+      }
+
+      return reactions.push(updater(fromJS({ name, count: 0 })));
+    });
+  }
+
+  return announcement;
+}));
+
+const updateReactionCount = (state, reaction) => updateReaction(state, reaction.announcement_id, reaction.name, x => x.set('count', reaction.count));
+
+const addReaction = (state, id, name) => updateReaction(state, id, name, x => x.set('me', true).update('count', y => y + 1));
+
+const removeReaction = (state, id, name) => updateReaction(state, id, name, x => x.set('me', false).update('count', y => y - 1));
+
+const sortAnnouncements = list => list.sortBy(x => x.get('starts_at') || x.get('published_at'));
+
+const updateAnnouncement = (state, announcement) => {
+  const idx = state.get('items').findIndex(x => x.get('id') === announcement.get('id'));
+
+  if (idx > -1) {
+    // Deep merge is used because announcements from the streaming API do not contain
+    // personalized data about which reactions have been selected by the given user,
+    // and that is information we want to preserve
+    return state.update('items', list => sortAnnouncements(list.update(idx, x => x.mergeDeep(announcement))));
+  }
+
+  return state.update('items', list => sortAnnouncements(list.unshift(announcement)));
+};
+
+export default function announcementsReducer(state = initialState, action) {
+  switch(action.type) {
+  case ANNOUNCEMENTS_TOGGLE_SHOW:
+    return state.withMutations(map => {
+      map.set('show', !map.get('show'));
+    });
+  case ANNOUNCEMENTS_FETCH_REQUEST:
+    return state.set('isLoading', true);
+  case ANNOUNCEMENTS_FETCH_SUCCESS:
+    return state.withMutations(map => {
+      const items = fromJS(action.announcements);
+
+      map.set('items', items);
+      map.set('isLoading', false);
+    });
+  case ANNOUNCEMENTS_FETCH_FAIL:
+    return state.set('isLoading', false);
+  case ANNOUNCEMENTS_UPDATE:
+    return updateAnnouncement(state, fromJS(action.announcement));
+  case ANNOUNCEMENTS_REACTION_UPDATE:
+    return updateReactionCount(state, action.reaction);
+  case ANNOUNCEMENTS_REACTION_ADD_REQUEST:
+  case ANNOUNCEMENTS_REACTION_REMOVE_FAIL:
+    return addReaction(state, action.id, action.name);
+  case ANNOUNCEMENTS_REACTION_REMOVE_REQUEST:
+  case ANNOUNCEMENTS_REACTION_ADD_FAIL:
+    return removeReaction(state, action.id, action.name);
+  case ANNOUNCEMENTS_DISMISS_SUCCESS:
+    return updateAnnouncement(state, fromJS({ 'id': action.id, 'read': true }));
+  case ANNOUNCEMENTS_DELETE:
+    return state.update('items', list => {
+      const idx = list.findIndex(x => x.get('id') === action.id);
+
+      if (idx > -1) {
+        return list.delete(idx);
+      }
+
+      return list;
+    });
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/blocks.js b/app/javascript/flavours/blobfox/reducers/blocks.js
new file mode 100644
index 00000000000000..1b65071634fb16
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/blocks.js
@@ -0,0 +1,22 @@
+import Immutable from 'immutable';
+
+import {
+  BLOCKS_INIT_MODAL,
+} from '../actions/blocks';
+
+const initialState = Immutable.Map({
+  new: Immutable.Map({
+    account_id: null,
+  }),
+});
+
+export default function mutes(state = initialState, action) {
+  switch (action.type) {
+  case BLOCKS_INIT_MODAL:
+    return state.withMutations((state) => {
+      state.setIn(['new', 'account_id'], action.account.get('id'));
+    });
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/boosts.js b/app/javascript/flavours/blobfox/reducers/boosts.js
new file mode 100644
index 00000000000000..9573748e752a25
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/boosts.js
@@ -0,0 +1,25 @@
+import Immutable from 'immutable';
+
+import {
+  BOOSTS_INIT_MODAL,
+  BOOSTS_CHANGE_PRIVACY,
+} from 'flavours/blobfox/actions/boosts';
+
+const initialState = Immutable.Map({
+  new: Immutable.Map({
+    privacy: 'public',
+  }),
+});
+
+export default function mutes(state = initialState, action) {
+  switch (action.type) {
+  case BOOSTS_INIT_MODAL:
+    return state.withMutations((state) => {
+      state.setIn(['new', 'privacy'], action.privacy);
+    });
+  case BOOSTS_CHANGE_PRIVACY:
+    return state.setIn(['new', 'privacy'], action.privacy);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/compose.js b/app/javascript/flavours/blobfox/reducers/compose.js
new file mode 100644
index 00000000000000..4722d3fdfeb445
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/compose.js
@@ -0,0 +1,660 @@
+import { Map as ImmutableMap, List as ImmutableList, OrderedSet as ImmutableOrderedSet, fromJS } from 'immutable';
+
+import {
+  COMPOSE_MOUNT,
+  COMPOSE_UNMOUNT,
+  COMPOSE_CHANGE,
+  COMPOSE_CYCLE_ELEFRIEND,
+  COMPOSE_REPLY,
+  COMPOSE_REPLY_CANCEL,
+  COMPOSE_DIRECT,
+  COMPOSE_MENTION,
+  COMPOSE_SUBMIT_REQUEST,
+  COMPOSE_SUBMIT_SUCCESS,
+  COMPOSE_SUBMIT_FAIL,
+  COMPOSE_UPLOAD_REQUEST,
+  COMPOSE_UPLOAD_SUCCESS,
+  COMPOSE_UPLOAD_FAIL,
+  COMPOSE_UPLOAD_UNDO,
+  COMPOSE_UPLOAD_PROGRESS,
+  COMPOSE_UPLOAD_PROCESSING,
+  THUMBNAIL_UPLOAD_REQUEST,
+  THUMBNAIL_UPLOAD_SUCCESS,
+  THUMBNAIL_UPLOAD_FAIL,
+  THUMBNAIL_UPLOAD_PROGRESS,
+  COMPOSE_SUGGESTIONS_CLEAR,
+  COMPOSE_SUGGESTIONS_READY,
+  COMPOSE_SUGGESTION_SELECT,
+  COMPOSE_SUGGESTION_IGNORE,
+  COMPOSE_SUGGESTION_TAGS_UPDATE,
+  COMPOSE_TAG_HISTORY_UPDATE,
+  COMPOSE_ADVANCED_OPTIONS_CHANGE,
+  COMPOSE_SENSITIVITY_CHANGE,
+  COMPOSE_SPOILERNESS_CHANGE,
+  COMPOSE_SPOILER_TEXT_CHANGE,
+  COMPOSE_VISIBILITY_CHANGE,
+  COMPOSE_LANGUAGE_CHANGE,
+  COMPOSE_CONTENT_TYPE_CHANGE,
+  COMPOSE_EMOJI_INSERT,
+  COMPOSE_UPLOAD_CHANGE_REQUEST,
+  COMPOSE_UPLOAD_CHANGE_SUCCESS,
+  COMPOSE_UPLOAD_CHANGE_FAIL,
+  COMPOSE_DOODLE_SET,
+  COMPOSE_RESET,
+  COMPOSE_POLL_ADD,
+  COMPOSE_POLL_REMOVE,
+  COMPOSE_POLL_OPTION_ADD,
+  COMPOSE_POLL_OPTION_CHANGE,
+  COMPOSE_POLL_OPTION_REMOVE,
+  COMPOSE_POLL_SETTINGS_CHANGE,
+  INIT_MEDIA_EDIT_MODAL,
+  COMPOSE_CHANGE_MEDIA_DESCRIPTION,
+  COMPOSE_CHANGE_MEDIA_FOCUS,
+  COMPOSE_SET_STATUS,
+  COMPOSE_FOCUS,
+} from '../actions/compose';
+import { REDRAFT } from '../actions/statuses';
+import { STORE_HYDRATE } from '../actions/store';
+import { TIMELINE_DELETE } from '../actions/timelines';
+import { me, defaultContentType } from '../initial_state';
+import { recoverHashtags } from '../utils/hashtag';
+import { unescapeHTML } from '../utils/html';
+import { overwrite } from '../utils/js_helpers';
+import { privacyPreference } from '../utils/privacy_preference';
+import { uuid } from '../uuid';
+
+const totalElefriends = 3;
+
+// ~4% chance you'll end up with an unexpected friend
+// blobfox-soc/mastodon repo created_at date: 2017-04-20T21:55:28Z
+const blobfoxProbability = 1 - 0.0420215528;
+
+const initialState = ImmutableMap({
+  mounted: 0,
+  advanced_options: ImmutableMap({
+    do_not_federate: false,
+    threaded_mode: false,
+  }),
+  sensitive: false,
+  elefriend: Math.random() < blobfoxProbability ? Math.floor(Math.random() * totalElefriends) : totalElefriends,
+  spoiler: false,
+  spoiler_text: '',
+  privacy: null,
+  id: null,
+  content_type: defaultContentType || 'text/plain',
+  text: '',
+  focusDate: null,
+  caretPosition: null,
+  preselectDate: null,
+  in_reply_to: null,
+  is_submitting: false,
+  is_uploading: false,
+  is_changing_upload: false,
+  progress: 0,
+  isUploadingThumbnail: false,
+  thumbnailProgress: 0,
+  media_attachments: ImmutableList(),
+  pending_media_attachments: 0,
+  poll: null,
+  suggestion_token: null,
+  suggestions: ImmutableList(),
+  default_advanced_options: ImmutableMap({
+    do_not_federate: false,
+    threaded_mode: null,  //  Do not reset
+  }),
+  default_privacy: 'public',
+  default_sensitive: false,
+  default_language: 'en',
+  resetFileKey: Math.floor((Math.random() * 0x10000)),
+  idempotencyKey: null,
+  tagHistory: ImmutableList(),
+  media_modal: ImmutableMap({
+    id: null,
+    description: '',
+    focusX: 0,
+    focusY: 0,
+    dirty: false,
+  }),
+  doodle: ImmutableMap({
+    fg: 'rgb(  0,    0,    0)',
+    bg: 'rgb(255,  255,  255)',
+    swapped: false,
+    mode: 'draw',
+    size: 'normal',
+    weight: 2,
+    opacity: 1,
+    adaptiveStroke: true,
+    smoothing: false,
+  }),
+});
+
+const initialPoll = ImmutableMap({
+  options: ImmutableList(['', '']),
+  expires_in: 24 * 3600,
+  multiple: false,
+});
+
+function statusToTextMentions(state, status) {
+  let set = ImmutableOrderedSet([]);
+
+  if (status.getIn(['account', 'id']) !== me) {
+    set = set.add(`@${status.getIn(['account', 'acct'])} `);
+  }
+
+  return set.union(status.get('mentions').filterNot(mention => mention.get('id') === me).map(mention => `@${mention.get('acct')} `)).join('');
+}
+
+function apiStatusToTextMentions (state, status) {
+  let set = ImmutableOrderedSet([]);
+
+  if (status.account.id !== me) {
+    set = set.add(`@${status.account.acct} `);
+  }
+
+  return set.union(status.mentions.filter(
+    mention => mention.id !== me,
+  ).map(
+    mention => `@${mention.acct} `,
+  )).join('');
+}
+
+function apiStatusToTextHashtags (state, status) {
+  const text = unescapeHTML(status.content);
+  return ImmutableOrderedSet([]).union(recoverHashtags(status.tags, text).map(
+    (name) => `#${name} `,
+  )).join('');
+}
+
+function clearAll(state) {
+  return state.withMutations(map => {
+    map.set('id', null);
+    map.set('text', '');
+    if (defaultContentType) map.set('content_type', defaultContentType);
+    map.set('spoiler', false);
+    map.set('spoiler_text', '');
+    map.set('is_submitting', false);
+    map.set('is_changing_upload', false);
+    map.set('in_reply_to', null);
+    map.update(
+      'advanced_options',
+      map => map.mergeWith(overwrite, state.get('default_advanced_options')),
+    );
+    map.set('privacy', state.get('default_privacy'));
+    map.set('sensitive', state.get('default_sensitive'));
+    map.set('language', state.get('default_language'));
+    map.update('media_attachments', list => list.clear());
+    map.set('poll', null);
+    map.set('idempotencyKey', uuid());
+  });
+}
+
+function continueThread (state, status) {
+  return state.withMutations(function (map) {
+    let text = apiStatusToTextMentions(state, status);
+    text = text + apiStatusToTextHashtags(state, status);
+    map.set('text', text);
+    if (status.spoiler_text) {
+      map.set('spoiler', true);
+      map.set('spoiler_text', status.spoiler_text);
+    } else {
+      map.set('spoiler', false);
+      map.set('spoiler_text', '');
+    }
+    map.set('is_submitting', false);
+    map.set('in_reply_to', status.id);
+    map.update(
+      'advanced_options',
+      map => map.merge(new ImmutableMap({ do_not_federate: status.local_only })),
+    );
+    map.set('privacy', status.visibility);
+    map.set('sensitive', false);
+    map.update('media_attachments', list => list.clear());
+    map.set('poll', null);
+    map.set('idempotencyKey', uuid());
+    map.set('focusDate', new Date());
+    map.set('caretPosition', null);
+    map.set('preselectDate', new Date());
+  });
+}
+
+function appendMedia(state, media, file) {
+  const prevSize = state.get('media_attachments').size;
+
+  return state.withMutations(map => {
+    if (media.get('type') === 'image') {
+      media = media.set('file', file);
+    }
+    map.update('media_attachments', list => list.push(media.set('unattached', true)));
+    map.set('is_uploading', false);
+    map.set('is_processing', false);
+    map.set('resetFileKey', Math.floor((Math.random() * 0x10000)));
+    map.set('idempotencyKey', uuid());
+    map.update('pending_media_attachments', n => n - 1);
+
+    if (prevSize === 0 && (state.get('default_sensitive') || state.get('spoiler'))) {
+      map.set('sensitive', true);
+    }
+  });
+}
+
+function removeMedia(state, mediaId) {
+  const prevSize = state.get('media_attachments').size;
+
+  return state.withMutations(map => {
+    map.update('media_attachments', list => list.filterNot(item => item.get('id') === mediaId));
+    map.set('idempotencyKey', uuid());
+
+    if (prevSize === 1) {
+      map.set('sensitive', false);
+    }
+  });
+}
+
+const insertSuggestion = (state, position, token, completion, path) => {
+  return state.withMutations(map => {
+    map.updateIn(path, oldText => `${oldText.slice(0, position)}${completion}${completion[0] === ':' ? '\u200B' : ' '}${oldText.slice(position + token.length)}`);
+    map.set('suggestion_token', null);
+    map.set('suggestions', ImmutableList());
+    if (path.length === 1 && path[0] === 'text') {
+      map.set('focusDate', new Date());
+      map.set('caretPosition', position + completion.length + 1);
+    }
+    map.set('idempotencyKey', uuid());
+  });
+};
+
+const ignoreSuggestion = (state, position, token, completion, path) => {
+  return state.withMutations(map => {
+    map.updateIn(path, oldText => `${oldText.slice(0, position + token.length)} ${oldText.slice(position + token.length)}`);
+    map.set('suggestion_token', null);
+    map.set('suggestions', ImmutableList());
+    map.set('focusDate', new Date());
+    map.set('caretPosition', position + token.length + 1);
+    map.set('idempotencyKey', uuid());
+  });
+};
+
+const sortHashtagsByUse = (state, tags) => {
+  const personalHistory = state.get('tagHistory').map(tag => tag.toLowerCase());
+
+  const tagsWithLowercase = tags.map(t => ({ ...t, lowerName: t.name.toLowerCase() }));
+  const sorted = tagsWithLowercase.sort((a, b) => {
+    const usedA = personalHistory.includes(a.lowerName);
+    const usedB = personalHistory.includes(b.lowerName);
+
+    if (usedA === usedB) {
+      return 0;
+    } else if (usedA && !usedB) {
+      return -1;
+    } else {
+      return 1;
+    }
+  });
+  sorted.forEach(tag => delete tag.lowerName);
+  return sorted;
+};
+
+const insertEmoji = (state, position, emojiData) => {
+  const emoji = emojiData.native;
+
+  return state.withMutations(map => {
+    map.update('text', oldText => `${oldText.slice(0, position)}${emoji}\u200B${oldText.slice(position)}`);
+    map.set('focusDate', new Date());
+    map.set('caretPosition', position + emoji.length + 1);
+    map.set('idempotencyKey', uuid());
+  });
+};
+
+const hydrate = (state, hydratedState) => {
+  state = clearAll(state.merge(hydratedState));
+
+  if (hydratedState.get('text')) {
+    state = state.set('text', hydratedState.get('text')).set('focusDate', new Date());
+  }
+
+  return state;
+};
+
+const domParser = new DOMParser();
+
+const expandMentions = status => {
+  const fragment = domParser.parseFromString(status.get('content'), 'text/html').documentElement;
+
+  status.get('mentions').forEach(mention => {
+    fragment.querySelector(`a[href="${mention.get('url')}"]`).textContent = `@${mention.get('acct')}`;
+  });
+
+  return fragment.innerHTML;
+};
+
+const expiresInFromExpiresAt = expires_at => {
+  if (!expires_at) return 24 * 3600;
+  const delta = (new Date(expires_at).getTime() - Date.now()) / 1000;
+  return [300, 1800, 3600, 21600, 86400, 259200, 604800].find(expires_in => expires_in >= delta) || 24 * 3600;
+};
+
+const mergeLocalHashtagResults = (suggestions, prefix, tagHistory) => {
+  prefix = prefix.toLowerCase();
+  if (suggestions.length < 4) {
+    const localTags = tagHistory.filter(tag => tag.toLowerCase().startsWith(prefix) && !suggestions.some(suggestion => suggestion.type === 'hashtag' && suggestion.name.toLowerCase() === tag.toLowerCase()));
+    return suggestions.concat(localTags.slice(0, 4 - suggestions.length).toJS().map(tag => ({ type: 'hashtag', name: tag })));
+  } else {
+    return suggestions;
+  }
+};
+
+const normalizeSuggestions = (state, { accounts, emojis, tags, token }) => {
+  if (accounts) {
+    return accounts.map(item => ({ id: item.id, type: 'account' }));
+  } else if (emojis) {
+    return emojis.map(item => ({ ...item, type: 'emoji' }));
+  } else {
+    return mergeLocalHashtagResults(sortHashtagsByUse(state, tags.map(item => ({ ...item, type: 'hashtag' }))), token.slice(1), state.get('tagHistory'));
+  }
+};
+
+const updateSuggestionTags = (state, token) => {
+  const prefix = token.slice(1);
+
+  const suggestions = state.get('suggestions').toJS();
+  return state.merge({
+    suggestions: ImmutableList(mergeLocalHashtagResults(suggestions, prefix, state.get('tagHistory'))),
+    suggestion_token: token,
+  });
+};
+
+export default function compose(state = initialState, action) {
+  switch(action.type) {
+  case STORE_HYDRATE:
+    return hydrate(state, action.state.get('compose'));
+  case COMPOSE_MOUNT:
+    return state.set('mounted', state.get('mounted') + 1);
+  case COMPOSE_UNMOUNT:
+    return state.set('mounted', Math.max(state.get('mounted') - 1, 0));
+  case COMPOSE_ADVANCED_OPTIONS_CHANGE:
+    return state
+      .set('advanced_options', state.get('advanced_options').set(action.option, !!overwrite(!state.getIn(['advanced_options', action.option]), action.value)))
+      .set('idempotencyKey', uuid());
+  case COMPOSE_SENSITIVITY_CHANGE:
+    return state.withMutations(map => {
+      if (!state.get('spoiler')) {
+        map.set('sensitive', !state.get('sensitive'));
+      }
+
+      map.set('idempotencyKey', uuid());
+    });
+  case COMPOSE_SPOILERNESS_CHANGE:
+    return state.withMutations(map => {
+      map.set('spoiler', !state.get('spoiler'));
+      map.set('idempotencyKey', uuid());
+
+      if (!state.get('sensitive') && state.get('media_attachments').size >= 1) {
+        map.set('sensitive', true);
+      }
+    });
+  case COMPOSE_SPOILER_TEXT_CHANGE:
+    return state
+      .set('spoiler_text', action.text)
+      .set('idempotencyKey', uuid());
+  case COMPOSE_VISIBILITY_CHANGE:
+    return state
+      .set('privacy', action.value)
+      .set('idempotencyKey', uuid());
+  case COMPOSE_CONTENT_TYPE_CHANGE:
+    return state
+      .set('content_type', action.value)
+      .set('idempotencyKey', uuid());
+  case COMPOSE_CHANGE:
+    return state
+      .set('text', action.text)
+      .set('idempotencyKey', uuid());
+  case COMPOSE_CYCLE_ELEFRIEND:
+    return state
+      .set('elefriend', (state.get('elefriend') + 1) % totalElefriends);
+  case COMPOSE_REPLY:
+    return state.withMutations(map => {
+      map.set('id', null);
+      map.set('in_reply_to', action.status.get('id'));
+      map.set('text', statusToTextMentions(state, action.status));
+      map.set('privacy', privacyPreference(action.status.get('visibility'), state.get('default_privacy')));
+      map.update(
+        'advanced_options',
+        map => map.merge(new ImmutableMap({ do_not_federate: !!action.status.get('local_only') })),
+      );
+      map.set('focusDate', new Date());
+      map.set('caretPosition', null);
+      map.set('preselectDate', new Date());
+      map.set('idempotencyKey', uuid());
+
+      map.update('media_attachments', list => list.filter(media => media.get('unattached')));
+
+      if (action.status.get('language') && !action.status.has('translation')) {
+        map.set('language', action.status.get('language'));
+      } else {
+        map.set('language', state.get('default_language'));
+      }
+
+      if (action.status.get('spoiler_text').length > 0) {
+        let spoiler_text = action.status.get('spoiler_text');
+        if (action.prependCWRe && !spoiler_text.match(/^re[: ]/i)) {
+          spoiler_text = 're: '.concat(spoiler_text);
+        }
+        map.set('spoiler', true);
+        map.set('spoiler_text', spoiler_text);
+      } else {
+        map.set('spoiler', false);
+        map.set('spoiler_text', '');
+      }
+    });
+  case COMPOSE_REPLY_CANCEL:
+    state = state.setIn(['advanced_options', 'threaded_mode'], false);
+    // eslint-disable-next-line no-fallthrough -- fall-through to `COMPOSE_RESET` is intended
+  case COMPOSE_RESET:
+    return state.withMutations(map => {
+      map.set('in_reply_to', null);
+      if (defaultContentType) map.set('content_type', defaultContentType);
+      map.set('text', '');
+      map.set('spoiler', false);
+      map.set('spoiler_text', '');
+      map.set('privacy', state.get('default_privacy'));
+      map.set('id', null);
+      map.set('poll', null);
+      map.set('language', state.get('default_language'));
+      map.update(
+        'advanced_options',
+        map => map.mergeWith(overwrite, state.get('default_advanced_options')),
+      );
+      map.set('idempotencyKey', uuid());
+    });
+  case COMPOSE_SUBMIT_REQUEST:
+    return state.set('is_submitting', true);
+  case COMPOSE_UPLOAD_CHANGE_REQUEST:
+    return state.set('is_changing_upload', true);
+  case COMPOSE_SUBMIT_SUCCESS:
+    return action.status && state.getIn(['advanced_options', 'threaded_mode']) ? continueThread(state, action.status) : clearAll(state);
+  case COMPOSE_SUBMIT_FAIL:
+    return state.set('is_submitting', false);
+  case COMPOSE_UPLOAD_CHANGE_FAIL:
+    return state.set('is_changing_upload', false);
+  case COMPOSE_UPLOAD_REQUEST:
+    return state.set('is_uploading', true).update('pending_media_attachments', n => n + 1);
+  case COMPOSE_UPLOAD_PROCESSING:
+    return state.set('is_processing', true);
+  case COMPOSE_UPLOAD_SUCCESS:
+    return appendMedia(state, fromJS(action.media), action.file);
+  case COMPOSE_UPLOAD_FAIL:
+    return state.set('is_uploading', false).set('is_processing', false).update('pending_media_attachments', n => n - 1);
+  case COMPOSE_UPLOAD_UNDO:
+    return removeMedia(state, action.media_id);
+  case COMPOSE_UPLOAD_PROGRESS:
+    return state.set('progress', Math.round((action.loaded / action.total) * 100));
+  case THUMBNAIL_UPLOAD_REQUEST:
+    return state.set('isUploadingThumbnail', true);
+  case THUMBNAIL_UPLOAD_PROGRESS:
+    return state.set('thumbnailProgress', Math.round((action.loaded / action.total) * 100));
+  case THUMBNAIL_UPLOAD_FAIL:
+    return state.set('isUploadingThumbnail', false);
+  case THUMBNAIL_UPLOAD_SUCCESS:
+    return state
+      .set('isUploadingThumbnail', false)
+      .update('media_attachments', list => list.map(item => {
+        if (item.get('id') === action.media.id) {
+          return fromJS(action.media);
+        }
+
+        return item;
+      }));
+  case INIT_MEDIA_EDIT_MODAL:
+    const media =  state.get('media_attachments').find(item => item.get('id') === action.id);
+    return state.set('media_modal', ImmutableMap({
+      id: action.id,
+      description: media.get('description') || '',
+      focusX: media.getIn(['meta', 'focus', 'x'], 0),
+      focusY: media.getIn(['meta', 'focus', 'y'], 0),
+      dirty: false,
+    }));
+  case COMPOSE_CHANGE_MEDIA_DESCRIPTION:
+    return state.setIn(['media_modal', 'description'], action.description).setIn(['media_modal', 'dirty'], true);
+  case COMPOSE_CHANGE_MEDIA_FOCUS:
+    return state.setIn(['media_modal', 'focusX'], action.focusX).setIn(['media_modal', 'focusY'], action.focusY).setIn(['media_modal', 'dirty'], true);
+  case COMPOSE_MENTION:
+    return state.withMutations(map => {
+      map.update('text', text => [text.trim(), `@${action.account.get('acct')} `].filter((str) => str.length !== 0).join(' '));
+      map.set('focusDate', new Date());
+      map.set('caretPosition', null);
+      map.set('idempotencyKey', uuid());
+    });
+  case COMPOSE_DIRECT:
+    return state.withMutations(map => {
+      map.update('text', text => [text.trim(), `@${action.account.get('acct')} `].filter((str) => str.length !== 0).join(' '));
+      map.set('privacy', 'direct');
+      map.set('focusDate', new Date());
+      map.set('caretPosition', null);
+      map.set('idempotencyKey', uuid());
+    });
+  case COMPOSE_SUGGESTIONS_CLEAR:
+    return state.update('suggestions', ImmutableList(), list => list.clear()).set('suggestion_token', null);
+  case COMPOSE_SUGGESTIONS_READY:
+    return state.set('suggestions', ImmutableList(normalizeSuggestions(state, action))).set('suggestion_token', action.token);
+  case COMPOSE_SUGGESTION_SELECT:
+    return insertSuggestion(state, action.position, action.token, action.completion, action.path);
+  case COMPOSE_SUGGESTION_IGNORE:
+    return ignoreSuggestion(state, action.position, action.token, action.completion, action.path);
+  case COMPOSE_SUGGESTION_TAGS_UPDATE:
+    return updateSuggestionTags(state, action.token);
+  case COMPOSE_TAG_HISTORY_UPDATE:
+    return state.set('tagHistory', fromJS(action.tags));
+  case TIMELINE_DELETE:
+    if (action.id === state.get('in_reply_to')) {
+      return state.set('in_reply_to', null);
+    } else if (action.id === state.get('id')) {
+      return state.set('id', null);
+    } else {
+      return state;
+    }
+  case COMPOSE_EMOJI_INSERT:
+    return insertEmoji(state, action.position, action.emoji);
+  case COMPOSE_UPLOAD_CHANGE_SUCCESS:
+    return state
+      .set('is_changing_upload', false)
+      .setIn(['media_modal', 'dirty'], false)
+      .update('media_attachments', list => list.map(item => {
+        if (item.get('id') === action.media.id) {
+          return fromJS(action.media).set('unattached', !action.attached);
+        }
+
+        return item;
+      }));
+  case COMPOSE_DOODLE_SET:
+    return state.mergeIn(['doodle'], action.options);
+  case REDRAFT:
+    const do_not_federate = !!action.status.get('local_only');
+    let text = action.raw_text || unescapeHTML(expandMentions(action.status));
+    if (do_not_federate) text = text.replace(/ ?👁\ufe0f?\u200b?$/, '');
+    return state.withMutations(map => {
+      map.set('text', text);
+      map.set('content_type', action.content_type || 'text/plain');
+      map.set('in_reply_to', action.status.get('in_reply_to_id'));
+      map.set('privacy', action.status.get('visibility'));
+      map.set('media_attachments', action.status.get('media_attachments').map((media) => media.set('unattached', true)));
+      map.set('focusDate', new Date());
+      map.set('caretPosition', null);
+      map.set('idempotencyKey', uuid());
+      map.set('sensitive', action.status.get('sensitive'));
+      map.set('language', action.status.get('language'));
+      map.update(
+        'advanced_options',
+        map => map.merge(new ImmutableMap({ do_not_federate })),
+      );
+      map.set('id', null);
+
+      if (action.status.get('spoiler_text').length > 0) {
+        map.set('spoiler', true);
+        map.set('spoiler_text', action.status.get('spoiler_text'));
+
+        if (map.get('media_attachments').size >= 1) {
+          map.set('sensitive', true);
+        }
+      } else {
+        map.set('spoiler', false);
+        map.set('spoiler_text', '');
+      }
+
+      if (action.status.get('poll')) {
+        map.set('poll', ImmutableMap({
+          options: action.status.getIn(['poll', 'options']).map(x => x.get('title')),
+          multiple: action.status.getIn(['poll', 'multiple']),
+          expires_in: expiresInFromExpiresAt(action.status.getIn(['poll', 'expires_at'])),
+        }));
+      }
+    });
+  case COMPOSE_SET_STATUS:
+    return state.withMutations(map => {
+      map.set('id', action.status.get('id'));
+      map.set('text', action.text);
+      map.set('content_type', action.content_type || 'text/plain');
+      map.set('in_reply_to', action.status.get('in_reply_to_id'));
+      map.set('privacy', action.status.get('visibility'));
+      map.set('media_attachments', action.status.get('media_attachments'));
+      map.set('focusDate', new Date());
+      map.set('caretPosition', null);
+      map.set('idempotencyKey', uuid());
+      map.set('sensitive', action.status.get('sensitive'));
+      map.set('language', action.status.get('language'));
+
+      if (action.spoiler_text.length > 0) {
+        map.set('spoiler', true);
+        map.set('spoiler_text', action.spoiler_text);
+      } else {
+        map.set('spoiler', false);
+        map.set('spoiler_text', '');
+      }
+
+      if (action.status.get('poll')) {
+        map.set('poll', ImmutableMap({
+          options: action.status.getIn(['poll', 'options']).map(x => x.get('title')),
+          multiple: action.status.getIn(['poll', 'multiple']),
+          expires_in: expiresInFromExpiresAt(action.status.getIn(['poll', 'expires_at'])),
+        }));
+      }
+    });
+  case COMPOSE_POLL_ADD:
+    return state.set('poll', initialPoll);
+  case COMPOSE_POLL_REMOVE:
+    return state.set('poll', null);
+  case COMPOSE_POLL_OPTION_ADD:
+    return state.updateIn(['poll', 'options'], options => options.push(action.title));
+  case COMPOSE_POLL_OPTION_CHANGE:
+    return state.setIn(['poll', 'options', action.index], action.title);
+  case COMPOSE_POLL_OPTION_REMOVE:
+    return state.updateIn(['poll', 'options'], options => options.delete(action.index));
+  case COMPOSE_POLL_SETTINGS_CHANGE:
+    return state.update('poll', poll => poll.set('expires_in', action.expiresIn).set('multiple', action.isMultiple));
+  case COMPOSE_LANGUAGE_CHANGE:
+    return state.set('language', action.language);
+  case COMPOSE_FOCUS:
+    return state.set('focusDate', new Date()).update('text', text => text.length > 0 ? text : action.defaultText);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/contexts.js b/app/javascript/flavours/blobfox/reducers/contexts.js
new file mode 100644
index 00000000000000..f7d7419a4e3ab9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/contexts.js
@@ -0,0 +1,107 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import {
+  blockAccountSuccess,
+  muteAccountSuccess,
+} from '../actions/accounts';
+import { CONTEXT_FETCH_SUCCESS } from '../actions/statuses';
+import { TIMELINE_DELETE, TIMELINE_UPDATE } from '../actions/timelines';
+import { compareId } from '../compare_id';
+
+const initialState = ImmutableMap({
+  inReplyTos: ImmutableMap(),
+  replies: ImmutableMap(),
+});
+
+const normalizeContext = (immutableState, id, ancestors, descendants) => immutableState.withMutations(state => {
+  state.update('inReplyTos', immutableAncestors => immutableAncestors.withMutations(inReplyTos => {
+    state.update('replies', immutableDescendants => immutableDescendants.withMutations(replies => {
+      function addReply({ id, in_reply_to_id }) {
+        if (in_reply_to_id && !inReplyTos.has(id)) {
+
+          replies.update(in_reply_to_id, ImmutableList(), siblings => {
+            const index = siblings.findLastIndex(sibling => compareId(sibling, id) < 0);
+            return siblings.insert(index + 1, id);
+          });
+
+          inReplyTos.set(id, in_reply_to_id);
+        }
+      }
+
+      // We know in_reply_to_id of statuses but `id` itself.
+      // So we assume that the status of the id replies to last ancestors.
+
+      ancestors.forEach(addReply);
+
+      if (ancestors[0]) {
+        addReply({ id, in_reply_to_id: ancestors[ancestors.length - 1].id });
+      }
+
+      descendants.forEach(addReply);
+    }));
+  }));
+});
+
+const deleteFromContexts = (immutableState, ids) => immutableState.withMutations(state => {
+  state.update('inReplyTos', immutableAncestors => immutableAncestors.withMutations(inReplyTos => {
+    state.update('replies', immutableDescendants => immutableDescendants.withMutations(replies => {
+      ids.forEach(id => {
+        const inReplyToIdOfId = inReplyTos.get(id);
+        const repliesOfId = replies.get(id);
+        const siblings = replies.get(inReplyToIdOfId);
+
+        if (siblings) {
+          replies.set(inReplyToIdOfId, siblings.filterNot(sibling => sibling === id));
+        }
+
+
+        if (repliesOfId) {
+          repliesOfId.forEach(reply => inReplyTos.delete(reply));
+        }
+
+        inReplyTos.delete(id);
+        replies.delete(id);
+      });
+    }));
+  }));
+});
+
+const filterContexts = (state, relationship, statuses) => {
+  const ownedStatusIds = statuses
+    .filter(status => status.get('account') === relationship.id)
+    .map(status => status.get('id'));
+
+  return deleteFromContexts(state, ownedStatusIds);
+};
+
+const updateContext = (state, status) => {
+  if (status.in_reply_to_id) {
+    return state.withMutations(mutable => {
+      const replies = mutable.getIn(['replies', status.in_reply_to_id], ImmutableList());
+
+      mutable.setIn(['inReplyTos', status.id], status.in_reply_to_id);
+
+      if (!replies.includes(status.id)) {
+        mutable.setIn(['replies', status.in_reply_to_id], replies.push(status.id));
+      }
+    });
+  }
+
+  return state;
+};
+
+export default function replies(state = initialState, action) {
+  switch(action.type) {
+  case blockAccountSuccess.type:
+  case muteAccountSuccess.type:
+    return filterContexts(state, action.payload.relationship, action.payload.statuses);
+  case CONTEXT_FETCH_SUCCESS:
+    return normalizeContext(state, action.id, action.ancestors, action.descendants);
+  case TIMELINE_DELETE:
+    return deleteFromContexts(state, [action.id]);
+  case TIMELINE_UPDATE:
+    return updateContext(state, action.status);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/conversations.js b/app/javascript/flavours/blobfox/reducers/conversations.js
new file mode 100644
index 00000000000000..f772305d71a457
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/conversations.js
@@ -0,0 +1,118 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import { blockAccountSuccess, muteAccountSuccess } from 'flavours/blobfox/actions/accounts';
+import { blockDomainSuccess } from 'flavours/blobfox/actions/domain_blocks';
+
+import {
+  CONVERSATIONS_MOUNT,
+  CONVERSATIONS_UNMOUNT,
+  CONVERSATIONS_FETCH_REQUEST,
+  CONVERSATIONS_FETCH_SUCCESS,
+  CONVERSATIONS_FETCH_FAIL,
+  CONVERSATIONS_UPDATE,
+  CONVERSATIONS_READ,
+  CONVERSATIONS_DELETE_SUCCESS,
+} from '../actions/conversations';
+import { compareId } from '../compare_id';
+
+const initialState = ImmutableMap({
+  items: ImmutableList(),
+  isLoading: false,
+  hasMore: true,
+  mounted: 0,
+});
+
+const conversationToMap = item => ImmutableMap({
+  id: item.id,
+  unread: item.unread,
+  accounts: ImmutableList(item.accounts.map(a => a.id)),
+  last_status: item.last_status ? item.last_status.id : null,
+});
+
+const updateConversation = (state, item) => state.update('items', list => {
+  const index   = list.findIndex(x => x.get('id') === item.id);
+  const newItem = conversationToMap(item);
+
+  if (index === -1) {
+    return list.unshift(newItem);
+  } else {
+    return list.set(index, newItem);
+  }
+});
+
+const expandNormalizedConversations = (state, conversations, next, isLoadingRecent) => {
+  let items = ImmutableList(conversations.map(conversationToMap));
+
+  return state.withMutations(mutable => {
+    if (!items.isEmpty()) {
+      mutable.update('items', list => {
+        list = list.map(oldItem => {
+          const newItemIndex = items.findIndex(x => x.get('id') === oldItem.get('id'));
+
+          if (newItemIndex === -1) {
+            return oldItem;
+          }
+
+          const newItem = items.get(newItemIndex);
+          items = items.delete(newItemIndex);
+
+          return newItem;
+        });
+
+        list = list.concat(items);
+
+        return list.sortBy(x => x.get('last_status'), (a, b) => {
+          if(a === null || b === null) {
+            return -1;
+          }
+
+          return compareId(a, b) * -1;
+        });
+      });
+    }
+
+    if (!next && !isLoadingRecent) {
+      mutable.set('hasMore', false);
+    }
+
+    mutable.set('isLoading', false);
+  });
+};
+
+const filterConversations = (state, accountIds) => {
+  return state.update('items', list => list.filterNot(item => item.get('accounts').some(accountId => accountIds.includes(accountId))));
+};
+
+export default function conversations(state = initialState, action) {
+  switch (action.type) {
+  case CONVERSATIONS_FETCH_REQUEST:
+    return state.set('isLoading', true);
+  case CONVERSATIONS_FETCH_FAIL:
+    return state.set('isLoading', false);
+  case CONVERSATIONS_FETCH_SUCCESS:
+    return expandNormalizedConversations(state, action.conversations, action.next, action.isLoadingRecent);
+  case CONVERSATIONS_UPDATE:
+    return updateConversation(state, action.conversation);
+  case CONVERSATIONS_MOUNT:
+    return state.update('mounted', count => count + 1);
+  case CONVERSATIONS_UNMOUNT:
+    return state.update('mounted', count => count - 1);
+  case CONVERSATIONS_READ:
+    return state.update('items', list => list.map(item => {
+      if (item.get('id') === action.id) {
+        return item.set('unread', false);
+      }
+
+      return item;
+    }));
+  case blockAccountSuccess.type:
+  case muteAccountSuccess.type:
+    return filterConversations(state, [action.payload.relationship.id]);
+  case blockDomainSuccess.type:
+    return filterConversations(state, action.payload.accounts);
+  case CONVERSATIONS_DELETE_SUCCESS:
+    return state.update('items', list => list.filterNot(item => item.get('id') === action.id));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/custom_emojis.js b/app/javascript/flavours/blobfox/reducers/custom_emojis.js
new file mode 100644
index 00000000000000..56ec80f2ffce22
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/custom_emojis.js
@@ -0,0 +1,16 @@
+import { List as ImmutableList, fromJS as ConvertToImmutable } from 'immutable';
+
+import { CUSTOM_EMOJIS_FETCH_SUCCESS } from '../actions/custom_emojis';
+import { buildCustomEmojis } from '../features/emoji/emoji';
+import { search as emojiSearch } from '../features/emoji/emoji_mart_search_light';
+
+const initialState = ImmutableList([]);
+
+export default function custom_emojis(state = initialState, action) {
+  if(action.type === CUSTOM_EMOJIS_FETCH_SUCCESS) {
+    state = ConvertToImmutable(action.custom_emojis);
+    emojiSearch('', { custom: buildCustomEmojis(state) });
+  }
+
+  return state;
+}
diff --git a/app/javascript/flavours/blobfox/reducers/domain_lists.js b/app/javascript/flavours/blobfox/reducers/domain_lists.js
new file mode 100644
index 00000000000000..5f63c77f5d4200
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/domain_lists.js
@@ -0,0 +1,26 @@
+import { Map as ImmutableMap, OrderedSet as ImmutableOrderedSet } from 'immutable';
+
+import {
+  DOMAIN_BLOCKS_FETCH_SUCCESS,
+  DOMAIN_BLOCKS_EXPAND_SUCCESS,
+  unblockDomainSuccess
+} from '../actions/domain_blocks';
+
+const initialState = ImmutableMap({
+  blocks: ImmutableMap({
+    items: ImmutableOrderedSet(),
+  }),
+});
+
+export default function domainLists(state = initialState, action) {
+  switch(action.type) {
+  case DOMAIN_BLOCKS_FETCH_SUCCESS:
+    return state.setIn(['blocks', 'items'], ImmutableOrderedSet(action.domains)).setIn(['blocks', 'next'], action.next);
+  case DOMAIN_BLOCKS_EXPAND_SUCCESS:
+    return state.updateIn(['blocks', 'items'], set => set.union(action.domains)).setIn(['blocks', 'next'], action.next);
+  case unblockDomainSuccess.type:
+    return state.updateIn(['blocks', 'items'], set => set.delete(action.payload.domain));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/dropdown_menu.ts b/app/javascript/flavours/blobfox/reducers/dropdown_menu.ts
new file mode 100644
index 00000000000000..59e19bb16d28c2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/dropdown_menu.ts
@@ -0,0 +1,33 @@
+import { createReducer } from '@reduxjs/toolkit';
+
+import { closeDropdownMenu, openDropdownMenu } from '../actions/dropdown_menu';
+
+interface DropdownMenuState {
+  openId: string | null;
+  keyboard: boolean;
+  scrollKey: string | null;
+}
+
+const initialState: DropdownMenuState = {
+  openId: null,
+  keyboard: false,
+  scrollKey: null,
+};
+
+export const dropdownMenuReducer = createReducer(initialState, (builder) => {
+  builder
+    .addCase(
+      openDropdownMenu,
+      (state, { payload: { id, keyboard, scrollKey } }) => {
+        state.openId = id;
+        state.keyboard = keyboard;
+        state.scrollKey = scrollKey;
+      },
+    )
+    .addCase(closeDropdownMenu, (state, { payload: { id } }) => {
+      if (state.openId === id) {
+        state.openId = null;
+        state.scrollKey = null;
+      }
+    });
+});
diff --git a/app/javascript/flavours/blobfox/reducers/filters.js b/app/javascript/flavours/blobfox/reducers/filters.js
new file mode 100644
index 00000000000000..566ad0c6ca3486
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/filters.js
@@ -0,0 +1,45 @@
+import { Map as ImmutableMap, is, fromJS } from 'immutable';
+
+import { FILTERS_FETCH_SUCCESS, FILTERS_CREATE_SUCCESS } from '../actions/filters';
+import { FILTERS_IMPORT } from '../actions/importer';
+
+const normalizeFilter = (state, filter) => {
+  const normalizedFilter = fromJS({
+    id: filter.id,
+    title: filter.title,
+    context: filter.context,
+    filter_action: filter.filter_action,
+    keywords: filter.keywords,
+    expires_at: filter.expires_at ? Date.parse(filter.expires_at) : null,
+  });
+
+  if (is(state.get(filter.id), normalizedFilter)) {
+    return state;
+  } else {
+    // Do not overwrite keywords when receiving a partial filter
+    return state.update(filter.id, ImmutableMap(), (old) => (
+      old.mergeWith(((old_value, new_value) => (new_value === undefined ? old_value : new_value)), normalizedFilter)
+    ));
+  }
+};
+
+const normalizeFilters = (state, filters) => {
+  filters.forEach(filter => {
+    state = normalizeFilter(state, filter);
+  });
+
+  return state;
+};
+
+export default function filters(state = ImmutableMap(), action) {
+  switch(action.type) {
+  case FILTERS_CREATE_SUCCESS:
+    return normalizeFilter(state, action.filter);
+  case FILTERS_FETCH_SUCCESS:
+    return normalizeFilters(ImmutableMap(), action.filters);
+  case FILTERS_IMPORT:
+    return normalizeFilters(state, action.filters);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/followed_tags.js b/app/javascript/flavours/blobfox/reducers/followed_tags.js
new file mode 100644
index 00000000000000..fa20c9ad6e4ce4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/followed_tags.js
@@ -0,0 +1,43 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import {
+  FOLLOWED_HASHTAGS_FETCH_REQUEST,
+  FOLLOWED_HASHTAGS_FETCH_SUCCESS,
+  FOLLOWED_HASHTAGS_FETCH_FAIL,
+  FOLLOWED_HASHTAGS_EXPAND_REQUEST,
+  FOLLOWED_HASHTAGS_EXPAND_SUCCESS,
+  FOLLOWED_HASHTAGS_EXPAND_FAIL,
+} from 'flavours/blobfox/actions/tags';
+
+const initialState = ImmutableMap({
+  items: ImmutableList(),
+  isLoading: false,
+  next: null,
+});
+
+export default function followed_tags(state = initialState, action) {
+  switch(action.type) {
+  case FOLLOWED_HASHTAGS_FETCH_REQUEST:
+    return state.set('isLoading', true);
+  case FOLLOWED_HASHTAGS_FETCH_SUCCESS:
+    return state.withMutations(map => {
+      map.set('items', fromJS(action.followed_tags));
+      map.set('isLoading', false);
+      map.set('next', action.next);
+    });
+  case FOLLOWED_HASHTAGS_FETCH_FAIL:
+    return state.set('isLoading', false);
+  case FOLLOWED_HASHTAGS_EXPAND_REQUEST:
+    return state.set('isLoading', true);
+  case FOLLOWED_HASHTAGS_EXPAND_SUCCESS:
+    return state.withMutations(map => {
+      map.update('items', set => set.concat(fromJS(action.followed_tags)));
+      map.set('isLoading', false);
+      map.set('next', action.next);
+    });
+  case FOLLOWED_HASHTAGS_EXPAND_FAIL:
+    return state.set('isLoading', false);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/height_cache.js b/app/javascript/flavours/blobfox/reducers/height_cache.js
new file mode 100644
index 00000000000000..2664d4f82463f7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/height_cache.js
@@ -0,0 +1,24 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import { HEIGHT_CACHE_SET, HEIGHT_CACHE_CLEAR } from '../actions/height_cache';
+
+const initialState = ImmutableMap();
+
+const setHeight = (state, key, id, height) => {
+  return state.update(key, ImmutableMap(), map => map.set(id, height));
+};
+
+const clearHeights = () => {
+  return ImmutableMap();
+};
+
+export default function statuses(state = initialState, action) {
+  switch(action.type) {
+  case HEIGHT_CACHE_SET:
+    return setHeight(state, action.key, action.id, action.height);
+  case HEIGHT_CACHE_CLEAR:
+    return clearHeights();
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/history.js b/app/javascript/flavours/blobfox/reducers/history.js
new file mode 100644
index 00000000000000..023e5ecedd8af0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/history.js
@@ -0,0 +1,29 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import { HISTORY_FETCH_REQUEST, HISTORY_FETCH_SUCCESS, HISTORY_FETCH_FAIL } from 'flavours/blobfox/actions/history';
+
+const initialHistory = ImmutableMap({
+  loading: false,
+  items: ImmutableList(),
+});
+
+const initialState = ImmutableMap();
+
+export default function history(state = initialState, action) {
+  switch(action.type) {
+  case HISTORY_FETCH_REQUEST:
+    return state.update(action.statusId, initialHistory, history => history.withMutations(map => {
+      map.set('loading', true);
+      map.set('items', ImmutableList());
+    }));
+  case HISTORY_FETCH_SUCCESS:
+    return state.update(action.statusId, initialHistory, history => history.withMutations(map => {
+      map.set('loading', false);
+      map.set('items', fromJS(action.history.map((x, i) => ({ ...x, account: x.account.id, original: i === 0 })).reverse()));
+    }));
+  case HISTORY_FETCH_FAIL:
+    return state.update(action.statusId, initialHistory, history => history.set('loading', false));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/index.ts b/app/javascript/flavours/blobfox/reducers/index.ts
new file mode 100644
index 00000000000000..4775c076e7837d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/index.ts
@@ -0,0 +1,111 @@
+import { Record as ImmutableRecord } from 'immutable';
+
+import { loadingBarReducer } from 'react-redux-loading-bar';
+import { combineReducers } from 'redux-immutable';
+
+import { accountsReducer } from './accounts';
+import accounts_map from './accounts_map';
+import alerts from './alerts';
+import announcements from './announcements';
+import blocks from './blocks';
+import boosts from './boosts';
+import compose from './compose';
+import contexts from './contexts';
+import conversations from './conversations';
+import custom_emojis from './custom_emojis';
+import domain_lists from './domain_lists';
+import { dropdownMenuReducer } from './dropdown_menu';
+import filters from './filters';
+import followed_tags from './followed_tags';
+import height_cache from './height_cache';
+import history from './history';
+import listAdder from './list_adder';
+import listEditor from './list_editor';
+import lists from './lists';
+import local_settings from './local_settings';
+import markers from './markers';
+import media_attachments from './media_attachments';
+import meta from './meta';
+import { modalReducer } from './modal';
+import mutes from './mutes';
+import notifications from './notifications';
+import picture_in_picture from './picture_in_picture';
+import pinnedAccountsEditor from './pinned_accounts_editor';
+import polls from './polls';
+import push_notifications from './push_notifications';
+import { relationshipsReducer } from './relationships';
+import search from './search';
+import server from './server';
+import settings from './settings';
+import status_lists from './status_lists';
+import statuses from './statuses';
+import suggestions from './suggestions';
+import tags from './tags';
+import timelines from './timelines';
+import trends from './trends';
+import user_lists from './user_lists';
+
+const reducers = {
+  announcements,
+  dropdownMenu: dropdownMenuReducer,
+  timelines,
+  meta,
+  alerts,
+  loadingBar: loadingBarReducer,
+  modal: modalReducer,
+  user_lists,
+  domain_lists,
+  status_lists,
+  accounts: accountsReducer,
+  accounts_map,
+  statuses,
+  relationships: relationshipsReducer,
+  settings,
+  local_settings,
+  push_notifications,
+  mutes,
+  blocks,
+  boosts,
+  server,
+  contexts,
+  compose,
+  search,
+  media_attachments,
+  notifications,
+  height_cache,
+  custom_emojis,
+  lists,
+  listEditor,
+  listAdder,
+  filters,
+  conversations,
+  suggestions,
+  pinnedAccountsEditor,
+  polls,
+  trends,
+  markers,
+  picture_in_picture,
+  history,
+  tags,
+  followed_tags,
+};
+
+// We want the root state to be an ImmutableRecord, which is an object with a defined list of keys,
+// so it is properly typed and keys can be accessed using `state.<key>` syntax.
+// This will allow an easy conversion to a plain object once we no longer call `get` or `getIn` on the root state
+
+// By default with `combineReducers` it is a Collection, so we provide our own implementation to get a Record
+const initialRootState = Object.fromEntries(
+  Object.entries(reducers).map(([name, reducer]) => [
+    name,
+    reducer(undefined, {
+      // empty action
+    }),
+  ]),
+);
+
+const RootStateRecord = ImmutableRecord(initialRootState, 'RootState');
+
+const rootReducer = combineReducers(reducers, RootStateRecord);
+
+export { rootReducer };
diff --git a/app/javascript/flavours/blobfox/reducers/list_adder.js b/app/javascript/flavours/blobfox/reducers/list_adder.js
new file mode 100644
index 00000000000000..0f61273aa6c574
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/list_adder.js
@@ -0,0 +1,48 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import {
+  LIST_ADDER_RESET,
+  LIST_ADDER_SETUP,
+  LIST_ADDER_LISTS_FETCH_REQUEST,
+  LIST_ADDER_LISTS_FETCH_SUCCESS,
+  LIST_ADDER_LISTS_FETCH_FAIL,
+  LIST_EDITOR_ADD_SUCCESS,
+  LIST_EDITOR_REMOVE_SUCCESS,
+} from '../actions/lists';
+
+const initialState = ImmutableMap({
+  accountId: null,
+
+  lists: ImmutableMap({
+    items: ImmutableList(),
+    loaded: false,
+    isLoading: false,
+  }),
+});
+
+export default function listAdderReducer(state = initialState, action) {
+  switch(action.type) {
+  case LIST_ADDER_RESET:
+    return initialState;
+  case LIST_ADDER_SETUP:
+    return state.withMutations(map => {
+      map.set('accountId', action.account.get('id'));
+    });
+  case LIST_ADDER_LISTS_FETCH_REQUEST:
+    return state.setIn(['lists', 'isLoading'], true);
+  case LIST_ADDER_LISTS_FETCH_FAIL:
+    return state.setIn(['lists', 'isLoading'], false);
+  case LIST_ADDER_LISTS_FETCH_SUCCESS:
+    return state.update('lists', lists => lists.withMutations(map => {
+      map.set('isLoading', false);
+      map.set('loaded', true);
+      map.set('items', ImmutableList(action.lists.map(item => item.id)));
+    }));
+  case LIST_EDITOR_ADD_SUCCESS:
+    return state.updateIn(['lists', 'items'], list => list.unshift(action.listId));
+  case LIST_EDITOR_REMOVE_SUCCESS:
+    return state.updateIn(['lists', 'items'], list => list.filterNot(item => item === action.listId));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/list_editor.js b/app/javascript/flavours/blobfox/reducers/list_editor.js
new file mode 100644
index 00000000000000..d3fd62adecbced
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/list_editor.js
@@ -0,0 +1,99 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import {
+  LIST_CREATE_REQUEST,
+  LIST_CREATE_FAIL,
+  LIST_CREATE_SUCCESS,
+  LIST_UPDATE_REQUEST,
+  LIST_UPDATE_FAIL,
+  LIST_UPDATE_SUCCESS,
+  LIST_EDITOR_RESET,
+  LIST_EDITOR_SETUP,
+  LIST_EDITOR_TITLE_CHANGE,
+  LIST_ACCOUNTS_FETCH_REQUEST,
+  LIST_ACCOUNTS_FETCH_SUCCESS,
+  LIST_ACCOUNTS_FETCH_FAIL,
+  LIST_EDITOR_SUGGESTIONS_READY,
+  LIST_EDITOR_SUGGESTIONS_CLEAR,
+  LIST_EDITOR_SUGGESTIONS_CHANGE,
+  LIST_EDITOR_ADD_SUCCESS,
+  LIST_EDITOR_REMOVE_SUCCESS,
+} from '../actions/lists';
+
+const initialState = ImmutableMap({
+  listId: null,
+  isSubmitting: false,
+  isChanged: false,
+  title: '',
+  isExclusive: false,
+
+  accounts: ImmutableMap({
+    items: ImmutableList(),
+    loaded: false,
+    isLoading: false,
+  }),
+
+  suggestions: ImmutableMap({
+    value: '',
+    items: ImmutableList(),
+  }),
+});
+
+export default function listEditorReducer(state = initialState, action) {
+  switch(action.type) {
+  case LIST_EDITOR_RESET:
+    return initialState;
+  case LIST_EDITOR_SETUP:
+    return state.withMutations(map => {
+      map.set('listId', action.list.get('id'));
+      map.set('title', action.list.get('title'));
+      map.set('isExclusive', action.list.get('is_exclusive'));
+      map.set('isSubmitting', false);
+    });
+  case LIST_EDITOR_TITLE_CHANGE:
+    return state.withMutations(map => {
+      map.set('title', action.value);
+      map.set('isChanged', true);
+    });
+  case LIST_CREATE_REQUEST:
+  case LIST_UPDATE_REQUEST:
+    return state.withMutations(map => {
+      map.set('isSubmitting', true);
+      map.set('isChanged', false);
+    });
+  case LIST_CREATE_FAIL:
+  case LIST_UPDATE_FAIL:
+    return state.set('isSubmitting', false);
+  case LIST_CREATE_SUCCESS:
+  case LIST_UPDATE_SUCCESS:
+    return state.withMutations(map => {
+      map.set('isSubmitting', false);
+      map.set('listId', action.list.id);
+    });
+  case LIST_ACCOUNTS_FETCH_REQUEST:
+    return state.setIn(['accounts', 'isLoading'], true);
+  case LIST_ACCOUNTS_FETCH_FAIL:
+    return state.setIn(['accounts', 'isLoading'], false);
+  case LIST_ACCOUNTS_FETCH_SUCCESS:
+    return state.update('accounts', accounts => accounts.withMutations(map => {
+      map.set('isLoading', false);
+      map.set('loaded', true);
+      map.set('items', ImmutableList(action.accounts.map(item => item.id)));
+    }));
+  case LIST_EDITOR_SUGGESTIONS_CHANGE:
+    return state.setIn(['suggestions', 'value'], action.value);
+  case LIST_EDITOR_SUGGESTIONS_READY:
+    return state.setIn(['suggestions', 'items'], ImmutableList(action.accounts.map(item => item.id)));
+  case LIST_EDITOR_SUGGESTIONS_CLEAR:
+    return state.update('suggestions', suggestions => suggestions.withMutations(map => {
+      map.set('items', ImmutableList());
+      map.set('value', '');
+    }));
+  case LIST_EDITOR_ADD_SUCCESS:
+    return state.updateIn(['accounts', 'items'], list => list.unshift(action.accountId));
+  case LIST_EDITOR_REMOVE_SUCCESS:
+    return state.updateIn(['accounts', 'items'], list => list.filterNot(item => item === action.accountId));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/lists.js b/app/javascript/flavours/blobfox/reducers/lists.js
new file mode 100644
index 00000000000000..2a797772b30437
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/lists.js
@@ -0,0 +1,38 @@
+import { Map as ImmutableMap, fromJS } from 'immutable';
+
+import {
+  LIST_FETCH_SUCCESS,
+  LIST_FETCH_FAIL,
+  LISTS_FETCH_SUCCESS,
+  LIST_CREATE_SUCCESS,
+  LIST_UPDATE_SUCCESS,
+  LIST_DELETE_SUCCESS,
+} from '../actions/lists';
+
+const initialState = ImmutableMap();
+
+const normalizeList = (state, list) => state.set(list.id, fromJS(list));
+
+const normalizeLists = (state, lists) => {
+  lists.forEach(list => {
+    state = normalizeList(state, list);
+  });
+
+  return state;
+};
+
+export default function lists(state = initialState, action) {
+  switch(action.type) {
+  case LIST_FETCH_SUCCESS:
+  case LIST_CREATE_SUCCESS:
+  case LIST_UPDATE_SUCCESS:
+    return normalizeList(state, action.list);
+  case LISTS_FETCH_SUCCESS:
+    return normalizeLists(state, action.lists);
+  case LIST_DELETE_SUCCESS:
+  case LIST_FETCH_FAIL:
+    return state.set(action.id, false);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/local_settings.js b/app/javascript/flavours/blobfox/reducers/local_settings.js
new file mode 100644
index 00000000000000..26da423839502e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/local_settings.js
@@ -0,0 +1,79 @@
+//  Package imports.
+import { Map as ImmutableMap } from 'immutable';
+
+//  Our imports.
+import { LOCAL_SETTING_CHANGE, LOCAL_SETTING_DELETE } from 'flavours/blobfox/actions/local_settings';
+import { STORE_HYDRATE } from 'flavours/blobfox/actions/store';
+
+const initialState = ImmutableMap({
+  stretch   : true,
+  side_arm  : 'none',
+  side_arm_reply_mode : 'keep',
+  show_reply_count : false,
+  always_show_spoilers_field: false,
+  confirm_missing_media_description: false,
+  confirm_boost_missing_media_description: false,
+  confirm_before_clearing_draft: true,
+  prepend_cw_re: true,
+  preselect_on_reply: true,
+  inline_preview_cards: true,
+  hicolor_privacy_icons: false,
+  show_content_type_choice: false,
+  tag_misleading_links: true,
+  rewrite_mentions: 'no',
+  content_warnings : ImmutableMap({
+    filter       : null,
+    media_outside: false,
+    shared_state : false,
+  }),
+  collapsed : ImmutableMap({
+    enabled     : true,
+    auto        : ImmutableMap({
+      all              : false,
+      notifications    : true,
+      lengthy          : true,
+      reblogs          : false,
+      replies          : false,
+      media            : false,
+      height           : 400,
+    }),
+    backgrounds : ImmutableMap({
+      user_backgrounds : false,
+      preview_images   : false,
+    }),
+    show_action_bar : true,
+  }),
+  media     : ImmutableMap({
+    letterbox        : true,
+    fullwidth        : true,
+    reveal_behind_cw : false,
+    pop_in_player    : true,
+    pop_in_position  : 'right',
+  }),
+  notifications : ImmutableMap({
+    favicon_badge : false,
+    tab_badge     : true,
+  }),
+  status_icons : ImmutableMap({
+    language:   true,
+    reply:      true,
+    local_only: true,
+    media:      true,
+    visibility: true,
+  }),
+});
+
+const hydrate = (state, localSettings) => state.mergeDeep(localSettings);
+
+export default function localSettings(state = initialState, action) {
+  switch(action.type) {
+  case STORE_HYDRATE:
+    return hydrate(state, action.state.get('local_settings'));
+  case LOCAL_SETTING_CHANGE:
+    return state.setIn(action.key, action.value);
+  case LOCAL_SETTING_DELETE:
+    return state.deleteIn(action.key);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/markers.js b/app/javascript/flavours/blobfox/reducers/markers.js
new file mode 100644
index 00000000000000..c7c5d99f6143f4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/markers.js
@@ -0,0 +1,26 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import {
+  MARKERS_SUBMIT_SUCCESS,
+} from '../actions/markers';
+
+
+const initialState = ImmutableMap({
+  home: '0',
+  notifications: '0',
+});
+
+export default function markers(state = initialState, action) {
+  switch(action.type) {
+  case MARKERS_SUBMIT_SUCCESS:
+    if (action.home) {
+      state = state.set('home', action.home);
+    }
+    if (action.notifications) {
+      state = state.set('notifications', action.notifications);
+    }
+    return state;
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/media_attachments.js b/app/javascript/flavours/blobfox/reducers/media_attachments.js
new file mode 100644
index 00000000000000..cbb4933bc7efa8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/media_attachments.js
@@ -0,0 +1,16 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import { STORE_HYDRATE } from '../actions/store';
+
+const initialState = ImmutableMap({
+  accept_content_types: [],
+});
+
+export default function meta(state = initialState, action) {
+  switch(action.type) {
+  case STORE_HYDRATE:
+    return state.merge(action.state.get('media_attachments'));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/meta.js b/app/javascript/flavours/blobfox/reducers/meta.js
new file mode 100644
index 00000000000000..2680969fc98f92
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/meta.js
@@ -0,0 +1,25 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import { changeLayout } from 'flavours/blobfox/actions/app';
+import { STORE_HYDRATE } from 'flavours/blobfox/actions/store';
+import { layoutFromWindow } from 'flavours/blobfox/is_mobile';
+
+const initialState = ImmutableMap({
+  streaming_api_base_url: null,
+  access_token: null,
+  layout: layoutFromWindow(),
+  permissions: '0',
+});
+
+export default function meta(state = initialState, action) {
+  switch(action.type) {
+  case STORE_HYDRATE:
+    return state.merge(action.state.get('meta'))
+      .set('permissions', action.state.getIn(['role', 'permissions']))
+      .set('layout', layoutFromWindow(action.state.getIn(['local_settings', 'layout'])));
+  case changeLayout.type:
+    return state.set('layout', action.payload.layout);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/modal.ts b/app/javascript/flavours/blobfox/reducers/modal.ts
new file mode 100644
index 00000000000000..73a2afb916c509
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/modal.ts
@@ -0,0 +1,83 @@
+import { Record as ImmutableRecord, Stack } from 'immutable';
+
+import type { Reducer } from '@reduxjs/toolkit';
+
+import { COMPOSE_UPLOAD_CHANGE_SUCCESS } from '../actions/compose';
+import type { ModalType } from '../actions/modal';
+import { openModal, closeModal } from '../actions/modal';
+import { TIMELINE_DELETE } from '../actions/timelines';
+
+export type ModalProps = Record<string, unknown>;
+interface Modal {
+  modalType: ModalType;
+  modalProps: ModalProps;
+}
+
+const Modal = ImmutableRecord<Modal>({
+  modalType: 'ACTIONS',
+  modalProps: ImmutableRecord({})(),
+});
+
+interface ModalState {
+  ignoreFocus: boolean;
+  stack: Stack<ImmutableRecord<Modal>>;
+}
+
+const initialState = ImmutableRecord<ModalState>({
+  ignoreFocus: false,
+  stack: Stack(),
+})();
+type State = typeof initialState;
+
+interface PopModalOption {
+  modalType: ModalType | undefined;
+  ignoreFocus: boolean;
+}
+const popModal = (
+  state: State,
+  { modalType, ignoreFocus }: PopModalOption,
+): State => {
+  if (
+    modalType === undefined ||
+    modalType === state.get('stack').get(0)?.get('modalType')
+  ) {
+    return state
+      .set('ignoreFocus', !!ignoreFocus)
+      .update('stack', (stack) => stack.shift());
+  } else {
+    return state;
+  }
+};
+
+const pushModal = (
+  state: State,
+  modalType: ModalType,
+  modalProps: ModalProps,
+): State => {
+  return state.withMutations((record) => {
+    record.set('ignoreFocus', false);
+    record.update('stack', (stack) =>
+      stack.unshift(Modal({ modalType, modalProps })),
+    );
+  });
+};
+
+export const modalReducer: Reducer<State> = (state = initialState, action) => {
+  if (openModal.match(action))
+    return pushModal(
+      state,
+      action.payload.modalType,
+      action.payload.modalProps,
+    );
+  else if (closeModal.match(action)) return popModal(state, action.payload);
+  // TODO: type those actions
+  else if (action.type === COMPOSE_UPLOAD_CHANGE_SUCCESS)
+    return popModal(state, { modalType: 'FOCAL_POINT', ignoreFocus: false });
+  else if (action.type === TIMELINE_DELETE)
+    return state.update('stack', (stack) =>
+      stack.filterNot(
+        (modal) => modal.get('modalProps').statusId === action.id,
+      ),
+    );
+  else return state;
+};
diff --git a/app/javascript/flavours/blobfox/reducers/mutes.js b/app/javascript/flavours/blobfox/reducers/mutes.js
new file mode 100644
index 00000000000000..a9eb61ff834cbc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/mutes.js
@@ -0,0 +1,31 @@
+import Immutable from 'immutable';
+
+import {
+  MUTES_INIT_MODAL,
+  MUTES_TOGGLE_HIDE_NOTIFICATIONS,
+  MUTES_CHANGE_DURATION,
+} from '../actions/mutes';
+
+const initialState = Immutable.Map({
+  new: Immutable.Map({
+    account: null,
+    notifications: true,
+    duration: 0,
+  }),
+});
+
+export default function mutes(state = initialState, action) {
+  switch (action.type) {
+  case MUTES_INIT_MODAL:
+    return state.withMutations((state) => {
+      state.setIn(['new', 'account'], action.account);
+      state.setIn(['new', 'notifications'], true);
+    });
+  case MUTES_TOGGLE_HIDE_NOTIFICATIONS:
+    return state.updateIn(['new', 'notifications'], (old) => !old);
+  case MUTES_CHANGE_DURATION:
+    return state.setIn(['new', 'duration'], Number(action.duration));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/notifications.js b/app/javascript/flavours/blobfox/reducers/notifications.js
new file mode 100644
index 00000000000000..9c377319e350cd
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/notifications.js
@@ -0,0 +1,376 @@
+import { fromJS, Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import { blockDomainSuccess } from 'flavours/blobfox/actions/domain_blocks';
+
+import {
+  authorizeFollowRequestSuccess,
+  blockAccountSuccess,
+  muteAccountSuccess,
+  rejectFollowRequestSuccess,
+} from '../actions/accounts';
+import {
+  MARKERS_FETCH_SUCCESS,
+} from '../actions/markers';
+import {
+  NOTIFICATIONS_MOUNT,
+  NOTIFICATIONS_UNMOUNT,
+  NOTIFICATIONS_SET_VISIBILITY,
+  notificationsUpdate,
+  NOTIFICATIONS_EXPAND_SUCCESS,
+  NOTIFICATIONS_EXPAND_REQUEST,
+  NOTIFICATIONS_EXPAND_FAIL,
+  NOTIFICATIONS_FILTER_SET,
+  NOTIFICATIONS_CLEAR,
+  NOTIFICATIONS_SCROLL_TOP,
+  NOTIFICATIONS_LOAD_PENDING,
+  NOTIFICATIONS_DELETE_MARKED_REQUEST,
+  NOTIFICATIONS_DELETE_MARKED_SUCCESS,
+  NOTIFICATION_MARK_FOR_DELETE,
+  NOTIFICATIONS_DELETE_MARKED_FAIL,
+  NOTIFICATIONS_ENTER_CLEARING_MODE,
+  NOTIFICATIONS_MARK_ALL_FOR_DELETE,
+  NOTIFICATIONS_MARK_AS_READ,
+  NOTIFICATIONS_SET_BROWSER_SUPPORT,
+  NOTIFICATIONS_SET_BROWSER_PERMISSION,
+} from '../actions/notifications';
+import { TIMELINE_DELETE, TIMELINE_DISCONNECT } from '../actions/timelines';
+import { compareId } from '../compare_id';
+
+const initialState = ImmutableMap({
+  pendingItems: ImmutableList(),
+  items: ImmutableList(),
+  hasMore: true,
+  top: false,
+  mounted: 0,
+  unread: 0,
+  lastReadId: '0',
+  readMarkerId: '0',
+  isLoading: 0,
+  cleaningMode: false,
+  isTabVisible: true,
+  browserSupport: false,
+  browserPermission: 'default',
+  // notification removal mark of new notifs loaded whilst cleaningMode is true.
+  markNewForDelete: false,
+});
+
+const notificationToMap = (notification, markForDelete) => ImmutableMap({
+  id: notification.id,
+  type: notification.type,
+  account: notification.account.id,
+  markedForDelete: markForDelete,
+  status: notification.status ? notification.status.id : null,
+  report: notification.report ? fromJS(notification.report) : null,
+});
+
+const normalizeNotification = (state, notification, usePendingItems) => {
+  const markNewForDelete = state.get('markNewForDelete');
+  const top = state.get('top');
+
+  // Under currently unknown conditions, the client may receive duplicates from the server
+  if (state.get('pendingItems').some((item) => item?.get('id') === notification.id) || state.get('items').some((item) => item?.get('id') === notification.id)) {
+    return state;
+  }
+
+  if (usePendingItems || !state.get('pendingItems').isEmpty()) {
+    return state.update('pendingItems', list => list.unshift(notificationToMap(notification, markNewForDelete))).update('unread', unread => unread + 1);
+  }
+
+  if (shouldCountUnreadNotifications(state)) {
+    state = state.update('unread', unread => unread + 1);
+  } else {
+    state = state.set('lastReadId', notification.id);
+  }
+
+  return state.update('items', list => {
+    if (top && list.size > 40) {
+      list = list.take(20);
+    }
+
+    return list.unshift(notificationToMap(notification, markNewForDelete));
+  });
+};
+
+const expandNormalizedNotifications = (state, notifications, next, isLoadingMore, isLoadingRecent, usePendingItems) => {
+  // This method is pretty tricky because:
+  // - existing notifications might be out of order
+  // - the existing notifications may have gaps, most often explicitly noted with a `null` item
+  // - ideally, we don't want it to reorder existing items
+  // - `notifications` may include items that are already included
+  // - this function can be called either to fill in a gap, or load newer items
+
+  const markNewForDelete = state.get('markNewForDelete');
+  const lastReadId = state.get('lastReadId');
+  const newItems = ImmutableList(notifications.map((notification) => notificationToMap(notification, markNewForDelete)));
+
+  return state.withMutations(mutable => {
+    if (!newItems.isEmpty()) {
+      usePendingItems = isLoadingRecent && (usePendingItems || !mutable.get('pendingItems').isEmpty());
+
+      mutable.update(usePendingItems ? 'pendingItems' : 'items', oldItems => {
+        // If called to poll *new* notifications, we just need to add them on top without duplicates
+        if (isLoadingRecent) {
+          const idsToCheck = oldItems.map(item => item?.get('id')).toSet();
+          const insertedItems = newItems.filterNot(item => idsToCheck.includes(item.get('id')));
+          return insertedItems.concat(oldItems);
+        }
+
+        // If called to expand more (presumably older than any known to the WebUI), we just have to
+        // add them to the bottom without duplicates
+        if (isLoadingMore) {
+          const idsToCheck = oldItems.map(item => item?.get('id')).toSet();
+          const insertedItems = newItems.filterNot(item => idsToCheck.includes(item.get('id')));
+          return oldItems.concat(insertedItems);
+        }
+
+        // Now this gets tricky, as we don't necessarily know for sure where the gap to fill is,
+        // and some items in the timeline may not be properly ordered.
+
+        // However, we know that `newItems.last()` is the oldest item that was requested and that
+        // there is no “hole” between `newItems.last()` and `newItems.first()`.
+
+        // First, find the furthest (if properly sorted, oldest) item in the notifications that is
+        // newer than the oldest fetched one, as it's most likely that it delimits the gap.
+        // Start the gap *after* that item.
+        const lastIndex = oldItems.findLastIndex(item => item !== null && compareId(item.get('id'), newItems.last().get('id')) >= 0) + 1;
+
+        // Then, try to find the furthest (if properly sorted, oldest) item in the notifications that
+        // is newer than the most recent fetched one, as it delimits a section comprised of only
+        // items older or within `newItems` (or that were deleted from the server, so should be removed
+        // anyway).
+        // Stop the gap *after* that item.
+        const firstIndex = oldItems.take(lastIndex).findLastIndex(item => item !== null && compareId(item.get('id'), newItems.first().get('id')) > 0) + 1;
+
+        // At this point:
+        // - no `oldItems` after `firstIndex` is newer than any of the `newItems`
+        // - all `oldItems` after `lastIndex` are older than every of the `newItems`
+        // - it is possible for items in the replaced slice to be older than every `newItems`
+        // - it is possible for items before `firstIndex` to be in the `newItems` range
+        // Therefore:
+        // - to avoid losing items, items from the replaced slice that are older than `newItems`
+        //   should be added in the back.
+        // - to avoid duplicates, `newItems` should be checked the first `firstIndex` items of
+        //   `oldItems`
+        const idsToCheck = oldItems.take(firstIndex).map(item => item?.get('id')).toSet();
+        const insertedItems = newItems.filterNot(item => idsToCheck.includes(item.get('id')));
+        const olderItems = oldItems.slice(firstIndex, lastIndex).filter(item => item !== null && compareId(item.get('id'), newItems.last().get('id')) < 0);
+
+        return oldItems.take(firstIndex).concat(
+          insertedItems,
+          olderItems,
+          oldItems.skip(lastIndex),
+        );
+      });
+    }
+
+    if (!next) {
+      mutable.set('hasMore', false);
+    }
+
+    if (shouldCountUnreadNotifications(state)) {
+      mutable.set('unread', mutable.get('pendingItems').count(item => item !== null) + mutable.get('items').count(item => item && compareId(item.get('id'), lastReadId) > 0));
+    } else {
+      const mostRecent = newItems.find(item => item !== null);
+      if (mostRecent && compareId(lastReadId, mostRecent.get('id')) < 0) {
+        mutable.set('lastReadId', mostRecent.get('id'));
+      }
+    }
+
+    mutable.update('isLoading', (nbLoading) => nbLoading - 1);
+  });
+};
+
+const filterNotifications = (state, accountIds, type) => {
+  const helper = list => list.filterNot(item => item !== null && accountIds.includes(item.get('account')) && (type === undefined || type === item.get('type')));
+  return state.update('items', helper).update('pendingItems', helper);
+};
+
+const clearUnread = (state) => {
+  state = state.set('unread', state.get('pendingItems').size);
+  const lastNotification = state.get('items').find(item => item !== null);
+  return state.set('lastReadId', lastNotification ? lastNotification.get('id') : '0');
+};
+
+const updateTop = (state, top) => {
+  state = state.set('top', top);
+
+  if (!shouldCountUnreadNotifications(state)) {
+    state = clearUnread(state);
+  }
+
+  return state;
+};
+
+const deleteByStatus = (state, statusId) => {
+  const lastReadId = state.get('lastReadId');
+
+  if (shouldCountUnreadNotifications(state)) {
+    const deletedUnread = state.get('items').filter(item => item !== null && item.get('status') === statusId && compareId(item.get('id'), lastReadId) > 0);
+    state = state.update('unread', unread => unread - deletedUnread.size);
+  }
+
+  const helper = list => list.filterNot(item => item !== null && item.get('status') === statusId);
+  const deletedUnread = state.get('pendingItems').filter(item => item !== null && item.get('status') === statusId && compareId(item.get('id'), lastReadId) > 0);
+  state = state.update('unread', unread => unread - deletedUnread.size);
+  return state.update('items', helper).update('pendingItems', helper);
+};
+
+const markForDelete = (state, notificationId, yes) => {
+  return state.update('items', list => list.map(item => {
+    if (item === null) {
+      return null;
+    } else if(item.get('id') === notificationId) {
+      return item.set('markedForDelete', yes);
+    } else {
+      return item;
+    }
+  }));
+};
+
+const markAllForDelete = (state, yes) => {
+  return state.update('items', list => list.map(item => {
+    if (item === null) {
+      return null;
+    } else if(yes !== null) {
+      return item.set('markedForDelete', yes);
+    } else {
+      return item.set('markedForDelete', !item.get('markedForDelete'));
+    }
+  }));
+};
+
+const unmarkAllForDelete = (state) => {
+  return state.update('items', list => list.map(item => item === null ? item : item.set('markedForDelete', false)));
+};
+
+const deleteMarkedNotifs = (state) => {
+  return state.update('items', list => list.filterNot(item => item === null ? item : item.get('markedForDelete')));
+};
+
+const updateMounted = (state) => {
+  state = state.update('mounted', count => count + 1);
+  if (!shouldCountUnreadNotifications(state, state.get('mounted') === 1)) {
+    state = state.set('readMarkerId', state.get('lastReadId'));
+    state = clearUnread(state);
+  }
+  return state;
+};
+
+const updateVisibility = (state, visibility) => {
+  state = state.set('isTabVisible', visibility);
+  if (!shouldCountUnreadNotifications(state)) {
+    state = state.set('readMarkerId', state.get('lastReadId'));
+    state = clearUnread(state);
+  }
+  return state;
+};
+
+const shouldCountUnreadNotifications = (state, ignoreScroll = false) => {
+  const isTabVisible   = state.get('isTabVisible');
+  const isOnTop        = state.get('top');
+  const isMounted      = state.get('mounted') > 0;
+  const lastReadId     = state.get('lastReadId');
+  const lastItem       = state.get('items').findLast(item => item !== null);
+  const lastItemReached = !state.get('hasMore') || lastReadId === '0' || (lastItem && compareId(lastItem.get('id'), lastReadId) <= 0);
+
+  return !(isTabVisible && (ignoreScroll || isOnTop) && isMounted && lastItemReached);
+};
+
+const recountUnread = (state, last_read_id) => {
+  return state.withMutations(mutable => {
+    if (compareId(last_read_id, mutable.get('lastReadId')) > 0) {
+      mutable.set('lastReadId', last_read_id);
+    }
+
+    if (compareId(last_read_id, mutable.get('readMarkerId')) > 0) {
+      mutable.set('readMarkerId', last_read_id);
+    }
+
+    if (state.get('unread') > 0 || shouldCountUnreadNotifications(state)) {
+      mutable.set('unread', mutable.get('pendingItems').count(item => item !== null) + mutable.get('items').count(item => item && compareId(item.get('id'), last_read_id) > 0));
+    }
+  });
+};
+
+export default function notifications(state = initialState, action) {
+  let st;
+
+  switch(action.type) {
+  case MARKERS_FETCH_SUCCESS:
+    return action.markers.notifications ? recountUnread(state, action.markers.notifications.last_read_id) : state;
+  case NOTIFICATIONS_MOUNT:
+    return updateMounted(state);
+  case NOTIFICATIONS_UNMOUNT:
+    return state.update('mounted', count => count - 1);
+  case NOTIFICATIONS_SET_VISIBILITY:
+    return updateVisibility(state, action.visibility);
+  case NOTIFICATIONS_LOAD_PENDING:
+    return state.update('items', list => state.get('pendingItems').concat(list.take(40))).set('pendingItems', ImmutableList()).set('unread', 0);
+  case NOTIFICATIONS_EXPAND_REQUEST:
+  case NOTIFICATIONS_DELETE_MARKED_REQUEST:
+    return state.update('isLoading', (nbLoading) => nbLoading + 1);
+  case NOTIFICATIONS_DELETE_MARKED_FAIL:
+  case NOTIFICATIONS_EXPAND_FAIL:
+    return state.update('isLoading', (nbLoading) => nbLoading - 1);
+  case NOTIFICATIONS_FILTER_SET:
+    return state.set('items', ImmutableList()).set('pendingItems', ImmutableList()).set('hasMore', true);
+  case NOTIFICATIONS_SCROLL_TOP:
+    return updateTop(state, action.top);
+  case notificationsUpdate.type:
+    return normalizeNotification(state, action.payload.notification, action.payload.usePendingItems);
+  case NOTIFICATIONS_EXPAND_SUCCESS:
+    return expandNormalizedNotifications(state, action.notifications, action.next, action.isLoadingMore, action.isLoadingRecent, action.usePendingItems);
+  case blockAccountSuccess.type:
+    return filterNotifications(state, [action.payload.relationship.id]);
+  case muteAccountSuccess.type:
+    return action.payload.relationship.muting_notifications ? filterNotifications(state, [action.payload.relationship.id]) : state;
+  case blockDomainSuccess.type:
+    return filterNotifications(state, action.payload.accounts);
+  case authorizeFollowRequestSuccess.type:
+  case rejectFollowRequestSuccess.type:
+    return filterNotifications(state, [action.payload.id], 'follow_request');
+  case NOTIFICATIONS_CLEAR:
+    return state.set('items', ImmutableList()).set('pendingItems', ImmutableList()).set('hasMore', false);
+  case TIMELINE_DELETE:
+    return deleteByStatus(state, action.id);
+  case TIMELINE_DISCONNECT:
+    return action.timeline === 'home' ?
+      state.update(action.usePendingItems ? 'pendingItems' : 'items', items => items.first() ? items.unshift(null) : items) :
+      state;
+  case NOTIFICATIONS_SET_BROWSER_SUPPORT:
+    return state.set('browserSupport', action.value);
+  case NOTIFICATIONS_SET_BROWSER_PERMISSION:
+    return state.set('browserPermission', action.value);
+
+  case NOTIFICATION_MARK_FOR_DELETE:
+    return markForDelete(state, action.id, action.yes);
+
+  case NOTIFICATIONS_DELETE_MARKED_SUCCESS:
+    return deleteMarkedNotifs(state).update('isLoading', (nbLoading) => nbLoading - 1);
+
+  case NOTIFICATIONS_ENTER_CLEARING_MODE:
+    st = state.set('cleaningMode', action.yes);
+    if (!action.yes) {
+      return unmarkAllForDelete(st).set('markNewForDelete', false);
+    } else {
+      return st;
+    }
+
+  case NOTIFICATIONS_MARK_ALL_FOR_DELETE:
+    st = state;
+    if (action.yes === null) {
+      // Toggle - this is a bit confusing, as it toggles the all-none mode
+      //st = st.set('markNewForDelete', !st.get('markNewForDelete'));
+    } else {
+      st = st.set('markNewForDelete', action.yes);
+    }
+    return markAllForDelete(st, action.yes);
+
+  case NOTIFICATIONS_MARK_AS_READ:
+    const lastNotification = state.get('items').find(item => item !== null);
+    return lastNotification ? recountUnread(state, lastNotification.get('id')) : state;
+
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/picture_in_picture.js b/app/javascript/flavours/blobfox/reducers/picture_in_picture.js
new file mode 100644
index 00000000000000..edf98eb9fefc10
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/picture_in_picture.js
@@ -0,0 +1,26 @@
+import { PICTURE_IN_PICTURE_DEPLOY, PICTURE_IN_PICTURE_REMOVE } from 'flavours/blobfox/actions/picture_in_picture';
+
+import { TIMELINE_DELETE } from '../actions/timelines';
+
+const initialState = {
+  statusId: null,
+  accountId: null,
+  type: null,
+  src: null,
+  muted: false,
+  volume: 0,
+  currentTime: 0,
+};
+
+export default function pictureInPicture(state = initialState, action) {
+  switch(action.type) {
+  case PICTURE_IN_PICTURE_DEPLOY:
+    return { statusId: action.statusId, accountId: action.accountId, type: action.playerType, ...action.props };
+  case PICTURE_IN_PICTURE_REMOVE:
+    return { ...initialState };
+  case TIMELINE_DELETE:
+    return (state.statusId === action.id) ? { ...initialState } : state;
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/pinned_accounts_editor.js b/app/javascript/flavours/blobfox/reducers/pinned_accounts_editor.js
new file mode 100644
index 00000000000000..352db5733bc2d6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/pinned_accounts_editor.js
@@ -0,0 +1,58 @@
+import { Map as ImmutableMap, List as ImmutableList } from 'immutable';
+
+import {
+  PINNED_ACCOUNTS_EDITOR_RESET,
+  PINNED_ACCOUNTS_FETCH_REQUEST,
+  PINNED_ACCOUNTS_FETCH_SUCCESS,
+  PINNED_ACCOUNTS_FETCH_FAIL,
+  PINNED_ACCOUNTS_SUGGESTIONS_FETCH_SUCCESS,
+  PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CLEAR,
+  PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CHANGE,
+  pinAccountSuccess,
+  unpinAccountSuccess,
+} from '../actions/accounts';
+
+const initialState = ImmutableMap({
+  accounts: ImmutableMap({
+    items: ImmutableList(),
+    loaded: false,
+    isLoading: false,
+  }),
+
+  suggestions: ImmutableMap({
+    value: '',
+    items: ImmutableList(),
+  }),
+});
+
+export default function listEditorReducer(state = initialState, action) {
+  switch(action.type) {
+  case PINNED_ACCOUNTS_EDITOR_RESET:
+    return initialState;
+  case PINNED_ACCOUNTS_FETCH_REQUEST:
+    return state.setIn(['accounts', 'isLoading'], true);
+  case PINNED_ACCOUNTS_FETCH_FAIL:
+    return state.setIn(['accounts', 'isLoading'], false);
+  case PINNED_ACCOUNTS_FETCH_SUCCESS:
+    return state.update('accounts', accounts => accounts.withMutations(map => {
+      map.set('isLoading', false);
+      map.set('loaded', true);
+      map.set('items', ImmutableList(action.accounts.map(item => item.id)));
+    }));
+  case PINNED_ACCOUNTS_SUGGESTIONS_FETCH_SUCCESS:
+    return state.setIn(['suggestions', 'items'], ImmutableList(action.accounts.map(item => item.id)));
+  case PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CHANGE:
+    return state.setIn(['suggestions', 'value'], action.value);
+  case PINNED_ACCOUNTS_EDITOR_SUGGESTIONS_CLEAR:
+    return state.update('suggestions', suggestions => suggestions.withMutations(map => {
+      map.set('items', ImmutableList());
+      map.set('value', '');
+    }));
+  case pinAccountSuccess.type:
+    return state.updateIn(['accounts', 'items'], list => list.unshift(action.payload.relationship.id));
+  case unpinAccountSuccess.type:
+    return state.updateIn(['accounts', 'items'], list => list.filterNot(item => item === action.payload.relationship.id));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/polls.js b/app/javascript/flavours/blobfox/reducers/polls.js
new file mode 100644
index 00000000000000..67e204c53c1fac
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/polls.js
@@ -0,0 +1,45 @@
+import { Map as ImmutableMap, fromJS } from 'immutable';
+
+import { POLLS_IMPORT } from 'flavours/blobfox/actions/importer';
+
+import { normalizePollOptionTranslation } from '../actions/importer/normalizer';
+import { STATUS_TRANSLATE_SUCCESS, STATUS_TRANSLATE_UNDO } from '../actions/statuses';
+
+const importPolls = (state, polls) => state.withMutations(map => polls.forEach(poll => map.set(poll.id, fromJS(poll))));
+
+const statusTranslateSuccess = (state, pollTranslation) => {
+  return state.withMutations(map => {
+    if (pollTranslation) {
+      const poll = state.get(pollTranslation.id);
+
+      pollTranslation.options.forEach((item, index) => {
+        map.setIn([pollTranslation.id, 'options', index, 'translation'], fromJS(normalizePollOptionTranslation(item, poll)));
+      });
+    }
+  });
+};
+
+const statusTranslateUndo = (state, id) => {
+  return state.withMutations(map => {
+    const options = map.getIn([id, 'options']);
+
+    if (options) {
+      options.forEach((item, index) => map.deleteIn([id, 'options', index, 'translation']));
+    }
+  });
+};
+
+const initialState = ImmutableMap();
+
+export default function polls(state = initialState, action) {
+  switch(action.type) {
+  case POLLS_IMPORT:
+    return importPolls(state, action.polls);
+  case STATUS_TRANSLATE_SUCCESS:
+    return statusTranslateSuccess(state, action.translation.poll);
+  case STATUS_TRANSLATE_UNDO:
+    return statusTranslateUndo(state, action.pollId);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/push_notifications.js b/app/javascript/flavours/blobfox/reducers/push_notifications.js
new file mode 100644
index 00000000000000..fa8af0e8ccbdaf
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/push_notifications.js
@@ -0,0 +1,54 @@
+import Immutable from 'immutable';
+
+import { SET_BROWSER_SUPPORT, SET_SUBSCRIPTION, CLEAR_SUBSCRIPTION, SET_ALERTS } from '../actions/push_notifications';
+import { STORE_HYDRATE } from '../actions/store';
+
+const initialState = Immutable.Map({
+  subscription: null,
+  alerts: new Immutable.Map({
+    follow: false,
+    follow_request: false,
+    favourite: false,
+    reblog: false,
+    mention: false,
+    poll: false,
+  }),
+  isSubscribed: false,
+  browserSupport: false,
+});
+
+export default function push_subscriptions(state = initialState, action) {
+  switch(action.type) {
+  case STORE_HYDRATE: {
+    const push_subscription = action.state.get('push_subscription');
+
+    if (push_subscription) {
+      return state
+        .set('subscription', new Immutable.Map({
+          id: push_subscription.get('id'),
+          endpoint: push_subscription.get('endpoint'),
+        }))
+        .set('alerts', push_subscription.get('alerts') || initialState.get('alerts'))
+        .set('isSubscribed', true);
+    }
+
+    return state;
+  }
+  case SET_SUBSCRIPTION:
+    return state
+      .set('subscription', new Immutable.Map({
+        id: action.subscription.id,
+        endpoint: action.subscription.endpoint,
+      }))
+      .set('alerts', new Immutable.Map(action.subscription.alerts))
+      .set('isSubscribed', true);
+  case SET_BROWSER_SUPPORT:
+    return state.set('browserSupport', action.value);
+  case CLEAR_SUBSCRIPTION:
+    return initialState;
+  case SET_ALERTS:
+    return state.setIn(action.path, action.value);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/relationships.ts b/app/javascript/flavours/blobfox/reducers/relationships.ts
new file mode 100644
index 00000000000000..997f7fef793e75
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/relationships.ts
@@ -0,0 +1,123 @@
+import { Map as ImmutableMap } from 'immutable';
+
+import { isFulfilled } from '@reduxjs/toolkit';
+import type { Reducer } from 'redux';
+
+import type { ApiRelationshipJSON } from 'flavours/blobfox/api_types/relationships';
+import type { Account } from 'flavours/blobfox/models/account';
+import { createRelationship } from 'flavours/blobfox/models/relationship';
+import type { Relationship } from 'flavours/blobfox/models/relationship';
+
+import { submitAccountNote } from '../actions/account_notes';
+import {
+  followAccountSuccess,
+  unfollowAccountSuccess,
+  authorizeFollowRequestSuccess,
+  rejectFollowRequestSuccess,
+  followAccountRequest,
+  followAccountFail,
+  unfollowAccountRequest,
+  unfollowAccountFail,
+  blockAccountSuccess,
+  unblockAccountSuccess,
+  muteAccountSuccess,
+  unmuteAccountSuccess,
+  pinAccountSuccess,
+  unpinAccountSuccess,
+  fetchRelationshipsSuccess,
+} from '../actions/accounts_typed';
+import {
+  blockDomainSuccess,
+  unblockDomainSuccess,
+} from '../actions/domain_blocks_typed';
+import { notificationsUpdate } from '../actions/notifications_typed';
+
+const initialState = ImmutableMap<string, Relationship>();
+type State = typeof initialState;
+
+const normalizeRelationship = (
+  state: State,
+  relationship: ApiRelationshipJSON,
+) => state.set(relationship.id, createRelationship(relationship));
+
+const normalizeRelationships = (
+  state: State,
+  relationships: ApiRelationshipJSON[],
+) => {
+  relationships.forEach((relationship) => {
+    state = normalizeRelationship(state, relationship);
+  });
+
+  return state;
+};
+
+const setDomainBlocking = (
+  state: State,
+  accounts: Account[],
+  blocking: boolean,
+) => {
+  return state.withMutations((map) => {
+    accounts.forEach((id) => {
+      map.setIn([id, 'domain_blocking'], blocking);
+    });
+  });
+};
+
+export const relationshipsReducer: Reducer<State> = (
+  state = initialState,
+  action,
+) => {
+  if (authorizeFollowRequestSuccess.match(action))
+    return state
+      .setIn([action.payload.id, 'followed_by'], true)
+      .setIn([action.payload.id, 'requested_by'], false);
+  else if (rejectFollowRequestSuccess.match(action))
+    return state
+      .setIn([action.payload.id, 'followed_by'], false)
+      .setIn([action.payload.id, 'requested_by'], false);
+  else if (notificationsUpdate.match(action))
+    return action.payload.notification.type === 'follow_request'
+      ? state.setIn(
+          [action.payload.notification.account.id, 'requested_by'],
+          true,
+        )
+      : state;
+  else if (followAccountRequest.match(action))
+    return state.getIn([action.payload.id, 'following'])
+      ? state
+      : state.setIn(
+          [
+            action.payload.id,
+            action.payload.locked ? 'requested' : 'following',
+          ],
+          true,
+        );
+  else if (followAccountFail.match(action))
+    return state.setIn(
+      [action.payload.id, action.payload.locked ? 'requested' : 'following'],
+      false,
+    );
+  else if (unfollowAccountRequest.match(action))
+    return state.setIn([action.payload.id, 'following'], false);
+  else if (unfollowAccountFail.match(action))
+    return state.setIn([action.payload.id, 'following'], true);
+  else if (
+    followAccountSuccess.match(action) ||
+    unfollowAccountSuccess.match(action) ||
+    blockAccountSuccess.match(action) ||
+    unblockAccountSuccess.match(action) ||
+    muteAccountSuccess.match(action) ||
+    unmuteAccountSuccess.match(action) ||
+    pinAccountSuccess.match(action) ||
+    unpinAccountSuccess.match(action) ||
+    isFulfilled(submitAccountNote)(action)
+  )
+    return normalizeRelationship(state, action.payload.relationship);
+  else if (fetchRelationshipsSuccess.match(action))
+    return normalizeRelationships(state, action.payload.relationships);
+  else if (blockDomainSuccess.match(action))
+    return setDomainBlocking(state, action.payload.accounts, true);
+  else if (unblockDomainSuccess.match(action))
+    return setDomainBlocking(state, action.payload.accounts, false);
+  else return state;
+};
diff --git a/app/javascript/flavours/blobfox/reducers/search.js b/app/javascript/flavours/blobfox/reducers/search.js
new file mode 100644
index 00000000000000..72835eb91745f3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/search.js
@@ -0,0 +1,82 @@
+import { Map as ImmutableMap, OrderedSet as ImmutableOrderedSet, fromJS } from 'immutable';
+
+import {
+  COMPOSE_MENTION,
+  COMPOSE_REPLY,
+  COMPOSE_DIRECT,
+} from '../actions/compose';
+import {
+  SEARCH_CHANGE,
+  SEARCH_CLEAR,
+  SEARCH_FETCH_REQUEST,
+  SEARCH_FETCH_FAIL,
+  SEARCH_FETCH_SUCCESS,
+  SEARCH_SHOW,
+  SEARCH_EXPAND_REQUEST,
+  SEARCH_EXPAND_SUCCESS,
+  SEARCH_EXPAND_FAIL,
+  SEARCH_HISTORY_UPDATE,
+} from '../actions/search';
+
+const initialState = ImmutableMap({
+  value: '',
+  submitted: false,
+  hidden: false,
+  results: ImmutableMap(),
+  isLoading: false,
+  searchTerm: '',
+  type: null,
+  recent: ImmutableOrderedSet(),
+});
+
+export default function search(state = initialState, action) {
+  switch(action.type) {
+  case SEARCH_CHANGE:
+    return state.set('value', action.value);
+  case SEARCH_CLEAR:
+    return state.withMutations(map => {
+      map.set('value', '');
+      map.set('results', ImmutableMap());
+      map.set('submitted', false);
+      map.set('hidden', false);
+      map.set('searchTerm', '');
+      map.set('type', null);
+    });
+  case SEARCH_SHOW:
+    return state.set('hidden', false);
+  case COMPOSE_REPLY:
+  case COMPOSE_MENTION:
+  case COMPOSE_DIRECT:
+    return state.set('hidden', true);
+  case SEARCH_FETCH_REQUEST:
+    return state.withMutations(map => {
+      map.set('isLoading', true);
+      map.set('submitted', true);
+      map.set('type', action.searchType);
+    });
+  case SEARCH_FETCH_FAIL:
+  case SEARCH_EXPAND_FAIL:
+    return state.set('isLoading', false);
+  case SEARCH_FETCH_SUCCESS:
+    return state.withMutations(map => {
+      map.set('results', ImmutableMap({
+        accounts: ImmutableOrderedSet(action.results.accounts.map(item => item.id)),
+        statuses: ImmutableOrderedSet(action.results.statuses.map(item => item.id)),
+        hashtags: ImmutableOrderedSet(fromJS(action.results.hashtags)),
+      }));
+
+      map.set('searchTerm', action.searchTerm);
+      map.set('type', action.searchType);
+      map.set('isLoading', false);
+    });
+  case SEARCH_EXPAND_REQUEST:
+    return state.set('type', action.searchType).set('isLoading', true);
+  case SEARCH_EXPAND_SUCCESS:
+    const results = action.searchType === 'hashtags' ? ImmutableOrderedSet(fromJS(action.results.hashtags)) : action.results[action.searchType].map(item => item.id);
+    return state.updateIn(['results', action.searchType], list => list.union(results)).set('isLoading', false);
+  case SEARCH_HISTORY_UPDATE:
+    return state.set('recent', ImmutableOrderedSet(fromJS(action.recent)));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/server.js b/app/javascript/flavours/blobfox/reducers/server.js
new file mode 100644
index 00000000000000..4442e1b1141094
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/server.js
@@ -0,0 +1,63 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import {
+  SERVER_FETCH_REQUEST,
+  SERVER_FETCH_SUCCESS,
+  SERVER_FETCH_FAIL,
+  SERVER_TRANSLATION_LANGUAGES_FETCH_REQUEST,
+  SERVER_TRANSLATION_LANGUAGES_FETCH_SUCCESS,
+  SERVER_TRANSLATION_LANGUAGES_FETCH_FAIL,
+  EXTENDED_DESCRIPTION_REQUEST,
+  EXTENDED_DESCRIPTION_SUCCESS,
+  EXTENDED_DESCRIPTION_FAIL,
+  SERVER_DOMAIN_BLOCKS_FETCH_REQUEST,
+  SERVER_DOMAIN_BLOCKS_FETCH_SUCCESS,
+  SERVER_DOMAIN_BLOCKS_FETCH_FAIL,
+} from 'flavours/blobfox/actions/server';
+
+const initialState = ImmutableMap({
+  server: ImmutableMap({
+    isLoading: false,
+  }),
+
+  extendedDescription: ImmutableMap({
+    isLoading: false,
+  }),
+
+  domainBlocks: ImmutableMap({
+    isLoading: false,
+    isAvailable: true,
+    items: ImmutableList(),
+  }),
+});
+
+export default function server(state = initialState, action) {
+  switch (action.type) {
+  case SERVER_FETCH_REQUEST:
+    return state.setIn(['server', 'isLoading'], true);
+  case SERVER_FETCH_SUCCESS:
+    return state.set('server', fromJS(action.server)).setIn(['server', 'isLoading'], false);
+  case SERVER_FETCH_FAIL:
+    return state.setIn(['server', 'isLoading'], false);
+  case SERVER_TRANSLATION_LANGUAGES_FETCH_REQUEST:
+    return state.setIn(['translationLanguages', 'isLoading'], true);
+  case SERVER_TRANSLATION_LANGUAGES_FETCH_SUCCESS:
+    return state.setIn(['translationLanguages', 'items'], fromJS(action.translationLanguages)).setIn(['translationLanguages', 'isLoading'], false);
+  case SERVER_TRANSLATION_LANGUAGES_FETCH_FAIL:
+    return state.setIn(['translationLanguages', 'isLoading'], false);
+  case EXTENDED_DESCRIPTION_REQUEST:
+    return state.setIn(['extendedDescription', 'isLoading'], true);
+  case EXTENDED_DESCRIPTION_SUCCESS:
+    return state.set('extendedDescription', fromJS(action.description)).setIn(['extendedDescription', 'isLoading'], false);
+  case EXTENDED_DESCRIPTION_FAIL:
+    return state.setIn(['extendedDescription', 'isLoading'], false);
+  case SERVER_DOMAIN_BLOCKS_FETCH_REQUEST:
+    return state.setIn(['domainBlocks', 'isLoading'], true);
+  case SERVER_DOMAIN_BLOCKS_FETCH_SUCCESS:
+    return state.setIn(['domainBlocks', 'items'], fromJS(action.blocks)).setIn(['domainBlocks', 'isLoading'], false).setIn(['domainBlocks', 'isAvailable'], action.isAvailable);
+  case SERVER_DOMAIN_BLOCKS_FETCH_FAIL:
+    return state.setIn(['domainBlocks', 'isLoading'], false);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/settings.js b/app/javascript/flavours/blobfox/reducers/settings.js
new file mode 100644
index 00000000000000..e695d3bf9d714b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/settings.js
@@ -0,0 +1,203 @@
+import { Map as ImmutableMap, fromJS } from 'immutable';
+
+import { COLUMN_ADD, COLUMN_REMOVE, COLUMN_MOVE, COLUMN_PARAMS_CHANGE } from '../actions/columns';
+import { EMOJI_USE } from '../actions/emojis';
+import { LANGUAGE_USE } from '../actions/languages';
+import { LIST_DELETE_SUCCESS, LIST_FETCH_FAIL } from '../actions/lists';
+import { NOTIFICATIONS_FILTER_SET } from '../actions/notifications';
+import { SETTING_CHANGE, SETTING_SAVE } from '../actions/settings';
+import { STORE_HYDRATE } from '../actions/store';
+import { uuid } from '../uuid';
+
+const initialState = ImmutableMap({
+  saved: true,
+
+  onboarded: false,
+  layout: 'auto',
+
+  skinTone: 1,
+
+  trends: ImmutableMap({
+    show: true,
+  }),
+
+  home: ImmutableMap({
+    shows: ImmutableMap({
+      reblog: true,
+      reply: true,
+      direct: true,
+    }),
+
+    regex: ImmutableMap({
+      body: '',
+    }),
+  }),
+
+  notifications: ImmutableMap({
+    alerts: ImmutableMap({
+      follow: false,
+      follow_request: false,
+      favourite: false,
+      reaction: false,
+      reblog: false,
+      mention: false,
+      poll: false,
+      status: false,
+      update: false,
+      'admin.sign_up': false,
+      'admin.report': false,
+    }),
+
+    quickFilter: ImmutableMap({
+      active: 'all',
+      show: true,
+      advanced: false,
+    }),
+
+    dismissPermissionBanner: false,
+    showUnread: true,
+
+    shows: ImmutableMap({
+      follow: true,
+      follow_request: false,
+      favourite: true,
+      reaction: true,
+      reblog: true,
+      reaction: true,
+      mention: true,
+      poll: true,
+      status: true,
+      update: true,
+      'admin.sign_up': true,
+      'admin.report': true,
+    }),
+
+    sounds: ImmutableMap({
+      follow: true,
+      follow_request: false,
+      favourite: true,
+      reaction: true,
+      reblog: true,
+      reaction: true,
+      mention: true,
+      poll: true,
+      status: true,
+      update: true,
+      'admin.sign_up': true,
+      'admin.report': true,
+    }),
+  }),
+
+  firehose: ImmutableMap({
+    onlyMedia: false,
+    allowLocalOnly: true,
+
+    regex: ImmutableMap({
+      body: '',
+    }),
+  }),
+
+  community: ImmutableMap({
+    regex: ImmutableMap({
+      body: '',
+    }),
+  }),
+
+  public: ImmutableMap({
+    regex: ImmutableMap({
+      body: '',
+    }),
+  }),
+
+  direct: ImmutableMap({
+    conversations: true,
+    regex: ImmutableMap({
+      body: '',
+    }),
+  }),
+
+  dismissed_banners: ImmutableMap({
+    'public_timeline': false,
+    'community_timeline': false,
+    'home.explore_prompt': false,
+    'explore/links': false,
+    'explore/statuses': false,
+    'explore/tags': false,
+  }),
+});
+
+const defaultColumns = fromJS([
+  { id: 'COMPOSE', uuid: uuid(), params: {} },
+  { id: 'HOME', uuid: uuid(), params: {} },
+  { id: 'NOTIFICATIONS', uuid: uuid(), params: {} },
+]);
+
+const hydrate = (state, settings) => state.mergeDeep(settings).update('columns', (val = defaultColumns) => val);
+
+const moveColumn = (state, uuid, direction) => {
+  const columns  = state.get('columns');
+  const index    = columns.findIndex(item => item.get('uuid') === uuid);
+  const newIndex = index + direction;
+
+  let newColumns;
+
+  newColumns = columns.splice(index, 1);
+  newColumns = newColumns.splice(newIndex, 0, columns.get(index));
+
+  return state
+    .set('columns', newColumns)
+    .set('saved', false);
+};
+
+const changeColumnParams = (state, uuid, path, value) => {
+  const columns = state.get('columns');
+  const index   = columns.findIndex(item => item.get('uuid') === uuid);
+
+  const newColumns = columns.update(index, column => column.updateIn(['params', ...path], () => value));
+
+  return state
+    .set('columns', newColumns)
+    .set('saved', false);
+};
+
+const updateFrequentEmojis = (state, emoji) => state.update('frequentlyUsedEmojis', ImmutableMap(), map => map.update(emoji.id, 0, count => count + 1)).set('saved', false);
+
+const updateFrequentLanguages = (state, language) => state.update('frequentlyUsedLanguages', ImmutableMap(), map => map.update(language, 0, count => count + 1)).set('saved', false);
+
+const filterDeadListColumns = (state, listId) => state.update('columns', columns => columns.filterNot(column => column.get('id') === 'LIST' && column.get('params').get('id') === listId));
+
+export default function settings(state = initialState, action) {
+  switch(action.type) {
+  case STORE_HYDRATE:
+    return hydrate(state, action.state.get('settings'));
+  case NOTIFICATIONS_FILTER_SET:
+  case SETTING_CHANGE:
+    return state
+      .setIn(action.path, action.value)
+      .set('saved', false);
+  case COLUMN_ADD:
+    return state
+      .update('columns', list => list.push(fromJS({ id: action.id, uuid: uuid(), params: action.params })))
+      .set('saved', false);
+  case COLUMN_REMOVE:
+    return state
+      .update('columns', list => list.filterNot(item => item.get('uuid') === action.uuid))
+      .set('saved', false);
+  case COLUMN_MOVE:
+    return moveColumn(state, action.uuid, action.direction);
+  case COLUMN_PARAMS_CHANGE:
+    return changeColumnParams(state, action.uuid, action.path, action.value);
+  case EMOJI_USE:
+    return updateFrequentEmojis(state, action.emoji);
+  case LANGUAGE_USE:
+    return updateFrequentLanguages(state, action.language);
+  case SETTING_SAVE:
+    return state.set('saved', true);
+  case LIST_FETCH_FAIL:
+    return action.error.response.status === 404 ? filterDeadListColumns(state, action.id) : state;
+  case LIST_DELETE_SUCCESS:
+    return filterDeadListColumns(state, action.id);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/status_lists.js b/app/javascript/flavours/blobfox/reducers/status_lists.js
new file mode 100644
index 00000000000000..6cb6a937bb915e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/status_lists.js
@@ -0,0 +1,151 @@
+import { Map as ImmutableMap, OrderedSet as ImmutableOrderedSet } from 'immutable';
+
+import {
+  blockAccountSuccess,
+  muteAccountSuccess,
+} from '../actions/accounts';
+import {
+  BOOKMARKED_STATUSES_FETCH_REQUEST,
+  BOOKMARKED_STATUSES_FETCH_SUCCESS,
+  BOOKMARKED_STATUSES_FETCH_FAIL,
+  BOOKMARKED_STATUSES_EXPAND_REQUEST,
+  BOOKMARKED_STATUSES_EXPAND_SUCCESS,
+  BOOKMARKED_STATUSES_EXPAND_FAIL,
+} from '../actions/bookmarks';
+import {
+  FAVOURITED_STATUSES_FETCH_REQUEST,
+  FAVOURITED_STATUSES_FETCH_SUCCESS,
+  FAVOURITED_STATUSES_FETCH_FAIL,
+  FAVOURITED_STATUSES_EXPAND_REQUEST,
+  FAVOURITED_STATUSES_EXPAND_SUCCESS,
+  FAVOURITED_STATUSES_EXPAND_FAIL,
+} from '../actions/favourites';
+import {
+  FAVOURITE_SUCCESS,
+  UNFAVOURITE_SUCCESS,
+  BOOKMARK_SUCCESS,
+  UNBOOKMARK_SUCCESS,
+  PIN_SUCCESS,
+  UNPIN_SUCCESS,
+} from '../actions/interactions';
+import {
+  PINNED_STATUSES_FETCH_SUCCESS,
+} from '../actions/pin_statuses';
+import {
+  TRENDS_STATUSES_FETCH_REQUEST,
+  TRENDS_STATUSES_FETCH_SUCCESS,
+  TRENDS_STATUSES_FETCH_FAIL,
+  TRENDS_STATUSES_EXPAND_REQUEST,
+  TRENDS_STATUSES_EXPAND_SUCCESS,
+  TRENDS_STATUSES_EXPAND_FAIL,
+} from '../actions/trends';
+
+
+
+const initialState = ImmutableMap({
+  favourites: ImmutableMap({
+    next: null,
+    loaded: false,
+    items: ImmutableOrderedSet(),
+  }),
+  bookmarks: ImmutableMap({
+    next: null,
+    loaded: false,
+    items: ImmutableOrderedSet(),
+  }),
+  pins: ImmutableMap({
+    next: null,
+    loaded: false,
+    items: ImmutableOrderedSet(),
+  }),
+  trending: ImmutableMap({
+    next: null,
+    loaded: false,
+    items: ImmutableOrderedSet(),
+  }),
+});
+
+const normalizeList = (state, listType, statuses, next) => {
+  return state.update(listType, listMap => listMap.withMutations(map => {
+    map.set('next', next);
+    map.set('loaded', true);
+    map.set('isLoading', false);
+    map.set('items', ImmutableOrderedSet(statuses.map(item => item.id)));
+  }));
+};
+
+const appendToList = (state, listType, statuses, next) => {
+  return state.update(listType, listMap => listMap.withMutations(map => {
+    map.set('next', next);
+    map.set('isLoading', false);
+    map.set('items', map.get('items').union(statuses.map(item => item.id)));
+  }));
+};
+
+const prependOneToList = (state, listType, status) => {
+  return state.updateIn([listType, 'items'], (list) => {
+    if (list.includes(status.get('id'))) {
+      return list;
+    } else {
+      return ImmutableOrderedSet([status.get('id')]).union(list);
+    }
+  });
+};
+
+const removeOneFromList = (state, listType, status) => {
+  return state.updateIn([listType, 'items'], (list) => list.delete(status.get('id')));
+};
+
+export default function statusLists(state = initialState, action) {
+  switch(action.type) {
+  case FAVOURITED_STATUSES_FETCH_REQUEST:
+  case FAVOURITED_STATUSES_EXPAND_REQUEST:
+    return state.setIn(['favourites', 'isLoading'], true);
+  case FAVOURITED_STATUSES_FETCH_FAIL:
+  case FAVOURITED_STATUSES_EXPAND_FAIL:
+    return state.setIn(['favourites', 'isLoading'], false);
+  case FAVOURITED_STATUSES_FETCH_SUCCESS:
+    return normalizeList(state, 'favourites', action.statuses, action.next);
+  case FAVOURITED_STATUSES_EXPAND_SUCCESS:
+    return appendToList(state, 'favourites', action.statuses, action.next);
+  case BOOKMARKED_STATUSES_FETCH_REQUEST:
+  case BOOKMARKED_STATUSES_EXPAND_REQUEST:
+    return state.setIn(['bookmarks', 'isLoading'], true);
+  case BOOKMARKED_STATUSES_FETCH_FAIL:
+  case BOOKMARKED_STATUSES_EXPAND_FAIL:
+    return state.setIn(['bookmarks', 'isLoading'], false);
+  case BOOKMARKED_STATUSES_FETCH_SUCCESS:
+    return normalizeList(state, 'bookmarks', action.statuses, action.next);
+  case BOOKMARKED_STATUSES_EXPAND_SUCCESS:
+    return appendToList(state, 'bookmarks', action.statuses, action.next);
+  case TRENDS_STATUSES_FETCH_REQUEST:
+  case TRENDS_STATUSES_EXPAND_REQUEST:
+    return state.setIn(['trending', 'isLoading'], true);
+  case TRENDS_STATUSES_FETCH_FAIL:
+  case TRENDS_STATUSES_EXPAND_FAIL:
+    return state.setIn(['trending', 'isLoading'], false);
+  case TRENDS_STATUSES_FETCH_SUCCESS:
+    return normalizeList(state, 'trending', action.statuses, action.next);
+  case TRENDS_STATUSES_EXPAND_SUCCESS:
+    return appendToList(state, 'trending', action.statuses, action.next);
+  case FAVOURITE_SUCCESS:
+    return prependOneToList(state, 'favourites', action.status);
+  case UNFAVOURITE_SUCCESS:
+    return removeOneFromList(state, 'favourites', action.status);
+  case BOOKMARK_SUCCESS:
+    return prependOneToList(state, 'bookmarks', action.status);
+  case UNBOOKMARK_SUCCESS:
+    return removeOneFromList(state, 'bookmarks', action.status);
+  case PINNED_STATUSES_FETCH_SUCCESS:
+    return normalizeList(state, 'pins', action.statuses, action.next);
+  case PIN_SUCCESS:
+    return prependOneToList(state, 'pins', action.status);
+  case UNPIN_SUCCESS:
+    return removeOneFromList(state, 'pins', action.status);
+  case blockAccountSuccess.type:
+  case muteAccountSuccess.type:
+    return state.updateIn(['trending', 'items'], ImmutableOrderedSet(), list => list.filterNot(statusId => action.payload.statuses.getIn([statusId, 'account']) === action.payload.relationship.id));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/statuses.js b/app/javascript/flavours/blobfox/reducers/statuses.js
new file mode 100644
index 00000000000000..340291594fa537
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/statuses.js
@@ -0,0 +1,191 @@
+import { Map as ImmutableMap, fromJS } from 'immutable';
+
+import { STATUS_IMPORT, STATUSES_IMPORT } from '../actions/importer';
+import { normalizeStatusTranslation } from '../actions/importer/normalizer';
+import {
+  REBLOG_REQUEST,
+  REBLOG_FAIL,
+  UNREBLOG_REQUEST,
+  UNREBLOG_FAIL,
+  FAVOURITE_REQUEST,
+  FAVOURITE_FAIL,
+  UNFAVOURITE_REQUEST,
+  UNFAVOURITE_FAIL,
+  BOOKMARK_REQUEST,
+  BOOKMARK_FAIL,
+  UNBOOKMARK_REQUEST,
+  UNBOOKMARK_FAIL,
+  REACTION_UPDATE,
+  REACTION_ADD_FAIL,
+  REACTION_REMOVE_FAIL,
+  REACTION_ADD_REQUEST,
+  REACTION_REMOVE_REQUEST,
+} from '../actions/interactions';
+import {
+  STATUS_MUTE_SUCCESS,
+  STATUS_UNMUTE_SUCCESS,
+  STATUS_REVEAL,
+  STATUS_HIDE,
+  STATUS_COLLAPSE,
+  STATUS_TRANSLATE_SUCCESS,
+  STATUS_TRANSLATE_UNDO,
+  STATUS_FETCH_REQUEST,
+  STATUS_FETCH_FAIL,
+} from '../actions/statuses';
+import { TIMELINE_DELETE } from '../actions/timelines';
+
+const importStatus = (state, status) => state.set(status.id, fromJS(status));
+
+const importStatuses = (state, statuses) =>
+  state.withMutations(mutable => statuses.forEach(status => importStatus(mutable, status)));
+
+const deleteStatus = (state, id, references) => {
+  references.forEach(ref => {
+    state = deleteStatus(state, ref, []);
+  });
+
+  return state.delete(id);
+};
+
+const statusTranslateSuccess = (state, id, translation) => {
+  return state.withMutations(map => {
+    map.setIn([id, 'translation'], fromJS(normalizeStatusTranslation(translation, map.get(id))));
+
+    const list = map.getIn([id, 'media_attachments']);
+    if (translation.media_attachments && list) {
+      translation.media_attachments.forEach(item => {
+        const index = list.findIndex(i => i.get('id') === item.id);
+        map.setIn([id, 'media_attachments', index, 'translation'], fromJS({ description: item.description }));
+      });
+    }
+  });
+};
+
+const statusTranslateUndo = (state, id) => {
+  return state.withMutations(map => {
+    map.deleteIn([id, 'translation']);
+    map.getIn([id, 'media_attachments']).forEach((item, index) => map.deleteIn([id, 'media_attachments', index, 'translation']));
+  });
+};
+
+const updateReaction = (state, id, name, updater) => state.update(
+  id,
+  status => status.update(
+    'reactions',
+    reactions => {
+      const index = reactions.findIndex(reaction => reaction.get('name') === name);
+      if (index > -1) {
+        return reactions.update(index, reaction => updater(reaction));
+      } else {
+        return reactions.push(updater(fromJS({ name, count: 0 })));
+      }
+    },
+  ),
+);
+
+const updateReactionCount = (state, reaction) => updateReaction(state, reaction.status_id, reaction.name, x => x.set('count', reaction.count));
+
+// The url parameter is only used when adding a new custom emoji reaction
+// (one that wasn't in the reactions list before) because we don't have its
+// URL yet.  In all other cases, it's undefined.
+const addReaction = (state, id, name, url) => updateReaction(
+  state,
+  id,
+  name,
+  x => x.set('me', true)
+    .update('count', n => n + 1)
+    .update('url', old => old ? old : url)
+    .update('static_url', old => old ? old : url),
+);
+
+const removeReaction = (state, id, name) => updateReaction(
+  state,
+  id,
+  name,
+  x => x.set('me', false).update('count', n => n - 1),
+);
+
+const initialState = ImmutableMap();
+
+export default function statuses(state = initialState, action) {
+  switch(action.type) {
+  case STATUS_FETCH_REQUEST:
+    return state.setIn([action.id, 'isLoading'], true);
+  case STATUS_FETCH_FAIL:
+    return state.delete(action.id);
+  case STATUS_IMPORT:
+    return importStatus(state, action.status);
+  case STATUSES_IMPORT:
+    return importStatuses(state, action.statuses);
+  case FAVOURITE_REQUEST:
+    return state.setIn([action.status.get('id'), 'favourited'], true);
+  case FAVOURITE_FAIL:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'favourited'], false);
+  case UNFAVOURITE_REQUEST:
+    return state.setIn([action.status.get('id'), 'favourited'], false);
+  case UNFAVOURITE_FAIL:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'favourited'], true);
+  case BOOKMARK_REQUEST:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'bookmarked'], true);
+  case BOOKMARK_FAIL:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'bookmarked'], false);
+  case UNBOOKMARK_REQUEST:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'bookmarked'], false);
+  case UNBOOKMARK_FAIL:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'bookmarked'], true);
+  case REBLOG_REQUEST:
+    return state.setIn([action.status.get('id'), 'reblogged'], true);
+  case REBLOG_FAIL:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'reblogged'], false);
+  case REACTION_UPDATE:
+    return updateReactionCount(state, action.reaction);
+  case REACTION_ADD_REQUEST:
+  case REACTION_REMOVE_FAIL:
+    return addReaction(state, action.id, action.name, action.url);
+  case REACTION_REMOVE_REQUEST:
+  case REACTION_ADD_FAIL:
+    return removeReaction(state, action.id, action.name);
+  case UNREBLOG_REQUEST:
+    return state.setIn([action.status.get('id'), 'reblogged'], false);
+  case UNREBLOG_FAIL:
+    return state.get(action.status.get('id')) === undefined ? state : state.setIn([action.status.get('id'), 'reblogged'], true);
+  case REACTION_UPDATE:
+    return updateReactionCount(state, action.reaction);
+  case REACTION_ADD_REQUEST:
+  case REACTION_REMOVE_FAIL:
+    return addReaction(state, action.id, action.name, action.url);
+  case REACTION_REMOVE_REQUEST:
+  case REACTION_ADD_FAIL:
+    return removeReaction(state, action.id, action.name);
+  case STATUS_MUTE_SUCCESS:
+    return state.setIn([action.id, 'muted'], true);
+  case STATUS_UNMUTE_SUCCESS:
+    return state.setIn([action.id, 'muted'], false);
+  case STATUS_REVEAL:
+    return state.withMutations(map => {
+      action.ids.forEach(id => {
+        if (!(state.get(id) === undefined)) {
+          map.setIn([id, 'hidden'], false);
+        }
+      });
+    });
+  case STATUS_HIDE:
+    return state.withMutations(map => {
+      action.ids.forEach(id => {
+        if (!(state.get(id) === undefined)) {
+          map.setIn([id, 'hidden'], true);
+        }
+      });
+    });
+  case STATUS_COLLAPSE:
+    return state.setIn([action.id, 'collapsed'], action.isCollapsed);
+  case TIMELINE_DELETE:
+    return deleteStatus(state, action.id, action.references);
+  case STATUS_TRANSLATE_SUCCESS:
+    return statusTranslateSuccess(state, action.id, action.translation);
+  case STATUS_TRANSLATE_UNDO:
+    return statusTranslateUndo(state, action.id);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/suggestions.js b/app/javascript/flavours/blobfox/reducers/suggestions.js
new file mode 100644
index 00000000000000..3ab190a9682b7f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/suggestions.js
@@ -0,0 +1,40 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import { blockAccountSuccess, muteAccountSuccess } from 'flavours/blobfox/actions/accounts';
+import { blockDomainSuccess } from 'flavours/blobfox/actions/domain_blocks';
+
+import {
+  SUGGESTIONS_FETCH_REQUEST,
+  SUGGESTIONS_FETCH_SUCCESS,
+  SUGGESTIONS_FETCH_FAIL,
+  SUGGESTIONS_DISMISS,
+} from '../actions/suggestions';
+
+
+const initialState = ImmutableMap({
+  items: ImmutableList(),
+  isLoading: false,
+});
+
+export default function suggestionsReducer(state = initialState, action) {
+  switch(action.type) {
+  case SUGGESTIONS_FETCH_REQUEST:
+    return state.set('isLoading', true);
+  case SUGGESTIONS_FETCH_SUCCESS:
+    return state.withMutations(map => {
+      map.set('items', fromJS(action.suggestions.map(x => ({ ...x, account: x.account.id }))));
+      map.set('isLoading', false);
+    });
+  case SUGGESTIONS_FETCH_FAIL:
+    return state.set('isLoading', false);
+  case SUGGESTIONS_DISMISS:
+    return state.update('items', list => list.filterNot(x => x.account === action.id));
+  case blockAccountSuccess.type:
+  case muteAccountSuccess.type:
+    return state.update('items', list => list.filterNot(x => x.account === action.payload.relationship.id));
+  case blockDomainSuccess.type:
+    return state.update('items', list => list.filterNot(x => action.payload.accounts.includes(x.account)));
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/tags.js b/app/javascript/flavours/blobfox/reducers/tags.js
new file mode 100644
index 00000000000000..e56dd9fd8046bb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/tags.js
@@ -0,0 +1,26 @@
+import { Map as ImmutableMap, fromJS } from 'immutable';
+
+import {
+  HASHTAG_FETCH_SUCCESS,
+  HASHTAG_FOLLOW_REQUEST,
+  HASHTAG_FOLLOW_FAIL,
+  HASHTAG_UNFOLLOW_REQUEST,
+  HASHTAG_UNFOLLOW_FAIL,
+} from 'flavours/blobfox/actions/tags';
+
+const initialState = ImmutableMap();
+
+export default function tags(state = initialState, action) {
+  switch(action.type) {
+  case HASHTAG_FETCH_SUCCESS:
+    return state.set(action.name, fromJS(action.tag));
+  case HASHTAG_FOLLOW_REQUEST:
+  case HASHTAG_UNFOLLOW_FAIL:
+    return state.setIn([action.name, 'following'], true);
+  case HASHTAG_FOLLOW_FAIL:
+  case HASHTAG_UNFOLLOW_REQUEST:
+    return state.setIn([action.name, 'following'], false);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/timelines.js b/app/javascript/flavours/blobfox/reducers/timelines.js
new file mode 100644
index 00000000000000..6ff83aa7f0f662
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/timelines.js
@@ -0,0 +1,233 @@
+import { Map as ImmutableMap, List as ImmutableList, OrderedSet as ImmutableOrderedSet, fromJS } from 'immutable';
+
+import {
+  blockAccountSuccess,
+  muteAccountSuccess,
+  unfollowAccountSuccess
+} from '../actions/accounts';
+import {
+  TIMELINE_UPDATE,
+  TIMELINE_DELETE,
+  TIMELINE_CLEAR,
+  TIMELINE_EXPAND_SUCCESS,
+  TIMELINE_EXPAND_REQUEST,
+  TIMELINE_EXPAND_FAIL,
+  TIMELINE_SCROLL_TOP,
+  TIMELINE_CONNECT,
+  TIMELINE_DISCONNECT,
+  TIMELINE_LOAD_PENDING,
+  TIMELINE_MARK_AS_PARTIAL,
+} from '../actions/timelines';
+import { compareId } from '../compare_id';
+
+const initialState = ImmutableMap();
+
+const initialTimeline = ImmutableMap({
+  unread: 0,
+  online: false,
+  top: true,
+  isLoading: false,
+  hasMore: true,
+  pendingItems: ImmutableList(),
+  items: ImmutableList(),
+});
+
+const expandNormalizedTimeline = (state, timeline, statuses, next, isPartial, isLoadingRecent, usePendingItems) => {
+  // This method is pretty tricky because:
+  // - existing items in the timeline might be out of order
+  // - the existing timeline may have gaps, most often explicitly noted with a `null` item
+  // - ideally, we don't want it to reorder existing items of the timeline
+  // - `statuses` may include items that are already included in the timeline
+  // - this function can be called either to fill in a gap, or load newer items
+
+  return state.update(timeline, initialTimeline, map => map.withMutations(mMap => {
+    mMap.set('isLoading', false);
+    mMap.set('isPartial', isPartial);
+
+    if (!next && !isLoadingRecent) mMap.set('hasMore', false);
+
+    if (timeline.endsWith(':pinned')) {
+      mMap.set('items', statuses.map(status => status.get('id')));
+    } else if (!statuses.isEmpty()) {
+      usePendingItems = isLoadingRecent && (usePendingItems || !mMap.get('pendingItems').isEmpty());
+
+      mMap.update(usePendingItems ? 'pendingItems' : 'items', ImmutableList(), oldIds => {
+        const newIds = statuses.map(status => status.get('id'));
+
+        // Now this gets tricky, as we don't necessarily know for sure where the gap to fill is
+        // and some items in the timeline may not be properly ordered.
+
+        // However, we know that `newIds.last()` is the oldest item that was requested and that
+        // there is no “hole” between `newIds.last()` and `newIds.first()`.
+
+        // First, find the furthest (if properly sorted, oldest) item in the timeline that is
+        // newer than the oldest fetched one, as it's most likely that it delimits the gap.
+        // Start the gap *after* that item.
+        const lastIndex = oldIds.findLastIndex(id => id !== null && compareId(id, newIds.last()) >= 0) + 1;
+
+        // Then, try to find the furthest (if properly sorted, oldest) item in the timeline that
+        // is newer than the most recent fetched one, as it delimits a section comprised of only
+        // items older or within `newIds` (or that were deleted from the server, so should be removed
+        // anyway).
+        // Stop the gap *after* that item.
+        const firstIndex = oldIds.take(lastIndex).findLastIndex(id => id !== null && compareId(id, newIds.first()) > 0) + 1;
+
+        let insertedIds = ImmutableOrderedSet(newIds).withMutations(insertedIds => {
+          // It is possible, though unlikely, that the slice we are replacing contains items older
+          // than the elements we got from the API. Get them and add them back at the back of the
+          // slice.
+          const olderIds = oldIds.slice(firstIndex, lastIndex).filter(id => id !== null && compareId(id, newIds.last()) < 0);
+          insertedIds.union(olderIds);
+
+          // Make sure we aren't inserting duplicates
+          insertedIds.subtract(oldIds.take(firstIndex), oldIds.skip(lastIndex));
+        }).toList();
+
+        // Finally, insert a gap marker if the data is marked as partial by the server
+        if (isPartial && (firstIndex === 0 || oldIds.get(firstIndex - 1) !== null)) {
+          insertedIds = insertedIds.unshift(null);
+        }
+
+        return oldIds.take(firstIndex).concat(
+          insertedIds,
+          oldIds.skip(lastIndex),
+        );
+      });
+    }
+  }));
+};
+
+const updateTimeline = (state, timeline, status, usePendingItems, filtered) => {
+  const top = state.getIn([timeline, 'top']);
+
+  if (usePendingItems || !state.getIn([timeline, 'pendingItems']).isEmpty()) {
+    if (state.getIn([timeline, 'pendingItems'], ImmutableList()).includes(status.get('id')) || state.getIn([timeline, 'items'], ImmutableList()).includes(status.get('id'))) {
+      return state;
+    }
+
+    state = state.update(timeline, initialTimeline, map => map.update('pendingItems', list => list.unshift(status.get('id'))));
+
+    if (!filtered) {
+      state = state.updateIn([timeline, 'unread'], unread => unread + 1);
+    }
+
+    return state;
+  }
+
+  const ids        = state.getIn([timeline, 'items'], ImmutableList());
+  const includesId = ids.includes(status.get('id'));
+  const unread     = state.getIn([timeline, 'unread'], 0);
+
+  if (includesId) {
+    return state;
+  }
+
+  let newIds = ids;
+
+  return state.update(timeline, initialTimeline, map => map.withMutations(mMap => {
+    if (!top && !filtered) mMap.set('unread', unread + 1);
+    if (top && ids.size > 40) newIds = newIds.take(20);
+    mMap.set('items', newIds.unshift(status.get('id')));
+  }));
+};
+
+const deleteStatus = (state, id, references, exclude_account = null) => {
+  state.keySeq().forEach(timeline => {
+    if (exclude_account === null || (timeline !== `account:${exclude_account}` && !timeline.startsWith(`account:${exclude_account}:`))) {
+      const helper = list => list.filterNot(item => item === id);
+      state = state.updateIn([timeline, 'items'], helper).updateIn([timeline, 'pendingItems'], helper);
+    }
+  });
+
+  // Remove reblogs of deleted status
+  references.forEach(ref => {
+    state = deleteStatus(state, ref, [], exclude_account);
+  });
+
+  return state;
+};
+
+const clearTimeline = (state, timeline) => {
+  return state.set(timeline, initialTimeline);
+};
+
+const filterTimelines = (state, relationship, statuses) => {
+  let references;
+
+  statuses.forEach(status => {
+    if (status.get('account') !== relationship.id) {
+      return;
+    }
+
+    references = statuses.filter(item => item.get('reblog') === status.get('id')).map(item => item.get('id'));
+    state      = deleteStatus(state, status.get('id'), references, relationship.id);
+  });
+
+  return state;
+};
+
+const filterTimeline = (timeline, state, relationship, statuses) => {
+  const helper = list => list.filterNot(statusId => statuses.getIn([statusId, 'account']) === relationship.id);
+  return state.updateIn([timeline, 'items'], ImmutableList(), helper).updateIn([timeline, 'pendingItems'], ImmutableList(), helper);
+};
+
+const updateTop = (state, timeline, top) => {
+  return state.update(timeline, initialTimeline, map => map.withMutations(mMap => {
+    if (top) mMap.set('unread', mMap.get('pendingItems').size);
+    mMap.set('top', top);
+  }));
+};
+
+const reconnectTimeline = (state, usePendingItems) => {
+  if (state.get('online')) {
+    return state;
+  }
+
+  return state.withMutations(mMap => {
+    mMap.update(usePendingItems ? 'pendingItems' : 'items', items => items.first() ? items.unshift(null) : items);
+    mMap.set('online', true);
+  });
+};
+
+export default function timelines(state = initialState, action) {
+  switch(action.type) {
+  case TIMELINE_LOAD_PENDING:
+    return state.update(action.timeline, initialTimeline, map =>
+      map.update('items', list => map.get('pendingItems').concat(list.take(40))).set('pendingItems', ImmutableList()).set('unread', 0));
+  case TIMELINE_EXPAND_REQUEST:
+    return state.update(action.timeline, initialTimeline, map => map.set('isLoading', true));
+  case TIMELINE_EXPAND_FAIL:
+    return state.update(action.timeline, initialTimeline, map => map.set('isLoading', false));
+  case TIMELINE_EXPAND_SUCCESS:
+    return expandNormalizedTimeline(state, action.timeline, fromJS(action.statuses), action.next, action.partial, action.isLoadingRecent, action.usePendingItems);
+  case TIMELINE_UPDATE:
+    return updateTimeline(state, action.timeline, fromJS(action.status), action.usePendingItems, action.filtered);
+  case TIMELINE_DELETE:
+    return deleteStatus(state, action.id, action.references, action.reblogOf);
+  case TIMELINE_CLEAR:
+    return clearTimeline(state, action.timeline);
+  case blockAccountSuccess.type:
+  case muteAccountSuccess.type:
+    return filterTimelines(state, action.payload.relationship, action.payload.statuses);
+  case unfollowAccountSuccess.type:
+    return filterTimeline('home', state, action.payload.relationship, action.payload.statuses);
+  case TIMELINE_SCROLL_TOP:
+    return updateTop(state, action.timeline, action.top);
+  case TIMELINE_CONNECT:
+    return state.update(action.timeline, initialTimeline, map => reconnectTimeline(map, action.usePendingItems));
+  case TIMELINE_DISCONNECT:
+    return state.update(
+      action.timeline,
+      initialTimeline,
+      map => map.set('online', false).update(action.usePendingItems ? 'pendingItems' : 'items', items => items.first() ? items.unshift(null) : items),
+    );
+  case TIMELINE_MARK_AS_PARTIAL:
+    return state.update(
+      action.timeline,
+      initialTimeline,
+      map => map.set('isPartial', true).set('items', ImmutableList()).set('pendingItems', ImmutableList()).set('unread', 0),
+    );
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/trends.js b/app/javascript/flavours/blobfox/reducers/trends.js
new file mode 100644
index 00000000000000..8f6733ceb310c6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/trends.js
@@ -0,0 +1,47 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import {
+  TRENDS_TAGS_FETCH_REQUEST,
+  TRENDS_TAGS_FETCH_SUCCESS,
+  TRENDS_TAGS_FETCH_FAIL,
+  TRENDS_LINKS_FETCH_REQUEST,
+  TRENDS_LINKS_FETCH_SUCCESS,
+  TRENDS_LINKS_FETCH_FAIL,
+} from 'flavours/blobfox/actions/trends';
+
+const initialState = ImmutableMap({
+  tags: ImmutableMap({
+    items: ImmutableList(),
+    isLoading: false,
+  }),
+
+  links: ImmutableMap({
+    items: ImmutableList(),
+    isLoading: false,
+  }),
+});
+
+export default function trendsReducer(state = initialState, action) {
+  switch(action.type) {
+  case TRENDS_TAGS_FETCH_REQUEST:
+    return state.setIn(['tags', 'isLoading'], true);
+  case TRENDS_TAGS_FETCH_SUCCESS:
+    return state.withMutations(map => {
+      map.setIn(['tags', 'items'], fromJS(action.trends));
+      map.setIn(['tags', 'isLoading'], false);
+    });
+  case TRENDS_TAGS_FETCH_FAIL:
+    return state.setIn(['tags', 'isLoading'], false);
+  case TRENDS_LINKS_FETCH_REQUEST:
+    return state.setIn(['links', 'isLoading'], true);
+  case TRENDS_LINKS_FETCH_SUCCESS:
+    return state.withMutations(map => {
+      map.setIn(['links', 'items'], fromJS(action.trends));
+      map.setIn(['links', 'isLoading'], false);
+    });
+  case TRENDS_LINKS_FETCH_FAIL:
+    return state.setIn(['links', 'isLoading'], false);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/reducers/user_lists.js b/app/javascript/flavours/blobfox/reducers/user_lists.js
new file mode 100644
index 00000000000000..a2bd2e8a17373f
--- /dev/null
+++ b/app/javascript/flavours/blobfox/reducers/user_lists.js
@@ -0,0 +1,216 @@
+import { Map as ImmutableMap, List as ImmutableList, fromJS } from 'immutable';
+
+import {
+  DIRECTORY_FETCH_REQUEST,
+  DIRECTORY_FETCH_SUCCESS,
+  DIRECTORY_FETCH_FAIL,
+  DIRECTORY_EXPAND_REQUEST,
+  DIRECTORY_EXPAND_SUCCESS,
+  DIRECTORY_EXPAND_FAIL,
+} from 'flavours/blobfox/actions/directory';
+import {
+  FEATURED_TAGS_FETCH_REQUEST,
+  FEATURED_TAGS_FETCH_SUCCESS,
+  FEATURED_TAGS_FETCH_FAIL,
+} from 'flavours/blobfox/actions/featured_tags';
+
+import {
+  FOLLOWERS_FETCH_REQUEST,
+  FOLLOWERS_FETCH_SUCCESS,
+  FOLLOWERS_FETCH_FAIL,
+  FOLLOWERS_EXPAND_REQUEST,
+  FOLLOWERS_EXPAND_SUCCESS,
+  FOLLOWERS_EXPAND_FAIL,
+  FOLLOWING_FETCH_REQUEST,
+  FOLLOWING_FETCH_SUCCESS,
+  FOLLOWING_FETCH_FAIL,
+  FOLLOWING_EXPAND_REQUEST,
+  FOLLOWING_EXPAND_SUCCESS,
+  FOLLOWING_EXPAND_FAIL,
+  FOLLOW_REQUESTS_FETCH_REQUEST,
+  FOLLOW_REQUESTS_FETCH_SUCCESS,
+  FOLLOW_REQUESTS_FETCH_FAIL,
+  FOLLOW_REQUESTS_EXPAND_REQUEST,
+  FOLLOW_REQUESTS_EXPAND_SUCCESS,
+  FOLLOW_REQUESTS_EXPAND_FAIL,
+  authorizeFollowRequestSuccess,
+  rejectFollowRequestSuccess,
+} from '../actions/accounts';
+import {
+  BLOCKS_FETCH_REQUEST,
+  BLOCKS_FETCH_SUCCESS,
+  BLOCKS_FETCH_FAIL,
+  BLOCKS_EXPAND_REQUEST,
+  BLOCKS_EXPAND_SUCCESS,
+  BLOCKS_EXPAND_FAIL,
+} from '../actions/blocks';
+import {
+  REBLOGS_FETCH_REQUEST,
+  REBLOGS_FETCH_SUCCESS,
+  REBLOGS_FETCH_FAIL,
+  REBLOGS_EXPAND_REQUEST,
+  REBLOGS_EXPAND_SUCCESS,
+  REBLOGS_EXPAND_FAIL,
+  FAVOURITES_FETCH_REQUEST,
+  FAVOURITES_FETCH_SUCCESS,
+  FAVOURITES_FETCH_FAIL,
+  FAVOURITES_EXPAND_REQUEST,
+  FAVOURITES_EXPAND_SUCCESS,
+  FAVOURITES_EXPAND_FAIL,
+} from '../actions/interactions';
+import {
+  MUTES_FETCH_REQUEST,
+  MUTES_FETCH_SUCCESS,
+  MUTES_FETCH_FAIL,
+  MUTES_EXPAND_REQUEST,
+  MUTES_EXPAND_SUCCESS,
+  MUTES_EXPAND_FAIL,
+} from '../actions/mutes';
+import { notificationsUpdate } from '../actions/notifications';
+
+const initialListState = ImmutableMap({
+  next: null,
+  isLoading: false,
+  items: ImmutableList(),
+});
+
+const initialState = ImmutableMap({
+  followers: initialListState,
+  following: initialListState,
+  reblogged_by: initialListState,
+  favourited_by: initialListState,
+  follow_requests: initialListState,
+  blocks: initialListState,
+  mutes: initialListState,
+  featured_tags: initialListState,
+});
+
+const normalizeList = (state, path, accounts, next) => {
+  return state.setIn(path, ImmutableMap({
+    next,
+    items: ImmutableList(accounts.map(item => item.id)),
+    isLoading: false,
+  }));
+};
+
+const appendToList = (state, path, accounts, next) => {
+  return state.updateIn(path, map => {
+    return map.set('next', next).set('isLoading', false).update('items', list => list.concat(accounts.map(item => item.id)));
+  });
+};
+
+const normalizeFollowRequest = (state, notification) => {
+  return state.updateIn(['follow_requests', 'items'], list => {
+    return list.filterNot(item => item === notification.account.id).unshift(notification.account.id);
+  });
+};
+
+const normalizeFeaturedTag = (featuredTags, accountId) => {
+  const normalizeFeaturedTag = { ...featuredTags, accountId: accountId };
+  return fromJS(normalizeFeaturedTag);
+};
+
+const normalizeFeaturedTags = (state, path, featuredTags, accountId) => {
+  return state.setIn(path, ImmutableMap({
+    items: ImmutableList(featuredTags.map(featuredTag => normalizeFeaturedTag(featuredTag, accountId)).sort((a, b) => b.get('statuses_count') - a.get('statuses_count'))),
+    isLoading: false,
+  }));
+};
+
+export default function userLists(state = initialState, action) {
+  switch(action.type) {
+  case FOLLOWERS_FETCH_SUCCESS:
+    return normalizeList(state, ['followers', action.id], action.accounts, action.next);
+  case FOLLOWERS_EXPAND_SUCCESS:
+    return appendToList(state, ['followers', action.id], action.accounts, action.next);
+  case FOLLOWERS_FETCH_REQUEST:
+  case FOLLOWERS_EXPAND_REQUEST:
+    return state.setIn(['followers', action.id, 'isLoading'], true);
+  case FOLLOWERS_FETCH_FAIL:
+  case FOLLOWERS_EXPAND_FAIL:
+    return state.setIn(['followers', action.id, 'isLoading'], false);
+  case FOLLOWING_FETCH_SUCCESS:
+    return normalizeList(state, ['following', action.id], action.accounts, action.next);
+  case FOLLOWING_EXPAND_SUCCESS:
+    return appendToList(state, ['following', action.id], action.accounts, action.next);
+  case FOLLOWING_FETCH_REQUEST:
+  case FOLLOWING_EXPAND_REQUEST:
+    return state.setIn(['following', action.id, 'isLoading'], true);
+  case FOLLOWING_FETCH_FAIL:
+  case FOLLOWING_EXPAND_FAIL:
+    return state.setIn(['following', action.id, 'isLoading'], false);
+  case REBLOGS_FETCH_SUCCESS:
+    return normalizeList(state, ['reblogged_by', action.id], action.accounts, action.next);
+  case REBLOGS_EXPAND_SUCCESS:
+    return appendToList(state, ['reblogged_by', action.id], action.accounts, action.next);
+  case REBLOGS_FETCH_REQUEST:
+  case REBLOGS_EXPAND_REQUEST:
+    return state.setIn(['reblogged_by', action.id, 'isLoading'], true);
+  case REBLOGS_FETCH_FAIL:
+  case REBLOGS_EXPAND_FAIL:
+    return state.setIn(['reblogged_by', action.id, 'isLoading'], false);
+  case FAVOURITES_FETCH_SUCCESS:
+    return normalizeList(state, ['favourited_by', action.id], action.accounts, action.next);
+  case FAVOURITES_EXPAND_SUCCESS:
+    return appendToList(state, ['favourited_by', action.id], action.accounts, action.next);
+  case FAVOURITES_FETCH_REQUEST:
+  case FAVOURITES_EXPAND_REQUEST:
+    return state.setIn(['favourited_by', action.id, 'isLoading'], true);
+  case FAVOURITES_FETCH_FAIL:
+  case FAVOURITES_EXPAND_FAIL:
+    return state.setIn(['favourited_by', action.id, 'isLoading'], false);
+  case notificationsUpdate.type:
+    return action.payload.notification.type === 'follow_request' ? normalizeFollowRequest(state, action.payload.notification) : state;
+  case FOLLOW_REQUESTS_FETCH_SUCCESS:
+    return normalizeList(state, ['follow_requests'], action.accounts, action.next);
+  case FOLLOW_REQUESTS_EXPAND_SUCCESS:
+    return appendToList(state, ['follow_requests'], action.accounts, action.next);
+  case FOLLOW_REQUESTS_FETCH_REQUEST:
+  case FOLLOW_REQUESTS_EXPAND_REQUEST:
+    return state.setIn(['follow_requests', 'isLoading'], true);
+  case FOLLOW_REQUESTS_FETCH_FAIL:
+  case FOLLOW_REQUESTS_EXPAND_FAIL:
+    return state.setIn(['follow_requests', 'isLoading'], false);
+  case authorizeFollowRequestSuccess.type:
+  case rejectFollowRequestSuccess.type:
+    return state.updateIn(['follow_requests', 'items'], list => list.filterNot(item => item === action.payload.id));
+  case BLOCKS_FETCH_SUCCESS:
+    return normalizeList(state, ['blocks'], action.accounts, action.next);
+  case BLOCKS_EXPAND_SUCCESS:
+    return appendToList(state, ['blocks'], action.accounts, action.next);
+  case BLOCKS_FETCH_REQUEST:
+  case BLOCKS_EXPAND_REQUEST:
+    return state.setIn(['blocks', 'isLoading'], true);
+  case BLOCKS_FETCH_FAIL:
+  case BLOCKS_EXPAND_FAIL:
+    return state.setIn(['blocks', 'isLoading'], false);
+  case MUTES_FETCH_SUCCESS:
+    return normalizeList(state, ['mutes'], action.accounts, action.next);
+  case MUTES_EXPAND_SUCCESS:
+    return appendToList(state, ['mutes'], action.accounts, action.next);
+  case MUTES_FETCH_REQUEST:
+  case MUTES_EXPAND_REQUEST:
+    return state.setIn(['mutes', 'isLoading'], true);
+  case MUTES_FETCH_FAIL:
+  case MUTES_EXPAND_FAIL:
+    return state.setIn(['mutes', 'isLoading'], false);
+  case DIRECTORY_FETCH_SUCCESS:
+    return normalizeList(state, ['directory'], action.accounts, action.next);
+  case DIRECTORY_EXPAND_SUCCESS:
+    return appendToList(state, ['directory'], action.accounts, action.next);
+  case DIRECTORY_FETCH_REQUEST:
+  case DIRECTORY_EXPAND_REQUEST:
+    return state.setIn(['directory', 'isLoading'], true);
+  case DIRECTORY_FETCH_FAIL:
+  case DIRECTORY_EXPAND_FAIL:
+    return state.setIn(['directory', 'isLoading'], false);
+  case FEATURED_TAGS_FETCH_SUCCESS:
+    return normalizeFeaturedTags(state, ['featured_tags', action.id], action.tags, action.id);
+  case FEATURED_TAGS_FETCH_REQUEST:
+    return state.setIn(['featured_tags', action.id, 'isLoading'], true);
+  case FEATURED_TAGS_FETCH_FAIL:
+    return state.setIn(['featured_tags', action.id, 'isLoading'], false);
+  default:
+    return state;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/scroll.ts b/app/javascript/flavours/blobfox/scroll.ts
new file mode 100644
index 00000000000000..35e13a4527d1d3
--- /dev/null
+++ b/app/javascript/flavours/blobfox/scroll.ts
@@ -0,0 +1,50 @@
+const easingOutQuint = (
+  x: number,
+  t: number,
+  b: number,
+  c: number,
+  d: number,
+) => c * ((t = t / d - 1) * t * t * t * t + 1) + b;
+const scroll = (
+  node: Element,
+  key: 'scrollTop' | 'scrollLeft',
+  target: number,
+) => {
+  const startTime = Date.now();
+  const offset = node[key];
+  const gap = target - offset;
+  const duration = 1000;
+  let interrupt = false;
+
+  const step = () => {
+    const elapsed = Date.now() - startTime;
+    const percentage = elapsed / duration;
+
+    if (percentage > 1 || interrupt) {
+      return;
+    }
+
+    node[key] = easingOutQuint(0, elapsed, offset, gap, duration);
+    requestAnimationFrame(step);
+  };
+
+  step();
+
+  return () => {
+    interrupt = true;
+  };
+};
+
+const isScrollBehaviorSupported =
+  'scrollBehavior' in document.documentElement.style;
+
+export const scrollRight = (node: Element, position: number) => {
+  if (isScrollBehaviorSupported)
+    node.scrollTo({ left: position, behavior: 'smooth' });
+  else scroll(node, 'scrollLeft', position);
+};
+
+export const scrollTop = (node: Element) => {
+  if (isScrollBehaviorSupported) node.scrollTo({ top: 0, behavior: 'smooth' });
+  else scroll(node, 'scrollTop', 0);
+};
diff --git a/app/javascript/flavours/blobfox/selectors/accounts.ts b/app/javascript/flavours/blobfox/selectors/accounts.ts
new file mode 100644
index 00000000000000..af2319405df219
--- /dev/null
+++ b/app/javascript/flavours/blobfox/selectors/accounts.ts
@@ -0,0 +1,47 @@
+import { Record as ImmutableRecord } from 'immutable';
+import { createSelector } from 'reselect';
+
+import { accountDefaultValues } from 'flavours/blobfox/models/account';
+import type { Account, AccountShape } from 'flavours/blobfox/models/account';
+import type { Relationship } from 'flavours/blobfox/models/relationship';
+import type { RootState } from 'flavours/blobfox/store';
+
+const getAccountBase = (state: RootState, id: string) =>
+  state.accounts.get(id, null);
+
+const getAccountRelationship = (state: RootState, id: string) =>
+  state.relationships.get(id, null);
+
+const getAccountMoved = (state: RootState, id: string) => {
+  const movedToId = state.accounts.get(id)?.moved;
+
+  if (!movedToId) return undefined;
+
+  return state.accounts.get(movedToId);
+};
+
+interface FullAccountShape extends Omit<AccountShape, 'moved'> {
+  relationship: Relationship | null;
+  moved: Account | null;
+}
+
+const FullAccountFactory = ImmutableRecord<FullAccountShape>({
+  ...accountDefaultValues,
+  moved: null,
+  relationship: null,
+});
+
+export function makeGetAccount() {
+  return createSelector(
+    [getAccountBase, getAccountRelationship, getAccountMoved],
+    (base, relationship, moved) => {
+      if (base === null) {
+        return null;
+      }
+
+      return FullAccountFactory(base)
+        .set('relationship', relationship)
+        .set('moved', moved ?? null);
+    },
+  );
+}
diff --git a/app/javascript/flavours/blobfox/selectors/index.js b/app/javascript/flavours/blobfox/selectors/index.js
new file mode 100644
index 00000000000000..ceb940031940b0
--- /dev/null
+++ b/app/javascript/flavours/blobfox/selectors/index.js
@@ -0,0 +1,118 @@
+import { List as ImmutableList, Map as ImmutableMap } from 'immutable';
+import { createSelector } from 'reselect';
+
+import { toServerSideType } from 'flavours/blobfox/utils/filters';
+
+import { me } from '../initial_state';
+
+export { makeGetAccount } from "./accounts";
+
+const getFilters = (state, { contextType }) => {
+  if (!contextType) return null;
+
+  const serverSideType = toServerSideType(contextType);
+  const now = new Date();
+
+  return state.get('filters').filter((filter) => filter.get('context').includes(serverSideType) && (filter.get('expires_at') === null || filter.get('expires_at') > now));
+};
+
+export const makeGetStatus = () => {
+  return createSelector(
+    [
+      (state, { id }) => state.getIn(['statuses', id]),
+      (state, { id }) => state.getIn(['statuses', state.getIn(['statuses', id, 'reblog'])]),
+      (state, { id }) => state.getIn(['accounts', state.getIn(['statuses', id, 'account'])]),
+      (state, { id }) => state.getIn(['accounts', state.getIn(['statuses', state.getIn(['statuses', id, 'reblog']), 'account'])]),
+      getFilters,
+    ],
+
+    (statusBase, statusReblog, accountBase, accountReblog, filters) => {
+      if (!statusBase || statusBase.get('isLoading')) {
+        return null;
+      }
+
+      let filtered = false;
+      if ((accountReblog || accountBase).get('id') !== me && filters) {
+        let filterResults = statusReblog?.get('filtered') || statusBase.get('filtered') || ImmutableList();
+        if (filterResults.some((result) => filters.getIn([result.get('filter'), 'filter_action']) === 'hide')) {
+          return null;
+        }
+        filterResults = filterResults.filter(result => filters.has(result.get('filter')));
+        if (!filterResults.isEmpty()) {
+          filtered = filterResults.map(result => filters.getIn([result.get('filter'), 'title']));
+        }
+      }
+
+      if (statusReblog) {
+        statusReblog = statusReblog.set('account', accountReblog);
+        statusReblog = statusReblog.set('matched_filters', filtered);
+      } else {
+        statusReblog = null;
+      }
+
+      return statusBase.withMutations(map => {
+        map.set('reblog', statusReblog);
+        map.set('account', accountBase);
+        map.set('matched_filters', filtered);
+      });
+    },
+  );
+};
+
+export const makeGetPictureInPicture = () => {
+  return createSelector([
+    (state, { id }) => state.get('picture_in_picture').statusId === id,
+    (state) => state.getIn(['meta', 'layout']) !== 'mobile',
+  ], (inUse, available) => ImmutableMap({
+    inUse: inUse && available,
+    available,
+  }));
+};
+
+const ALERT_DEFAULTS = {
+  dismissAfter: 5000,
+  style: false,
+};
+
+export const getAlerts = createSelector(state => state.get('alerts'), alerts =>
+  alerts.map(item => ({
+    ...ALERT_DEFAULTS,
+    ...item,
+  })).toArray());
+
+export const makeGetNotification = () => createSelector([
+  (_, base)             => base,
+  (state, _, accountId) => state.getIn(['accounts', accountId]),
+], (base, account) => base.set('account', account));
+
+export const makeGetReport = () => createSelector([
+  (_, base) => base,
+  (state, _, targetAccountId) => state.getIn(['accounts', targetAccountId]),
+], (base, targetAccount) => base.set('target_account', targetAccount));
+
+export const getAccountGallery = createSelector([
+  (state, id) => state.getIn(['timelines', `account:${id}:media`, 'items'], ImmutableList()),
+  state       => state.get('statuses'),
+  (state, id) => state.getIn(['accounts', id]),
+], (statusIds, statuses, account) => {
+  let medias = ImmutableList();
+
+  statusIds.forEach(statusId => {
+    const status = statuses.get(statusId);
+    medias = medias.concat(status.get('media_attachments').map(media => media.set('status', status).set('account', account)));
+  });
+
+  return medias;
+});
+
+export const getAccountHidden = createSelector([
+  (state, id) => state.getIn(['accounts', id, 'hidden']),
+  (state, id) => state.getIn(['relationships', id, 'following']) || state.getIn(['relationships', id, 'requested']),
+  (state, id) => id === me,
+], (hidden, followingOrRequested, isSelf) => {
+  return hidden && !(isSelf || followingOrRequested);
+});
+
+export const getStatusList = createSelector([
+  (state, type) => state.getIn(['status_lists', type, 'items']),
+], (items) => items.toList());
diff --git a/app/javascript/flavours/blobfox/settings.js b/app/javascript/flavours/blobfox/settings.js
new file mode 100644
index 00000000000000..f31aee0afc0725
--- /dev/null
+++ b/app/javascript/flavours/blobfox/settings.js
@@ -0,0 +1,50 @@
+export default class Settings {
+
+  constructor(keyBase = null) {
+    this.keyBase = keyBase;
+  }
+
+  generateKey(id) {
+    return this.keyBase ? [this.keyBase, `id${id}`].join('.') : id;
+  }
+
+  set(id, data) {
+    const key = this.generateKey(id);
+    try {
+      const encodedData = JSON.stringify(data);
+      localStorage.setItem(key, encodedData);
+      return data;
+    } catch (e) {
+      return null;
+    }
+  }
+
+  get(id) {
+    const key = this.generateKey(id);
+    try {
+      const rawData = localStorage.getItem(key);
+      return JSON.parse(rawData);
+    } catch (e) {
+      return null;
+    }
+  }
+
+  remove(id) {
+    const data = this.get(id);
+    if (data) {
+      const key = this.generateKey(id);
+      try {
+        localStorage.removeItem(key);
+      } catch (e) {
+      }
+    }
+    return data;
+  }
+
+}
+
+export const pushNotificationsSetting = new Settings('mastodon_push_notification_data');
+export const tagHistory = new Settings('mastodon_tag_history');
+export const bannerSettings = new Settings('mastodon_banner_settings');
+export const searchHistory = new Settings('mastodon_search_history');
+export const playerSettings = new Settings('mastodon_player');
diff --git a/app/javascript/flavours/blobfox/store/index.ts b/app/javascript/flavours/blobfox/store/index.ts
new file mode 100644
index 00000000000000..c2629b0ed74050
--- /dev/null
+++ b/app/javascript/flavours/blobfox/store/index.ts
@@ -0,0 +1,8 @@
+export { store } from './store';
+export type { GetState, AppDispatch, RootState } from './store';
+
+export {
+  createAppAsyncThunk,
+  useAppDispatch,
+  useAppSelector,
+} from './typed_functions';
diff --git a/app/javascript/flavours/blobfox/store/middlewares/errors.ts b/app/javascript/flavours/blobfox/store/middlewares/errors.ts
new file mode 100644
index 00000000000000..9f28f5ff53f92b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/store/middlewares/errors.ts
@@ -0,0 +1,21 @@
+import type { AnyAction, Middleware } from 'redux';
+
+import type { RootState } from '..';
+import { showAlertForError } from '../../actions/alerts';
+
+const defaultFailSuffix = 'FAIL';
+
+export const errorsMiddleware: Middleware<unknown, RootState> =
+  ({ dispatch }) =>
+  (next) =>
+  (action: AnyAction & { skipAlert?: boolean; skipNotFound?: boolean }) => {
+    if (action.type && !action.skipAlert) {
+      const isFail = new RegExp(`${defaultFailSuffix}$`, 'g');
+
+      if (typeof action.type === 'string' && action.type.match(isFail)) {
+        dispatch(showAlertForError(action.error, action.skipNotFound));
+      }
+    }
+
+    return next(action);
+  };
diff --git a/app/javascript/flavours/blobfox/store/middlewares/loading_bar.ts b/app/javascript/flavours/blobfox/store/middlewares/loading_bar.ts
new file mode 100644
index 00000000000000..83056ee49f44a9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/store/middlewares/loading_bar.ts
@@ -0,0 +1,69 @@
+import {
+  isAsyncThunkAction,
+  isPending as isThunkActionPending,
+  isFulfilled as isThunkActionFulfilled,
+  isRejected as isThunkActionRejected,
+} from '@reduxjs/toolkit';
+import { showLoading, hideLoading } from 'react-redux-loading-bar';
+import type { AnyAction, Middleware } from 'redux';
+
+import type { RootState } from '..';
+
+interface Config {
+  promiseTypeSuffixes?: string[];
+}
+
+const defaultTypeSuffixes: Config['promiseTypeSuffixes'] = [
+  'PENDING',
+  'FULFILLED',
+  'REJECTED',
+];
+
+export const loadingBarMiddleware = (
+  config: Config = {},
+): Middleware<unknown, RootState> => {
+  const promiseTypeSuffixes = config.promiseTypeSuffixes ?? defaultTypeSuffixes;
+
+  return ({ dispatch }) =>
+    (next) =>
+    (action: AnyAction) => {
+      let isPending = false;
+      let isFulfilled = false;
+      let isRejected = false;
+
+      if (
+        isAsyncThunkAction(action)
+        // TODO: once we get the first use-case for it, add a check for skipLoading
+      ) {
+        if (isThunkActionPending(action)) isPending = true;
+        else if (isThunkActionFulfilled(action)) isFulfilled = true;
+        else if (isThunkActionRejected(action)) isRejected = true;
+      } else if (
+        action.type &&
+        !action.skipLoading &&
+        typeof action.type === 'string'
+      ) {
+        const [PENDING, FULFILLED, REJECTED] = promiseTypeSuffixes;
+
+        const isPendingRegexp = new RegExp(`${PENDING}$`, 'g');
+        const isFulfilledRegexp = new RegExp(`${FULFILLED}$`, 'g');
+        const isRejectedRegexp = new RegExp(`${REJECTED}$`, 'g');
+
+        if (action.type.match(isPendingRegexp)) {
+          isPending = true;
+        } else if (action.type.match(isFulfilledRegexp)) {
+          isFulfilled = true;
+        } else if (action.type.match(isRejectedRegexp)) {
+          isRejected = true;
+        }
+      }
+
+      if (isPending) {
+        dispatch(showLoading());
+      } else if (isFulfilled || isRejected) {
+        dispatch(hideLoading());
+      }
+
+      return next(action);
+    };
+};
diff --git a/app/javascript/flavours/blobfox/store/middlewares/sounds.ts b/app/javascript/flavours/blobfox/store/middlewares/sounds.ts
new file mode 100644
index 00000000000000..3b0556c9c29db4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/store/middlewares/sounds.ts
@@ -0,0 +1,64 @@
+import type { Middleware, AnyAction } from 'redux';
+
+import ready from 'flavours/blobfox/ready';
+import { assetHost } from 'flavours/blobfox/utils/config';
+
+import type { RootState } from '..';
+
+interface AudioSource {
+  src: string;
+  type: string;
+}
+
+const createAudio = (sources: AudioSource[]) => {
+  const audio = new Audio();
+  sources.forEach(({ type, src }) => {
+    const source = document.createElement('source');
+    source.type = type;
+    source.src = src;
+    audio.appendChild(source);
+  });
+  return audio;
+};
+
+const play = (audio: HTMLAudioElement) => {
+  if (!audio.paused) {
+    audio.pause();
+    if (typeof audio.fastSeek === 'function') {
+      audio.fastSeek(0);
+    } else {
+      audio.currentTime = 0;
+    }
+  }
+
+  void audio.play();
+};
+
+export const soundsMiddleware = (): Middleware<unknown, RootState> => {
+  const soundCache: Record<string, HTMLAudioElement> = {};
+
+  void ready(() => {
+    soundCache.boop = createAudio([
+      {
+        src: `${assetHost}/sounds/boop.ogg`,
+        type: 'audio/ogg',
+      },
+      {
+        src: `${assetHost}/sounds/boop.mp3`,
+        type: 'audio/mpeg',
+      },
+    ]);
+  });
+
+  return () =>
+    (next) =>
+    (action: AnyAction & { meta?: { sound?: string } }) => {
+      const sound = action.meta?.sound;
+
+      if (sound && Object.hasOwn(soundCache, sound)) {
+        play(soundCache[sound]);
+      }
+
+      return next(action);
+    };
+};
diff --git a/app/javascript/flavours/blobfox/store/store.ts b/app/javascript/flavours/blobfox/store/store.ts
new file mode 100644
index 00000000000000..9f43f58a43dfa4
--- /dev/null
+++ b/app/javascript/flavours/blobfox/store/store.ts
@@ -0,0 +1,39 @@
+import { configureStore } from '@reduxjs/toolkit';
+
+import { rootReducer } from '../reducers';
+
+import { errorsMiddleware } from './middlewares/errors';
+import { loadingBarMiddleware } from './middlewares/loading_bar';
+import { soundsMiddleware } from './middlewares/sounds';
+
+export const store = configureStore({
+  reducer: rootReducer,
+  middleware: (getDefaultMiddleware) =>
+    getDefaultMiddleware({
+      // In development, Redux Toolkit enables 2 default middlewares to detect
+      // common issues with states. Unfortunately, our use of ImmutableJS for state
+      // triggers both, so lets disable them until our state is fully refactored
+
+      // https://redux-toolkit.js.org/api/serializabilityMiddleware
+      // This checks recursively that every values in the state are serializable in JSON
+      // Which is not the case, as we use ImmutableJS structures, but also File objects
+      serializableCheck: false,
+
+      // https://redux-toolkit.js.org/api/immutabilityMiddleware
+      // This checks recursively if every value in the state is immutable (ie, a JS primitive type)
+      // But this is not the case, as our Root State is an ImmutableJS map, which is an object
+      immutableCheck: false,
+    })
+      .concat(
+        loadingBarMiddleware({
+          promiseTypeSuffixes: ['REQUEST', 'SUCCESS', 'FAIL'],
+        }),
+      )
+      .concat(errorsMiddleware)
+      .concat(soundsMiddleware()),
+});
+
+// Infer the `RootState` and `AppDispatch` types from the store itself
+export type RootState = ReturnType<typeof rootReducer>;
+export type AppDispatch = typeof store.dispatch;
+export type GetState = typeof store.getState;
diff --git a/app/javascript/flavours/blobfox/store/typed_functions.ts b/app/javascript/flavours/blobfox/store/typed_functions.ts
new file mode 100644
index 00000000000000..f1e71385a88ff5
--- /dev/null
+++ b/app/javascript/flavours/blobfox/store/typed_functions.ts
@@ -0,0 +1,15 @@
+import type { TypedUseSelectorHook } from 'react-redux';
+import { useDispatch, useSelector } from 'react-redux';
+
+import { createAsyncThunk } from '@reduxjs/toolkit';
+
+import type { AppDispatch, RootState } from './store';
+
+export const useAppDispatch: () => AppDispatch = useDispatch;
+export const useAppSelector: TypedUseSelectorHook<RootState> = useSelector;
+
+export const createAppAsyncThunk = createAsyncThunk.withTypes<{
+  state: RootState;
+  dispatch: AppDispatch;
+  rejectValue: string;
+}>();
diff --git a/app/javascript/flavours/blobfox/stream.js b/app/javascript/flavours/blobfox/stream.js
new file mode 100644
index 00000000000000..ff3af5fd885c96
--- /dev/null
+++ b/app/javascript/flavours/blobfox/stream.js
@@ -0,0 +1,275 @@
+// @ts-check
+
+import WebSocketClient from '@gamestdio/websocket';
+
+/**
+ * @type {WebSocketClient | undefined}
+ */
+let sharedConnection;
+
+/**
+ * @typedef Subscription
+ * @property {string} channelName
+ * @property {Object.<string, string>} params
+ * @property {function(): void} onConnect
+ * @property {function(StreamEvent): void} onReceive
+ * @property {function(): void} onDisconnect
+ */
+
+/**
+ * @typedef StreamEvent
+ * @property {string} event
+ * @property {object} payload
+ */
+
+/**
+ * @type {Array.<Subscription>}
+ */
+const subscriptions = [];
+
+/**
+ * @type {Object.<string, number>}
+ */
+const subscriptionCounters = {};
+
+/**
+ * @param {Subscription} subscription
+ */
+const addSubscription = subscription => {
+  subscriptions.push(subscription);
+};
+
+/**
+ * @param {Subscription} subscription
+ */
+const removeSubscription = subscription => {
+  const index = subscriptions.indexOf(subscription);
+
+  if (index !== -1) {
+    subscriptions.splice(index, 1);
+  }
+};
+
+/**
+ * @param {Subscription} subscription
+ */
+const subscribe = ({ channelName, params, onConnect }) => {
+  const key = channelNameWithInlineParams(channelName, params);
+
+  subscriptionCounters[key] = subscriptionCounters[key] || 0;
+
+  if (subscriptionCounters[key] === 0) {
+    // @ts-expect-error
+    sharedConnection.send(JSON.stringify({ type: 'subscribe', stream: channelName, ...params }));
+  }
+
+  subscriptionCounters[key] += 1;
+  onConnect();
+};
+
+/**
+ * @param {Subscription} subscription
+ */
+const unsubscribe = ({ channelName, params, onDisconnect }) => {
+  const key = channelNameWithInlineParams(channelName, params);
+
+  subscriptionCounters[key] = subscriptionCounters[key] || 1;
+
+  // @ts-expect-error
+  if (subscriptionCounters[key] === 1 && sharedConnection.readyState === WebSocketClient.OPEN) {
+    // @ts-expect-error
+    sharedConnection.send(JSON.stringify({ type: 'unsubscribe', stream: channelName, ...params }));
+  }
+
+  subscriptionCounters[key] -= 1;
+  onDisconnect();
+};
+
+const sharedCallbacks = {
+  connected() {
+    subscriptions.forEach(subscription => subscribe(subscription));
+  },
+
+  // @ts-expect-error
+  received(data) {
+    const { stream } = data;
+
+    subscriptions.filter(({ channelName, params }) => {
+      const streamChannelName = stream[0];
+
+      if (stream.length === 1) {
+        return channelName === streamChannelName;
+      }
+
+      const streamIdentifier = stream[1];
+
+      if (['hashtag', 'hashtag:local'].includes(channelName)) {
+        return channelName === streamChannelName && params.tag === streamIdentifier;
+      } else if (channelName === 'list') {
+        return channelName === streamChannelName && params.list === streamIdentifier;
+      }
+
+      return false;
+    }).forEach(subscription => {
+      subscription.onReceive(data);
+    });
+  },
+
+  disconnected() {
+    subscriptions.forEach(subscription => unsubscribe(subscription));
+  },
+
+  reconnected() {
+  },
+};
+
+/**
+ * @param {string} channelName
+ * @param {Object.<string, string>} params
+ * @returns {string}
+ */
+const channelNameWithInlineParams = (channelName, params) => {
+  if (Object.keys(params).length === 0) {
+    return channelName;
+  }
+
+  return `${channelName}&${Object.keys(params).map(key => `${key}=${params[key]}`).join('&')}`;
+};
+
+/**
+ * @param {string} channelName
+ * @param {Object.<string, string>} params
+ * @param {function(Function, Function): { onConnect: (function(): void), onReceive: (function(StreamEvent): void), onDisconnect: (function(): void) }} callbacks
+ * @returns {function(): void}
+ */
+// @ts-expect-error
+export const connectStream = (channelName, params, callbacks) => (dispatch, getState) => {
+  const streamingAPIBaseURL = getState().getIn(['meta', 'streaming_api_base_url']);
+  const accessToken = getState().getIn(['meta', 'access_token']);
+  const { onConnect, onReceive, onDisconnect } = callbacks(dispatch, getState);
+
+  // If we cannot use a websockets connection, we must fall back
+  // to using individual connections for each channel
+  if (!streamingAPIBaseURL.startsWith('ws')) {
+    const connection = createConnection(streamingAPIBaseURL, accessToken, channelNameWithInlineParams(channelName, params), {
+      connected() {
+        onConnect();
+      },
+
+      received(data) {
+        onReceive(data);
+      },
+
+      disconnected() {
+        onDisconnect();
+      },
+
+      reconnected() {
+        onConnect();
+      },
+    });
+
+    return () => {
+      connection.close();
+    };
+  }
+
+  const subscription = {
+    channelName,
+    params,
+    onConnect,
+    onReceive,
+    onDisconnect,
+  };
+
+  addSubscription(subscription);
+
+  // If a connection is open, we can execute the subscription right now. Otherwise,
+  // because we have already registered it, it will be executed on connect
+
+  if (!sharedConnection) {
+    sharedConnection = /** @type {WebSocketClient} */ (createConnection(streamingAPIBaseURL, accessToken, '', sharedCallbacks));
+  } else if (sharedConnection.readyState === WebSocketClient.OPEN) {
+    subscribe(subscription);
+  }
+
+  return () => {
+    removeSubscription(subscription);
+    unsubscribe(subscription);
+  };
+};
+
+const KNOWN_EVENT_TYPES = [
+  'update',
+  'delete',
+  'notification',
+  'conversation',
+  'filters_changed',
+  'encrypted_message',
+  'announcement',
+  'announcement.delete',
+  'announcement.reaction',
+];
+
+/**
+ * @param {MessageEvent} e
+ * @param {function(StreamEvent): void} received
+ */
+const handleEventSourceMessage = (e, received) => {
+  received({
+    event: e.type,
+    payload: e.data,
+  });
+};
+
+/**
+ * @param {string} streamingAPIBaseURL
+ * @param {string} accessToken
+ * @param {string} channelName
+ * @param {{ connected: Function, received: function(StreamEvent): void, disconnected: Function, reconnected: Function }} callbacks
+ * @returns {WebSocketClient | EventSource}
+ */
+const createConnection = (streamingAPIBaseURL, accessToken, channelName, { connected, received, disconnected, reconnected }) => {
+  const params = channelName.split('&');
+
+  // @ts-expect-error
+  channelName = params.shift();
+
+  if (streamingAPIBaseURL.startsWith('ws')) {
+    // @ts-expect-error
+    const ws = new WebSocketClient(`${streamingAPIBaseURL}/api/v1/streaming/?${params.join('&')}`, accessToken);
+
+    // @ts-expect-error
+    ws.onopen = connected;
+    ws.onmessage = e => received(JSON.parse(e.data));
+    // @ts-expect-error
+    ws.onclose = disconnected;
+    // @ts-expect-error
+    ws.onreconnect = reconnected;
+
+    return ws;
+  }
+
+  channelName = channelName.replace(/:/g, '/');
+
+  if (channelName.endsWith(':media')) {
+    channelName = channelName.replace('/media', '');
+    params.push('only_media=true');
+  }
+
+  params.push(`access_token=${accessToken}`);
+
+  const es = new EventSource(`${streamingAPIBaseURL}/api/v1/streaming/${channelName}?${params.join('&')}`);
+
+  es.onopen = () => {
+    connected();
+  };
+
+  KNOWN_EVENT_TYPES.forEach(type => {
+    es.addEventListener(type, e => handleEventSourceMessage(/** @type {MessageEvent} */(e), received));
+  });
+
+  es.onerror = /** @type {function(): void} */ (disconnected);
+
+  return es;
+};
diff --git a/app/javascript/flavours/blobfox/styles/_mixins.scss b/app/javascript/flavours/blobfox/styles/_mixins.scss
new file mode 100644
index 00000000000000..6643cd1aa515da
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/_mixins.scss
@@ -0,0 +1,97 @@
+@mixin avatar-radius() {
+  border-radius: $ui-avatar-border-size;
+  background-position: 50%;
+  background-clip: padding-box;
+}
+
+@mixin avatar-size($size: 48px) {
+  width: $size;
+  height: $size;
+  background-size: $size $size;
+}
+
+@mixin single-column($media, $parent: '&') {
+  .auto-columns #{$parent} {
+    @media #{$media} {
+      @content;
+    }
+  }
+  .single-column #{$parent} {
+    @content;
+  }
+}
+
+@mixin limited-single-column($media, $parent: '&') {
+  .auto-columns #{$parent},
+  .single-column #{$parent} {
+    @media #{$media} {
+      @content;
+    }
+  }
+}
+
+@mixin multi-columns($media, $parent: '&') {
+  .auto-columns #{$parent} {
+    @media #{$media} {
+      @content;
+    }
+  }
+  .multi-columns #{$parent} {
+    @content;
+  }
+}
+
+@mixin fullwidth-gallery {
+  &.full-width {
+    margin-left: -14px;
+    margin-right: -14px;
+    width: inherit;
+    max-width: none;
+    border-radius: 0;
+  }
+}
+
+@mixin search-input() {
+  outline: 0;
+  box-sizing: border-box;
+  width: 100%;
+  border: 0;
+  box-shadow: none;
+  font-family: inherit;
+  background: $ui-base-color;
+  color: $darker-text-color;
+  border-radius: 4px;
+  font-size: 14px;
+  margin: 0;
+}
+
+@mixin search-popout() {
+  background: $simple-background-color;
+  border-radius: 4px;
+  padding: 10px 14px;
+  padding-bottom: 14px;
+  margin-top: 10px;
+  color: $light-text-color;
+  box-shadow: 2px 4px 15px rgba($base-shadow-color, 0.4);
+
+  h4 {
+    text-transform: uppercase;
+    color: $light-text-color;
+    font-size: 13px;
+    font-weight: 500;
+    margin-bottom: 10px;
+  }
+
+  li {
+    padding: 4px 0;
+  }
+
+  ul {
+    margin-bottom: 10px;
+  }
+
+  em {
+    font-weight: 500;
+    color: $inverted-text-color;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/about.scss b/app/javascript/flavours/blobfox/styles/about.scss
new file mode 100644
index 00000000000000..0f02563b48d31e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/about.scss
@@ -0,0 +1,56 @@
+$maximum-width: 1235px;
+$fluid-breakpoint: $maximum-width + 20px;
+
+.container {
+  box-sizing: border-box;
+  max-width: $maximum-width;
+  margin: 0 auto;
+  position: relative;
+
+  @media screen and (max-width: $fluid-breakpoint) {
+    width: 100%;
+    padding: 0 10px;
+  }
+}
+
+.brand {
+  position: relative;
+  text-decoration: none;
+}
+
+.rules-list {
+  font-size: 15px;
+  line-height: 22px;
+  color: $primary-text-color;
+  counter-reset: list-counter;
+
+  li {
+    position: relative;
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+    padding: 1em 1.75em;
+    padding-inline-start: 3em;
+    font-weight: 500;
+    counter-increment: list-counter;
+
+    &::before {
+      content: counter(list-counter);
+      position: absolute;
+      inset-inline-start: 0;
+      top: 50%;
+      transform: translateY(-50%);
+      background: $highlight-text-color;
+      color: $ui-base-color;
+      border-radius: 50%;
+      width: 4ch;
+      height: 4ch;
+      font-weight: 500;
+      display: flex;
+      justify-content: center;
+      align-items: center;
+    }
+
+    &:last-child {
+      border-bottom: 0;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/accessibility.scss b/app/javascript/flavours/blobfox/styles/accessibility.scss
new file mode 100644
index 00000000000000..68f4f8f1e2b3fb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/accessibility.scss
@@ -0,0 +1,55 @@
+$emojis-requiring-inversion: 'back' 'copyright' 'curly_loop' 'currency_exchange'
+  'end' 'heavy_check_mark' 'heavy_division_sign' 'heavy_dollar_sign'
+  'heavy_minus_sign' 'heavy_multiplication_x' 'heavy_plus_sign' 'on'
+  'registered' 'soon' 'spider' 'telephone_receiver' 'tm' 'top' 'wavy_dash' !default;
+
+%emoji-color-inversion {
+  filter: invert(1);
+}
+
+.emojione {
+  @each $emoji in $emojis-requiring-inversion {
+    &[title=':#{$emoji}:'] {
+      @extend %emoji-color-inversion;
+    }
+  }
+}
+
+// Display a checkmark on active UI elements otherwise differing only by color
+.status__action-bar-button,
+.detailed-status__button .icon-button {
+  position: relative;
+
+  &.active::after {
+    position: absolute;
+    content: '\F00C';
+    font-size: 50%;
+    inset-inline-end: -0.55em;
+    top: -0.44em;
+
+    /* stylelint-disable-next-line font-family-no-missing-generic-family-keyword -- this is an icon font, this can't use a generic font */
+    font-family: FontAwesome;
+  }
+}
+
+.hicolor-privacy-icons {
+  .status__visibility-icon.fa-globe,
+  .privacy-dropdown__option .fa-globe {
+    color: #1976d2;
+  }
+
+  .status__visibility-icon.fa-unlock,
+  .privacy-dropdown__option .fa-unlock {
+    color: #388e3c;
+  }
+
+  .status__visibility-icon.fa-lock,
+  .privacy-dropdown__option .fa-lock {
+    color: #ffa000;
+  }
+
+  .status__visibility-icon.fa-envelope,
+  .privacy-dropdown__option .fa-envelope {
+    color: #d32f2f;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/accounts.scss b/app/javascript/flavours/blobfox/styles/accounts.scss
new file mode 100644
index 00000000000000..2bc0150ef45a90
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/accounts.scss
@@ -0,0 +1,381 @@
+.card {
+  & > a {
+    display: block;
+    text-decoration: none;
+    color: inherit;
+    overflow: hidden;
+    border-radius: 4px;
+
+    &:hover,
+    &:active,
+    &:focus {
+      .card__bar {
+        background: lighten($ui-base-color, 8%);
+      }
+    }
+  }
+
+  &__img {
+    height: 130px;
+    position: relative;
+    background: darken($ui-base-color, 12%);
+
+    img {
+      display: block;
+      width: 100%;
+      height: 100%;
+      margin: 0;
+      object-fit: cover;
+    }
+
+    @media screen and (width <= 600px) {
+      height: 200px;
+    }
+  }
+
+  &__bar {
+    position: relative;
+    padding: 15px;
+    display: flex;
+    justify-content: flex-start;
+    align-items: center;
+    background: lighten($ui-base-color, 4%);
+
+    .avatar {
+      flex: 0 0 auto;
+      width: 48px;
+      height: 48px;
+      @include avatar-size(48px);
+
+      padding-top: 2px;
+
+      img {
+        width: 100%;
+        height: 100%;
+        display: block;
+        margin: 0;
+        border-radius: 4px;
+        @include avatar-radius;
+
+        background: darken($ui-base-color, 8%);
+        object-fit: cover;
+      }
+    }
+
+    .display-name {
+      margin-inline-start: 15px;
+      text-align: start;
+
+      i[data-hidden] {
+        display: none;
+      }
+
+      strong {
+        font-size: 15px;
+        color: $primary-text-color;
+        font-weight: 500;
+        overflow: hidden;
+        text-overflow: ellipsis;
+      }
+
+      span {
+        display: block;
+        font-size: 14px;
+        color: $darker-text-color;
+        font-weight: 400;
+        overflow: hidden;
+        text-overflow: ellipsis;
+      }
+    }
+  }
+}
+
+.pagination {
+  padding: 30px 0;
+  text-align: center;
+  overflow: hidden;
+
+  a,
+  .current,
+  .newer,
+  .older,
+  .page,
+  .gap {
+    font-size: 14px;
+    color: $primary-text-color;
+    font-weight: 500;
+    display: inline-block;
+    padding: 6px 10px;
+    text-decoration: none;
+  }
+
+  .current {
+    background: $simple-background-color;
+    border-radius: 100px;
+    color: $inverted-text-color;
+    cursor: default;
+    margin: 0 10px;
+  }
+
+  .gap {
+    cursor: default;
+  }
+
+  .older,
+  .newer {
+    text-transform: uppercase;
+    color: $secondary-text-color;
+  }
+
+  .older {
+    float: left;
+    padding-inline-start: 0;
+
+    .fa {
+      display: inline-block;
+      margin-inline-end: 5px;
+    }
+  }
+
+  .newer {
+    float: right;
+    padding-inline-start: 0;
+
+    .fa {
+      display: inline-block;
+      margin-inline-start: 5px;
+    }
+  }
+
+  .disabled {
+    cursor: default;
+    color: lighten($inverted-text-color, 10%);
+  }
+
+  @media screen and (width <= 700px) {
+    padding: 30px 20px;
+
+    .page {
+      display: none;
+    }
+
+    .newer,
+    .older {
+      display: inline-block;
+    }
+  }
+}
+
+.nothing-here {
+  background: $ui-base-color;
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+  color: $light-text-color;
+  font-size: 14px;
+  font-weight: 500;
+  text-align: center;
+  display: flex;
+  justify-content: center;
+  align-items: center;
+  cursor: default;
+  border-radius: 4px;
+  padding: 20px;
+  min-height: 30vh;
+
+  &--under-tabs {
+    border-radius: 0 0 4px 4px;
+  }
+
+  &--flexible {
+    box-sizing: border-box;
+    min-height: 100%;
+  }
+}
+
+.account-role,
+.information-badge,
+.simple_form .overridden,
+.simple_form .recommended,
+.simple_form .not_recommended,
+.simple_form .blobfox_only {
+  display: inline-block;
+  padding: 4px 6px;
+  cursor: default;
+  border-radius: 3px;
+  font-size: 12px;
+  line-height: 12px;
+  font-weight: 500;
+  color: $ui-secondary-color;
+  background-color: var(--user-role-background, rgba($ui-secondary-color, 0.1));
+  border: 1px solid var(--user-role-border, rgba($ui-secondary-color, 0.5));
+
+  &.moderator {
+    color: $success-green;
+    background-color: rgba($success-green, 0.1);
+    border-color: rgba($success-green, 0.5);
+  }
+
+  &.admin {
+    color: lighten($error-red, 12%);
+    background-color: rgba(lighten($error-red, 12%), 0.1);
+    border-color: rgba(lighten($error-red, 12%), 0.5);
+  }
+}
+
+.simple_form .not_recommended {
+  color: lighten($error-red, 12%);
+  background-color: rgba(lighten($error-red, 12%), 0.1);
+  border-color: rgba(lighten($error-red, 12%), 0.5);
+}
+
+.simple_form .blobfox_only {
+  color: lighten($warning-red, 12%);
+  background-color: rgba(lighten($warning-red, 12%), 0.1);
+  border-color: rgba(lighten($warning-red, 12%), 0.5);
+}
+
+.account__header__fields {
+  max-width: 100vw;
+  padding: 0;
+  margin: 15px -15px -15px;
+  border: 0 none;
+  border-top: 1px solid lighten($ui-base-color, 12%);
+  border-bottom: 1px solid lighten($ui-base-color, 12%);
+  font-size: 14px;
+  line-height: 20px;
+
+  dl {
+    display: flex;
+    border-bottom: 1px solid lighten($ui-base-color, 12%);
+  }
+
+  dt,
+  dd {
+    box-sizing: border-box;
+    padding: 14px;
+    text-align: center;
+    max-height: 48px;
+    overflow: hidden;
+    white-space: nowrap;
+    text-overflow: ellipsis;
+  }
+
+  dt {
+    font-weight: 500;
+    width: 120px;
+    flex: 0 0 auto;
+    color: $secondary-text-color;
+    background: rgba(darken($ui-base-color, 8%), 0.5);
+  }
+
+  dd {
+    flex: 1 1 auto;
+    color: $darker-text-color;
+  }
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: none;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: underline;
+    }
+  }
+
+  .verified {
+    border: 1px solid rgba($valid-value-color, 0.5);
+    background: rgba($valid-value-color, 0.25);
+
+    a {
+      color: $valid-value-color;
+      font-weight: 500;
+    }
+
+    &__mark {
+      color: $valid-value-color;
+    }
+  }
+
+  dl:last-child {
+    border-bottom: 0;
+  }
+}
+
+.directory__tag .trends__item__current {
+  width: auto;
+}
+
+.pending-account {
+  &__header {
+    color: $darker-text-color;
+
+    a {
+      color: $ui-secondary-color;
+      text-decoration: none;
+
+      &:hover,
+      &:active,
+      &:focus {
+        text-decoration: underline;
+      }
+    }
+
+    strong {
+      color: $primary-text-color;
+      font-weight: 700;
+    }
+  }
+
+  &__body {
+    margin-top: 10px;
+  }
+}
+
+.batch-table__row--muted {
+  color: lighten($ui-base-color, 26%);
+}
+
+.batch-table__row--muted .pending-account__header,
+.batch-table__row--muted .accounts-table,
+.batch-table__row--muted .name-tag {
+  &,
+  a,
+  strong {
+    color: lighten($ui-base-color, 26%);
+  }
+}
+
+.batch-table__row--muted .name-tag .avatar {
+  opacity: 0.5;
+}
+
+.batch-table__row--muted .accounts-table {
+  tbody td.accounts-table__extra,
+  &__count,
+  &__count small {
+    color: lighten($ui-base-color, 26%);
+  }
+}
+
+.batch-table__row--attention {
+  color: $gold-star;
+}
+
+.batch-table__row--attention .pending-account__header,
+.batch-table__row--attention .accounts-table,
+.batch-table__row--attention .name-tag {
+  &,
+  a,
+  strong {
+    color: $gold-star;
+  }
+}
+
+.batch-table__row--attention .accounts-table {
+  tbody td.accounts-table__extra,
+  &__count,
+  &__count small {
+    color: $gold-star;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/admin.scss b/app/javascript/flavours/blobfox/styles/admin.scss
new file mode 100644
index 00000000000000..2f4027b03fb6ea
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/admin.scss
@@ -0,0 +1,1884 @@
+@use 'sass:math';
+
+$no-columns-breakpoint: 600px;
+$sidebar-width: 240px;
+$content-width: 840px;
+
+.admin-wrapper {
+  display: flex;
+  justify-content: center;
+  width: 100%;
+  min-height: 100vh;
+
+  .sidebar-wrapper {
+    min-height: 100vh;
+    overflow: hidden;
+    pointer-events: none;
+    flex: 1 1 auto;
+
+    &__inner {
+      display: flex;
+      justify-content: flex-end;
+      background: $ui-base-color;
+      height: 100%;
+    }
+  }
+
+  .sidebar {
+    width: $sidebar-width;
+    padding: 0;
+    pointer-events: auto;
+
+    &__toggle {
+      display: none;
+      background: darken($ui-base-color, 4%);
+      border-bottom: 1px solid lighten($ui-base-color, 4%);
+      align-items: center;
+
+      &__logo {
+        flex: 1 1 auto;
+
+        a {
+          display: block;
+          padding: 15px;
+        }
+      }
+
+      &__icon {
+        display: block;
+        color: $darker-text-color;
+        text-decoration: none;
+        flex: 0 0 auto;
+        font-size: 18px;
+        padding: 10px;
+        margin: 5px 10px;
+        border-radius: 4px;
+
+        &:focus {
+          background: $ui-base-color;
+        }
+
+        .fa-times {
+          display: none;
+        }
+
+        &.active {
+          .fa-times {
+            display: block;
+          }
+
+          .fa-bars {
+            display: none;
+          }
+        }
+      }
+    }
+
+    .logo {
+      display: block;
+      margin: 40px auto;
+      width: 100px;
+      height: 100px;
+    }
+
+    .logo--wordmark {
+      display: inherit;
+      margin: inherit;
+      width: inherit;
+      height: 25px;
+    }
+
+    @media screen and (max-width: $no-columns-breakpoint) {
+      & > a:first-child {
+        display: none;
+      }
+    }
+
+    ul {
+      list-style: none;
+      border-radius: 4px 0 0 4px;
+      overflow: hidden;
+      margin-bottom: 20px;
+
+      @media screen and (max-width: $no-columns-breakpoint) {
+        margin-bottom: 0;
+      }
+
+      a {
+        display: block;
+        padding: 15px;
+        color: $darker-text-color;
+        text-decoration: none;
+        transition: all 200ms linear;
+        transition-property: color, background-color;
+        border-radius: 4px 0 0 4px;
+        white-space: nowrap;
+        overflow: hidden;
+        text-overflow: ellipsis;
+
+        i.fa {
+          margin-inline-end: 5px;
+        }
+
+        &:hover {
+          color: $primary-text-color;
+          background-color: darken($ui-base-color, 5%);
+          transition: all 100ms linear;
+          transition-property: color, background-color;
+        }
+
+        &.selected {
+          border-radius: 4px 0 0;
+        }
+      }
+
+      ul {
+        background: darken($ui-base-color, 4%);
+        border-radius: 0 0 0 4px;
+        margin: 0;
+
+        a {
+          border: 0;
+          padding: 15px 35px;
+        }
+      }
+
+      .warning a {
+        color: $gold-star;
+        font-weight: 700;
+      }
+
+      .simple-navigation-active-leaf a {
+        color: $primary-text-color;
+        background-color: $ui-highlight-color;
+        border-bottom: 0;
+        border-radius: 0;
+      }
+    }
+
+    & > ul > .simple-navigation-active-leaf a {
+      border-radius: 4px 0 0 4px;
+    }
+  }
+
+  .content-wrapper {
+    box-sizing: border-box;
+    width: 100%;
+    max-width: $content-width;
+    flex: 1 1 auto;
+  }
+
+  @media screen and (max-width: $content-width + $sidebar-width) {
+    .sidebar-wrapper--empty {
+      display: none;
+    }
+
+    .sidebar-wrapper {
+      width: $sidebar-width;
+      flex: 0 0 auto;
+    }
+  }
+
+  @media screen and (max-width: $no-columns-breakpoint) {
+    .sidebar-wrapper {
+      width: 100%;
+    }
+  }
+
+  .content {
+    padding-top: 55px;
+    padding-bottom: 20px;
+    padding-inline-start: 25px;
+    padding-inline-end: 15px;
+
+    @media screen and (max-width: $no-columns-breakpoint) {
+      max-width: none;
+      padding: 15px;
+      padding-top: 30px;
+    }
+
+    &__heading {
+      margin-bottom: 45px;
+
+      &__row {
+        display: flex;
+        flex-wrap: wrap;
+        align-items: center;
+        justify-content: space-between;
+        margin-top: -15px;
+        margin-inline-end: -15px;
+
+        & > * {
+          margin-top: 15px;
+          margin-inline-end: 15px;
+        }
+      }
+
+      &__tabs {
+        margin-top: 30px;
+        width: 100%;
+
+        & > div {
+          display: flex;
+          flex-wrap: wrap;
+          gap: 5px;
+        }
+
+        a {
+          font-size: 14px;
+          display: inline-flex;
+          align-items: center;
+          padding: 7px 10px;
+          border-radius: 4px;
+          color: $darker-text-color;
+          text-decoration: none;
+          font-weight: 500;
+          gap: 5px;
+          white-space: nowrap;
+
+          &:hover,
+          &:focus,
+          &:active {
+            background: lighten($ui-base-color, 4%);
+          }
+
+          &.selected {
+            font-weight: 700;
+            color: $primary-text-color;
+            background: $ui-highlight-color;
+          }
+        }
+      }
+
+      &__actions {
+        display: inline-flex;
+        flex-flow: wrap;
+        gap: 5px;
+      }
+
+      h2 small {
+        font-size: 12px;
+        display: block;
+        font-weight: 500;
+        color: $darker-text-color;
+        line-height: 18px;
+      }
+
+      @media screen and (max-width: $no-columns-breakpoint) {
+        border-bottom: 0;
+        padding-bottom: 0;
+      }
+    }
+
+    h2 {
+      color: $secondary-text-color;
+      font-size: 24px;
+      line-height: 36px;
+      font-weight: 700;
+    }
+
+    h3 {
+      color: $secondary-text-color;
+      font-size: 20px;
+      line-height: 28px;
+      font-weight: 400;
+      margin-bottom: 30px;
+    }
+
+    h4 {
+      text-transform: uppercase;
+      font-size: 13px;
+      font-weight: 700;
+      color: $darker-text-color;
+      padding-bottom: 8px;
+      margin-bottom: 8px;
+      border-bottom: 1px solid lighten($ui-base-color, 8%);
+    }
+
+    h6 {
+      font-size: 16px;
+      color: $secondary-text-color;
+      line-height: 28px;
+      font-weight: 500;
+    }
+
+    .fields-group h6 {
+      color: $primary-text-color;
+      font-weight: 500;
+    }
+
+    .directory__tag > a,
+    .directory__tag > div {
+      box-shadow: none;
+    }
+
+    .directory__tag .table-action-link .fa {
+      color: inherit;
+    }
+
+    .directory__tag h4 {
+      font-size: 18px;
+      font-weight: 700;
+      color: $primary-text-color;
+      text-transform: none;
+      padding-bottom: 0;
+      margin-bottom: 0;
+      border-bottom: 0;
+    }
+
+    & > p {
+      font-size: 14px;
+      line-height: 21px;
+      color: $secondary-text-color;
+      margin-bottom: 20px;
+
+      strong {
+        color: $primary-text-color;
+        font-weight: 500;
+
+        @each $lang in $cjk-langs {
+          &:lang(#{$lang}) {
+            font-weight: 700;
+          }
+        }
+      }
+    }
+
+    hr {
+      width: 100%;
+      height: 0;
+      border: 0;
+      border-bottom: 1px solid rgba($ui-base-lighter-color, 0.6);
+      margin: 20px 0;
+
+      &.spacer {
+        height: 1px;
+        border: 0;
+      }
+    }
+  }
+
+  @media screen and (max-width: $no-columns-breakpoint) {
+    display: block;
+
+    .sidebar-wrapper {
+      min-height: 0;
+    }
+
+    .sidebar {
+      width: 100%;
+      padding: 0;
+      height: auto;
+
+      &__toggle {
+        display: flex;
+      }
+
+      & > ul {
+        display: none;
+
+        &.visible {
+          display: block;
+          position: fixed;
+          z-index: 10;
+          width: 100%;
+          height: calc(100% - 56px);
+          inset-inline-start: 0;
+          bottom: 0;
+          overflow-y: auto;
+          background: $ui-base-color;
+        }
+      }
+
+      ul a,
+      ul ul a {
+        border-radius: 0;
+        border-bottom: 1px solid lighten($ui-base-color, 4%);
+        transition: none;
+
+        &:hover {
+          transition: none;
+        }
+      }
+
+      ul ul {
+        border-radius: 0;
+      }
+
+      ul .simple-navigation-active-leaf a {
+        border-bottom-color: $ui-highlight-color;
+      }
+    }
+  }
+}
+
+hr.spacer {
+  width: 100%;
+  border: 0;
+  margin: 20px 0;
+  height: 1px;
+}
+
+body,
+.admin-wrapper .content {
+  .muted-hint {
+    color: $darker-text-color;
+
+    a {
+      color: $highlight-text-color;
+    }
+  }
+
+  .positive-hint,
+  .negative-hint,
+  .neutral-hint {
+    a {
+      color: inherit;
+      text-decoration: underline;
+
+      &:focus,
+      &:hover,
+      &:active {
+        text-decoration: none;
+      }
+    }
+  }
+
+  .positive-hint {
+    color: $valid-value-color;
+    font-weight: 500;
+  }
+
+  .negative-hint {
+    color: $error-value-color;
+    font-weight: 500;
+  }
+
+  .neutral-hint {
+    color: $dark-text-color;
+    font-weight: 500;
+  }
+
+  .warning-hint {
+    color: $gold-star;
+    font-weight: 500;
+  }
+}
+
+.filters {
+  display: flex;
+  flex-wrap: wrap;
+  gap: 40px;
+
+  .filter-subset {
+    flex: 0 0 auto;
+    margin-bottom: 20px;
+
+    &:last-child {
+      margin-bottom: 30px;
+    }
+
+    ul {
+      margin-top: 5px;
+      list-style: none;
+
+      li {
+        display: inline-block;
+        margin-inline-end: 5px;
+      }
+    }
+
+    & > div {
+      display: flex;
+      gap: 5px;
+    }
+
+    strong {
+      font-weight: 500;
+      text-transform: uppercase;
+      font-size: 12px;
+
+      @each $lang in $cjk-langs {
+        &:lang(#{$lang}) {
+          font-weight: 700;
+        }
+      }
+    }
+
+    &--with-select strong {
+      display: block;
+      margin-bottom: 10px;
+    }
+
+    a {
+      display: inline-block;
+      color: $darker-text-color;
+      text-decoration: none;
+      text-transform: uppercase;
+      font-size: 12px;
+      font-weight: 500;
+      border-bottom: 2px solid $ui-base-color;
+
+      &:hover {
+        color: $primary-text-color;
+        border-bottom: 2px solid lighten($ui-base-color, 5%);
+      }
+
+      &.selected {
+        color: $highlight-text-color;
+        border-bottom: 2px solid $ui-highlight-color;
+      }
+    }
+  }
+}
+
+.flavour-screen {
+  display: block;
+  margin: 10px auto;
+  max-width: 100%;
+}
+
+.flavour-description {
+  display: block;
+  font-size: 16px;
+  margin: 10px 0;
+
+  & > p {
+    margin: 10px 0;
+  }
+}
+
+.report-accounts {
+  display: flex;
+  flex-wrap: wrap;
+  margin-bottom: 20px;
+}
+
+.report-accounts__item {
+  display: flex;
+  flex: 250px;
+  flex-direction: column;
+  margin: 0 5px;
+
+  & > strong {
+    display: block;
+    margin-top: 0;
+    margin-bottom: 10px;
+    margin-inline-end: 0;
+    margin-inline-start: -5px;
+    font-weight: 500;
+    font-size: 14px;
+    line-height: 18px;
+    color: $secondary-text-color;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  .account-card {
+    flex: 1 1 auto;
+  }
+}
+
+.report-status,
+.account-status {
+  display: flex;
+  margin-bottom: 10px;
+
+  .activity-stream {
+    flex: 2 0 0;
+    margin-inline-end: 20px;
+    max-width: calc(100% - 60px);
+
+    .entry {
+      border-radius: 4px;
+    }
+  }
+}
+
+.report-status__actions,
+.account-status__actions {
+  flex: 0 0 auto;
+  display: flex;
+  flex-direction: column;
+
+  .icon-button {
+    font-size: 24px;
+    width: 24px;
+    text-align: center;
+    margin-bottom: 10px;
+  }
+}
+
+.simple_form.new_report_note,
+.simple_form.new_account_moderation_note {
+  max-width: 100%;
+}
+
+.simple_form {
+  .actions {
+    margin-top: 15px;
+  }
+
+  .button {
+    font-size: 15px;
+  }
+}
+
+.batch-form-box {
+  display: flex;
+  flex-wrap: wrap;
+  margin-bottom: 5px;
+
+  #form_status_batch_action {
+    margin-bottom: 5px;
+    margin-inline-end: 5px;
+    font-size: 14px;
+  }
+
+  input.button {
+    margin-bottom: 5px;
+    margin-inline-end: 5px;
+  }
+
+  .media-spoiler-toggle-buttons {
+    margin-inline-start: auto;
+
+    .button {
+      overflow: visible;
+      margin-bottom: 5px;
+      margin-inline-start: 5px;
+      float: right;
+    }
+  }
+}
+
+.back-link {
+  margin-bottom: 10px;
+  font-size: 14px;
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+  }
+}
+
+.special-action-button,
+.back-link {
+  text-align: end;
+  flex: 1 1 auto;
+}
+
+.action-buttons {
+  display: flex;
+  overflow: hidden;
+  justify-content: space-between;
+}
+
+.spacer {
+  flex: 1 1 auto;
+}
+
+.log-entry {
+  display: block;
+  line-height: 20px;
+  padding: 15px;
+  padding-inline-start: 15px * 2 + 40px;
+  background: $ui-base-color;
+  border-bottom: 1px solid darken($ui-base-color, 8%);
+  position: relative;
+  text-decoration: none;
+  color: $darker-text-color;
+  font-size: 14px;
+
+  &:first-child {
+    border-top-left-radius: 4px;
+    border-top-right-radius: 4px;
+  }
+
+  &:last-child {
+    border-bottom-left-radius: 4px;
+    border-bottom-right-radius: 4px;
+    border-bottom: 0;
+  }
+
+  &:hover,
+  &:focus,
+  &:active {
+    background: lighten($ui-base-color, 4%);
+  }
+
+  &__avatar {
+    position: absolute;
+    inset-inline-start: 15px;
+    top: 15px;
+
+    .avatar {
+      border-radius: 4px;
+      width: 40px;
+      height: 40px;
+    }
+  }
+
+  &__title {
+    word-wrap: break-word;
+  }
+
+  &__timestamp {
+    color: $dark-text-color;
+  }
+
+  a,
+  .username,
+  .target {
+    color: $secondary-text-color;
+    text-decoration: none;
+    font-weight: 500;
+  }
+
+  a {
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: underline;
+    }
+  }
+}
+
+a.name-tag,
+.name-tag,
+a.inline-name-tag,
+.inline-name-tag {
+  text-decoration: none;
+  color: $secondary-text-color;
+
+  .username {
+    font-weight: 500;
+  }
+
+  &.suspended {
+    .username {
+      text-decoration: line-through;
+      color: lighten($error-red, 12%);
+    }
+
+    .avatar {
+      filter: grayscale(100%);
+      opacity: 0.8;
+    }
+  }
+}
+
+a.name-tag,
+.name-tag {
+  display: inline-flex;
+  align-items: center;
+  vertical-align: top;
+
+  .avatar {
+    display: block;
+    margin: 0;
+    margin-inline-end: 5px;
+    border-radius: 50%;
+  }
+
+  &.suspended {
+    .avatar {
+      filter: grayscale(100%);
+      opacity: 0.8;
+    }
+  }
+}
+
+.speech-bubble {
+  margin-bottom: 20px;
+  border-inline-start: 4px solid $ui-highlight-color;
+
+  &.positive {
+    border-left-color: $success-green;
+  }
+
+  &.negative {
+    border-left-color: lighten($error-red, 12%);
+  }
+
+  &.warning {
+    border-left-color: $gold-star;
+  }
+
+  &__bubble {
+    padding: 16px;
+    padding-inline-start: 14px;
+    font-size: 15px;
+    line-height: 20px;
+    border-radius: 4px 4px 4px 0;
+    position: relative;
+    font-weight: 500;
+
+    a {
+      color: $darker-text-color;
+    }
+  }
+
+  &__owner {
+    padding: 8px;
+    padding-inline-start: 12px;
+  }
+
+  time {
+    color: $dark-text-color;
+  }
+}
+
+.report-card {
+  background: $ui-base-color;
+  border-radius: 4px;
+  margin-bottom: 20px;
+
+  &__profile {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    padding: 15px;
+
+    .account {
+      padding: 0;
+      border: 0;
+
+      &__avatar-wrapper {
+        margin-inline-start: 0;
+      }
+    }
+
+    &__stats {
+      flex: 0 0 auto;
+      font-weight: 500;
+      color: $darker-text-color;
+      text-transform: uppercase;
+      text-align: end;
+
+      a {
+        color: inherit;
+        text-decoration: none;
+
+        &:focus,
+        &:hover,
+        &:active {
+          color: lighten($darker-text-color, 8%);
+        }
+      }
+
+      .red {
+        color: $error-value-color;
+      }
+    }
+  }
+
+  &__summary {
+    &__item {
+      display: flex;
+      justify-content: flex-start;
+      border-top: 1px solid darken($ui-base-color, 4%);
+
+      &:hover {
+        background: lighten($ui-base-color, 2%);
+      }
+
+      &__reported-by,
+      &__assigned {
+        padding: 15px;
+        flex: 0 0 auto;
+        box-sizing: border-box;
+        width: 150px;
+        color: $darker-text-color;
+
+        &,
+        .username {
+          white-space: nowrap;
+          overflow: hidden;
+          text-overflow: ellipsis;
+        }
+      }
+
+      &__content {
+        flex: 1 1 auto;
+        max-width: calc(100% - 300px);
+
+        &__icon {
+          color: $dark-text-color;
+          margin-inline-end: 4px;
+          font-weight: 500;
+        }
+      }
+
+      &__content a {
+        display: block;
+        box-sizing: border-box;
+        width: 100%;
+        padding: 15px;
+        text-decoration: none;
+        color: $darker-text-color;
+      }
+    }
+  }
+}
+
+.one-line {
+  white-space: nowrap;
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
+
+.ellipsized-ip {
+  display: inline-block;
+  max-width: 120px;
+  overflow: hidden;
+  text-overflow: ellipsis;
+  vertical-align: middle;
+}
+
+.admin-account-bio {
+  display: flex;
+  flex-wrap: wrap;
+  margin: 0 -5px;
+  margin-top: 20px;
+
+  > div {
+    box-sizing: border-box;
+    padding: 0 5px;
+    margin-bottom: 10px;
+    flex: 1 0 50%;
+    max-width: 100%;
+  }
+
+  .account__header__fields,
+  .account__header__content {
+    background: lighten($ui-base-color, 8%);
+    border-radius: 4px;
+    height: 100%;
+  }
+
+  .account__header__fields {
+    margin: 0;
+    border: 0;
+
+    a {
+      color: $highlight-text-color;
+    }
+
+    dl:first-child .verified {
+      border-radius: 0 4px 0 0;
+    }
+
+    .verified a {
+      color: $valid-value-color;
+    }
+  }
+
+  .account__header__content {
+    box-sizing: border-box;
+    padding: 20px;
+    color: $primary-text-color;
+  }
+}
+
+.center-text {
+  text-align: center;
+}
+
+.applications-list__item,
+.filters-list__item {
+  padding: 15px 0;
+  background: $ui-base-color;
+  border: 1px solid lighten($ui-base-color, 4%);
+  border-radius: 4px;
+  margin-top: 15px;
+}
+
+.user-role {
+  color: var(--user-role-accent);
+}
+
+.announcements-list,
+.filters-list {
+  border: 1px solid lighten($ui-base-color, 4%);
+  border-radius: 4px;
+
+  &__item {
+    padding: 15px 0;
+    background: $ui-base-color;
+    border-bottom: 1px solid lighten($ui-base-color, 4%);
+
+    &__title {
+      padding: 0 15px;
+      display: block;
+      font-weight: 500;
+      font-size: 18px;
+      line-height: 1.5;
+      color: $secondary-text-color;
+      text-decoration: none;
+      margin-bottom: 10px;
+
+      .account-role {
+        vertical-align: middle;
+      }
+    }
+
+    a.announcements-list__item__title {
+      &:hover,
+      &:focus,
+      &:active {
+        color: $primary-text-color;
+      }
+    }
+
+    &__meta {
+      padding: 0 15px;
+      color: $dark-text-color;
+
+      a {
+        color: inherit;
+        text-decoration: underline;
+
+        &:hover,
+        &:focus,
+        &:active {
+          text-decoration: none;
+        }
+      }
+    }
+
+    &__action-bar {
+      display: flex;
+      justify-content: space-between;
+      align-items: center;
+    }
+
+    &__permissions {
+      margin-top: 10px;
+    }
+
+    &:last-child {
+      border-bottom: 0;
+    }
+  }
+}
+
+.filters-list__item {
+  &__title {
+    display: flex;
+    justify-content: space-between;
+    margin-bottom: 0;
+  }
+
+  &__permissions {
+    margin-top: 0;
+    margin-bottom: 10px;
+  }
+
+  .expiration {
+    font-size: 13px;
+  }
+
+  &.expired {
+    .expiration {
+      color: lighten($error-red, 12%);
+    }
+
+    .permissions-list__item__icon {
+      color: $dark-text-color;
+    }
+  }
+}
+
+.dashboard__counters.admin-account-counters {
+  margin-top: 10px;
+}
+
+.account-badges {
+  margin: -2px 0;
+}
+
+.retention {
+  overflow: auto;
+
+  > h4 {
+    position: sticky;
+    inset-inline-start: 0;
+  }
+
+  &__table {
+    &__number {
+      color: $secondary-text-color;
+      padding: 10px;
+    }
+
+    &__date {
+      white-space: nowrap;
+      padding: 10px 0;
+      text-align: start;
+      min-width: 120px;
+
+      &.retention__table__average {
+        font-weight: 700;
+      }
+    }
+
+    &__size {
+      text-align: center;
+      padding: 10px;
+    }
+
+    &__label {
+      font-weight: 700;
+      color: $darker-text-color;
+    }
+
+    &__box {
+      box-sizing: border-box;
+      background: $ui-highlight-color;
+      padding: 10px;
+      font-weight: 500;
+      color: $primary-text-color;
+      width: 52px;
+      margin: 1px;
+
+      @for $i from 0 through 10 {
+        &--#{10 * $i} {
+          background-color: rgba(
+            $ui-highlight-color,
+            1 * (math.div(max(1, $i), 10))
+          );
+        }
+      }
+    }
+  }
+}
+
+.sparkline {
+  display: block;
+  text-decoration: none;
+  background: lighten($ui-base-color, 4%);
+  border-radius: 4px;
+  padding: 0;
+  position: relative;
+  padding-bottom: 55px + 20px;
+  overflow: hidden;
+
+  &__value {
+    display: flex;
+    line-height: 33px;
+    align-items: flex-end;
+    padding: 20px;
+    padding-bottom: 10px;
+
+    &__total {
+      display: block;
+      margin-inline-end: 10px;
+      font-weight: 500;
+      font-size: 28px;
+      color: $primary-text-color;
+    }
+
+    &__change {
+      display: block;
+      font-weight: 500;
+      font-size: 18px;
+      color: $darker-text-color;
+      margin-bottom: -3px;
+
+      &.positive {
+        color: $valid-value-color;
+      }
+
+      &.negative {
+        color: $error-value-color;
+      }
+    }
+  }
+
+  &__label {
+    padding: 0 20px;
+    padding-bottom: 10px;
+    text-transform: uppercase;
+    color: $darker-text-color;
+    font-weight: 500;
+  }
+
+  &__graph {
+    position: absolute;
+    bottom: 0;
+    width: 100%;
+
+    svg {
+      display: block;
+      margin: 0;
+    }
+
+    path:first-child {
+      fill: rgba($highlight-text-color, 0.25) !important;
+      fill-opacity: 1 !important;
+    }
+
+    path:last-child {
+      stroke: lighten($highlight-text-color, 6%) !important;
+      fill: none !important;
+    }
+  }
+}
+
+a.sparkline {
+  &:hover,
+  &:focus,
+  &:active {
+    background: lighten($ui-base-color, 6%);
+  }
+}
+
+.skeleton {
+  background-color: lighten($ui-base-color, 8%);
+  background-image: linear-gradient(
+    90deg,
+    lighten($ui-base-color, 8%),
+    lighten($ui-base-color, 12%),
+    lighten($ui-base-color, 8%)
+  );
+  background-size: 200px 100%;
+  background-repeat: no-repeat;
+  border-radius: 4px;
+  display: inline-block;
+  line-height: 1;
+  width: 100%;
+  animation: skeleton 1.2s ease-in-out infinite;
+}
+
+@keyframes skeleton {
+  0% {
+    background-position: -200px 0;
+  }
+
+  100% {
+    background-position: calc(200px + 100%) 0;
+  }
+}
+
+.dimension {
+  table {
+    width: 100%;
+  }
+
+  &__item {
+    border-bottom: 1px solid lighten($ui-base-color, 4%);
+
+    &__key {
+      font-weight: 500;
+      padding: 11px 10px;
+    }
+
+    &__value {
+      text-align: end;
+      color: $darker-text-color;
+      padding: 11px 10px;
+    }
+
+    &__indicator {
+      display: inline-block;
+      width: 8px;
+      height: 8px;
+      border-radius: 50%;
+      background: $ui-highlight-color;
+      margin-inline-end: 10px;
+
+      @for $i from 0 through 10 {
+        &--#{10 * $i} {
+          background-color: rgba(
+            $ui-highlight-color,
+            1 * (math.div(max(1, $i), 10))
+          );
+        }
+      }
+    }
+
+    &:last-child {
+      border-bottom: 0;
+    }
+
+    &.negative {
+      color: $error-value-color;
+      font-weight: 700;
+
+      .dimension__item__value {
+        color: $error-value-color;
+      }
+    }
+  }
+}
+
+.report-reason-selector {
+  border-radius: 4px;
+  background: $ui-base-color;
+  margin-bottom: 20px;
+
+  &__category {
+    cursor: pointer;
+    border-bottom: 1px solid darken($ui-base-color, 8%);
+
+    &:last-child {
+      border-bottom: 0;
+    }
+
+    &__label {
+      padding: 15px;
+    }
+
+    &__rules {
+      margin-inline-start: 30px;
+    }
+  }
+
+  &__rule {
+    cursor: pointer;
+    padding: 15px;
+  }
+}
+
+.report-header {
+  display: grid;
+  grid-gap: 15px;
+  grid-template-columns: minmax(0, 1fr) 300px;
+
+  &__details {
+    &__item {
+      border-bottom: 1px solid lighten($ui-base-color, 8%);
+      padding: 15px 0;
+
+      &:last-child {
+        border-bottom: 0;
+      }
+
+      &__header {
+        font-weight: 600;
+        padding: 4px 0;
+      }
+    }
+
+    &--horizontal {
+      display: grid;
+      grid-auto-columns: minmax(0, 1fr);
+      grid-auto-flow: column;
+
+      .report-header__details__item {
+        border-bottom: 0;
+      }
+    }
+  }
+
+  @media screen and (width <= 930px) {
+    grid-template-columns: minmax(0, 1fr);
+  }
+}
+
+.account-card {
+  background: $ui-base-color;
+  border-radius: 4px;
+
+  &__permalink {
+    color: inherit;
+    text-decoration: none;
+  }
+
+  &__header {
+    padding: 4px;
+    border-radius: 4px;
+    height: 128px;
+
+    img {
+      display: block;
+      margin: 0;
+      width: 100%;
+      height: 100%;
+      object-fit: cover;
+      background: darken($ui-base-color, 8%);
+    }
+  }
+
+  &__title {
+    margin-top: -(15px + 8px);
+    display: flex;
+    align-items: flex-end;
+
+    &__avatar {
+      padding: 14px;
+
+      img,
+      .account__avatar {
+        display: block;
+        margin: 0;
+        width: 56px;
+        height: 56px;
+        background-color: darken($ui-base-color, 8%);
+        border-radius: 8px;
+        border: 1px solid $ui-base-color;
+      }
+    }
+
+    .display-name {
+      color: $darker-text-color;
+      padding-bottom: 15px;
+      font-size: 15px;
+      line-height: 20px;
+
+      bdi {
+        display: block;
+        color: $primary-text-color;
+        font-weight: 700;
+      }
+    }
+  }
+
+  &__bio {
+    padding: 0 15px;
+    margin: 8px 0;
+    overflow: hidden;
+    text-overflow: ellipsis;
+    word-wrap: break-word;
+    max-height: 21px * 2;
+    position: relative;
+    font-size: 15px;
+    line-height: 21px;
+
+    &::after {
+      display: block;
+      content: '';
+      width: 50px;
+      height: 21px;
+      position: absolute;
+      bottom: 0;
+      inset-inline-end: 15px;
+      background: linear-gradient(to left, $ui-base-color, transparent);
+      pointer-events: none;
+    }
+
+    a {
+      color: $secondary-text-color;
+      text-decoration: none;
+      unicode-bidi: isolate;
+
+      &:hover {
+        text-decoration: underline;
+      }
+
+      &.mention {
+        &:hover {
+          text-decoration: none;
+
+          span {
+            text-decoration: underline;
+          }
+        }
+      }
+    }
+  }
+
+  &__actions {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+
+    &__button {
+      flex-shrink: 1;
+      padding: 0 15px;
+      overflow: hidden;
+
+      .button {
+        min-width: 0;
+        white-space: nowrap;
+        text-overflow: ellipsis;
+        overflow: hidden;
+        max-width: 100%;
+      }
+    }
+  }
+
+  &__counters {
+    flex: 1 1 auto;
+    display: grid;
+    grid-auto-columns: minmax(0, 1fr);
+    grid-auto-flow: column;
+    max-width: 340px;
+    min-width: 65px * 3;
+
+    &__item {
+      padding: 15px 0;
+      text-align: center;
+      color: $primary-text-color;
+      font-weight: 600;
+      font-size: 15px;
+      line-height: 21px;
+
+      small {
+        display: block;
+        color: $darker-text-color;
+        font-weight: 400;
+        font-size: 13px;
+        line-height: 18px;
+      }
+    }
+  }
+}
+
+.report-notes {
+  margin-bottom: 20px;
+
+  &__item {
+    background: $ui-base-color;
+    position: relative;
+    padding: 15px;
+    padding-inline-start: 15px * 2 + 40px;
+    border-bottom: 1px solid darken($ui-base-color, 8%);
+
+    &:first-child {
+      border-top-left-radius: 4px;
+      border-top-right-radius: 4px;
+    }
+
+    &:last-child {
+      border-bottom-left-radius: 4px;
+      border-bottom-right-radius: 4px;
+      border-bottom: 0;
+    }
+
+    &:hover {
+      background-color: lighten($ui-base-color, 4%);
+    }
+
+    &__avatar {
+      position: absolute;
+      inset-inline-start: 15px;
+      top: 15px;
+      border-radius: 4px;
+      width: 40px;
+      height: 40px;
+    }
+
+    &__header {
+      color: $darker-text-color;
+      font-size: 15px;
+      line-height: 20px;
+      margin-bottom: 4px;
+
+      .username {
+        color: $primary-text-color;
+        font-weight: 500;
+        margin-inline-end: 5px;
+
+        a {
+          color: inherit;
+          text-decoration: none;
+
+          &:hover,
+          &:focus,
+          &:active {
+            text-decoration: underline;
+          }
+        }
+      }
+
+      time {
+        margin-inline-start: 5px;
+        vertical-align: baseline;
+      }
+    }
+
+    &__content {
+      font-size: 15px;
+      line-height: 20px;
+      word-wrap: break-word;
+      font-weight: 400;
+      color: $primary-text-color;
+
+      p {
+        margin-bottom: 20px;
+        white-space: pre-wrap;
+        unicode-bidi: plaintext;
+
+        &:last-child {
+          margin-bottom: 0;
+        }
+      }
+
+      a {
+        color: $highlight-text-color;
+        text-decoration: none;
+
+        &:hover {
+          text-decoration: underline;
+        }
+      }
+    }
+
+    &__actions {
+      position: absolute;
+      top: 15px;
+      inset-inline-end: 15px;
+      text-align: end;
+    }
+  }
+}
+
+.report-actions {
+  border: 1px solid darken($ui-base-color, 8%);
+
+  &__item {
+    display: flex;
+    align-items: center;
+    line-height: 18px;
+    border-bottom: 1px solid darken($ui-base-color, 8%);
+
+    &:last-child {
+      border-bottom: 0;
+    }
+
+    &__button {
+      box-sizing: border-box;
+      flex: 0 0 auto;
+      width: 200px;
+      padding: 15px;
+      padding-inline-end: 0;
+
+      .button {
+        display: block;
+        width: 100%;
+      }
+    }
+
+    &__description {
+      padding: 15px;
+      font-size: 14px;
+      color: $dark-text-color;
+    }
+  }
+
+  @media screen and (width <= 800px) {
+    border: 0;
+
+    &__item {
+      flex-direction: column;
+      border: 0;
+
+      &__button {
+        width: 100%;
+        padding: 15px 0;
+      }
+
+      &__description {
+        padding: 0;
+        padding-bottom: 15px;
+      }
+    }
+  }
+}
+
+.section-skip-link {
+  float: right;
+
+  a {
+    color: $ui-highlight-color;
+    text-decoration: none;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: underline;
+    }
+  }
+}
+
+.strike-card {
+  padding: 15px;
+  border-radius: 4px;
+  background: $ui-base-color;
+  font-size: 15px;
+  line-height: 20px;
+  word-wrap: break-word;
+  font-weight: 400;
+  color: $primary-text-color;
+  box-sizing: border-box;
+  min-height: 100%;
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+  }
+
+  p {
+    margin-bottom: 20px;
+    unicode-bidi: plaintext;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+
+    strong {
+      font-weight: 700;
+    }
+  }
+
+  &__rules {
+    list-style: disc;
+    padding-inline-start: 15px;
+    margin-bottom: 20px;
+    color: $darker-text-color;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+
+    &__text {
+      color: $primary-text-color;
+    }
+  }
+
+  &__statuses-list {
+    border-radius: 4px;
+    border: 1px solid darken($ui-base-color, 8%);
+    font-size: 13px;
+    line-height: 18px;
+    overflow: hidden;
+
+    &__item {
+      padding: 16px;
+      background: lighten($ui-base-color, 2%);
+      border-bottom: 1px solid darken($ui-base-color, 8%);
+
+      &:last-child {
+        border-bottom: 0;
+      }
+
+      &__meta {
+        color: $darker-text-color;
+      }
+
+      a {
+        color: inherit;
+        text-decoration: none;
+
+        &:hover,
+        &:focus,
+        &:active {
+          text-decoration: underline;
+        }
+      }
+    }
+  }
+}
+
+.availability-indicator {
+  display: flex;
+  align-items: center;
+  margin-bottom: 30px;
+  font-size: 14px;
+  line-height: 21px;
+
+  &__hint {
+    padding: 0 15px;
+  }
+
+  &__graphic {
+    display: flex;
+    margin: 0 -2px;
+
+    &__item {
+      display: block;
+      flex: 0 0 auto;
+      width: 4px;
+      height: 21px;
+      background: lighten($ui-base-color, 8%);
+      margin: 0 2px;
+      border-radius: 2px;
+
+      &.positive {
+        background: $valid-value-color;
+      }
+
+      &.negative {
+        background: $error-value-color;
+      }
+    }
+  }
+}
+
+.history {
+  counter-reset: step 0;
+  font-size: 15px;
+  line-height: 22px;
+
+  li {
+    counter-increment: step 1;
+    padding-inline-start: 2.5rem;
+    padding-bottom: 8px;
+    position: relative;
+    margin-bottom: 8px;
+
+    &::before {
+      position: absolute;
+      content: counter(step);
+      font-size: 0.625rem;
+      font-weight: 500;
+      inset-inline-start: 0;
+      display: flex;
+      justify-content: center;
+      align-items: center;
+      width: calc(1.375rem + 1px);
+      height: calc(1.375rem + 1px);
+      background: $ui-base-color;
+      border: 1px solid $highlight-text-color;
+      color: $highlight-text-color;
+      border-radius: 8px;
+    }
+
+    &::after {
+      position: absolute;
+      content: '';
+      width: 1px;
+      background: $highlight-text-color;
+      bottom: 0;
+      top: calc(1.875rem + 1px);
+      inset-inline-start: 0.6875rem;
+    }
+
+    &:last-child {
+      margin-bottom: 0;
+
+      &::after {
+        display: none;
+      }
+    }
+  }
+
+  &__entry {
+    h5 {
+      font-weight: 500;
+      color: $primary-text-color;
+      line-height: 25px;
+      margin-bottom: 16px;
+    }
+
+    .status {
+      border: 1px solid lighten($ui-base-color, 4%);
+      background: $ui-base-color;
+      border-radius: 4px;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/basics.scss b/app/javascript/flavours/blobfox/styles/basics.scss
new file mode 100644
index 00000000000000..ff00c797c8e4dc
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/basics.scss
@@ -0,0 +1,294 @@
+@function hex-color($color) {
+  @if type-of($color) == 'color' {
+    $color: str-slice(ie-hex-str($color), 4);
+  }
+
+  @return '%23' + unquote($color);
+}
+
+body {
+  font-family: $font-sans-serif, sans-serif;
+  background: darken($ui-base-color, 7%);
+  font-size: 13px;
+  line-height: 18px;
+  font-weight: 400;
+  color: $primary-text-color;
+  text-rendering: optimizelegibility;
+  font-feature-settings: 'kern';
+  text-size-adjust: none;
+  -webkit-tap-highlight-color: rgba(0, 0, 0, 0%);
+  -webkit-tap-highlight-color: transparent;
+
+  &.system-font {
+    // system-ui => standard property (Chrome/Android WebView 56+, Opera 43+, Safari 11+)
+    // -apple-system => Safari <11 specific
+    // BlinkMacSystemFont => Chrome <56 on macOS specific
+    // Segoe UI => Windows 7/8/10
+    // Oxygen => KDE
+    // Ubuntu => Unity/Ubuntu
+    // Cantarell => GNOME
+    // Fira Sans => Firefox OS
+    // Droid Sans => Older Androids (<4.0)
+    // Helvetica Neue => Older macOS <10.11
+    // $font-sans-serif => web-font (Roboto) fallback and newer Androids (>=4.0)
+    font-family:
+      system-ui,
+      -apple-system,
+      BlinkMacSystemFont,
+      'Segoe UI',
+      Oxygen,
+      Ubuntu,
+      Cantarell,
+      'Fira Sans',
+      'Droid Sans',
+      'Helvetica Neue',
+      $font-sans-serif,
+      sans-serif;
+  }
+
+  &.app-body {
+    padding: 0;
+
+    &.layout-single-column {
+      height: auto;
+      min-height: 100vh;
+      overflow-y: scroll;
+    }
+
+    &.layout-multiple-columns {
+      position: absolute;
+      width: 100%;
+      height: 100%;
+    }
+
+    &.with-modals--active {
+      overflow-y: hidden;
+    }
+  }
+
+  &.lighter {
+    background: $ui-base-color;
+  }
+
+  &.with-modals {
+    overflow-x: hidden;
+    overflow-y: scroll;
+
+    &--active {
+      overflow-y: hidden;
+    }
+  }
+
+  &.player {
+    padding: 0;
+    margin: 0;
+    position: absolute;
+    width: 100%;
+    height: 100%;
+    overflow: hidden;
+
+    & > div {
+      height: 100%;
+    }
+
+    .video-player video {
+      width: 100%;
+      height: 100%;
+      max-height: 100vh;
+    }
+
+    .media-gallery {
+      margin-top: 0;
+      height: 100% !important;
+      border-radius: 0;
+    }
+
+    .media-gallery__item {
+      border-radius: 0;
+    }
+  }
+
+  &.embed {
+    background: lighten($ui-base-color, 4%);
+    margin: 0;
+    padding-bottom: 0;
+
+    .container {
+      position: absolute;
+      width: 100%;
+      height: 100%;
+      overflow: hidden;
+    }
+  }
+
+  &.admin {
+    background: darken($ui-base-color, 4%);
+    padding: 0;
+  }
+
+  &.error {
+    position: absolute;
+    text-align: center;
+    color: $darker-text-color;
+    background: $ui-base-color;
+    width: 100%;
+    height: 100%;
+    padding: 0;
+    display: flex;
+    justify-content: center;
+    align-items: center;
+
+    .dialog {
+      vertical-align: middle;
+      margin: 20px;
+
+      &__illustration {
+        img {
+          display: block;
+          max-width: 470px;
+          width: 100%;
+          height: auto;
+          margin-top: -120px;
+        }
+      }
+
+      h1 {
+        font-size: 20px;
+        line-height: 28px;
+        font-weight: 400;
+      }
+    }
+  }
+}
+
+button {
+  font-family: inherit;
+  cursor: pointer;
+
+  &:focus {
+    outline: none;
+  }
+}
+
+.app-holder {
+  &,
+  & > div,
+  & > noscript {
+    display: flex;
+    width: 100%;
+    align-items: center;
+    justify-content: center;
+    outline: 0 !important;
+  }
+
+  & > noscript {
+    height: 100vh;
+  }
+}
+
+.layout-single-column .app-holder {
+  &,
+  & > div {
+    min-height: 100vh;
+  }
+}
+
+.layout-multiple-columns .app-holder {
+  &,
+  & > div {
+    height: 100%;
+  }
+}
+
+.error-boundary,
+.app-holder noscript {
+  flex-direction: column;
+  font-size: 16px;
+  font-weight: 400;
+  line-height: 1.7;
+  color: lighten($error-red, 4%);
+  text-align: center;
+
+  & > div {
+    max-width: 500px;
+  }
+
+  p {
+    margin-bottom: 0.85em;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  a {
+    color: $highlight-text-color;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: none;
+    }
+  }
+
+  &__footer {
+    color: $dark-text-color;
+    font-size: 13px;
+
+    a {
+      color: $dark-text-color;
+    }
+  }
+
+  button {
+    display: inline;
+    border: 0;
+    background: transparent;
+    color: $dark-text-color;
+    font: inherit;
+    padding: 0;
+    margin: 0;
+    line-height: inherit;
+    cursor: pointer;
+    outline: 0;
+    transition: color 300ms linear;
+    text-decoration: underline;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: none;
+    }
+
+    &.copied {
+      color: $valid-value-color;
+      transition: none;
+    }
+  }
+}
+
+.logo-resources {
+  // Not using display: none because of https://bugs.chromium.org/p/chromium/issues/detail?id=258029
+  visibility: hidden;
+  user-select: none;
+  pointer-events: none;
+  width: 0;
+  height: 0;
+  overflow: hidden;
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  z-index: -1000;
+}
+
+// NoScript adds a __ns__pop2top class to the full ancestry of blocked elements,
+// to set the z-index to a high value, which messes with modals and dropdowns.
+// Blocked elements can in theory only be media and frames/embeds, so they
+// should only appear in statuses, under divs and articles.
+body,
+div,
+article {
+  .__ns__pop2top {
+    z-index: unset !important;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/branding.scss b/app/javascript/flavours/blobfox/styles/branding.scss
new file mode 100644
index 00000000000000..d1bddc68b0d7f9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/branding.scss
@@ -0,0 +1,3 @@
+.logo {
+  color: $primary-text-color;
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/about.scss b/app/javascript/flavours/blobfox/styles/components/about.scss
new file mode 100644
index 00000000000000..98ff91ad18c207
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/about.scss
@@ -0,0 +1,295 @@
+.image {
+  position: relative;
+  overflow: hidden;
+
+  &__preview {
+    position: absolute;
+    top: 0;
+    inset-inline-start: 0;
+    width: 100%;
+    height: 100%;
+    object-fit: cover;
+  }
+
+  &.loaded &__preview {
+    display: none;
+  }
+
+  img {
+    display: block;
+    width: 100%;
+    height: 100%;
+    object-fit: cover;
+    border: 0;
+    background: transparent;
+    opacity: 0;
+  }
+
+  &.loaded img {
+    opacity: 1;
+  }
+}
+
+.link-footer {
+  flex: 0 0 auto;
+  padding: 10px;
+  padding-top: 20px;
+  z-index: 1;
+  font-size: 13px;
+
+  p {
+    color: $dark-text-color;
+    margin-bottom: 20px;
+
+    .version {
+      white-space: nowrap;
+    }
+
+    strong {
+      font-weight: 500;
+    }
+
+    a {
+      color: $dark-text-color;
+      text-decoration: underline;
+
+      &:hover,
+      &:focus,
+      &:active {
+        text-decoration: none;
+      }
+    }
+  }
+}
+
+.about {
+  padding: 20px;
+
+  @media screen and (min-width: $no-gap-breakpoint) {
+    border-radius: 4px;
+  }
+
+  &__footer {
+    color: $dark-text-color;
+    text-align: center;
+    font-size: 15px;
+    line-height: 22px;
+    margin-top: 20px;
+  }
+
+  &__header {
+    margin-bottom: 30px;
+
+    &__hero {
+      width: 100%;
+      height: auto;
+      aspect-ratio: 1.9;
+      background: lighten($ui-base-color, 4%);
+      border-radius: 8px;
+      margin-bottom: 30px;
+    }
+
+    h1,
+    p {
+      text-align: center;
+    }
+
+    h1 {
+      font-size: 24px;
+      line-height: 1.5;
+      font-weight: 700;
+      margin-bottom: 10px;
+    }
+
+    p {
+      font-size: 16px;
+      line-height: 24px;
+      font-weight: 400;
+      color: $darker-text-color;
+    }
+  }
+
+  &__meta {
+    background: lighten($ui-base-color, 4%);
+    border-radius: 4px;
+    display: flex;
+    margin-bottom: 30px;
+    font-size: 15px;
+
+    &__column {
+      box-sizing: border-box;
+      width: 50%;
+      padding: 20px;
+    }
+
+    &__divider {
+      width: 0;
+      border: 0;
+      border-style: solid;
+      border-color: lighten($ui-base-color, 8%);
+      border-left-width: 1px;
+      min-height: calc(100% - 60px);
+      flex: 0 0 auto;
+    }
+
+    h4 {
+      font-size: 15px;
+      text-transform: uppercase;
+      color: $darker-text-color;
+      font-weight: 500;
+      margin-bottom: 20px;
+    }
+
+    @media screen and (width <= 600px) {
+      display: block;
+
+      h4 {
+        text-align: center;
+      }
+
+      &__column {
+        width: 100%;
+        display: flex;
+        flex-direction: column;
+        align-items: center;
+      }
+
+      &__divider {
+        min-height: 0;
+        width: 100%;
+        border-left-width: 0;
+        border-top-width: 1px;
+      }
+    }
+
+    .layout-multiple-columns & {
+      display: block;
+
+      h4 {
+        text-align: center;
+      }
+
+      &__column {
+        width: 100%;
+        display: flex;
+        flex-direction: column;
+        align-items: center;
+      }
+
+      &__divider {
+        min-height: 0;
+        width: 100%;
+        border-left-width: 0;
+        border-top-width: 1px;
+      }
+    }
+  }
+
+  &__mail {
+    color: $primary-text-color;
+    text-decoration: none;
+    font-weight: 500;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: underline;
+    }
+  }
+
+  .link-footer {
+    padding: 0;
+    margin-top: 60px;
+    text-align: center;
+    font-size: 15px;
+    line-height: 22px;
+
+    @media screen and (min-width: $no-gap-breakpoint) {
+      display: none;
+    }
+  }
+
+  .account {
+    padding: 0;
+    border: 0;
+  }
+
+  .account__avatar-wrapper {
+    margin-inline-start: 0;
+  }
+
+  .account__relationship {
+    display: none;
+  }
+
+  &__section {
+    margin-bottom: 10px;
+
+    &__title {
+      font-size: 17px;
+      font-weight: 600;
+      line-height: 22px;
+      padding: 20px;
+      border-radius: 4px;
+      background: lighten($ui-base-color, 4%);
+      color: $highlight-text-color;
+      cursor: pointer;
+    }
+
+    &.active &__title {
+      border-radius: 4px 4px 0 0;
+    }
+
+    &__body {
+      border: 1px solid lighten($ui-base-color, 4%);
+      border-top: 0;
+      padding: 20px;
+      font-size: 15px;
+      line-height: 22px;
+    }
+  }
+
+  &__domain-blocks {
+    margin-top: 30px;
+    background: darken($ui-base-color, 4%);
+    border: 1px solid lighten($ui-base-color, 4%);
+    border-radius: 4px;
+
+    &__domain {
+      border-bottom: 1px solid lighten($ui-base-color, 4%);
+      padding: 10px;
+      font-size: 15px;
+      color: $darker-text-color;
+
+      &:nth-child(2n) {
+        background: darken($ui-base-color, 2%);
+      }
+
+      &:last-child {
+        border-bottom: 0;
+      }
+
+      &__header {
+        display: flex;
+        gap: 10px;
+        justify-content: space-between;
+        font-weight: 500;
+        margin-bottom: 4px;
+      }
+
+      h6 {
+        color: $secondary-text-color;
+        font-size: inherit;
+        white-space: nowrap;
+        overflow: hidden;
+        text-overflow: ellipsis;
+      }
+
+      p {
+        white-space: nowrap;
+        overflow: hidden;
+        text-overflow: ellipsis;
+      }
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/accounts.scss b/app/javascript/flavours/blobfox/styles/components/accounts.scss
new file mode 100644
index 00000000000000..685421c0a87ae8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/accounts.scss
@@ -0,0 +1,807 @@
+.account {
+  padding: 10px;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  color: inherit;
+  text-decoration: none;
+
+  .account__display-name {
+    flex: 1 1 auto;
+    display: flex;
+    align-items: center;
+    gap: 10px;
+    color: $darker-text-color;
+    overflow: hidden;
+    text-decoration: none;
+    font-size: 14px;
+
+    .display-name {
+      margin-bottom: 4px;
+    }
+
+    .display-name strong {
+      display: inline;
+    }
+  }
+
+  &--minimal {
+    .account__display-name {
+      .display-name {
+        margin-bottom: 0;
+      }
+
+      .display-name strong {
+        display: block;
+      }
+    }
+  }
+
+  &__note {
+    font-size: 14px;
+    font-weight: 400;
+    overflow: hidden;
+    text-overflow: ellipsis;
+    display: -webkit-box;
+    -webkit-line-clamp: 1;
+    -webkit-box-orient: vertical;
+    margin-top: 10px;
+    color: $darker-text-color;
+
+    &--missing {
+      color: $dark-text-color;
+    }
+
+    p {
+      margin-bottom: 10px;
+
+      &:last-child {
+        margin-bottom: 0;
+      }
+    }
+
+    a {
+      color: inherit;
+
+      &:hover,
+      &:focus,
+      &:active {
+        text-decoration: none;
+      }
+    }
+  }
+}
+
+.account__wrapper {
+  display: flex;
+  gap: 10px;
+  align-items: center;
+}
+
+.account__avatar-wrapper {
+  float: left;
+}
+
+.account__avatar {
+  @include avatar-radius;
+
+  display: block;
+  position: relative;
+  overflow: hidden;
+
+  img {
+    display: block;
+    width: 100%;
+    height: 100%;
+    object-fit: cover;
+  }
+
+  &-inline {
+    display: inline-block;
+    vertical-align: middle;
+    margin-inline-end: 5px;
+  }
+
+  &-composite {
+    @include avatar-radius;
+
+    overflow: hidden;
+    position: relative;
+
+    & > div {
+      @include avatar-radius;
+
+      float: left;
+      position: relative;
+      box-sizing: border-box;
+    }
+
+    .account__avatar {
+      width: 100% !important;
+      height: 100% !important;
+    }
+
+    &__label {
+      display: block;
+      position: absolute;
+      top: 50%;
+      inset-inline-start: 50%;
+      transform: translate(-50%, -50%);
+      color: $primary-text-color;
+      text-shadow: 1px 1px 2px $base-shadow-color;
+      font-weight: 700;
+      font-size: 15px;
+    }
+  }
+}
+
+.account__avatar-overlay {
+  position: relative;
+
+  &-overlay {
+    position: absolute;
+    bottom: 0;
+    inset-inline-end: 0;
+    z-index: 1;
+  }
+}
+
+.account__relationship {
+  white-space: nowrap;
+  display: flex;
+  align-items: center;
+  gap: 4px;
+}
+
+.account__header__wrapper {
+  flex: 0 0 auto;
+  background: lighten($ui-base-color, 4%);
+}
+
+.account__disclaimer {
+  padding: 10px;
+  color: $dark-text-color;
+
+  strong {
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  a {
+    font-weight: 500;
+    color: inherit;
+    text-decoration: underline;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: none;
+    }
+  }
+}
+
+.account__action-bar {
+  border-top: 1px solid lighten($ui-base-color, 8%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  line-height: 36px;
+  overflow: hidden;
+  flex: 0 0 auto;
+  display: flex;
+}
+
+.account__action-bar-links {
+  display: flex;
+  flex: 1 1 auto;
+  line-height: 18px;
+  text-align: center;
+}
+
+.account__action-bar__tab {
+  text-decoration: none;
+  overflow: hidden;
+  flex: 0 1 100%;
+  border-inline-start: 1px solid lighten($ui-base-color, 8%);
+  padding: 10px 0;
+  border-bottom: 4px solid transparent;
+
+  &:first-child {
+    border-inline-start: 0;
+  }
+
+  &.active {
+    border-bottom: 4px solid $ui-highlight-color;
+  }
+
+  & > span {
+    display: block;
+    text-transform: uppercase;
+    font-size: 11px;
+    color: $darker-text-color;
+  }
+
+  strong {
+    display: block;
+    font-size: 15px;
+    font-weight: 500;
+    color: $primary-text-color;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  abbr {
+    color: $highlight-text-color;
+  }
+}
+
+.account-authorize {
+  padding: 14px 10px;
+
+  .detailed-status__display-name {
+    display: block;
+    margin-bottom: 15px;
+    overflow: hidden;
+  }
+}
+
+.account-authorize__avatar {
+  float: left;
+  margin-inline-end: 10px;
+}
+
+.notification__report {
+  padding: 8px 10px;
+  padding-inline-start: 68px;
+  position: relative;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  min-height: 54px;
+
+  &__details {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    color: $darker-text-color;
+    font-size: 15px;
+    line-height: 22px;
+
+    strong {
+      font-weight: 500;
+    }
+  }
+
+  &__avatar {
+    position: absolute;
+    inset-inline-start: 10px;
+    top: 10px;
+  }
+}
+
+.notification__message {
+  margin-inline-start: 42px;
+  padding-top: 8px;
+  padding-inline-start: 26px;
+  cursor: default;
+  color: $darker-text-color;
+  font-size: 15px;
+  position: relative;
+
+  .fa {
+    color: $highlight-text-color;
+  }
+
+  > span {
+    display: block;
+    overflow: hidden;
+    text-overflow: ellipsis;
+  }
+}
+
+.account--panel {
+  background: lighten($ui-base-color, 4%);
+  border-top: 1px solid lighten($ui-base-color, 8%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  display: flex;
+  flex-direction: row;
+  padding: 10px 0;
+}
+
+.account--panel__button,
+.detailed-status__button {
+  flex: 1 1 auto;
+  text-align: center;
+}
+
+.detailed-status__button .emoji-button {
+  padding: 0;
+}
+
+.relationship-tag {
+  color: $white;
+  margin-bottom: 4px;
+  display: block;
+  background-color: rgba($black, 0.45);
+  text-transform: uppercase;
+  font-size: 11px;
+  font-weight: 500;
+  padding: 4px;
+  border-radius: 4px;
+  opacity: 0.7;
+
+  &:hover {
+    opacity: 1;
+  }
+}
+
+.account-gallery__container {
+  display: flex;
+  flex-wrap: wrap;
+  padding: 4px 2px;
+}
+
+.account-gallery__item {
+  border: 0;
+  box-sizing: border-box;
+  display: block;
+  position: relative;
+  border-radius: 4px;
+  overflow: hidden;
+  margin: 2px;
+
+  &__icons {
+    position: absolute;
+    top: 50%;
+    inset-inline-start: 50%;
+    transform: translate(-50%, -50%);
+    font-size: 24px;
+  }
+}
+
+.notification__filter-bar,
+.account__section-headline {
+  background: darken($ui-base-color, 4%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  cursor: default;
+  display: flex;
+  flex-shrink: 0;
+
+  button {
+    background: transparent;
+    border: 0;
+    margin: 0;
+  }
+
+  button,
+  a {
+    display: block;
+    flex: 1 1 auto;
+    color: $darker-text-color;
+    padding: 15px 0;
+    font-size: 14px;
+    font-weight: 500;
+    text-align: center;
+    text-decoration: none;
+    position: relative;
+
+    &.active {
+      color: $primary-text-color;
+
+      &::before {
+        display: block;
+        content: '';
+        position: absolute;
+        bottom: -1px;
+        left: 0;
+        width: 100%;
+        height: 3px;
+        border-radius: 4px;
+        background: $highlight-text-color;
+      }
+    }
+  }
+
+  &.directory__section-headline {
+    background: darken($ui-base-color, 2%);
+    border-bottom-color: transparent;
+
+    a,
+    button {
+      &.active {
+        &::before {
+          display: none;
+        }
+
+        &::after {
+          border-color: transparent transparent darken($ui-base-color, 7%);
+        }
+      }
+    }
+  }
+}
+
+.account__moved-note {
+  padding: 14px 10px;
+  padding-bottom: 16px;
+  background: lighten($ui-base-color, 4%);
+  border-top: 1px solid lighten($ui-base-color, 8%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+
+  &__message {
+    position: relative;
+    margin-inline-start: 58px;
+    color: $dark-text-color;
+    padding: 8px 0;
+    padding-top: 0;
+    padding-bottom: 4px;
+    font-size: 14px;
+
+    > span {
+      display: block;
+      overflow: hidden;
+      text-overflow: ellipsis;
+    }
+  }
+
+  &__icon-wrapper {
+    inset-inline-start: -26px;
+    position: absolute;
+  }
+
+  .detailed-status__display-avatar {
+    position: relative;
+  }
+
+  .detailed-status__display-name {
+    margin-bottom: 0;
+  }
+}
+
+.account__header__content {
+  color: $darker-text-color;
+  font-size: 14px;
+  font-weight: 400;
+  overflow: hidden;
+  word-break: normal;
+  word-wrap: break-word;
+
+  p {
+    margin-bottom: 20px;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  a {
+    color: inherit;
+    text-decoration: underline;
+
+    &:hover {
+      text-decoration: none;
+    }
+  }
+}
+
+.account__header {
+  overflow: hidden;
+
+  &.inactive {
+    opacity: 0.5;
+
+    .account__header__image,
+    .account__avatar {
+      filter: grayscale(100%);
+    }
+  }
+
+  &__info {
+    position: absolute;
+    top: 10px;
+    inset-inline-start: 10px;
+  }
+
+  &__image {
+    overflow: hidden;
+    height: 145px;
+    position: relative;
+    background: darken($ui-base-color, 4%);
+
+    img {
+      object-fit: cover;
+      display: block;
+      width: 100%;
+      height: 100%;
+      margin: 0;
+    }
+  }
+
+  &__bar {
+    position: relative;
+    background: lighten($ui-base-color, 4%);
+    padding: 5px;
+    border-bottom: 1px solid lighten($ui-base-color, 12%);
+
+    .avatar {
+      display: block;
+      flex: 0 0 auto;
+      width: 94px;
+
+      .account__avatar {
+        background: darken($ui-base-color, 8%);
+        border: 2px solid lighten($ui-base-color, 4%);
+      }
+    }
+  }
+
+  &__tabs {
+    display: flex;
+    align-items: flex-end;
+    justify-content: space-between;
+    padding: 7px 10px;
+    margin-top: -81px;
+    height: 130px;
+    overflow: hidden;
+    margin-inline-start: -2px; // aligns the pfp with content below
+
+    .account-role {
+      margin: 0 2px;
+      padding: 4px 0;
+      box-sizing: border-box;
+      min-width: 90px;
+      text-align: center;
+    }
+
+    &__buttons {
+      display: flex;
+      align-items: center;
+      gap: 8px;
+      padding-top: 55px;
+      overflow: hidden;
+
+      .button {
+        flex-shrink: 1;
+        white-space: nowrap;
+
+        @media screen and (max-width: $no-gap-breakpoint) {
+          min-width: 0;
+        }
+      }
+
+      .icon-button {
+        border: 1px solid lighten($ui-base-color, 12%);
+        border-radius: 4px;
+        box-sizing: content-box;
+        padding: 2px;
+      }
+    }
+
+    &__name {
+      padding: 5px 10px;
+
+      .account-role {
+        vertical-align: top;
+      }
+
+      .emojione {
+        width: 22px;
+        height: 22px;
+      }
+
+      h1 {
+        font-size: 16px;
+        line-height: 24px;
+        color: $primary-text-color;
+        font-weight: 500;
+        overflow: hidden;
+        white-space: nowrap;
+        text-overflow: ellipsis;
+
+        small {
+          display: block;
+          font-size: 14px;
+          color: $darker-text-color;
+          font-weight: 400;
+          overflow: hidden;
+          text-overflow: ellipsis;
+
+          span {
+            user-select: all;
+          }
+        }
+      }
+    }
+
+    .spacer {
+      flex: 1 1 auto;
+    }
+  }
+
+  &__bio {
+    overflow: hidden;
+    margin: 0 -5px;
+
+    .account__header__content {
+      padding: 20px 15px;
+      padding-bottom: 5px;
+      color: $primary-text-color;
+    }
+
+    .account__header__joined {
+      font-size: 14px;
+      padding: 5px 15px;
+      color: $darker-text-color;
+
+      .columns-area--mobile & {
+        padding-inline-start: 20px;
+        padding-inline-end: 20px;
+      }
+    }
+
+    .account__header__fields {
+      margin: 0;
+      border-top: 1px solid lighten($ui-base-color, 12%);
+
+      a {
+        color: lighten($ui-highlight-color, 8%);
+      }
+
+      dl:first-child .verified {
+        border-radius: 0 4px 0 0;
+      }
+
+      .verified a {
+        color: $valid-value-color;
+      }
+    }
+  }
+
+  &__extra {
+    margin-top: 4px;
+
+    &__links {
+      font-size: 14px;
+      color: $darker-text-color;
+      padding: 10px 0;
+
+      a {
+        display: inline-block;
+        color: $darker-text-color;
+        text-decoration: none;
+        padding: 5px 10px;
+        font-weight: 500;
+
+        strong {
+          font-weight: 700;
+          color: $primary-text-color;
+        }
+      }
+    }
+  }
+
+  &__account-note {
+    margin: 0 -5px;
+    padding: 10px 15px;
+    display: flex;
+    flex-direction: column;
+    font-size: 14px;
+    font-weight: 400;
+    border-top: 1px solid lighten($ui-base-color, 12%);
+
+    label {
+      display: block;
+      font-size: 12px;
+      font-weight: 500;
+      color: $darker-text-color;
+      text-transform: uppercase;
+      margin-bottom: 5px;
+    }
+
+    &__content {
+      white-space: pre-wrap;
+      padding: 10px 0;
+    }
+
+    strong {
+      font-size: 12px;
+      font-weight: 500;
+      text-transform: uppercase;
+    }
+
+    textarea {
+      display: block;
+      box-sizing: border-box;
+      width: calc(100% + 20px);
+      color: $secondary-text-color;
+      background: $ui-base-color;
+      padding: 10px;
+      margin: 0 -10px;
+      font-family: inherit;
+      font-size: 14px;
+      resize: none;
+      border: 0;
+      outline: 0;
+      border-radius: 4px;
+
+      &::placeholder {
+        color: $dark-text-color;
+        opacity: 1;
+      }
+    }
+  }
+}
+
+.account__contents {
+  overflow: hidden;
+}
+
+.account__details {
+  display: flex;
+  flex-wrap: wrap;
+  column-gap: 1em;
+}
+
+.verified-badge {
+  display: inline-flex;
+  align-items: center;
+  color: $valid-value-color;
+  gap: 4px;
+  overflow: hidden;
+  white-space: nowrap;
+
+  > span {
+    overflow: hidden;
+    text-overflow: ellipsis;
+  }
+
+  a {
+    color: inherit;
+    font-weight: 500;
+    text-decoration: none;
+  }
+}
+
+.moved-account-banner,
+.follow-request-banner,
+.account-memorial-banner {
+  padding: 20px;
+  background: lighten($ui-base-color, 4%);
+  display: flex;
+  align-items: center;
+  flex-direction: column;
+
+  &__message {
+    color: $darker-text-color;
+    padding: 8px 0;
+    padding-top: 0;
+    padding-bottom: 4px;
+    font-size: 14px;
+    font-weight: 500;
+    text-align: center;
+    margin-bottom: 16px;
+  }
+
+  &__action {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    gap: 15px;
+    width: 100%;
+  }
+
+  .detailed-status__display-name {
+    margin-bottom: 0;
+  }
+}
+
+.follow-request-banner .button {
+  width: 100%;
+}
+
+.account-memorial-banner__message {
+  margin-bottom: 0;
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/announcements.scss b/app/javascript/flavours/blobfox/styles/components/announcements.scss
new file mode 100644
index 00000000000000..be27120a7d21ce
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/announcements.scss
@@ -0,0 +1,233 @@
+.announcements__item__content {
+  word-wrap: break-word;
+  overflow-y: auto;
+
+  .emojione {
+    width: 20px;
+    height: 20px;
+    margin: -3px 0 0;
+  }
+
+  p {
+    margin-bottom: 10px;
+    white-space: pre-wrap;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  a {
+    color: $secondary-text-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+
+    &.mention {
+      &:hover {
+        text-decoration: none;
+
+        span {
+          text-decoration: underline;
+        }
+      }
+    }
+
+    &.unhandled-link {
+      color: $highlight-text-color;
+    }
+  }
+}
+
+.announcements {
+  background: lighten($ui-base-color, 8%);
+  font-size: 13px;
+  display: flex;
+  align-items: flex-end;
+
+  &__mastodon {
+    width: 124px;
+    flex: 0 0 auto;
+
+    @media screen and (max-width: 124px + 300px) {
+      display: none;
+    }
+  }
+
+  &__container {
+    width: calc(100% - 124px);
+    flex: 0 0 auto;
+    position: relative;
+
+    @media screen and (max-width: 124px + 300px) {
+      width: 100%;
+    }
+  }
+
+  &__item {
+    box-sizing: border-box;
+    width: 100%;
+    padding: 15px;
+    position: relative;
+    font-size: 15px;
+    line-height: 20px;
+    word-wrap: break-word;
+    font-weight: 400;
+    max-height: 50vh;
+    overflow: hidden;
+    display: flex;
+    flex-direction: column;
+
+    &__range {
+      display: block;
+      font-weight: 500;
+      margin-bottom: 10px;
+      padding-inline-end: 18px;
+    }
+
+    &__unread {
+      position: absolute;
+      top: 19px;
+      inset-inline-end: 19px;
+      display: block;
+      background: $highlight-text-color;
+      border-radius: 50%;
+      width: 0.625rem;
+      height: 0.625rem;
+    }
+  }
+
+  &__pagination {
+    padding: 15px;
+    color: $darker-text-color;
+    position: absolute;
+    bottom: 3px;
+    inset-inline-end: 0;
+  }
+}
+
+.layout-multiple-columns .announcements__mastodon {
+  display: none;
+}
+
+.layout-multiple-columns .announcements__container {
+  width: 100%;
+}
+
+.reactions-bar {
+  display: flex;
+  flex-wrap: wrap;
+  align-items: center;
+  margin-top: 15px;
+  margin-inline-start: -2px;
+  width: calc(100% - (90px - 33px));
+
+  &__item {
+    flex-shrink: 0;
+    background: lighten($ui-base-color, 12%);
+    border: 0;
+    border-radius: 3px;
+    margin: 2px;
+    cursor: pointer;
+    user-select: none;
+    padding: 0 6px;
+    display: flex;
+    align-items: center;
+    transition: all 100ms ease-in;
+    transition-property: background-color, color;
+
+    &__emoji {
+      display: block;
+      margin: 3px 0;
+      width: 16px;
+      height: 16px;
+
+      img {
+        display: block;
+        margin: 0;
+        width: 100%;
+        height: 100%;
+        min-width: auto;
+        min-height: auto;
+        vertical-align: bottom;
+        object-fit: contain;
+      }
+    }
+
+    &__count {
+      display: block;
+      min-width: 9px;
+      font-size: 13px;
+      font-weight: 500;
+      text-align: center;
+      margin-inline-start: 6px;
+      color: $darker-text-color;
+    }
+
+    &:hover,
+    &:focus,
+    &:active {
+      background: lighten($ui-base-color, 16%);
+      transition: all 200ms ease-out;
+      transition-property: background-color, color;
+
+      &__count {
+        color: lighten($darker-text-color, 4%);
+      }
+    }
+
+    &.active {
+      transition: all 100ms ease-in;
+      transition-property: background-color, color;
+      background-color: mix(
+        lighten($ui-base-color, 12%),
+        $ui-highlight-color,
+        80%
+      );
+
+      .reactions-bar__item__count {
+        color: lighten($highlight-text-color, 8%);
+      }
+    }
+  }
+
+  .emoji-picker-dropdown {
+    margin: 2px;
+  }
+
+  &:hover .emoji-button {
+    opacity: 0.85;
+  }
+
+  .emoji-button {
+    color: $darker-text-color;
+    margin: 0;
+    font-size: 16px;
+    width: auto;
+    flex-shrink: 0;
+    padding: 0 6px;
+    height: 22px;
+    display: flex;
+    align-items: center;
+    opacity: 0.5;
+    transition: all 100ms ease-in;
+    transition-property: background-color, color;
+
+    &:hover,
+    &:active,
+    &:focus {
+      opacity: 1;
+      color: lighten($darker-text-color, 4%);
+      transition: all 200ms ease-out;
+      transition-property: background-color, color;
+    }
+  }
+
+  &--empty {
+    .emoji-button {
+      padding: 0;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/boost.scss b/app/javascript/flavours/blobfox/styles/components/boost.scss
new file mode 100644
index 00000000000000..2969958e227261
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/boost.scss
@@ -0,0 +1,44 @@
+button.icon-button {
+  i.fa-retweet {
+    background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='22' height='209'><path d='M4.97 3.16c-.1.03-.17.1-.22.18L.8 8.24c-.2.3.03.78.4.8H3.6v2.68c0 4.26-.55 3.62 3.66 3.62h7.66l-2.3-2.84c-.03-.02-.03-.04-.05-.06H7.27c-.44 0-.72-.3-.72-.72v-2.7h2.5c.37.03.63-.48.4-.77L5.5 3.35c-.12-.17-.34-.25-.53-.2zm12.16.43c-.55-.02-1.32.02-2.4.02H7.1l2.32 2.85.03.06h5.25c.42 0 .72.28.72.72v2.7h-2.5c-.36.02-.56.54-.3.8l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.26-.28 0-.83-.37-.8H18.4v-2.7c0-3.15.4-3.62-1.25-3.66z' fill='#{hex-color($action-button-color)}' stroke-width='0'/><path d='M7.78 19.66c-.24.02-.44.25-.44.5v2.46h-.06c-1.08 0-1.86-.03-2.4-.03-1.64 0-1.25.43-1.25 3.65v4.47c0 4.26-.56 3.62 3.65 3.62H8.5l-1.3-1.06c-.1-.08-.18-.2-.2-.3-.02-.17.06-.35.2-.45l1.33-1.1H7.28c-.44 0-.72-.3-.72-.7v-4.48c0-.44.28-.72.72-.72h.06v2.5c0 .38.54.63.82.38l4.9-3.93c.25-.18.25-.6 0-.78l-4.9-3.92c-.1-.1-.24-.14-.38-.12zm9.34 2.93c-.54-.02-1.3.02-2.4.02h-1.25l1.3 1.07c.1.07.18.2.2.33.02.16-.06.3-.2.4l-1.33 1.1h1.28c.42 0 .72.28.72.72v4.47c0 .42-.3.72-.72.72h-.1v-2.47c0-.3-.3-.53-.6-.47-.07 0-.14.05-.2.1l-4.9 3.93c-.26.18-.26.6 0 .78l4.9 3.92c.27.25.82 0 .8-.38v-2.5h.1c4.27 0 3.65.67 3.65-3.62v-4.47c0-3.15.4-3.62-1.25-3.66zM10.34 38.66c-.24.02-.44.25-.43.5v2.47H7.3c-1.08 0-1.86-.04-2.4-.04-1.64 0-1.25.43-1.25 3.65v4.47c0 3.66-.23 3.7 2.34 3.66l-1.34-1.1c-.1-.08-.18-.2-.2-.3 0-.17.07-.35.2-.45l1.96-1.6c-.03-.06-.04-.13-.04-.2v-4.48c0-.44.28-.72.72-.72H9.9v2.5c0 .36.5.6.8.38l4.93-3.93c.24-.18.24-.6 0-.78l-4.94-3.92c-.1-.08-.23-.13-.36-.12zm5.63 2.93l1.34 1.1c.1.07.18.2.2.33.02.16-.03.3-.16.4l-1.96 1.6c.02.07.06.13.06.22v4.47c0 .42-.3.72-.72.72h-2.66v-2.47c0-.3-.3-.53-.6-.47-.06.02-.12.05-.18.1l-4.94 3.93c-.24.18-.24.6 0 .78l4.94 3.92c.28.22.78-.02.78-.38v-2.5h2.66c4.27 0 3.65.67 3.65-3.62v-4.47c0-3.66.34-3.7-2.4-3.66zM13.06 57.66c-.23.03-.4.26-.4.5v2.47H7.28c-1.08 0-1.86-.04-2.4-.04-1.64 0-1.25.43-1.25 3.65v4.87l2.93-2.37v-2.5c0-.44.28-.72.72-.72h5.38v2.5c0 .36.5.6.78.38l4.94-3.93c.24-.18.24-.6 0-.78l-4.94-3.92c-.1-.1-.24-.14-.38-.12zm5.3 6.15l-2.92 2.4v2.52c0 .42-.3.72-.72.72h-5.4v-2.47c0-.3-.32-.53-.6-.47-.07.02-.13.05-.2.1L3.6 70.52c-.25.18-.25.6 0 .78l4.93 3.92c.28.22.78-.02.78-.38v-2.5h5.42c4.27 0 3.65.67 3.65-3.62v-4.47-.44zM19.25 78.8c-.1.03-.2.1-.28.17l-.9.9c-.44-.3-1.36-.25-3.35-.25H7.28c-1.08 0-1.86-.03-2.4-.03-1.64 0-1.25.43-1.25 3.65v.7l2.93.3v-1c0-.44.28-.72.72-.72h7.44c.2 0 .37.08.5.2l-1.8 1.8c-.25.26-.08.76.27.8l6.27.7c.28.03.56-.25.53-.53l-.7-6.25c0-.27-.3-.48-.55-.44zm-17.2 6.1c-.2.07-.36.3-.33.54l.7 6.25c.02.36.58.55.83.27l.8-.8c.02 0 .04-.02.04 0 .46.24 1.37.17 3.18.17h7.44c4.27 0 3.65.67 3.65-3.62v-.75l-2.93-.3v1.05c0 .42-.3.72-.72.72H7.28c-.15 0-.3-.03-.4-.1L8.8 86.4c.3-.24.1-.8-.27-.84l-6.28-.65h-.2zM4.88 98.6c-1.33 0-1.34.48-1.3 2.3l1.14-1.37c.08-.1.22-.17.34-.2.16 0 .34.08.44.2l1.66 2.03c.04 0 .07-.03.12-.03h7.44c.34 0 .57.2.65.5h-2.43c-.34.05-.53.52-.3.78l3.92 4.95c.18.24.6.24.78 0l3.94-4.94c.22-.27-.02-.76-.37-.77H18.4c.02-3.9.6-3.4-3.66-3.4H7.28c-1.08 0-1.86-.04-2.4-.04zm.15 2.46c-.1.03-.2.1-.28.2l-3.94 4.9c-.2.28.03.77.4.78H3.6c-.02 3.94-.45 3.4 3.66 3.4h7.44c3.65 0 3.74.3 3.7-2.25l-1.1 1.34c-.1.1-.2.17-.32.2-.16 0-.34-.08-.44-.2l-1.65-2.03c-.06.02-.1.04-.18.04H7.28c-.35 0-.57-.2-.66-.5h2.44c.37 0 .63-.5.4-.78l-3.96-4.9c-.1-.15-.3-.23-.47-.2zM4.88 117.6c-1.16 0-1.3.3-1.3 1.56l1.14-1.38c.08-.1.22-.14.34-.16.16 0 .34.04.44.16l2.22 2.75h7c.42 0 .72.28.72.72v.53h-2.6c-.3.1-.43.54-.2.78l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-.53c0-4.2.72-3.63-3.66-3.63H7.28c-1.08 0-1.86-.03-2.4-.03zm.1 1.74c-.1.03-.17.1-.23.16L.8 124.44c-.2.28.03.77.4.78H3.6v.5c0 4.26-.55 3.62 3.66 3.62h7.44c1.03 0 1.74.02 2.28 0-.16.02-.34-.03-.44-.15l-2.22-2.76H7.28c-.44 0-.72-.3-.72-.72v-.5h2.5c.37.02.63-.5.4-.78L5.5 119.5c-.12-.15-.34-.22-.53-.16zm12.02 10c1.2-.02 1.4-.25 1.4-1.53l-1.1 1.36c-.07.1-.17.17-.3.18zM5.94 136.6l2.37 2.93h6.42c.42 0 .72.28.72.72v1.25h-2.6c-.3.1-.43.54-.2.78l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-1.25c0-4.2.72-3.63-3.66-3.63H7.28c-.6 0-.92-.02-1.34-.03zm-1.72.06c-.4.08-.54.3-.6.75l.6-.74zm.84.93c-.12 0-.24.08-.3.18l-3.95 4.9c-.24.3 0 .83.4.82H3.6v1.22c0 4.26-.55 3.62 3.66 3.62h7.44c.63 0 .97.02 1.4.03l-2.37-2.93H7.28c-.44 0-.72-.3-.72-.72v-1.22h2.5c.4.04.67-.53.4-.8l-3.96-4.92c-.1-.13-.27-.2-.44-.2zm13.28 10.03l-.56.7c.36-.07.5-.3.56-.7zM17.13 155.6c-.55-.02-1.32.03-2.4.03h-8.2l2.38 2.9h5.82c.42 0 .72.28.72.72v1.97H12.9c-.32.06-.48.52-.28.78l3.94 4.94c.2.23.6.22.78-.03l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-1.97c0-3.15.4-3.62-1.25-3.66zm-12.1.28c-.1.02-.2.1-.28.18l-3.94 4.9c-.2.3.03.78.4.8H3.6v1.96c0 4.26-.55 3.62 3.66 3.62h8.24l-2.36-2.9H7.28c-.44 0-.72-.3-.72-.72v-1.97h2.5c.37.02.63-.5.4-.78l-3.96-4.9c-.1-.15-.3-.22-.47-.2zM5.13 174.5c-.15 0-.3.07-.38.2L.8 179.6c-.24.27 0 .82.4.8H3.6v2.32c0 4.26-.55 3.62 3.66 3.62h7.94l-2.35-2.9h-5.6c-.43 0-.7-.3-.7-.72v-2.3h2.5c.38.03.66-.54.4-.83l-3.97-4.9c-.1-.13-.23-.2-.38-.2zm12 .1c-.55-.02-1.32.03-2.4.03H6.83l2.35 2.9h5.52c.42 0 .72.28.72.72v2.34h-2.6c-.3.1-.43.53-.2.78l3.92 4.9c.18.24.6.24.78 0l3.94-4.9c.22-.3-.02-.78-.37-.8H18.4v-2.33c0-3.15.4-3.62-1.25-3.66zM4.97 193.16c-.1.03-.17.1-.22.18l-3.94 4.9c-.2.3.03.78.4.8H3.6v2.68c0 4.26-.55 3.62 3.66 3.62h7.66l-2.3-2.84c-.03-.02-.03-.04-.05-.06H7.27c-.44 0-.72-.3-.72-.72v-2.7h2.5c.37.03.63-.48.4-.77l-3.96-4.9c-.12-.17-.34-.25-.53-.2zm12.16.43c-.55-.02-1.32.03-2.4.03H7.1l2.32 2.84.03.06h5.25c.42 0 .72.28.72.72v2.7h-2.5c-.36.02-.56.54-.3.8l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.26-.28 0-.83-.37-.8H18.4v-2.7c0-3.15.4-3.62-1.25-3.66z' fill='#{hex-color($highlight-text-color)}' stroke-width='0'/></svg>");
+  }
+
+  &:hover i.fa-retweet {
+    background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='22' height='209'><path d='M4.97 3.16c-.1.03-.17.1-.22.18L.8 8.24c-.2.3.03.78.4.8H3.6v2.68c0 4.26-.55 3.62 3.66 3.62h7.66l-2.3-2.84c-.03-.02-.03-.04-.05-.06H7.27c-.44 0-.72-.3-.72-.72v-2.7h2.5c.37.03.63-.48.4-.77L5.5 3.35c-.12-.17-.34-.25-.53-.2zm12.16.43c-.55-.02-1.32.02-2.4.02H7.1l2.32 2.85.03.06h5.25c.42 0 .72.28.72.72v2.7h-2.5c-.36.02-.56.54-.3.8l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.26-.28 0-.83-.37-.8H18.4v-2.7c0-3.15.4-3.62-1.25-3.66z' fill='#{hex-color(lighten($action-button-color, 7%))}' stroke-width='0'/><path d='M7.78 19.66c-.24.02-.44.25-.44.5v2.46h-.06c-1.08 0-1.86-.03-2.4-.03-1.64 0-1.25.43-1.25 3.65v4.47c0 4.26-.56 3.62 3.65 3.62H8.5l-1.3-1.06c-.1-.08-.18-.2-.2-.3-.02-.17.06-.35.2-.45l1.33-1.1H7.28c-.44 0-.72-.3-.72-.7v-4.48c0-.44.28-.72.72-.72h.06v2.5c0 .38.54.63.82.38l4.9-3.93c.25-.18.25-.6 0-.78l-4.9-3.92c-.1-.1-.24-.14-.38-.12zm9.34 2.93c-.54-.02-1.3.02-2.4.02h-1.25l1.3 1.07c.1.07.18.2.2.33.02.16-.06.3-.2.4l-1.33 1.1h1.28c.42 0 .72.28.72.72v4.47c0 .42-.3.72-.72.72h-.1v-2.47c0-.3-.3-.53-.6-.47-.07 0-.14.05-.2.1l-4.9 3.93c-.26.18-.26.6 0 .78l4.9 3.92c.27.25.82 0 .8-.38v-2.5h.1c4.27 0 3.65.67 3.65-3.62v-4.47c0-3.15.4-3.62-1.25-3.66zM10.34 38.66c-.24.02-.44.25-.43.5v2.47H7.3c-1.08 0-1.86-.04-2.4-.04-1.64 0-1.25.43-1.25 3.65v4.47c0 3.66-.23 3.7 2.34 3.66l-1.34-1.1c-.1-.08-.18-.2-.2-.3 0-.17.07-.35.2-.45l1.96-1.6c-.03-.06-.04-.13-.04-.2v-4.48c0-.44.28-.72.72-.72H9.9v2.5c0 .36.5.6.8.38l4.93-3.93c.24-.18.24-.6 0-.78l-4.94-3.92c-.1-.08-.23-.13-.36-.12zm5.63 2.93l1.34 1.1c.1.07.18.2.2.33.02.16-.03.3-.16.4l-1.96 1.6c.02.07.06.13.06.22v4.47c0 .42-.3.72-.72.72h-2.66v-2.47c0-.3-.3-.53-.6-.47-.06.02-.12.05-.18.1l-4.94 3.93c-.24.18-.24.6 0 .78l4.94 3.92c.28.22.78-.02.78-.38v-2.5h2.66c4.27 0 3.65.67 3.65-3.62v-4.47c0-3.66.34-3.7-2.4-3.66zM13.06 57.66c-.23.03-.4.26-.4.5v2.47H7.28c-1.08 0-1.86-.04-2.4-.04-1.64 0-1.25.43-1.25 3.65v4.87l2.93-2.37v-2.5c0-.44.28-.72.72-.72h5.38v2.5c0 .36.5.6.78.38l4.94-3.93c.24-.18.24-.6 0-.78l-4.94-3.92c-.1-.1-.24-.14-.38-.12zm5.3 6.15l-2.92 2.4v2.52c0 .42-.3.72-.72.72h-5.4v-2.47c0-.3-.32-.53-.6-.47-.07.02-.13.05-.2.1L3.6 70.52c-.25.18-.25.6 0 .78l4.93 3.92c.28.22.78-.02.78-.38v-2.5h5.42c4.27 0 3.65.67 3.65-3.62v-4.47-.44zM19.25 78.8c-.1.03-.2.1-.28.17l-.9.9c-.44-.3-1.36-.25-3.35-.25H7.28c-1.08 0-1.86-.03-2.4-.03-1.64 0-1.25.43-1.25 3.65v.7l2.93.3v-1c0-.44.28-.72.72-.72h7.44c.2 0 .37.08.5.2l-1.8 1.8c-.25.26-.08.76.27.8l6.27.7c.28.03.56-.25.53-.53l-.7-6.25c0-.27-.3-.48-.55-.44zm-17.2 6.1c-.2.07-.36.3-.33.54l.7 6.25c.02.36.58.55.83.27l.8-.8c.02 0 .04-.02.04 0 .46.24 1.37.17 3.18.17h7.44c4.27 0 3.65.67 3.65-3.62v-.75l-2.93-.3v1.05c0 .42-.3.72-.72.72H7.28c-.15 0-.3-.03-.4-.1L8.8 86.4c.3-.24.1-.8-.27-.84l-6.28-.65h-.2zM4.88 98.6c-1.33 0-1.34.48-1.3 2.3l1.14-1.37c.08-.1.22-.17.34-.2.16 0 .34.08.44.2l1.66 2.03c.04 0 .07-.03.12-.03h7.44c.34 0 .57.2.65.5h-2.43c-.34.05-.53.52-.3.78l3.92 4.95c.18.24.6.24.78 0l3.94-4.94c.22-.27-.02-.76-.37-.77H18.4c.02-3.9.6-3.4-3.66-3.4H7.28c-1.08 0-1.86-.04-2.4-.04zm.15 2.46c-.1.03-.2.1-.28.2l-3.94 4.9c-.2.28.03.77.4.78H3.6c-.02 3.94-.45 3.4 3.66 3.4h7.44c3.65 0 3.74.3 3.7-2.25l-1.1 1.34c-.1.1-.2.17-.32.2-.16 0-.34-.08-.44-.2l-1.65-2.03c-.06.02-.1.04-.18.04H7.28c-.35 0-.57-.2-.66-.5h2.44c.37 0 .63-.5.4-.78l-3.96-4.9c-.1-.15-.3-.23-.47-.2zM4.88 117.6c-1.16 0-1.3.3-1.3 1.56l1.14-1.38c.08-.1.22-.14.34-.16.16 0 .34.04.44.16l2.22 2.75h7c.42 0 .72.28.72.72v.53h-2.6c-.3.1-.43.54-.2.78l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-.53c0-4.2.72-3.63-3.66-3.63H7.28c-1.08 0-1.86-.03-2.4-.03zm.1 1.74c-.1.03-.17.1-.23.16L.8 124.44c-.2.28.03.77.4.78H3.6v.5c0 4.26-.55 3.62 3.66 3.62h7.44c1.03 0 1.74.02 2.28 0-.16.02-.34-.03-.44-.15l-2.22-2.76H7.28c-.44 0-.72-.3-.72-.72v-.5h2.5c.37.02.63-.5.4-.78L5.5 119.5c-.12-.15-.34-.22-.53-.16zm12.02 10c1.2-.02 1.4-.25 1.4-1.53l-1.1 1.36c-.07.1-.17.17-.3.18zM5.94 136.6l2.37 2.93h6.42c.42 0 .72.28.72.72v1.25h-2.6c-.3.1-.43.54-.2.78l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-1.25c0-4.2.72-3.63-3.66-3.63H7.28c-.6 0-.92-.02-1.34-.03zm-1.72.06c-.4.08-.54.3-.6.75l.6-.74zm.84.93c-.12 0-.24.08-.3.18l-3.95 4.9c-.24.3 0 .83.4.82H3.6v1.22c0 4.26-.55 3.62 3.66 3.62h7.44c.63 0 .97.02 1.4.03l-2.37-2.93H7.28c-.44 0-.72-.3-.72-.72v-1.22h2.5c.4.04.67-.53.4-.8l-3.96-4.92c-.1-.13-.27-.2-.44-.2zm13.28 10.03l-.56.7c.36-.07.5-.3.56-.7zM17.13 155.6c-.55-.02-1.32.03-2.4.03h-8.2l2.38 2.9h5.82c.42 0 .72.28.72.72v1.97H12.9c-.32.06-.48.52-.28.78l3.94 4.94c.2.23.6.22.78-.03l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-1.97c0-3.15.4-3.62-1.25-3.66zm-12.1.28c-.1.02-.2.1-.28.18l-3.94 4.9c-.2.3.03.78.4.8H3.6v1.96c0 4.26-.55 3.62 3.66 3.62h8.24l-2.36-2.9H7.28c-.44 0-.72-.3-.72-.72v-1.97h2.5c.37.02.63-.5.4-.78l-3.96-4.9c-.1-.15-.3-.22-.47-.2zM5.13 174.5c-.15 0-.3.07-.38.2L.8 179.6c-.24.27 0 .82.4.8H3.6v2.32c0 4.26-.55 3.62 3.66 3.62h7.94l-2.35-2.9h-5.6c-.43 0-.7-.3-.7-.72v-2.3h2.5c.38.03.66-.54.4-.83l-3.97-4.9c-.1-.13-.23-.2-.38-.2zm12 .1c-.55-.02-1.32.03-2.4.03H6.83l2.35 2.9h5.52c.42 0 .72.28.72.72v2.34h-2.6c-.3.1-.43.53-.2.78l3.92 4.9c.18.24.6.24.78 0l3.94-4.9c.22-.3-.02-.78-.37-.8H18.4v-2.33c0-3.15.4-3.62-1.25-3.66zM4.97 193.16c-.1.03-.17.1-.22.18l-3.94 4.9c-.2.3.03.78.4.8H3.6v2.68c0 4.26-.55 3.62 3.66 3.62h7.66l-2.3-2.84c-.03-.02-.03-.04-.05-.06H7.27c-.44 0-.72-.3-.72-.72v-2.7h2.5c.37.03.63-.48.4-.77l-3.96-4.9c-.12-.17-.34-.25-.53-.2zm12.16.43c-.55-.02-1.32.03-2.4.03H7.1l2.32 2.84.03.06h5.25c.42 0 .72.28.72.72v2.7h-2.5c-.36.02-.56.54-.3.8l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.26-.28 0-.83-.37-.8H18.4v-2.7c0-3.15.4-3.62-1.25-3.66z' fill='#{hex-color($highlight-text-color)}' stroke-width='0'/></svg>");
+  }
+
+  &.reblogPrivate {
+    i.fa-retweet {
+      background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' height='209' width='22'><path d='M 4.9707031 3.1503906 L 4.9707031 3.1601562 C 4.8707031 3.1901563 4.8 3.2598438 4.75 3.3398438 L 0.80078125 8.2402344 C 0.60078125 8.5402344 0.8292187 9.0190625 1.1992188 9.0390625 L 3.5996094 9.0390625 L 3.5996094 11.720703 C 3.5996094 15.980703 3.0497656 15.339844 7.2597656 15.339844 L 11.869141 15.339844 L 11.869141 14.119141 L 11.869141 13.523438 L 11.869141 12.441406 C 11.869141 12.441406 11.869141 12.439453 11.869141 12.439453 L 7.2695312 12.439453 C 6.8295312 12.439453 6.5507814 12.140703 6.5507812 11.720703 L 6.5507812 9.0195312 L 9.0507812 9.0195312 C 9.4207813 9.0495313 9.6792188 8.54 9.4492188 8.25 L 5.5 3.3496094 C 5.38 3.1796094 5.1607031 3.1003906 4.9707031 3.1503906 z M 17.150391 3.5800781 L 17.130859 3.5898438 C 16.580859 3.5698436 15.810469 3.609375 14.730469 3.609375 L 7.0996094 3.609375 L 9.4199219 6.4609375 L 9.4492188 6.5195312 L 14.699219 6.5195312 C 15.106887 6.5195312 15.397113 6.7872181 15.414062 7.2050781 C 15.738375 7.0991315 16.077769 7.0273437 16.435547 7.0273438 L 16.578125 7.0273438 C 17.24903 7.0273438 17.874081 7.2325787 18.400391 7.578125 L 18.400391 7.2402344 C 18.400391 4.0902344 18.800391 3.6200781 17.150391 3.5800781 z M 16.435547 8.0273438 C 15.143818 8.0273438 14.083984 9.0851838 14.083984 10.376953 L 14.083984 11.607422 L 13.570312 11.607422 C 13.375448 11.607422 13.210603 11.704118 13.119141 11.791016 C 13.027691 11.877916 12.983569 11.958238 12.951172 12.03125 C 12.886382 12.177277 12.867187 12.304789 12.867188 12.441406 L 12.867188 13.523438 L 12.867188 14.119141 L 12.867188 15.677734 L 12.867188 16.509766 L 13.570312 16.509766 L 19.472656 16.509766 L 20.173828 16.509766 L 20.173828 15.677734 L 20.173828 13.523438 L 20.173828 12.441406 C 20.173828 12.304794 20.156597 12.177281 20.091797 12.03125 C 20.059397 11.95824 20.015299 11.877916 19.923828 11.791016 C 19.832368 11.704116 19.667509 11.607422 19.472656 11.607422 L 18.927734 11.607422 L 18.927734 10.376953 C 18.927734 9.0851838 17.867902 8.0273438 16.576172 8.0273438 L 16.435547 8.0273438 z M 16.435547 9.2207031 L 16.576172 9.2207031 C 17.22782 9.2207031 17.734375 9.7251013 17.734375 10.376953 L 17.734375 11.607422 L 15.277344 11.607422 L 15.277344 10.376953 C 15.277344 9.7251013 15.7839 9.2207031 16.435547 9.2207031 z M 12.919922 9.9394531 C 12.559922 9.9594531 12.359141 10.480234 12.619141 10.740234 L 12.751953 10.904297 C 12.862211 10.870135 12.980058 10.842244 13.085938 10.802734 L 13.085938 10.378906 C 13.085938 10.228632 13.111295 10.084741 13.130859 9.9394531 L 12.919922 9.9394531 z M 19.882812 9.9394531 C 19.902378 10.084741 19.927734 10.228632 19.927734 10.378906 L 19.927734 10.791016 C 20.168811 10.875098 20.455966 10.916935 20.613281 11.066406 C 20.691227 11.140457 20.749315 11.223053 20.806641 11.302734 L 21.259766 10.740234 C 21.519766 10.460234 21.260625 9.9094531 20.890625 9.9394531 L 19.882812 9.9394531 z M 16.435547 10.220703 C 16.301234 10.220703 16.277344 10.244432 16.277344 10.378906 L 16.277344 10.607422 L 16.734375 10.607422 L 16.734375 10.378906 C 16.734375 10.244433 16.712442 10.220703 16.578125 10.220703 L 16.435547 10.220703 z ' fill='#{hex-color($action-button-color)}' stroke-width='0'/><path d='M 7.7792969 19.650391 L 7.7792969 19.660156 C 7.5392969 19.680156 7.3398437 19.910156 7.3398438 20.160156 L 7.3398438 22.619141 L 7.2792969 22.619141 C 6.1992969 22.619141 5.4208594 22.589844 4.8808594 22.589844 C 3.2408594 22.589844 3.6308594 23.020234 3.6308594 26.240234 L 3.6308594 30.710938 C 3.6308594 34.970937 3.0692969 34.330078 7.2792969 34.330078 L 8.5 34.330078 L 7.1992188 33.269531 C 7.0992188 33.189531 7.02 33.070703 7 32.970703 C 6.98 32.800703 7.0592186 32.619531 7.1992188 32.519531 L 8.5292969 31.419922 L 7.2792969 31.419922 C 6.8392969 31.419922 6.5605469 31.120703 6.5605469 30.720703 L 6.5605469 26.240234 C 6.5605469 25.800234 6.8392969 25.519531 7.2792969 25.519531 L 7.3398438 25.519531 L 7.3398438 28.019531 C 7.3398438 28.399531 7.8801564 28.650391 8.1601562 28.400391 L 13.060547 24.470703 C 13.310547 24.290703 13.310547 23.869453 13.060547 23.689453 L 8.1601562 19.769531 C 8.0601563 19.669531 7.9192969 19.630391 7.7792969 19.650391 z M 17.119141 22.580078 L 17.119141 22.589844 C 16.579141 22.569844 15.820703 22.609375 14.720703 22.609375 L 13.470703 22.609375 L 14.769531 23.679688 C 14.869531 23.749688 14.950703 23.879766 14.970703 24.009766 C 14.990703 24.169766 14.909531 24.310156 14.769531 24.410156 L 13.439453 25.509766 L 14.720703 25.509766 C 15.129702 25.509766 15.41841 25.778986 15.433594 26.199219 C 15.752266 26.097283 16.084896 26.027344 16.435547 26.027344 L 16.578125 26.027344 C 17.236645 26.027344 17.848901 26.228565 18.369141 26.5625 L 18.369141 26.240234 C 18.369141 23.090234 18.769141 22.620078 17.119141 22.580078 z M 16.435547 27.027344 C 15.143818 27.027344 14.083984 28.085184 14.083984 29.376953 L 14.083984 30.607422 L 13.570312 30.607422 C 13.375452 30.607422 13.210603 30.704118 13.119141 30.791016 C 13.027691 30.877916 12.983569 30.958238 12.951172 31.03125 C 12.886382 31.177277 12.867184 31.304789 12.867188 31.441406 L 12.867188 32.523438 L 12.867188 33.119141 L 12.867188 34.677734 L 12.867188 35.509766 L 13.570312 35.509766 L 19.472656 35.509766 L 20.173828 35.509766 L 20.173828 34.677734 L 20.173828 32.523438 L 20.173828 31.441406 C 20.173828 31.304794 20.156597 31.177281 20.091797 31.03125 C 20.059397 30.95824 20.015299 30.877916 19.923828 30.791016 C 19.832368 30.704116 19.667509 30.607422 19.472656 30.607422 L 18.927734 30.607422 L 18.927734 29.376953 C 18.927734 28.085184 17.867902 27.027344 16.576172 27.027344 L 16.435547 27.027344 z M 16.435547 28.220703 L 16.576172 28.220703 C 17.22782 28.220703 17.734375 28.725101 17.734375 29.376953 L 17.734375 30.607422 L 15.277344 30.607422 L 15.277344 29.376953 C 15.277344 28.725101 15.7839 28.220703 16.435547 28.220703 z M 13.109375 29.150391 L 8.9199219 32.509766 C 8.6599219 32.689766 8.6599219 33.109063 8.9199219 33.289062 L 11.869141 35.648438 L 11.869141 34.677734 L 11.869141 33.119141 L 11.869141 32.523438 L 11.869141 31.441406 C 11.869141 31.217489 11.912641 30.907486 12.037109 30.626953 C 12.093758 30.499284 12.228597 30.257492 12.429688 30.066406 C 12.580253 29.92335 12.859197 29.887344 13.085938 29.802734 L 13.085938 29.378906 C 13.085938 29.300761 13.104 29.227272 13.109375 29.150391 z M 16.435547 29.220703 C 16.301234 29.220703 16.277344 29.244432 16.277344 29.378906 L 16.277344 29.607422 L 16.734375 29.607422 L 16.734375 29.378906 C 16.734375 29.244433 16.712442 29.220703 16.578125 29.220703 L 16.435547 29.220703 z M 12.943359 36.509766 L 13.820312 37.210938 C 14.090314 37.460938 14.639141 37.210078 14.619141 36.830078 L 14.619141 36.509766 L 13.570312 36.509766 L 12.943359 36.509766 z M 10.330078 38.650391 L 10.339844 38.660156 C 10.099844 38.680156 9.9001562 38.910156 9.9101562 39.160156 L 9.9101562 41.630859 L 7.3007812 41.630859 C 6.2207812 41.630859 5.4403906 41.589844 4.9003906 41.589844 C 3.2603906 41.589844 3.6503906 42.020234 3.6503906 45.240234 L 3.6503906 49.710938 C 3.6503906 53.370936 3.4202344 53.409141 5.9902344 53.369141 L 4.6503906 52.269531 C 4.5503906 52.189531 4.4692187 52.070703 4.4492188 51.970703 C 4.4492188 51.800703 4.5203906 51.619531 4.6503906 51.519531 L 6.609375 49.919922 C 6.579375 49.859922 6.5703125 49.790703 6.5703125 49.720703 L 6.5703125 45.240234 C 6.5703125 44.800234 6.8490625 44.519531 7.2890625 44.519531 L 9.9003906 44.519531 L 9.9003906 47.019531 C 9.9003906 47.379531 10.399219 47.620391 10.699219 47.400391 L 15.630859 43.470703 C 15.870859 43.290703 15.870859 42.869453 15.630859 42.689453 L 10.689453 38.769531 C 10.589453 38.689531 10.460078 38.640391 10.330078 38.650391 z M 16.869141 41.585938 C 16.616211 41.581522 16.322969 41.584844 15.980469 41.589844 L 15.970703 41.589844 L 17.310547 42.689453 C 17.410547 42.759453 17.489766 42.889531 17.509766 43.019531 C 17.529766 43.179531 17.479609 43.319922 17.349609 43.419922 L 15.390625 45.019531 C 15.406724 45.075878 15.427133 45.132837 15.4375 45.197266 C 15.754974 45.096169 16.086404 45.027344 16.435547 45.027344 L 16.578125 45.027344 C 17.24129 45.027344 17.858323 45.230088 18.380859 45.568359 L 18.380859 45.25 C 18.380859 42.0475 18.639648 41.616836 16.869141 41.585938 z M 16.435547 46.027344 C 15.143818 46.027344 14.083984 47.085184 14.083984 48.376953 L 14.083984 49.607422 L 13.570312 49.607422 C 13.375448 49.607422 13.210603 49.704118 13.119141 49.791016 C 13.027691 49.877916 12.983569 49.958238 12.951172 50.03125 C 12.886382 50.177277 12.867187 50.304789 12.867188 50.441406 L 12.867188 51.523438 L 12.867188 52.119141 L 12.867188 53.677734 L 12.867188 54.509766 L 13.570312 54.509766 L 19.472656 54.509766 L 20.173828 54.509766 L 20.173828 53.677734 L 20.173828 51.523438 L 20.173828 50.441406 C 20.173828 50.304794 20.156597 50.177281 20.091797 50.03125 C 20.059397 49.95824 20.015299 49.877916 19.923828 49.791016 C 19.832368 49.704116 19.667509 49.607422 19.472656 49.607422 L 18.927734 49.607422 L 18.927734 48.376953 C 18.927734 47.085184 17.867902 46.027344 16.576172 46.027344 L 16.435547 46.027344 z M 16.435547 47.220703 L 16.576172 47.220703 C 17.22782 47.220703 17.734375 47.725101 17.734375 48.376953 L 17.734375 49.607422 L 15.277344 49.607422 L 15.277344 48.376953 C 15.277344 47.725101 15.7839 47.220703 16.435547 47.220703 z M 11.470703 47.490234 C 11.410703 47.510234 11.349063 47.539844 11.289062 47.589844 L 6.3496094 51.519531 C 6.1096094 51.699531 6.1096094 52.120781 6.3496094 52.300781 L 11.289062 56.220703 C 11.569064 56.440703 12.070312 56.199844 12.070312 55.839844 L 12.070312 55.509766 L 11.869141 55.509766 L 11.869141 53.677734 L 11.869141 52.119141 L 11.869141 51.523438 L 11.869141 50.441406 C 11.869141 50.217489 11.912641 49.907486 12.037109 49.626953 C 12.043809 49.611855 12.061451 49.584424 12.070312 49.566406 L 12.070312 47.960938 C 12.070312 47.660938 11.770703 47.430234 11.470703 47.490234 z M 16.435547 48.220703 C 16.301234 48.220703 16.277344 48.244432 16.277344 48.378906 L 16.277344 48.607422 L 16.734375 48.607422 L 16.734375 48.378906 C 16.734375 48.244433 16.712442 48.220703 16.578125 48.220703 L 16.435547 48.220703 z M 13.060547 57.650391 L 13.060547 57.660156 C 12.830547 57.690156 12.660156 57.920156 12.660156 58.160156 L 12.660156 60.630859 L 7.2792969 60.630859 C 6.1992969 60.630859 5.4208594 60.589844 4.8808594 60.589844 C 3.2408594 60.589844 3.6308594 61.020234 3.6308594 64.240234 L 3.6308594 69.109375 L 6.5605469 66.740234 L 6.5605469 64.240234 C 6.5605469 63.800234 6.8392969 63.519531 7.2792969 63.519531 L 12.660156 63.519531 L 12.660156 66.019531 C 12.660156 66.299799 12.960394 66.500006 13.226562 66.474609 C 13.625751 65.076914 14.904956 64.035678 16.421875 64.029297 L 18.380859 62.470703 C 18.620859 62.290703 18.620859 61.869453 18.380859 61.689453 L 13.439453 57.769531 C 13.339453 57.669531 13.200547 57.630391 13.060547 57.650391 z M 18.359375 63.810547 L 17.800781 64.269531 C 18.004793 64.350836 18.198411 64.450249 18.380859 64.568359 L 18.380859 64.25 L 18.380859 63.810547 L 18.359375 63.810547 z M 16.435547 65.027344 C 15.143818 65.027344 14.083984 66.085184 14.083984 67.376953 L 14.083984 68.607422 L 13.570312 68.607422 C 13.375448 68.607422 13.210603 68.704118 13.119141 68.791016 C 13.027691 68.877916 12.983569 68.958238 12.951172 69.03125 C 12.886382 69.177277 12.867187 69.304789 12.867188 69.441406 L 12.867188 70.523438 L 12.867188 71.119141 L 12.867188 72.677734 L 12.867188 73.509766 L 13.570312 73.509766 L 19.472656 73.509766 L 20.173828 73.509766 L 20.173828 72.677734 L 20.173828 70.523438 L 20.173828 69.441406 C 20.173828 69.304794 20.156597 69.177281 20.091797 69.03125 C 20.059397 68.95824 20.015299 68.877916 19.923828 68.791016 C 19.832368 68.704116 19.667509 68.607422 19.472656 68.607422 L 18.927734 68.607422 L 18.927734 67.376953 C 18.927734 66.085184 17.867902 65.027344 16.576172 65.027344 L 16.435547 65.027344 z M 16.435547 66.220703 L 16.576172 66.220703 C 17.22782 66.220703 17.734375 66.725101 17.734375 67.376953 L 17.734375 68.607422 L 15.277344 68.607422 L 15.277344 67.376953 C 15.277344 66.725101 15.7839 66.220703 16.435547 66.220703 z M 8.7207031 66.509766 C 8.6507031 66.529766 8.5895312 66.559375 8.5195312 66.609375 L 3.5996094 70.519531 C 3.3496094 70.699531 3.3496094 71.120781 3.5996094 71.300781 L 8.5292969 75.220703 C 8.8092969 75.440703 9.3105469 75.199844 9.3105469 74.839844 L 9.3105469 72.339844 L 11.869141 72.339844 L 11.869141 71.119141 L 11.869141 70.523438 L 11.869141 69.449219 L 9.3203125 69.449219 L 9.3203125 66.980469 C 9.3203125 66.680469 9.0007031 66.449766 8.7207031 66.509766 z M 16.435547 67.220703 C 16.301234 67.220703 16.277344 67.244432 16.277344 67.378906 L 16.277344 67.607422 L 16.734375 67.607422 L 16.734375 67.378906 C 16.734375 67.244433 16.712442 67.220703 16.578125 67.220703 L 16.435547 67.220703 z M 19.248047 78.800781 C 19.148558 78.831033 19.050295 78.90106 18.970703 78.970703 L 18.070312 79.869141 C 17.630312 79.569141 16.710703 79.619141 14.720703 79.619141 L 7.2792969 79.619141 C 6.1992969 79.619141 5.4208594 79.589844 4.8808594 79.589844 C 3.2408594 79.589844 3.6308594 80.020234 3.6308594 83.240234 L 3.6308594 83.939453 L 6.5605469 84.240234 L 6.5605469 83.240234 C 6.5605469 82.800234 6.8392969 82.519531 7.2792969 82.519531 L 14.720703 82.519531 C 14.920703 82.519531 15.090703 82.600703 15.220703 82.720703 L 13.419922 84.519531 C 13.279464 84.665607 13.281282 84.881022 13.363281 85.054688 C 13.880838 83.867655 15.067337 83.027344 16.435547 83.027344 L 16.578125 83.027344 C 18.290465 83.027344 19.703357 84.345788 19.890625 86.011719 L 19.960938 86.019531 C 20.240938 86.049531 20.520234 85.770234 20.490234 85.490234 L 19.789062 79.240234 C 19.789062 78.973661 19.498025 78.767523 19.25 78.800781 L 19.248047 78.800781 z M 16.435547 84.027344 C 15.143818 84.027344 14.083984 85.085184 14.083984 86.376953 L 14.083984 87.607422 L 13.570312 87.607422 C 13.375448 87.607422 13.210603 87.704118 13.119141 87.791016 C 13.027691 87.877916 12.983569 87.958238 12.951172 88.03125 C 12.886382 88.177277 12.867187 88.304789 12.867188 88.441406 L 12.867188 89.523438 L 12.867188 90.119141 L 12.867188 91.677734 L 12.867188 92.509766 L 13.570312 92.509766 L 19.472656 92.509766 L 20.173828 92.509766 L 20.173828 91.677734 L 20.173828 89.523438 L 20.173828 88.441406 C 20.173828 88.304794 20.156597 88.177281 20.091797 88.03125 C 20.059397 87.95824 20.015299 87.877916 19.923828 87.791016 C 19.832368 87.704116 19.667509 87.607422 19.472656 87.607422 L 18.927734 87.607422 L 18.927734 86.376953 C 18.927734 85.085184 17.867902 84.027344 16.576172 84.027344 L 16.435547 84.027344 z M 2.0507812 84.900391 C 1.8507824 84.970391 1.6907031 85.199453 1.7207031 85.439453 L 2.4199219 91.689453 C 2.4399219 92.049453 3 92.240929 3.25 91.960938 L 4.0507812 91.160156 C 4.0707812 91.160156 4.0898437 91.140156 4.0898438 91.160156 C 4.5498437 91.400156 5.4595313 91.330078 7.2695312 91.330078 L 11.869141 91.330078 L 11.869141 90.119141 L 11.869141 89.523438 L 11.869141 88.441406 C 11.869141 88.437991 11.871073 88.433136 11.871094 88.429688 L 7.2792969 88.429688 C 7.1292969 88.429688 6.9808594 88.400078 6.8808594 88.330078 L 8.8007812 86.400391 C 9.1007822 86.160391 8.8992969 85.600547 8.5292969 85.560547 L 2.25 84.910156 L 2.0507812 84.910156 L 2.0507812 84.900391 z M 16.435547 85.220703 L 16.576172 85.220703 C 17.22782 85.220703 17.734375 85.725101 17.734375 86.376953 L 17.734375 87.607422 L 15.277344 87.607422 L 15.277344 86.376953 C 15.277344 85.725101 15.7839 85.220703 16.435547 85.220703 z M 4.8808594 98.599609 C 3.5508594 98.599609 3.5400781 99.080402 3.5800781 100.90039 L 4.7207031 99.529297 C 4.8007031 99.429297 4.9405469 99.360078 5.0605469 99.330078 C 5.2205469 99.330078 5.4 99.409297 5.5 99.529297 L 7.1601562 101.56055 C 7.2001563 101.56055 7.2292969 101.5293 7.2792969 101.5293 L 14.720703 101.5293 C 15.060703 101.5293 15.289141 101.7293 15.369141 102.0293 L 12.939453 102.0293 C 12.599453 102.0793 12.410625 102.55055 12.640625 102.81055 L 13.470703 103.85742 C 14.029941 102.77899 15.146801 102.02734 16.435547 102.02734 L 16.578125 102.02734 C 18.158418 102.02734 19.491598 103.14879 19.835938 104.63086 L 21.279297 102.82031 C 21.499297 102.55031 21.260156 102.06078 20.910156 102.05078 L 18.400391 102.05078 C 18.420391 98.150792 19.000234 98.650391 14.740234 98.650391 L 7.2792969 98.650391 C 6.1992969 98.650391 5.4208594 98.609375 4.8808594 98.609375 L 4.8808594 98.599609 z M 5.0292969 101.06055 C 4.9292969 101.09055 4.83 101.15977 4.75 101.25977 L 0.81054688 106.16016 C 0.61054688 106.44016 0.8409375 106.92945 1.2109375 106.93945 L 3.5996094 106.93945 C 3.5796094 110.87945 3.1497656 110.33984 7.2597656 110.33984 L 11.869141 110.33984 L 11.869141 109.11914 L 11.869141 108.52344 L 11.869141 107.44141 L 11.869141 107.43945 L 7.2792969 107.43945 C 6.9292969 107.43945 6.7091406 107.23945 6.6191406 106.93945 L 9.0605469 106.93945 C 9.4305469 106.93945 9.6909375 106.44016 9.4609375 106.16016 L 5.5 101.25977 C 5.4 101.10977 5.1992969 101.03055 5.0292969 101.06055 z M 16.435547 103.02734 C 15.143818 103.02734 14.083984 104.08518 14.083984 105.37695 L 14.083984 106.60742 L 13.570312 106.60742 C 13.375448 106.60742 13.210603 106.70409 13.119141 106.79102 C 13.027691 106.87792 12.983569 106.95823 12.951172 107.03125 C 12.886382 107.17727 12.867187 107.30479 12.867188 107.44141 L 12.867188 108.52344 L 12.867188 109.11914 L 12.867188 110.67773 L 12.867188 111.50977 L 13.570312 111.50977 L 19.472656 111.50977 L 20.173828 111.50977 L 20.173828 110.67773 L 20.173828 108.52344 L 20.173828 107.44141 C 20.173828 107.3048 20.156597 107.17728 20.091797 107.03125 C 20.059397 106.95825 20.015299 106.87792 19.923828 106.79102 C 19.832368 106.70412 19.667509 106.60742 19.472656 106.60742 L 18.927734 106.60742 L 18.927734 105.37695 C 18.927734 104.08518 17.867902 103.02734 16.576172 103.02734 L 16.435547 103.02734 z M 16.435547 104.2207 L 16.576172 104.2207 C 17.22782 104.2207 17.734375 104.7251 17.734375 105.37695 L 17.734375 106.60742 L 15.277344 106.60742 L 15.277344 105.37695 C 15.277344 104.7251 15.7839 104.2207 16.435547 104.2207 z M 16.435547 105.2207 C 16.301234 105.2207 16.277344 105.24444 16.277344 105.37891 L 16.277344 105.60742 L 16.734375 105.60742 L 16.734375 105.37891 C 16.734375 105.24441 16.712442 105.2207 16.578125 105.2207 L 16.435547 105.2207 z M 4.8808594 117.58984 L 4.8808594 117.59961 C 3.7208594 117.59961 3.5800781 117.90016 3.5800781 119.16016 L 4.7207031 117.7793 C 4.8007031 117.6793 4.9405469 117.63914 5.0605469 117.61914 C 5.2205469 117.61914 5.4 117.6593 5.5 117.7793 L 7.7207031 120.5293 L 14.720703 120.5293 C 15.123595 120.5293 15.408576 120.79174 15.431641 121.20117 C 15.750992 121.09876 16.08404 121.02734 16.435547 121.02734 L 16.578125 121.02734 C 17.24903 121.02734 17.874081 121.23262 18.400391 121.57812 L 18.400391 121.25 C 18.400391 117.05 19.120234 117.61914 14.740234 117.61914 L 7.2792969 117.61914 C 6.1992969 117.61914 5.4208594 117.58984 4.8808594 117.58984 z M 4.9804688 119.33984 C 4.8804688 119.36984 4.81 119.44 4.75 119.5 L 0.80078125 124.43945 C 0.60078125 124.71945 0.8292182 125.2107 1.1992188 125.2207 L 3.5996094 125.2207 L 3.5996094 125.7207 C 3.5996094 129.9807 3.0497656 129.33984 7.2597656 129.33984 L 11.869141 129.33984 L 11.869141 128.11914 L 11.869141 127.52344 L 11.869141 126.44141 C 11.869141 126.43799 11.871073 126.43314 11.871094 126.42969 L 7.2792969 126.42969 C 6.8392969 126.42969 6.5605469 126.13094 6.5605469 125.71094 L 6.5605469 125.21094 L 9.0605469 125.21094 C 9.4305469 125.23094 9.6909375 124.70969 9.4609375 124.42969 L 5.5 119.5 C 5.3820133 119.35252 5.1682348 119.28513 4.9804688 119.33984 z M 12.839844 121.7793 C 12.539844 121.8793 12.410625 122.32055 12.640625 122.56055 L 13.267578 123.34375 C 13.473522 122.72168 13.852237 122.1828 14.353516 121.7793 L 12.839844 121.7793 z M 18.658203 121.7793 C 19.393958 122.37155 19.878978 123.25738 19.916016 124.25781 L 21.279297 122.56055 C 21.499297 122.28055 21.260156 121.7893 20.910156 121.7793 L 18.658203 121.7793 z M 16.435547 122.02734 C 15.143818 122.02734 14.083984 123.08518 14.083984 124.37695 L 14.083984 125.60742 L 13.570312 125.60742 C 13.375448 125.60742 13.210603 125.70409 13.119141 125.79102 C 13.027691 125.87792 12.983569 125.95823 12.951172 126.03125 C 12.886382 126.17727 12.867187 126.30479 12.867188 126.44141 L 12.867188 127.52344 L 12.867188 128.11914 L 12.867188 129.67773 L 12.867188 130.50977 L 13.570312 130.50977 L 19.472656 130.50977 L 20.173828 130.50977 L 20.173828 129.67773 L 20.173828 127.52344 L 20.173828 126.44141 C 20.173828 126.3048 20.156597 126.17728 20.091797 126.03125 C 20.059397 125.95825 20.015299 125.87792 19.923828 125.79102 C 19.832368 125.70412 19.667509 125.60742 19.472656 125.60742 L 18.927734 125.60742 L 18.927734 124.37695 C 18.927734 123.08518 17.867902 122.02734 16.576172 122.02734 L 16.435547 122.02734 z M 16.435547 123.2207 L 16.576172 123.2207 C 17.22782 123.2207 17.734375 123.7251 17.734375 124.37695 L 17.734375 125.60742 L 15.277344 125.60742 L 15.277344 124.37695 C 15.277344 123.7251 15.7839 123.2207 16.435547 123.2207 z M 16.435547 124.2207 C 16.301234 124.2207 16.277344 124.24444 16.277344 124.37891 L 16.277344 124.60742 L 16.734375 124.60742 L 16.734375 124.37891 C 16.734375 124.24441 16.712442 124.2207 16.578125 124.2207 L 16.435547 124.2207 z M 5.9394531 136.58984 L 5.9394531 136.59961 L 8.3105469 139.5293 L 14.730469 139.5293 C 15.131912 139.5293 15.414551 139.79039 15.439453 140.19727 C 15.756409 140.09653 16.087055 140.02734 16.435547 140.02734 L 16.578125 140.02734 C 17.24903 140.02734 17.874081 140.23261 18.400391 140.57812 L 18.400391 140.25 C 18.400391 136.05 19.120234 136.61914 14.740234 136.61914 L 7.2792969 136.61914 C 6.6792969 136.61914 6.3594531 136.59984 5.9394531 136.58984 z M 4.2207031 136.66016 C 3.8207031 136.74016 3.6791406 136.96016 3.6191406 137.41016 L 4.2207031 136.66992 L 4.2207031 136.66016 z M 5.0605469 137.57031 L 5.0605469 137.58984 C 4.9405469 137.58984 4.8197656 137.66953 4.7597656 137.76953 L 0.81054688 142.66992 C 0.57054688 142.96992 0.8109375 143.50023 1.2109375 143.49023 L 3.5996094 143.49023 L 3.5996094 144.71094 C 3.5996094 148.97094 3.0497656 148.33008 7.2597656 148.33008 L 11.869141 148.33008 L 11.869141 147.11914 L 11.869141 146.52344 L 11.869141 145.44141 C 11.869141 145.43799 11.871073 145.43314 11.871094 145.42969 L 7.2792969 145.42969 C 6.8392969 145.42969 6.5605469 145.13094 6.5605469 144.71094 L 6.5605469 143.49023 L 9.0605469 143.49023 C 9.4605469 143.53023 9.7309375 142.95945 9.4609375 142.68945 L 5.5 137.76953 C 5.4 137.63953 5.2305469 137.57031 5.0605469 137.57031 z M 16.435547 141.02734 C 15.143818 141.02734 14.083984 142.08518 14.083984 143.37695 L 14.083984 144.60742 L 13.570312 144.60742 C 13.375448 144.60742 13.210603 144.70409 13.119141 144.79102 C 13.027691 144.87792 12.983569 144.95823 12.951172 145.03125 C 12.886382 145.17727 12.867187 145.30479 12.867188 145.44141 L 12.867188 146.52344 L 12.867188 147.11914 L 12.867188 148.67773 L 12.867188 149.50977 L 13.570312 149.50977 L 19.472656 149.50977 L 20.173828 149.50977 L 20.173828 148.67773 L 20.173828 146.52344 L 20.173828 145.44141 C 20.173828 145.3048 20.156597 145.17728 20.091797 145.03125 C 20.059397 144.95825 20.015299 144.87792 19.923828 144.79102 C 19.832368 144.70412 19.667509 144.60742 19.472656 144.60742 L 18.927734 144.60742 L 18.927734 143.37695 C 18.927734 142.08518 17.867902 141.02734 16.576172 141.02734 L 16.435547 141.02734 z M 12.849609 141.5 C 12.549609 141.6 12.420391 142.0393 12.650391 142.2793 L 13.136719 142.88672 C 13.213026 142.38119 13.390056 141.90696 13.667969 141.5 L 12.849609 141.5 z M 19.34375 141.5 C 19.710704 142.03735 19.927734 142.68522 19.927734 143.37891 L 19.927734 143.79102 C 19.965561 143.80421 20.005506 143.81448 20.044922 143.82617 L 21.289062 142.2793 C 21.509062 141.9993 21.269922 141.51 20.919922 141.5 L 19.34375 141.5 z M 16.435547 142.2207 L 16.576172 142.2207 C 17.22782 142.2207 17.734375 142.7251 17.734375 143.37695 L 17.734375 144.60742 L 15.277344 144.60742 L 15.277344 143.37695 C 15.277344 142.7251 15.7839 142.2207 16.435547 142.2207 z M 16.435547 143.2207 C 16.301234 143.2207 16.277344 143.24444 16.277344 143.37891 L 16.277344 143.60742 L 16.734375 143.60742 L 16.734375 143.37891 C 16.734375 143.24441 16.712442 143.2207 16.578125 143.2207 L 16.435547 143.2207 z M 17.130859 155.59961 C 16.580859 155.57961 15.810469 155.63086 14.730469 155.63086 L 6.5292969 155.63086 L 8.9101562 158.5293 L 14.730469 158.5293 C 15.131912 158.5293 15.414551 158.79039 15.439453 159.19727 C 15.756409 159.09653 16.087055 159.02734 16.435547 159.02734 L 16.578125 159.02734 C 17.24903 159.02734 17.874081 159.23261 18.400391 159.57812 L 18.400391 159.25977 C 18.400391 156.10977 18.800391 155.63961 17.150391 155.59961 L 17.130859 155.59961 z M 5.0292969 155.86914 L 5.0292969 155.88086 C 4.9292969 155.90086 4.83 155.98055 4.75 156.06055 L 0.81054688 160.96094 C 0.61054688 161.26094 0.8409375 161.73977 1.2109375 161.75977 L 3.5996094 161.75977 L 3.5996094 163.7207 C 3.5996094 167.9807 3.0497656 167.33984 7.2597656 167.33984 L 11.869141 167.33984 L 11.869141 166.11914 L 11.869141 165.52344 L 11.869141 164.44141 L 11.869141 164.43945 L 7.2792969 164.43945 C 6.8392969 164.43945 6.5605469 164.1407 6.5605469 163.7207 L 6.5605469 161.75 L 9.0605469 161.75 C 9.4305469 161.77 9.6909375 161.2507 9.4609375 160.9707 L 5.5 156.07031 C 5.4 155.92031 5.1992969 155.84914 5.0292969 155.86914 z M 16.435547 160.02734 C 15.143818 160.02734 14.083984 161.08518 14.083984 162.37695 L 14.083984 163.60742 L 13.570312 163.60742 C 13.375448 163.60742 13.210603 163.70409 13.119141 163.79102 C 13.027691 163.87792 12.983569 163.95823 12.951172 164.03125 C 12.886382 164.17727 12.867187 164.30479 12.867188 164.44141 L 12.867188 165.52344 L 12.867188 166.11914 L 12.867188 167.67773 L 12.867188 168.50977 L 13.570312 168.50977 L 19.472656 168.50977 L 20.173828 168.50977 L 20.173828 167.67773 L 20.173828 165.52344 L 20.173828 164.44141 C 20.173828 164.3048 20.156597 164.17728 20.091797 164.03125 C 20.059397 163.95825 20.015299 163.87792 19.923828 163.79102 C 19.832368 163.70412 19.667509 163.60742 19.472656 163.60742 L 18.927734 163.60742 L 18.927734 162.37695 C 18.927734 161.08518 17.867902 160.02734 16.576172 160.02734 L 16.435547 160.02734 z M 12.900391 161.2207 C 12.580391 161.2807 12.419141 161.74 12.619141 162 L 13.085938 162.58594 L 13.085938 162.37891 C 13.085938 161.97087 13.170592 161.58376 13.306641 161.2207 L 12.900391 161.2207 z M 16.435547 161.2207 L 16.576172 161.2207 C 17.22782 161.2207 17.734375 161.7251 17.734375 162.37695 L 17.734375 163.60742 L 15.277344 163.60742 L 15.277344 162.37695 C 15.277344 161.7251 15.7839 161.2207 16.435547 161.2207 z M 19.708984 161.23047 C 19.842743 161.59081 19.927734 161.97449 19.927734 162.37891 L 19.927734 162.79102 C 20.119162 162.85779 20.322917 162.91147 20.484375 163 L 21.279297 162.00977 C 21.499297 161.72977 21.260156 161.24047 20.910156 161.23047 L 19.708984 161.23047 z M 16.435547 162.2207 C 16.301234 162.2207 16.277344 162.24444 16.277344 162.37891 L 16.277344 162.60742 L 16.734375 162.60742 L 16.734375 162.37891 C 16.734375 162.24441 16.712442 162.2207 16.578125 162.2207 L 16.435547 162.2207 z M 5.0996094 174.49023 L 5.1308594 174.5 C 4.9808594 174.5 4.83 174.56922 4.75 174.69922 L 0.80078125 179.59961 C 0.56078125 179.86961 0.7992182 180.42039 1.1992188 180.40039 L 3.5996094 180.40039 L 3.5996094 182.7207 C 3.5996094 186.9807 3.0497656 186.33984 7.2597656 186.33984 L 11.869141 186.33984 L 11.869141 185.11914 L 11.869141 184.52344 L 11.869141 183.44141 L 11.869141 183.43945 L 7.25 183.43945 C 6.82 183.43945 6.5507814 183.1407 6.5507812 182.7207 L 6.5507812 180.41992 L 9.0507812 180.41992 C 9.4307824 180.44992 9.7092187 179.87984 9.4492188 179.58984 L 5.4804688 174.68945 C 5.3804688 174.55945 5.2496094 174.49023 5.0996094 174.49023 z M 17.150391 174.58008 L 17.130859 174.59961 C 16.580859 174.57961 15.810469 174.63086 14.730469 174.63086 L 6.8300781 174.63086 L 9.1796875 177.5293 L 14.699219 177.5293 C 15.104107 177.5293 15.391475 177.79407 15.412109 178.20703 C 15.737096 178.1006 16.076913 178.02734 16.435547 178.02734 L 16.578125 178.02734 C 17.24903 178.02734 17.874081 178.2326 18.400391 178.57812 L 18.400391 178.24023 C 18.400391 175.09023 18.800391 174.62008 17.150391 174.58008 z M 16.435547 179.02734 C 15.143818 179.02734 14.083984 180.08518 14.083984 181.37695 L 14.083984 182.60742 L 13.570312 182.60742 C 13.375448 182.60742 13.210603 182.70409 13.119141 182.79102 C 13.027691 182.87792 12.983569 182.95823 12.951172 183.03125 C 12.886382 183.17727 12.867187 183.30479 12.867188 183.44141 L 12.867188 184.52344 L 12.867188 185.11914 L 12.867188 186.67773 L 12.867188 187.50977 L 13.570312 187.50977 L 19.472656 187.50977 L 20.173828 187.50977 L 20.173828 186.67773 L 20.173828 184.52344 L 20.173828 183.44141 C 20.173828 183.3048 20.156597 183.17728 20.091797 183.03125 C 20.059397 182.95825 20.015299 182.87792 19.923828 182.79102 C 19.832368 182.70412 19.667509 182.60742 19.472656 182.60742 L 18.927734 182.60742 L 18.927734 181.37695 C 18.927734 180.08518 17.867902 179.02734 16.576172 179.02734 L 16.435547 179.02734 z M 16.435547 180.2207 L 16.576172 180.2207 C 17.22782 180.2207 17.734375 180.7251 17.734375 181.37695 L 17.734375 182.60742 L 15.277344 182.60742 L 15.277344 181.37695 C 15.277344 180.7251 15.7839 180.2207 16.435547 180.2207 z M 19.816406 180.57031 C 19.882311 180.83091 19.927734 181.09907 19.927734 181.37891 L 19.927734 181.79102 C 20.168811 181.87511 20.455966 181.91694 20.613281 182.06641 C 20.630645 182.0829 20.639883 182.10199 20.65625 182.11914 L 21.259766 181.36914 C 21.479766 181.06914 21.240625 180.59031 20.890625 180.57031 L 19.816406 180.57031 z M 12.820312 180.58984 C 12.520316 180.68984 12.389141 181.11914 12.619141 181.36914 L 12.990234 181.83203 C 13.022029 181.82207 13.055579 181.81406 13.085938 181.80273 L 13.085938 181.37891 C 13.085938 181.10616 13.128698 180.84442 13.191406 180.58984 L 12.820312 180.58984 z M 16.435547 181.2207 C 16.301234 181.2207 16.277344 181.24444 16.277344 181.37891 L 16.277344 181.60742 L 16.734375 181.60742 L 16.734375 181.37891 C 16.734375 181.24441 16.712442 181.2207 16.578125 181.2207 L 16.435547 181.2207 z M 4.9609375 193.15039 L 4.9707031 193.16016 C 4.8707031 193.19016 4.8 193.25984 4.75 193.33984 L 0.81054688 198.24023 C 0.61054688 198.54023 0.8409375 199.01906 1.2109375 199.03906 L 3.5996094 199.03906 L 3.5996094 201.7207 C 3.5996094 205.9807 3.0497656 205.33984 7.2597656 205.33984 L 11.869141 205.33984 L 11.869141 204.11914 L 11.869141 203.52344 L 11.869141 202.44141 C 11.869141 202.44141 11.869141 202.43945 11.869141 202.43945 L 7.2695312 202.43945 C 6.8295312 202.43945 6.5507814 202.1407 6.5507812 201.7207 L 6.5507812 199.01953 L 9.0507812 199.01953 C 9.4207814 199.04953 9.6792188 198.54 9.4492188 198.25 L 5.4902344 193.34961 C 5.3702344 193.17961 5.1509375 193.10039 4.9609375 193.15039 z M 17.150391 193.58008 L 17.130859 193.58984 C 16.580859 193.56984 15.810469 193.61914 14.730469 193.61914 L 7.0996094 193.61914 L 9.4199219 196.46094 L 9.4492188 196.51953 L 14.699219 196.51953 C 15.106887 196.51953 15.397075 196.78718 15.414062 197.20508 C 15.738375 197.09913 16.077769 197.02734 16.435547 197.02734 L 16.578125 197.02734 C 17.24903 197.02734 17.874081 197.23259 18.400391 197.57812 L 18.400391 197.24023 C 18.400391 194.09023 18.800391 193.62008 17.150391 193.58008 z M 16.435547 198.02734 C 15.143818 198.02734 14.083984 199.08518 14.083984 200.37695 L 14.083984 201.60742 L 13.570312 201.60742 C 13.375448 201.60742 13.210603 201.70409 13.119141 201.79102 C 13.027691 201.87792 12.983569 201.95823 12.951172 202.03125 C 12.886382 202.17727 12.867187 202.30479 12.867188 202.44141 L 12.867188 203.52344 L 12.867188 204.11914 L 12.867188 205.67773 L 12.867188 206.50977 L 13.570312 206.50977 L 19.472656 206.50977 L 20.173828 206.50977 L 20.173828 205.67773 L 20.173828 203.52344 L 20.173828 202.44141 C 20.173828 202.3048 20.156597 202.17728 20.091797 202.03125 C 20.059397 201.95825 20.015299 201.87792 19.923828 201.79102 C 19.832368 201.70412 19.667509 201.60742 19.472656 201.60742 L 18.927734 201.60742 L 18.927734 200.37695 C 18.927734 199.08518 17.867902 198.02734 16.576172 198.02734 L 16.435547 198.02734 z M 16.435547 199.2207 L 16.576172 199.2207 C 17.22782 199.2207 17.734375 199.7251 17.734375 200.37695 L 17.734375 201.60742 L 15.277344 201.60742 L 15.277344 200.37695 C 15.277344 199.7251 15.7839 199.2207 16.435547 199.2207 z M 12.919922 199.93945 C 12.559922 199.95945 12.359141 200.48023 12.619141 200.74023 L 12.751953 200.9043 C 12.862211 200.87013 12.980058 200.84224 13.085938 200.80273 L 13.085938 200.37891 C 13.085938 200.22863 13.111295 200.08474 13.130859 199.93945 L 12.919922 199.93945 z M 19.882812 199.93945 C 19.902378 200.08474 19.927734 200.22863 19.927734 200.37891 L 19.927734 200.79102 C 20.168811 200.87511 20.455966 200.91694 20.613281 201.06641 C 20.691227 201.14046 20.749315 201.22305 20.806641 201.30273 L 21.259766 200.74023 C 21.519766 200.46023 21.260625 199.90945 20.890625 199.93945 L 19.882812 199.93945 z M 16.435547 200.2207 C 16.301234 200.2207 16.277344 200.24444 16.277344 200.37891 L 16.277344 200.60742 L 16.734375 200.60742 L 16.734375 200.37891 C 16.734375 200.24441 16.712442 200.2207 16.578125 200.2207 L 16.435547 200.2207 z ' fill='#{hex-color($highlight-text-color)}' stroke-width='0' /></svg>");
+    }
+
+    &:hover i.fa-retweet {
+      background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' height='209' width='22'><path d='M 4.9707031 3.1503906 L 4.9707031 3.1601562 C 4.8707031 3.1901563 4.8 3.2598438 4.75 3.3398438 L 0.80078125 8.2402344 C 0.60078125 8.5402344 0.8292187 9.0190625 1.1992188 9.0390625 L 3.5996094 9.0390625 L 3.5996094 11.720703 C 3.5996094 15.980703 3.0497656 15.339844 7.2597656 15.339844 L 11.869141 15.339844 L 11.869141 14.119141 L 11.869141 13.523438 L 11.869141 12.441406 C 11.869141 12.441406 11.869141 12.439453 11.869141 12.439453 L 7.2695312 12.439453 C 6.8295312 12.439453 6.5507814 12.140703 6.5507812 11.720703 L 6.5507812 9.0195312 L 9.0507812 9.0195312 C 9.4207813 9.0495313 9.6792188 8.54 9.4492188 8.25 L 5.5 3.3496094 C 5.38 3.1796094 5.1607031 3.1003906 4.9707031 3.1503906 z M 17.150391 3.5800781 L 17.130859 3.5898438 C 16.580859 3.5698436 15.810469 3.609375 14.730469 3.609375 L 7.0996094 3.609375 L 9.4199219 6.4609375 L 9.4492188 6.5195312 L 14.699219 6.5195312 C 15.106887 6.5195312 15.397113 6.7872181 15.414062 7.2050781 C 15.738375 7.0991315 16.077769 7.0273437 16.435547 7.0273438 L 16.578125 7.0273438 C 17.24903 7.0273438 17.874081 7.2325787 18.400391 7.578125 L 18.400391 7.2402344 C 18.400391 4.0902344 18.800391 3.6200781 17.150391 3.5800781 z M 16.435547 8.0273438 C 15.143818 8.0273438 14.083984 9.0851838 14.083984 10.376953 L 14.083984 11.607422 L 13.570312 11.607422 C 13.375448 11.607422 13.210603 11.704118 13.119141 11.791016 C 13.027691 11.877916 12.983569 11.958238 12.951172 12.03125 C 12.886382 12.177277 12.867187 12.304789 12.867188 12.441406 L 12.867188 13.523438 L 12.867188 14.119141 L 12.867188 15.677734 L 12.867188 16.509766 L 13.570312 16.509766 L 19.472656 16.509766 L 20.173828 16.509766 L 20.173828 15.677734 L 20.173828 13.523438 L 20.173828 12.441406 C 20.173828 12.304794 20.156597 12.177281 20.091797 12.03125 C 20.059397 11.95824 20.015299 11.877916 19.923828 11.791016 C 19.832368 11.704116 19.667509 11.607422 19.472656 11.607422 L 18.927734 11.607422 L 18.927734 10.376953 C 18.927734 9.0851838 17.867902 8.0273438 16.576172 8.0273438 L 16.435547 8.0273438 z M 16.435547 9.2207031 L 16.576172 9.2207031 C 17.22782 9.2207031 17.734375 9.7251013 17.734375 10.376953 L 17.734375 11.607422 L 15.277344 11.607422 L 15.277344 10.376953 C 15.277344 9.7251013 15.7839 9.2207031 16.435547 9.2207031 z M 12.919922 9.9394531 C 12.559922 9.9594531 12.359141 10.480234 12.619141 10.740234 L 12.751953 10.904297 C 12.862211 10.870135 12.980058 10.842244 13.085938 10.802734 L 13.085938 10.378906 C 13.085938 10.228632 13.111295 10.084741 13.130859 9.9394531 L 12.919922 9.9394531 z M 19.882812 9.9394531 C 19.902378 10.084741 19.927734 10.228632 19.927734 10.378906 L 19.927734 10.791016 C 20.168811 10.875098 20.455966 10.916935 20.613281 11.066406 C 20.691227 11.140457 20.749315 11.223053 20.806641 11.302734 L 21.259766 10.740234 C 21.519766 10.460234 21.260625 9.9094531 20.890625 9.9394531 L 19.882812 9.9394531 z M 16.435547 10.220703 C 16.301234 10.220703 16.277344 10.244432 16.277344 10.378906 L 16.277344 10.607422 L 16.734375 10.607422 L 16.734375 10.378906 C 16.734375 10.244433 16.712442 10.220703 16.578125 10.220703 L 16.435547 10.220703 z ' fill='#{hex-color(lighten($action-button-color, 7%))}' stroke-width='0'/><path d='M 7.7792969 19.650391 L 7.7792969 19.660156 C 7.5392969 19.680156 7.3398437 19.910156 7.3398438 20.160156 L 7.3398438 22.619141 L 7.2792969 22.619141 C 6.1992969 22.619141 5.4208594 22.589844 4.8808594 22.589844 C 3.2408594 22.589844 3.6308594 23.020234 3.6308594 26.240234 L 3.6308594 30.710938 C 3.6308594 34.970937 3.0692969 34.330078 7.2792969 34.330078 L 8.5 34.330078 L 7.1992188 33.269531 C 7.0992188 33.189531 7.02 33.070703 7 32.970703 C 6.98 32.800703 7.0592186 32.619531 7.1992188 32.519531 L 8.5292969 31.419922 L 7.2792969 31.419922 C 6.8392969 31.419922 6.5605469 31.120703 6.5605469 30.720703 L 6.5605469 26.240234 C 6.5605469 25.800234 6.8392969 25.519531 7.2792969 25.519531 L 7.3398438 25.519531 L 7.3398438 28.019531 C 7.3398438 28.399531 7.8801564 28.650391 8.1601562 28.400391 L 13.060547 24.470703 C 13.310547 24.290703 13.310547 23.869453 13.060547 23.689453 L 8.1601562 19.769531 C 8.0601563 19.669531 7.9192969 19.630391 7.7792969 19.650391 z M 17.119141 22.580078 L 17.119141 22.589844 C 16.579141 22.569844 15.820703 22.609375 14.720703 22.609375 L 13.470703 22.609375 L 14.769531 23.679688 C 14.869531 23.749688 14.950703 23.879766 14.970703 24.009766 C 14.990703 24.169766 14.909531 24.310156 14.769531 24.410156 L 13.439453 25.509766 L 14.720703 25.509766 C 15.129702 25.509766 15.41841 25.778986 15.433594 26.199219 C 15.752266 26.097283 16.084896 26.027344 16.435547 26.027344 L 16.578125 26.027344 C 17.236645 26.027344 17.848901 26.228565 18.369141 26.5625 L 18.369141 26.240234 C 18.369141 23.090234 18.769141 22.620078 17.119141 22.580078 z M 16.435547 27.027344 C 15.143818 27.027344 14.083984 28.085184 14.083984 29.376953 L 14.083984 30.607422 L 13.570312 30.607422 C 13.375452 30.607422 13.210603 30.704118 13.119141 30.791016 C 13.027691 30.877916 12.983569 30.958238 12.951172 31.03125 C 12.886382 31.177277 12.867184 31.304789 12.867188 31.441406 L 12.867188 32.523438 L 12.867188 33.119141 L 12.867188 34.677734 L 12.867188 35.509766 L 13.570312 35.509766 L 19.472656 35.509766 L 20.173828 35.509766 L 20.173828 34.677734 L 20.173828 32.523438 L 20.173828 31.441406 C 20.173828 31.304794 20.156597 31.177281 20.091797 31.03125 C 20.059397 30.95824 20.015299 30.877916 19.923828 30.791016 C 19.832368 30.704116 19.667509 30.607422 19.472656 30.607422 L 18.927734 30.607422 L 18.927734 29.376953 C 18.927734 28.085184 17.867902 27.027344 16.576172 27.027344 L 16.435547 27.027344 z M 16.435547 28.220703 L 16.576172 28.220703 C 17.22782 28.220703 17.734375 28.725101 17.734375 29.376953 L 17.734375 30.607422 L 15.277344 30.607422 L 15.277344 29.376953 C 15.277344 28.725101 15.7839 28.220703 16.435547 28.220703 z M 13.109375 29.150391 L 8.9199219 32.509766 C 8.6599219 32.689766 8.6599219 33.109063 8.9199219 33.289062 L 11.869141 35.648438 L 11.869141 34.677734 L 11.869141 33.119141 L 11.869141 32.523438 L 11.869141 31.441406 C 11.869141 31.217489 11.912641 30.907486 12.037109 30.626953 C 12.093758 30.499284 12.228597 30.257492 12.429688 30.066406 C 12.580253 29.92335 12.859197 29.887344 13.085938 29.802734 L 13.085938 29.378906 C 13.085938 29.300761 13.104 29.227272 13.109375 29.150391 z M 16.435547 29.220703 C 16.301234 29.220703 16.277344 29.244432 16.277344 29.378906 L 16.277344 29.607422 L 16.734375 29.607422 L 16.734375 29.378906 C 16.734375 29.244433 16.712442 29.220703 16.578125 29.220703 L 16.435547 29.220703 z M 12.943359 36.509766 L 13.820312 37.210938 C 14.090314 37.460938 14.639141 37.210078 14.619141 36.830078 L 14.619141 36.509766 L 13.570312 36.509766 L 12.943359 36.509766 z M 10.330078 38.650391 L 10.339844 38.660156 C 10.099844 38.680156 9.9001562 38.910156 9.9101562 39.160156 L 9.9101562 41.630859 L 7.3007812 41.630859 C 6.2207812 41.630859 5.4403906 41.589844 4.9003906 41.589844 C 3.2603906 41.589844 3.6503906 42.020234 3.6503906 45.240234 L 3.6503906 49.710938 C 3.6503906 53.370936 3.4202344 53.409141 5.9902344 53.369141 L 4.6503906 52.269531 C 4.5503906 52.189531 4.4692187 52.070703 4.4492188 51.970703 C 4.4492188 51.800703 4.5203906 51.619531 4.6503906 51.519531 L 6.609375 49.919922 C 6.579375 49.859922 6.5703125 49.790703 6.5703125 49.720703 L 6.5703125 45.240234 C 6.5703125 44.800234 6.8490625 44.519531 7.2890625 44.519531 L 9.9003906 44.519531 L 9.9003906 47.019531 C 9.9003906 47.379531 10.399219 47.620391 10.699219 47.400391 L 15.630859 43.470703 C 15.870859 43.290703 15.870859 42.869453 15.630859 42.689453 L 10.689453 38.769531 C 10.589453 38.689531 10.460078 38.640391 10.330078 38.650391 z M 16.869141 41.585938 C 16.616211 41.581522 16.322969 41.584844 15.980469 41.589844 L 15.970703 41.589844 L 17.310547 42.689453 C 17.410547 42.759453 17.489766 42.889531 17.509766 43.019531 C 17.529766 43.179531 17.479609 43.319922 17.349609 43.419922 L 15.390625 45.019531 C 15.406724 45.075878 15.427133 45.132837 15.4375 45.197266 C 15.754974 45.096169 16.086404 45.027344 16.435547 45.027344 L 16.578125 45.027344 C 17.24129 45.027344 17.858323 45.230088 18.380859 45.568359 L 18.380859 45.25 C 18.380859 42.0475 18.639648 41.616836 16.869141 41.585938 z M 16.435547 46.027344 C 15.143818 46.027344 14.083984 47.085184 14.083984 48.376953 L 14.083984 49.607422 L 13.570312 49.607422 C 13.375448 49.607422 13.210603 49.704118 13.119141 49.791016 C 13.027691 49.877916 12.983569 49.958238 12.951172 50.03125 C 12.886382 50.177277 12.867187 50.304789 12.867188 50.441406 L 12.867188 51.523438 L 12.867188 52.119141 L 12.867188 53.677734 L 12.867188 54.509766 L 13.570312 54.509766 L 19.472656 54.509766 L 20.173828 54.509766 L 20.173828 53.677734 L 20.173828 51.523438 L 20.173828 50.441406 C 20.173828 50.304794 20.156597 50.177281 20.091797 50.03125 C 20.059397 49.95824 20.015299 49.877916 19.923828 49.791016 C 19.832368 49.704116 19.667509 49.607422 19.472656 49.607422 L 18.927734 49.607422 L 18.927734 48.376953 C 18.927734 47.085184 17.867902 46.027344 16.576172 46.027344 L 16.435547 46.027344 z M 16.435547 47.220703 L 16.576172 47.220703 C 17.22782 47.220703 17.734375 47.725101 17.734375 48.376953 L 17.734375 49.607422 L 15.277344 49.607422 L 15.277344 48.376953 C 15.277344 47.725101 15.7839 47.220703 16.435547 47.220703 z M 11.470703 47.490234 C 11.410703 47.510234 11.349063 47.539844 11.289062 47.589844 L 6.3496094 51.519531 C 6.1096094 51.699531 6.1096094 52.120781 6.3496094 52.300781 L 11.289062 56.220703 C 11.569064 56.440703 12.070312 56.199844 12.070312 55.839844 L 12.070312 55.509766 L 11.869141 55.509766 L 11.869141 53.677734 L 11.869141 52.119141 L 11.869141 51.523438 L 11.869141 50.441406 C 11.869141 50.217489 11.912641 49.907486 12.037109 49.626953 C 12.043809 49.611855 12.061451 49.584424 12.070312 49.566406 L 12.070312 47.960938 C 12.070312 47.660938 11.770703 47.430234 11.470703 47.490234 z M 16.435547 48.220703 C 16.301234 48.220703 16.277344 48.244432 16.277344 48.378906 L 16.277344 48.607422 L 16.734375 48.607422 L 16.734375 48.378906 C 16.734375 48.244433 16.712442 48.220703 16.578125 48.220703 L 16.435547 48.220703 z M 13.060547 57.650391 L 13.060547 57.660156 C 12.830547 57.690156 12.660156 57.920156 12.660156 58.160156 L 12.660156 60.630859 L 7.2792969 60.630859 C 6.1992969 60.630859 5.4208594 60.589844 4.8808594 60.589844 C 3.2408594 60.589844 3.6308594 61.020234 3.6308594 64.240234 L 3.6308594 69.109375 L 6.5605469 66.740234 L 6.5605469 64.240234 C 6.5605469 63.800234 6.8392969 63.519531 7.2792969 63.519531 L 12.660156 63.519531 L 12.660156 66.019531 C 12.660156 66.299799 12.960394 66.500006 13.226562 66.474609 C 13.625751 65.076914 14.904956 64.035678 16.421875 64.029297 L 18.380859 62.470703 C 18.620859 62.290703 18.620859 61.869453 18.380859 61.689453 L 13.439453 57.769531 C 13.339453 57.669531 13.200547 57.630391 13.060547 57.650391 z M 18.359375 63.810547 L 17.800781 64.269531 C 18.004793 64.350836 18.198411 64.450249 18.380859 64.568359 L 18.380859 64.25 L 18.380859 63.810547 L 18.359375 63.810547 z M 16.435547 65.027344 C 15.143818 65.027344 14.083984 66.085184 14.083984 67.376953 L 14.083984 68.607422 L 13.570312 68.607422 C 13.375448 68.607422 13.210603 68.704118 13.119141 68.791016 C 13.027691 68.877916 12.983569 68.958238 12.951172 69.03125 C 12.886382 69.177277 12.867187 69.304789 12.867188 69.441406 L 12.867188 70.523438 L 12.867188 71.119141 L 12.867188 72.677734 L 12.867188 73.509766 L 13.570312 73.509766 L 19.472656 73.509766 L 20.173828 73.509766 L 20.173828 72.677734 L 20.173828 70.523438 L 20.173828 69.441406 C 20.173828 69.304794 20.156597 69.177281 20.091797 69.03125 C 20.059397 68.95824 20.015299 68.877916 19.923828 68.791016 C 19.832368 68.704116 19.667509 68.607422 19.472656 68.607422 L 18.927734 68.607422 L 18.927734 67.376953 C 18.927734 66.085184 17.867902 65.027344 16.576172 65.027344 L 16.435547 65.027344 z M 16.435547 66.220703 L 16.576172 66.220703 C 17.22782 66.220703 17.734375 66.725101 17.734375 67.376953 L 17.734375 68.607422 L 15.277344 68.607422 L 15.277344 67.376953 C 15.277344 66.725101 15.7839 66.220703 16.435547 66.220703 z M 8.7207031 66.509766 C 8.6507031 66.529766 8.5895312 66.559375 8.5195312 66.609375 L 3.5996094 70.519531 C 3.3496094 70.699531 3.3496094 71.120781 3.5996094 71.300781 L 8.5292969 75.220703 C 8.8092969 75.440703 9.3105469 75.199844 9.3105469 74.839844 L 9.3105469 72.339844 L 11.869141 72.339844 L 11.869141 71.119141 L 11.869141 70.523438 L 11.869141 69.449219 L 9.3203125 69.449219 L 9.3203125 66.980469 C 9.3203125 66.680469 9.0007031 66.449766 8.7207031 66.509766 z M 16.435547 67.220703 C 16.301234 67.220703 16.277344 67.244432 16.277344 67.378906 L 16.277344 67.607422 L 16.734375 67.607422 L 16.734375 67.378906 C 16.734375 67.244433 16.712442 67.220703 16.578125 67.220703 L 16.435547 67.220703 z M 19.248047 78.800781 C 19.148558 78.831033 19.050295 78.90106 18.970703 78.970703 L 18.070312 79.869141 C 17.630312 79.569141 16.710703 79.619141 14.720703 79.619141 L 7.2792969 79.619141 C 6.1992969 79.619141 5.4208594 79.589844 4.8808594 79.589844 C 3.2408594 79.589844 3.6308594 80.020234 3.6308594 83.240234 L 3.6308594 83.939453 L 6.5605469 84.240234 L 6.5605469 83.240234 C 6.5605469 82.800234 6.8392969 82.519531 7.2792969 82.519531 L 14.720703 82.519531 C 14.920703 82.519531 15.090703 82.600703 15.220703 82.720703 L 13.419922 84.519531 C 13.279464 84.665607 13.281282 84.881022 13.363281 85.054688 C 13.880838 83.867655 15.067337 83.027344 16.435547 83.027344 L 16.578125 83.027344 C 18.290465 83.027344 19.703357 84.345788 19.890625 86.011719 L 19.960938 86.019531 C 20.240938 86.049531 20.520234 85.770234 20.490234 85.490234 L 19.789062 79.240234 C 19.789062 78.973661 19.498025 78.767523 19.25 78.800781 L 19.248047 78.800781 z M 16.435547 84.027344 C 15.143818 84.027344 14.083984 85.085184 14.083984 86.376953 L 14.083984 87.607422 L 13.570312 87.607422 C 13.375448 87.607422 13.210603 87.704118 13.119141 87.791016 C 13.027691 87.877916 12.983569 87.958238 12.951172 88.03125 C 12.886382 88.177277 12.867187 88.304789 12.867188 88.441406 L 12.867188 89.523438 L 12.867188 90.119141 L 12.867188 91.677734 L 12.867188 92.509766 L 13.570312 92.509766 L 19.472656 92.509766 L 20.173828 92.509766 L 20.173828 91.677734 L 20.173828 89.523438 L 20.173828 88.441406 C 20.173828 88.304794 20.156597 88.177281 20.091797 88.03125 C 20.059397 87.95824 20.015299 87.877916 19.923828 87.791016 C 19.832368 87.704116 19.667509 87.607422 19.472656 87.607422 L 18.927734 87.607422 L 18.927734 86.376953 C 18.927734 85.085184 17.867902 84.027344 16.576172 84.027344 L 16.435547 84.027344 z M 2.0507812 84.900391 C 1.8507824 84.970391 1.6907031 85.199453 1.7207031 85.439453 L 2.4199219 91.689453 C 2.4399219 92.049453 3 92.240929 3.25 91.960938 L 4.0507812 91.160156 C 4.0707812 91.160156 4.0898437 91.140156 4.0898438 91.160156 C 4.5498437 91.400156 5.4595313 91.330078 7.2695312 91.330078 L 11.869141 91.330078 L 11.869141 90.119141 L 11.869141 89.523438 L 11.869141 88.441406 C 11.869141 88.437991 11.871073 88.433136 11.871094 88.429688 L 7.2792969 88.429688 C 7.1292969 88.429688 6.9808594 88.400078 6.8808594 88.330078 L 8.8007812 86.400391 C 9.1007822 86.160391 8.8992969 85.600547 8.5292969 85.560547 L 2.25 84.910156 L 2.0507812 84.910156 L 2.0507812 84.900391 z M 16.435547 85.220703 L 16.576172 85.220703 C 17.22782 85.220703 17.734375 85.725101 17.734375 86.376953 L 17.734375 87.607422 L 15.277344 87.607422 L 15.277344 86.376953 C 15.277344 85.725101 15.7839 85.220703 16.435547 85.220703 z M 4.8808594 98.599609 C 3.5508594 98.599609 3.5400781 99.080402 3.5800781 100.90039 L 4.7207031 99.529297 C 4.8007031 99.429297 4.9405469 99.360078 5.0605469 99.330078 C 5.2205469 99.330078 5.4 99.409297 5.5 99.529297 L 7.1601562 101.56055 C 7.2001563 101.56055 7.2292969 101.5293 7.2792969 101.5293 L 14.720703 101.5293 C 15.060703 101.5293 15.289141 101.7293 15.369141 102.0293 L 12.939453 102.0293 C 12.599453 102.0793 12.410625 102.55055 12.640625 102.81055 L 13.470703 103.85742 C 14.029941 102.77899 15.146801 102.02734 16.435547 102.02734 L 16.578125 102.02734 C 18.158418 102.02734 19.491598 103.14879 19.835938 104.63086 L 21.279297 102.82031 C 21.499297 102.55031 21.260156 102.06078 20.910156 102.05078 L 18.400391 102.05078 C 18.420391 98.150792 19.000234 98.650391 14.740234 98.650391 L 7.2792969 98.650391 C 6.1992969 98.650391 5.4208594 98.609375 4.8808594 98.609375 L 4.8808594 98.599609 z M 5.0292969 101.06055 C 4.9292969 101.09055 4.83 101.15977 4.75 101.25977 L 0.81054688 106.16016 C 0.61054688 106.44016 0.8409375 106.92945 1.2109375 106.93945 L 3.5996094 106.93945 C 3.5796094 110.87945 3.1497656 110.33984 7.2597656 110.33984 L 11.869141 110.33984 L 11.869141 109.11914 L 11.869141 108.52344 L 11.869141 107.44141 L 11.869141 107.43945 L 7.2792969 107.43945 C 6.9292969 107.43945 6.7091406 107.23945 6.6191406 106.93945 L 9.0605469 106.93945 C 9.4305469 106.93945 9.6909375 106.44016 9.4609375 106.16016 L 5.5 101.25977 C 5.4 101.10977 5.1992969 101.03055 5.0292969 101.06055 z M 16.435547 103.02734 C 15.143818 103.02734 14.083984 104.08518 14.083984 105.37695 L 14.083984 106.60742 L 13.570312 106.60742 C 13.375448 106.60742 13.210603 106.70409 13.119141 106.79102 C 13.027691 106.87792 12.983569 106.95823 12.951172 107.03125 C 12.886382 107.17727 12.867187 107.30479 12.867188 107.44141 L 12.867188 108.52344 L 12.867188 109.11914 L 12.867188 110.67773 L 12.867188 111.50977 L 13.570312 111.50977 L 19.472656 111.50977 L 20.173828 111.50977 L 20.173828 110.67773 L 20.173828 108.52344 L 20.173828 107.44141 C 20.173828 107.3048 20.156597 107.17728 20.091797 107.03125 C 20.059397 106.95825 20.015299 106.87792 19.923828 106.79102 C 19.832368 106.70412 19.667509 106.60742 19.472656 106.60742 L 18.927734 106.60742 L 18.927734 105.37695 C 18.927734 104.08518 17.867902 103.02734 16.576172 103.02734 L 16.435547 103.02734 z M 16.435547 104.2207 L 16.576172 104.2207 C 17.22782 104.2207 17.734375 104.7251 17.734375 105.37695 L 17.734375 106.60742 L 15.277344 106.60742 L 15.277344 105.37695 C 15.277344 104.7251 15.7839 104.2207 16.435547 104.2207 z M 16.435547 105.2207 C 16.301234 105.2207 16.277344 105.24444 16.277344 105.37891 L 16.277344 105.60742 L 16.734375 105.60742 L 16.734375 105.37891 C 16.734375 105.24441 16.712442 105.2207 16.578125 105.2207 L 16.435547 105.2207 z M 4.8808594 117.58984 L 4.8808594 117.59961 C 3.7208594 117.59961 3.5800781 117.90016 3.5800781 119.16016 L 4.7207031 117.7793 C 4.8007031 117.6793 4.9405469 117.63914 5.0605469 117.61914 C 5.2205469 117.61914 5.4 117.6593 5.5 117.7793 L 7.7207031 120.5293 L 14.720703 120.5293 C 15.123595 120.5293 15.408576 120.79174 15.431641 121.20117 C 15.750992 121.09876 16.08404 121.02734 16.435547 121.02734 L 16.578125 121.02734 C 17.24903 121.02734 17.874081 121.23262 18.400391 121.57812 L 18.400391 121.25 C 18.400391 117.05 19.120234 117.61914 14.740234 117.61914 L 7.2792969 117.61914 C 6.1992969 117.61914 5.4208594 117.58984 4.8808594 117.58984 z M 4.9804688 119.33984 C 4.8804688 119.36984 4.81 119.44 4.75 119.5 L 0.80078125 124.43945 C 0.60078125 124.71945 0.8292182 125.2107 1.1992188 125.2207 L 3.5996094 125.2207 L 3.5996094 125.7207 C 3.5996094 129.9807 3.0497656 129.33984 7.2597656 129.33984 L 11.869141 129.33984 L 11.869141 128.11914 L 11.869141 127.52344 L 11.869141 126.44141 C 11.869141 126.43799 11.871073 126.43314 11.871094 126.42969 L 7.2792969 126.42969 C 6.8392969 126.42969 6.5605469 126.13094 6.5605469 125.71094 L 6.5605469 125.21094 L 9.0605469 125.21094 C 9.4305469 125.23094 9.6909375 124.70969 9.4609375 124.42969 L 5.5 119.5 C 5.3820133 119.35252 5.1682348 119.28513 4.9804688 119.33984 z M 12.839844 121.7793 C 12.539844 121.8793 12.410625 122.32055 12.640625 122.56055 L 13.267578 123.34375 C 13.473522 122.72168 13.852237 122.1828 14.353516 121.7793 L 12.839844 121.7793 z M 18.658203 121.7793 C 19.393958 122.37155 19.878978 123.25738 19.916016 124.25781 L 21.279297 122.56055 C 21.499297 122.28055 21.260156 121.7893 20.910156 121.7793 L 18.658203 121.7793 z M 16.435547 122.02734 C 15.143818 122.02734 14.083984 123.08518 14.083984 124.37695 L 14.083984 125.60742 L 13.570312 125.60742 C 13.375448 125.60742 13.210603 125.70409 13.119141 125.79102 C 13.027691 125.87792 12.983569 125.95823 12.951172 126.03125 C 12.886382 126.17727 12.867187 126.30479 12.867188 126.44141 L 12.867188 127.52344 L 12.867188 128.11914 L 12.867188 129.67773 L 12.867188 130.50977 L 13.570312 130.50977 L 19.472656 130.50977 L 20.173828 130.50977 L 20.173828 129.67773 L 20.173828 127.52344 L 20.173828 126.44141 C 20.173828 126.3048 20.156597 126.17728 20.091797 126.03125 C 20.059397 125.95825 20.015299 125.87792 19.923828 125.79102 C 19.832368 125.70412 19.667509 125.60742 19.472656 125.60742 L 18.927734 125.60742 L 18.927734 124.37695 C 18.927734 123.08518 17.867902 122.02734 16.576172 122.02734 L 16.435547 122.02734 z M 16.435547 123.2207 L 16.576172 123.2207 C 17.22782 123.2207 17.734375 123.7251 17.734375 124.37695 L 17.734375 125.60742 L 15.277344 125.60742 L 15.277344 124.37695 C 15.277344 123.7251 15.7839 123.2207 16.435547 123.2207 z M 16.435547 124.2207 C 16.301234 124.2207 16.277344 124.24444 16.277344 124.37891 L 16.277344 124.60742 L 16.734375 124.60742 L 16.734375 124.37891 C 16.734375 124.24441 16.712442 124.2207 16.578125 124.2207 L 16.435547 124.2207 z M 5.9394531 136.58984 L 5.9394531 136.59961 L 8.3105469 139.5293 L 14.730469 139.5293 C 15.131912 139.5293 15.414551 139.79039 15.439453 140.19727 C 15.756409 140.09653 16.087055 140.02734 16.435547 140.02734 L 16.578125 140.02734 C 17.24903 140.02734 17.874081 140.23261 18.400391 140.57812 L 18.400391 140.25 C 18.400391 136.05 19.120234 136.61914 14.740234 136.61914 L 7.2792969 136.61914 C 6.6792969 136.61914 6.3594531 136.59984 5.9394531 136.58984 z M 4.2207031 136.66016 C 3.8207031 136.74016 3.6791406 136.96016 3.6191406 137.41016 L 4.2207031 136.66992 L 4.2207031 136.66016 z M 5.0605469 137.57031 L 5.0605469 137.58984 C 4.9405469 137.58984 4.8197656 137.66953 4.7597656 137.76953 L 0.81054688 142.66992 C 0.57054688 142.96992 0.8109375 143.50023 1.2109375 143.49023 L 3.5996094 143.49023 L 3.5996094 144.71094 C 3.5996094 148.97094 3.0497656 148.33008 7.2597656 148.33008 L 11.869141 148.33008 L 11.869141 147.11914 L 11.869141 146.52344 L 11.869141 145.44141 C 11.869141 145.43799 11.871073 145.43314 11.871094 145.42969 L 7.2792969 145.42969 C 6.8392969 145.42969 6.5605469 145.13094 6.5605469 144.71094 L 6.5605469 143.49023 L 9.0605469 143.49023 C 9.4605469 143.53023 9.7309375 142.95945 9.4609375 142.68945 L 5.5 137.76953 C 5.4 137.63953 5.2305469 137.57031 5.0605469 137.57031 z M 16.435547 141.02734 C 15.143818 141.02734 14.083984 142.08518 14.083984 143.37695 L 14.083984 144.60742 L 13.570312 144.60742 C 13.375448 144.60742 13.210603 144.70409 13.119141 144.79102 C 13.027691 144.87792 12.983569 144.95823 12.951172 145.03125 C 12.886382 145.17727 12.867187 145.30479 12.867188 145.44141 L 12.867188 146.52344 L 12.867188 147.11914 L 12.867188 148.67773 L 12.867188 149.50977 L 13.570312 149.50977 L 19.472656 149.50977 L 20.173828 149.50977 L 20.173828 148.67773 L 20.173828 146.52344 L 20.173828 145.44141 C 20.173828 145.3048 20.156597 145.17728 20.091797 145.03125 C 20.059397 144.95825 20.015299 144.87792 19.923828 144.79102 C 19.832368 144.70412 19.667509 144.60742 19.472656 144.60742 L 18.927734 144.60742 L 18.927734 143.37695 C 18.927734 142.08518 17.867902 141.02734 16.576172 141.02734 L 16.435547 141.02734 z M 12.849609 141.5 C 12.549609 141.6 12.420391 142.0393 12.650391 142.2793 L 13.136719 142.88672 C 13.213026 142.38119 13.390056 141.90696 13.667969 141.5 L 12.849609 141.5 z M 19.34375 141.5 C 19.710704 142.03735 19.927734 142.68522 19.927734 143.37891 L 19.927734 143.79102 C 19.965561 143.80421 20.005506 143.81448 20.044922 143.82617 L 21.289062 142.2793 C 21.509062 141.9993 21.269922 141.51 20.919922 141.5 L 19.34375 141.5 z M 16.435547 142.2207 L 16.576172 142.2207 C 17.22782 142.2207 17.734375 142.7251 17.734375 143.37695 L 17.734375 144.60742 L 15.277344 144.60742 L 15.277344 143.37695 C 15.277344 142.7251 15.7839 142.2207 16.435547 142.2207 z M 16.435547 143.2207 C 16.301234 143.2207 16.277344 143.24444 16.277344 143.37891 L 16.277344 143.60742 L 16.734375 143.60742 L 16.734375 143.37891 C 16.734375 143.24441 16.712442 143.2207 16.578125 143.2207 L 16.435547 143.2207 z M 17.130859 155.59961 C 16.580859 155.57961 15.810469 155.63086 14.730469 155.63086 L 6.5292969 155.63086 L 8.9101562 158.5293 L 14.730469 158.5293 C 15.131912 158.5293 15.414551 158.79039 15.439453 159.19727 C 15.756409 159.09653 16.087055 159.02734 16.435547 159.02734 L 16.578125 159.02734 C 17.24903 159.02734 17.874081 159.23261 18.400391 159.57812 L 18.400391 159.25977 C 18.400391 156.10977 18.800391 155.63961 17.150391 155.59961 L 17.130859 155.59961 z M 5.0292969 155.86914 L 5.0292969 155.88086 C 4.9292969 155.90086 4.83 155.98055 4.75 156.06055 L 0.81054688 160.96094 C 0.61054688 161.26094 0.8409375 161.73977 1.2109375 161.75977 L 3.5996094 161.75977 L 3.5996094 163.7207 C 3.5996094 167.9807 3.0497656 167.33984 7.2597656 167.33984 L 11.869141 167.33984 L 11.869141 166.11914 L 11.869141 165.52344 L 11.869141 164.44141 L 11.869141 164.43945 L 7.2792969 164.43945 C 6.8392969 164.43945 6.5605469 164.1407 6.5605469 163.7207 L 6.5605469 161.75 L 9.0605469 161.75 C 9.4305469 161.77 9.6909375 161.2507 9.4609375 160.9707 L 5.5 156.07031 C 5.4 155.92031 5.1992969 155.84914 5.0292969 155.86914 z M 16.435547 160.02734 C 15.143818 160.02734 14.083984 161.08518 14.083984 162.37695 L 14.083984 163.60742 L 13.570312 163.60742 C 13.375448 163.60742 13.210603 163.70409 13.119141 163.79102 C 13.027691 163.87792 12.983569 163.95823 12.951172 164.03125 C 12.886382 164.17727 12.867187 164.30479 12.867188 164.44141 L 12.867188 165.52344 L 12.867188 166.11914 L 12.867188 167.67773 L 12.867188 168.50977 L 13.570312 168.50977 L 19.472656 168.50977 L 20.173828 168.50977 L 20.173828 167.67773 L 20.173828 165.52344 L 20.173828 164.44141 C 20.173828 164.3048 20.156597 164.17728 20.091797 164.03125 C 20.059397 163.95825 20.015299 163.87792 19.923828 163.79102 C 19.832368 163.70412 19.667509 163.60742 19.472656 163.60742 L 18.927734 163.60742 L 18.927734 162.37695 C 18.927734 161.08518 17.867902 160.02734 16.576172 160.02734 L 16.435547 160.02734 z M 12.900391 161.2207 C 12.580391 161.2807 12.419141 161.74 12.619141 162 L 13.085938 162.58594 L 13.085938 162.37891 C 13.085938 161.97087 13.170592 161.58376 13.306641 161.2207 L 12.900391 161.2207 z M 16.435547 161.2207 L 16.576172 161.2207 C 17.22782 161.2207 17.734375 161.7251 17.734375 162.37695 L 17.734375 163.60742 L 15.277344 163.60742 L 15.277344 162.37695 C 15.277344 161.7251 15.7839 161.2207 16.435547 161.2207 z M 19.708984 161.23047 C 19.842743 161.59081 19.927734 161.97449 19.927734 162.37891 L 19.927734 162.79102 C 20.119162 162.85779 20.322917 162.91147 20.484375 163 L 21.279297 162.00977 C 21.499297 161.72977 21.260156 161.24047 20.910156 161.23047 L 19.708984 161.23047 z M 16.435547 162.2207 C 16.301234 162.2207 16.277344 162.24444 16.277344 162.37891 L 16.277344 162.60742 L 16.734375 162.60742 L 16.734375 162.37891 C 16.734375 162.24441 16.712442 162.2207 16.578125 162.2207 L 16.435547 162.2207 z M 5.0996094 174.49023 L 5.1308594 174.5 C 4.9808594 174.5 4.83 174.56922 4.75 174.69922 L 0.80078125 179.59961 C 0.56078125 179.86961 0.7992182 180.42039 1.1992188 180.40039 L 3.5996094 180.40039 L 3.5996094 182.7207 C 3.5996094 186.9807 3.0497656 186.33984 7.2597656 186.33984 L 11.869141 186.33984 L 11.869141 185.11914 L 11.869141 184.52344 L 11.869141 183.44141 L 11.869141 183.43945 L 7.25 183.43945 C 6.82 183.43945 6.5507814 183.1407 6.5507812 182.7207 L 6.5507812 180.41992 L 9.0507812 180.41992 C 9.4307824 180.44992 9.7092187 179.87984 9.4492188 179.58984 L 5.4804688 174.68945 C 5.3804688 174.55945 5.2496094 174.49023 5.0996094 174.49023 z M 17.150391 174.58008 L 17.130859 174.59961 C 16.580859 174.57961 15.810469 174.63086 14.730469 174.63086 L 6.8300781 174.63086 L 9.1796875 177.5293 L 14.699219 177.5293 C 15.104107 177.5293 15.391475 177.79407 15.412109 178.20703 C 15.737096 178.1006 16.076913 178.02734 16.435547 178.02734 L 16.578125 178.02734 C 17.24903 178.02734 17.874081 178.2326 18.400391 178.57812 L 18.400391 178.24023 C 18.400391 175.09023 18.800391 174.62008 17.150391 174.58008 z M 16.435547 179.02734 C 15.143818 179.02734 14.083984 180.08518 14.083984 181.37695 L 14.083984 182.60742 L 13.570312 182.60742 C 13.375448 182.60742 13.210603 182.70409 13.119141 182.79102 C 13.027691 182.87792 12.983569 182.95823 12.951172 183.03125 C 12.886382 183.17727 12.867187 183.30479 12.867188 183.44141 L 12.867188 184.52344 L 12.867188 185.11914 L 12.867188 186.67773 L 12.867188 187.50977 L 13.570312 187.50977 L 19.472656 187.50977 L 20.173828 187.50977 L 20.173828 186.67773 L 20.173828 184.52344 L 20.173828 183.44141 C 20.173828 183.3048 20.156597 183.17728 20.091797 183.03125 C 20.059397 182.95825 20.015299 182.87792 19.923828 182.79102 C 19.832368 182.70412 19.667509 182.60742 19.472656 182.60742 L 18.927734 182.60742 L 18.927734 181.37695 C 18.927734 180.08518 17.867902 179.02734 16.576172 179.02734 L 16.435547 179.02734 z M 16.435547 180.2207 L 16.576172 180.2207 C 17.22782 180.2207 17.734375 180.7251 17.734375 181.37695 L 17.734375 182.60742 L 15.277344 182.60742 L 15.277344 181.37695 C 15.277344 180.7251 15.7839 180.2207 16.435547 180.2207 z M 19.816406 180.57031 C 19.882311 180.83091 19.927734 181.09907 19.927734 181.37891 L 19.927734 181.79102 C 20.168811 181.87511 20.455966 181.91694 20.613281 182.06641 C 20.630645 182.0829 20.639883 182.10199 20.65625 182.11914 L 21.259766 181.36914 C 21.479766 181.06914 21.240625 180.59031 20.890625 180.57031 L 19.816406 180.57031 z M 12.820312 180.58984 C 12.520316 180.68984 12.389141 181.11914 12.619141 181.36914 L 12.990234 181.83203 C 13.022029 181.82207 13.055579 181.81406 13.085938 181.80273 L 13.085938 181.37891 C 13.085938 181.10616 13.128698 180.84442 13.191406 180.58984 L 12.820312 180.58984 z M 16.435547 181.2207 C 16.301234 181.2207 16.277344 181.24444 16.277344 181.37891 L 16.277344 181.60742 L 16.734375 181.60742 L 16.734375 181.37891 C 16.734375 181.24441 16.712442 181.2207 16.578125 181.2207 L 16.435547 181.2207 z M 4.9609375 193.15039 L 4.9707031 193.16016 C 4.8707031 193.19016 4.8 193.25984 4.75 193.33984 L 0.81054688 198.24023 C 0.61054688 198.54023 0.8409375 199.01906 1.2109375 199.03906 L 3.5996094 199.03906 L 3.5996094 201.7207 C 3.5996094 205.9807 3.0497656 205.33984 7.2597656 205.33984 L 11.869141 205.33984 L 11.869141 204.11914 L 11.869141 203.52344 L 11.869141 202.44141 C 11.869141 202.44141 11.869141 202.43945 11.869141 202.43945 L 7.2695312 202.43945 C 6.8295312 202.43945 6.5507814 202.1407 6.5507812 201.7207 L 6.5507812 199.01953 L 9.0507812 199.01953 C 9.4207814 199.04953 9.6792188 198.54 9.4492188 198.25 L 5.4902344 193.34961 C 5.3702344 193.17961 5.1509375 193.10039 4.9609375 193.15039 z M 17.150391 193.58008 L 17.130859 193.58984 C 16.580859 193.56984 15.810469 193.61914 14.730469 193.61914 L 7.0996094 193.61914 L 9.4199219 196.46094 L 9.4492188 196.51953 L 14.699219 196.51953 C 15.106887 196.51953 15.397075 196.78718 15.414062 197.20508 C 15.738375 197.09913 16.077769 197.02734 16.435547 197.02734 L 16.578125 197.02734 C 17.24903 197.02734 17.874081 197.23259 18.400391 197.57812 L 18.400391 197.24023 C 18.400391 194.09023 18.800391 193.62008 17.150391 193.58008 z M 16.435547 198.02734 C 15.143818 198.02734 14.083984 199.08518 14.083984 200.37695 L 14.083984 201.60742 L 13.570312 201.60742 C 13.375448 201.60742 13.210603 201.70409 13.119141 201.79102 C 13.027691 201.87792 12.983569 201.95823 12.951172 202.03125 C 12.886382 202.17727 12.867187 202.30479 12.867188 202.44141 L 12.867188 203.52344 L 12.867188 204.11914 L 12.867188 205.67773 L 12.867188 206.50977 L 13.570312 206.50977 L 19.472656 206.50977 L 20.173828 206.50977 L 20.173828 205.67773 L 20.173828 203.52344 L 20.173828 202.44141 C 20.173828 202.3048 20.156597 202.17728 20.091797 202.03125 C 20.059397 201.95825 20.015299 201.87792 19.923828 201.79102 C 19.832368 201.70412 19.667509 201.60742 19.472656 201.60742 L 18.927734 201.60742 L 18.927734 200.37695 C 18.927734 199.08518 17.867902 198.02734 16.576172 198.02734 L 16.435547 198.02734 z M 16.435547 199.2207 L 16.576172 199.2207 C 17.22782 199.2207 17.734375 199.7251 17.734375 200.37695 L 17.734375 201.60742 L 15.277344 201.60742 L 15.277344 200.37695 C 15.277344 199.7251 15.7839 199.2207 16.435547 199.2207 z M 12.919922 199.93945 C 12.559922 199.95945 12.359141 200.48023 12.619141 200.74023 L 12.751953 200.9043 C 12.862211 200.87013 12.980058 200.84224 13.085938 200.80273 L 13.085938 200.37891 C 13.085938 200.22863 13.111295 200.08474 13.130859 199.93945 L 12.919922 199.93945 z M 19.882812 199.93945 C 19.902378 200.08474 19.927734 200.22863 19.927734 200.37891 L 19.927734 200.79102 C 20.168811 200.87511 20.455966 200.91694 20.613281 201.06641 C 20.691227 201.14046 20.749315 201.22305 20.806641 201.30273 L 21.259766 200.74023 C 21.519766 200.46023 21.260625 199.90945 20.890625 199.93945 L 19.882812 199.93945 z M 16.435547 200.2207 C 16.301234 200.2207 16.277344 200.24444 16.277344 200.37891 L 16.277344 200.60742 L 16.734375 200.60742 L 16.734375 200.37891 C 16.734375 200.24441 16.712442 200.2207 16.578125 200.2207 L 16.435547 200.2207 z ' fill='#{hex-color($highlight-text-color)}' stroke-width='0' /></svg>");
+    }
+  }
+
+  &.disabled {
+    i.fa-retweet,
+    &:hover i.fa-retweet {
+      background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' height='209' width='22'><path d='M 18.972656 1.2011719 C 18.829825 1.1881782 18.685932 1.2302188 18.572266 1.3300781 L 15.990234 3.5996094 C 15.58109 3.6070661 15.297269 3.609375 14.730469 3.609375 L 7.0996094 3.609375 L 9.4199219 6.4609375 L 9.4492188 6.5195312 L 12.664062 6.5195312 L 6.5761719 11.867188 C 6.5674697 11.818249 6.5507813 11.773891 6.5507812 11.720703 L 6.5507812 9.0195312 L 9.0507812 9.0195312 C 9.4207813 9.0495313 9.6792188 8.54 9.4492188 8.25 L 5.5 3.3496094 C 5.38 3.1796094 5.1607031 3.1003906 4.9707031 3.1503906 L 4.9707031 3.1601562 C 4.8707031 3.1901563 4.8 3.2598438 4.75 3.3398438 L 0.80078125 8.2402344 C 0.60078125 8.5402344 0.8292187 9.0190625 1.1992188 9.0390625 L 3.5996094 9.0390625 L 3.5996094 11.720703 C 3.5996094 13.045739 3.5690668 13.895038 3.6503906 14.4375 L 2.6152344 15.347656 C 2.3879011 15.547375 2.3754917 15.901081 2.5859375 16.140625 L 3.1464844 16.78125 C 3.3569308 17.020794 3.7101667 17.053234 3.9375 16.853516 L 19.892578 2.8359375 C 20.119911 2.6362188 20.134275 2.282513 19.923828 2.0429688 L 19.361328 1.4023438 C 19.256105 1.282572 19.115488 1.2141655 18.972656 1.2011719 z M 18.410156 6.7753906 L 15.419922 9.4042969 L 15.419922 9.9394531 L 14.810547 9.9394531 L 13.148438 11.400391 L 16.539062 15.640625 C 16.719062 15.890625 17.140313 15.890625 17.320312 15.640625 L 21.259766 10.740234 C 21.519766 10.460234 21.260625 9.9094531 20.890625 9.9394531 L 18.400391 9.9394531 L 18.400391 7.2402344 C 18.400391 7.0470074 18.407711 6.9489682 18.410156 6.7753906 z M 11.966797 12.439453 L 8.6679688 15.339844 L 14.919922 15.339844 L 12.619141 12.5 C 12.589141 12.48 12.590313 12.459453 12.570312 12.439453 L 11.966797 12.439453 z' fill='#{hex-color(darken($action-button-color, 13%))}' stroke-width='0'/></svg>");
+    }
+  }
+
+  .media-modal__overlay .picture-in-picture__footer & {
+    i.fa-retweet {
+      background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' width='22' height='209'><path d='M4.97 3.16c-.1.03-.17.1-.22.18L.8 8.24c-.2.3.03.78.4.8H3.6v2.68c0 4.26-.55 3.62 3.66 3.62h7.66l-2.3-2.84c-.03-.02-.03-.04-.05-.06H7.27c-.44 0-.72-.3-.72-.72v-2.7h2.5c.37.03.63-.48.4-.77L5.5 3.35c-.12-.17-.34-.25-.53-.2zm12.16.43c-.55-.02-1.32.02-2.4.02H7.1l2.32 2.85.03.06h5.25c.42 0 .72.28.72.72v2.7h-2.5c-.36.02-.56.54-.3.8l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.26-.28 0-.83-.37-.8H18.4v-2.7c0-3.15.4-3.62-1.25-3.66z' fill='#{hex-color($white)}' stroke-width='0'/><path d='M7.78 19.66c-.24.02-.44.25-.44.5v2.46h-.06c-1.08 0-1.86-.03-2.4-.03-1.64 0-1.25.43-1.25 3.65v4.47c0 4.26-.56 3.62 3.65 3.62H8.5l-1.3-1.06c-.1-.08-.18-.2-.2-.3-.02-.17.06-.35.2-.45l1.33-1.1H7.28c-.44 0-.72-.3-.72-.7v-4.48c0-.44.28-.72.72-.72h.06v2.5c0 .38.54.63.82.38l4.9-3.93c.25-.18.25-.6 0-.78l-4.9-3.92c-.1-.1-.24-.14-.38-.12zm9.34 2.93c-.54-.02-1.3.02-2.4.02h-1.25l1.3 1.07c.1.07.18.2.2.33.02.16-.06.3-.2.4l-1.33 1.1h1.28c.42 0 .72.28.72.72v4.47c0 .42-.3.72-.72.72h-.1v-2.47c0-.3-.3-.53-.6-.47-.07 0-.14.05-.2.1l-4.9 3.93c-.26.18-.26.6 0 .78l4.9 3.92c.27.25.82 0 .8-.38v-2.5h.1c4.27 0 3.65.67 3.65-3.62v-4.47c0-3.15.4-3.62-1.25-3.66zM10.34 38.66c-.24.02-.44.25-.43.5v2.47H7.3c-1.08 0-1.86-.04-2.4-.04-1.64 0-1.25.43-1.25 3.65v4.47c0 3.66-.23 3.7 2.34 3.66l-1.34-1.1c-.1-.08-.18-.2-.2-.3 0-.17.07-.35.2-.45l1.96-1.6c-.03-.06-.04-.13-.04-.2v-4.48c0-.44.28-.72.72-.72H9.9v2.5c0 .36.5.6.8.38l4.93-3.93c.24-.18.24-.6 0-.78l-4.94-3.92c-.1-.08-.23-.13-.36-.12zm5.63 2.93l1.34 1.1c.1.07.18.2.2.33.02.16-.03.3-.16.4l-1.96 1.6c.02.07.06.13.06.22v4.47c0 .42-.3.72-.72.72h-2.66v-2.47c0-.3-.3-.53-.6-.47-.06.02-.12.05-.18.1l-4.94 3.93c-.24.18-.24.6 0 .78l4.94 3.92c.28.22.78-.02.78-.38v-2.5h2.66c4.27 0 3.65.67 3.65-3.62v-4.47c0-3.66.34-3.7-2.4-3.66zM13.06 57.66c-.23.03-.4.26-.4.5v2.47H7.28c-1.08 0-1.86-.04-2.4-.04-1.64 0-1.25.43-1.25 3.65v4.87l2.93-2.37v-2.5c0-.44.28-.72.72-.72h5.38v2.5c0 .36.5.6.78.38l4.94-3.93c.24-.18.24-.6 0-.78l-4.94-3.92c-.1-.1-.24-.14-.38-.12zm5.3 6.15l-2.92 2.4v2.52c0 .42-.3.72-.72.72h-5.4v-2.47c0-.3-.32-.53-.6-.47-.07.02-.13.05-.2.1L3.6 70.52c-.25.18-.25.6 0 .78l4.93 3.92c.28.22.78-.02.78-.38v-2.5h5.42c4.27 0 3.65.67 3.65-3.62v-4.47-.44zM19.25 78.8c-.1.03-.2.1-.28.17l-.9.9c-.44-.3-1.36-.25-3.35-.25H7.28c-1.08 0-1.86-.03-2.4-.03-1.64 0-1.25.43-1.25 3.65v.7l2.93.3v-1c0-.44.28-.72.72-.72h7.44c.2 0 .37.08.5.2l-1.8 1.8c-.25.26-.08.76.27.8l6.27.7c.28.03.56-.25.53-.53l-.7-6.25c0-.27-.3-.48-.55-.44zm-17.2 6.1c-.2.07-.36.3-.33.54l.7 6.25c.02.36.58.55.83.27l.8-.8c.02 0 .04-.02.04 0 .46.24 1.37.17 3.18.17h7.44c4.27 0 3.65.67 3.65-3.62v-.75l-2.93-.3v1.05c0 .42-.3.72-.72.72H7.28c-.15 0-.3-.03-.4-.1L8.8 86.4c.3-.24.1-.8-.27-.84l-6.28-.65h-.2zM4.88 98.6c-1.33 0-1.34.48-1.3 2.3l1.14-1.37c.08-.1.22-.17.34-.2.16 0 .34.08.44.2l1.66 2.03c.04 0 .07-.03.12-.03h7.44c.34 0 .57.2.65.5h-2.43c-.34.05-.53.52-.3.78l3.92 4.95c.18.24.6.24.78 0l3.94-4.94c.22-.27-.02-.76-.37-.77H18.4c.02-3.9.6-3.4-3.66-3.4H7.28c-1.08 0-1.86-.04-2.4-.04zm.15 2.46c-.1.03-.2.1-.28.2l-3.94 4.9c-.2.28.03.77.4.78H3.6c-.02 3.94-.45 3.4 3.66 3.4h7.44c3.65 0 3.74.3 3.7-2.25l-1.1 1.34c-.1.1-.2.17-.32.2-.16 0-.34-.08-.44-.2l-1.65-2.03c-.06.02-.1.04-.18.04H7.28c-.35 0-.57-.2-.66-.5h2.44c.37 0 .63-.5.4-.78l-3.96-4.9c-.1-.15-.3-.23-.47-.2zM4.88 117.6c-1.16 0-1.3.3-1.3 1.56l1.14-1.38c.08-.1.22-.14.34-.16.16 0 .34.04.44.16l2.22 2.75h7c.42 0 .72.28.72.72v.53h-2.6c-.3.1-.43.54-.2.78l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-.53c0-4.2.72-3.63-3.66-3.63H7.28c-1.08 0-1.86-.03-2.4-.03zm.1 1.74c-.1.03-.17.1-.23.16L.8 124.44c-.2.28.03.77.4.78H3.6v.5c0 4.26-.55 3.62 3.66 3.62h7.44c1.03 0 1.74.02 2.28 0-.16.02-.34-.03-.44-.15l-2.22-2.76H7.28c-.44 0-.72-.3-.72-.72v-.5h2.5c.37.02.63-.5.4-.78L5.5 119.5c-.12-.15-.34-.22-.53-.16zm12.02 10c1.2-.02 1.4-.25 1.4-1.53l-1.1 1.36c-.07.1-.17.17-.3.18zM5.94 136.6l2.37 2.93h6.42c.42 0 .72.28.72.72v1.25h-2.6c-.3.1-.43.54-.2.78l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-1.25c0-4.2.72-3.63-3.66-3.63H7.28c-.6 0-.92-.02-1.34-.03zm-1.72.06c-.4.08-.54.3-.6.75l.6-.74zm.84.93c-.12 0-.24.08-.3.18l-3.95 4.9c-.24.3 0 .83.4.82H3.6v1.22c0 4.26-.55 3.62 3.66 3.62h7.44c.63 0 .97.02 1.4.03l-2.37-2.93H7.28c-.44 0-.72-.3-.72-.72v-1.22h2.5c.4.04.67-.53.4-.8l-3.96-4.92c-.1-.13-.27-.2-.44-.2zm13.28 10.03l-.56.7c.36-.07.5-.3.56-.7zM17.13 155.6c-.55-.02-1.32.03-2.4.03h-8.2l2.38 2.9h5.82c.42 0 .72.28.72.72v1.97H12.9c-.32.06-.48.52-.28.78l3.94 4.94c.2.23.6.22.78-.03l3.94-4.9c.22-.28-.02-.77-.37-.78H18.4v-1.97c0-3.15.4-3.62-1.25-3.66zm-12.1.28c-.1.02-.2.1-.28.18l-3.94 4.9c-.2.3.03.78.4.8H3.6v1.96c0 4.26-.55 3.62 3.66 3.62h8.24l-2.36-2.9H7.28c-.44 0-.72-.3-.72-.72v-1.97h2.5c.37.02.63-.5.4-.78l-3.96-4.9c-.1-.15-.3-.22-.47-.2zM5.13 174.5c-.15 0-.3.07-.38.2L.8 179.6c-.24.27 0 .82.4.8H3.6v2.32c0 4.26-.55 3.62 3.66 3.62h7.94l-2.35-2.9h-5.6c-.43 0-.7-.3-.7-.72v-2.3h2.5c.38.03.66-.54.4-.83l-3.97-4.9c-.1-.13-.23-.2-.38-.2zm12 .1c-.55-.02-1.32.03-2.4.03H6.83l2.35 2.9h5.52c.42 0 .72.28.72.72v2.34h-2.6c-.3.1-.43.53-.2.78l3.92 4.9c.18.24.6.24.78 0l3.94-4.9c.22-.3-.02-.78-.37-.8H18.4v-2.33c0-3.15.4-3.62-1.25-3.66zM4.97 193.16c-.1.03-.17.1-.22.18l-3.94 4.9c-.2.3.03.78.4.8H3.6v2.68c0 4.26-.55 3.62 3.66 3.62h7.66l-2.3-2.84c-.03-.02-.03-.04-.05-.06H7.27c-.44 0-.72-.3-.72-.72v-2.7h2.5c.37.03.63-.48.4-.77l-3.96-4.9c-.12-.17-.34-.25-.53-.2zm12.16.43c-.55-.02-1.32.03-2.4.03H7.1l2.32 2.84.03.06h5.25c.42 0 .72.28.72.72v2.7h-2.5c-.36.02-.56.54-.3.8l3.92 4.9c.18.25.6.25.78 0l3.94-4.9c.26-.28 0-.83-.37-.8H18.4v-2.7c0-3.15.4-3.62-1.25-3.66z' fill='#{hex-color($highlight-text-color)}' stroke-width='0'/></svg>");
+    }
+
+    &.reblogPrivate {
+      i.fa-retweet {
+        background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' height='209' width='22'><path d='M 4.9707031 3.1503906 L 4.9707031 3.1601562 C 4.8707031 3.1901563 4.8 3.2598438 4.75 3.3398438 L 0.80078125 8.2402344 C 0.60078125 8.5402344 0.8292187 9.0190625 1.1992188 9.0390625 L 3.5996094 9.0390625 L 3.5996094 11.720703 C 3.5996094 15.980703 3.0497656 15.339844 7.2597656 15.339844 L 11.869141 15.339844 L 11.869141 14.119141 L 11.869141 13.523438 L 11.869141 12.441406 C 11.869141 12.441406 11.869141 12.439453 11.869141 12.439453 L 7.2695312 12.439453 C 6.8295312 12.439453 6.5507814 12.140703 6.5507812 11.720703 L 6.5507812 9.0195312 L 9.0507812 9.0195312 C 9.4207813 9.0495313 9.6792188 8.54 9.4492188 8.25 L 5.5 3.3496094 C 5.38 3.1796094 5.1607031 3.1003906 4.9707031 3.1503906 z M 17.150391 3.5800781 L 17.130859 3.5898438 C 16.580859 3.5698436 15.810469 3.609375 14.730469 3.609375 L 7.0996094 3.609375 L 9.4199219 6.4609375 L 9.4492188 6.5195312 L 14.699219 6.5195312 C 15.106887 6.5195312 15.397113 6.7872181 15.414062 7.2050781 C 15.738375 7.0991315 16.077769 7.0273437 16.435547 7.0273438 L 16.578125 7.0273438 C 17.24903 7.0273438 17.874081 7.2325787 18.400391 7.578125 L 18.400391 7.2402344 C 18.400391 4.0902344 18.800391 3.6200781 17.150391 3.5800781 z M 16.435547 8.0273438 C 15.143818 8.0273438 14.083984 9.0851838 14.083984 10.376953 L 14.083984 11.607422 L 13.570312 11.607422 C 13.375448 11.607422 13.210603 11.704118 13.119141 11.791016 C 13.027691 11.877916 12.983569 11.958238 12.951172 12.03125 C 12.886382 12.177277 12.867187 12.304789 12.867188 12.441406 L 12.867188 13.523438 L 12.867188 14.119141 L 12.867188 15.677734 L 12.867188 16.509766 L 13.570312 16.509766 L 19.472656 16.509766 L 20.173828 16.509766 L 20.173828 15.677734 L 20.173828 13.523438 L 20.173828 12.441406 C 20.173828 12.304794 20.156597 12.177281 20.091797 12.03125 C 20.059397 11.95824 20.015299 11.877916 19.923828 11.791016 C 19.832368 11.704116 19.667509 11.607422 19.472656 11.607422 L 18.927734 11.607422 L 18.927734 10.376953 C 18.927734 9.0851838 17.867902 8.0273438 16.576172 8.0273438 L 16.435547 8.0273438 z M 16.435547 9.2207031 L 16.576172 9.2207031 C 17.22782 9.2207031 17.734375 9.7251013 17.734375 10.376953 L 17.734375 11.607422 L 15.277344 11.607422 L 15.277344 10.376953 C 15.277344 9.7251013 15.7839 9.2207031 16.435547 9.2207031 z M 12.919922 9.9394531 C 12.559922 9.9594531 12.359141 10.480234 12.619141 10.740234 L 12.751953 10.904297 C 12.862211 10.870135 12.980058 10.842244 13.085938 10.802734 L 13.085938 10.378906 C 13.085938 10.228632 13.111295 10.084741 13.130859 9.9394531 L 12.919922 9.9394531 z M 19.882812 9.9394531 C 19.902378 10.084741 19.927734 10.228632 19.927734 10.378906 L 19.927734 10.791016 C 20.168811 10.875098 20.455966 10.916935 20.613281 11.066406 C 20.691227 11.140457 20.749315 11.223053 20.806641 11.302734 L 21.259766 10.740234 C 21.519766 10.460234 21.260625 9.9094531 20.890625 9.9394531 L 19.882812 9.9394531 z M 16.435547 10.220703 C 16.301234 10.220703 16.277344 10.244432 16.277344 10.378906 L 16.277344 10.607422 L 16.734375 10.607422 L 16.734375 10.378906 C 16.734375 10.244433 16.712442 10.220703 16.578125 10.220703 L 16.435547 10.220703 z ' fill='#{hex-color($white)}' stroke-width='0'/><path d='M 7.7792969 19.650391 L 7.7792969 19.660156 C 7.5392969 19.680156 7.3398437 19.910156 7.3398438 20.160156 L 7.3398438 22.619141 L 7.2792969 22.619141 C 6.1992969 22.619141 5.4208594 22.589844 4.8808594 22.589844 C 3.2408594 22.589844 3.6308594 23.020234 3.6308594 26.240234 L 3.6308594 30.710938 C 3.6308594 34.970937 3.0692969 34.330078 7.2792969 34.330078 L 8.5 34.330078 L 7.1992188 33.269531 C 7.0992188 33.189531 7.02 33.070703 7 32.970703 C 6.98 32.800703 7.0592186 32.619531 7.1992188 32.519531 L 8.5292969 31.419922 L 7.2792969 31.419922 C 6.8392969 31.419922 6.5605469 31.120703 6.5605469 30.720703 L 6.5605469 26.240234 C 6.5605469 25.800234 6.8392969 25.519531 7.2792969 25.519531 L 7.3398438 25.519531 L 7.3398438 28.019531 C 7.3398438 28.399531 7.8801564 28.650391 8.1601562 28.400391 L 13.060547 24.470703 C 13.310547 24.290703 13.310547 23.869453 13.060547 23.689453 L 8.1601562 19.769531 C 8.0601563 19.669531 7.9192969 19.630391 7.7792969 19.650391 z M 17.119141 22.580078 L 17.119141 22.589844 C 16.579141 22.569844 15.820703 22.609375 14.720703 22.609375 L 13.470703 22.609375 L 14.769531 23.679688 C 14.869531 23.749688 14.950703 23.879766 14.970703 24.009766 C 14.990703 24.169766 14.909531 24.310156 14.769531 24.410156 L 13.439453 25.509766 L 14.720703 25.509766 C 15.129702 25.509766 15.41841 25.778986 15.433594 26.199219 C 15.752266 26.097283 16.084896 26.027344 16.435547 26.027344 L 16.578125 26.027344 C 17.236645 26.027344 17.848901 26.228565 18.369141 26.5625 L 18.369141 26.240234 C 18.369141 23.090234 18.769141 22.620078 17.119141 22.580078 z M 16.435547 27.027344 C 15.143818 27.027344 14.083984 28.085184 14.083984 29.376953 L 14.083984 30.607422 L 13.570312 30.607422 C 13.375452 30.607422 13.210603 30.704118 13.119141 30.791016 C 13.027691 30.877916 12.983569 30.958238 12.951172 31.03125 C 12.886382 31.177277 12.867184 31.304789 12.867188 31.441406 L 12.867188 32.523438 L 12.867188 33.119141 L 12.867188 34.677734 L 12.867188 35.509766 L 13.570312 35.509766 L 19.472656 35.509766 L 20.173828 35.509766 L 20.173828 34.677734 L 20.173828 32.523438 L 20.173828 31.441406 C 20.173828 31.304794 20.156597 31.177281 20.091797 31.03125 C 20.059397 30.95824 20.015299 30.877916 19.923828 30.791016 C 19.832368 30.704116 19.667509 30.607422 19.472656 30.607422 L 18.927734 30.607422 L 18.927734 29.376953 C 18.927734 28.085184 17.867902 27.027344 16.576172 27.027344 L 16.435547 27.027344 z M 16.435547 28.220703 L 16.576172 28.220703 C 17.22782 28.220703 17.734375 28.725101 17.734375 29.376953 L 17.734375 30.607422 L 15.277344 30.607422 L 15.277344 29.376953 C 15.277344 28.725101 15.7839 28.220703 16.435547 28.220703 z M 13.109375 29.150391 L 8.9199219 32.509766 C 8.6599219 32.689766 8.6599219 33.109063 8.9199219 33.289062 L 11.869141 35.648438 L 11.869141 34.677734 L 11.869141 33.119141 L 11.869141 32.523438 L 11.869141 31.441406 C 11.869141 31.217489 11.912641 30.907486 12.037109 30.626953 C 12.093758 30.499284 12.228597 30.257492 12.429688 30.066406 C 12.580253 29.92335 12.859197 29.887344 13.085938 29.802734 L 13.085938 29.378906 C 13.085938 29.300761 13.104 29.227272 13.109375 29.150391 z M 16.435547 29.220703 C 16.301234 29.220703 16.277344 29.244432 16.277344 29.378906 L 16.277344 29.607422 L 16.734375 29.607422 L 16.734375 29.378906 C 16.734375 29.244433 16.712442 29.220703 16.578125 29.220703 L 16.435547 29.220703 z M 12.943359 36.509766 L 13.820312 37.210938 C 14.090314 37.460938 14.639141 37.210078 14.619141 36.830078 L 14.619141 36.509766 L 13.570312 36.509766 L 12.943359 36.509766 z M 10.330078 38.650391 L 10.339844 38.660156 C 10.099844 38.680156 9.9001562 38.910156 9.9101562 39.160156 L 9.9101562 41.630859 L 7.3007812 41.630859 C 6.2207812 41.630859 5.4403906 41.589844 4.9003906 41.589844 C 3.2603906 41.589844 3.6503906 42.020234 3.6503906 45.240234 L 3.6503906 49.710938 C 3.6503906 53.370936 3.4202344 53.409141 5.9902344 53.369141 L 4.6503906 52.269531 C 4.5503906 52.189531 4.4692187 52.070703 4.4492188 51.970703 C 4.4492188 51.800703 4.5203906 51.619531 4.6503906 51.519531 L 6.609375 49.919922 C 6.579375 49.859922 6.5703125 49.790703 6.5703125 49.720703 L 6.5703125 45.240234 C 6.5703125 44.800234 6.8490625 44.519531 7.2890625 44.519531 L 9.9003906 44.519531 L 9.9003906 47.019531 C 9.9003906 47.379531 10.399219 47.620391 10.699219 47.400391 L 15.630859 43.470703 C 15.870859 43.290703 15.870859 42.869453 15.630859 42.689453 L 10.689453 38.769531 C 10.589453 38.689531 10.460078 38.640391 10.330078 38.650391 z M 16.869141 41.585938 C 16.616211 41.581522 16.322969 41.584844 15.980469 41.589844 L 15.970703 41.589844 L 17.310547 42.689453 C 17.410547 42.759453 17.489766 42.889531 17.509766 43.019531 C 17.529766 43.179531 17.479609 43.319922 17.349609 43.419922 L 15.390625 45.019531 C 15.406724 45.075878 15.427133 45.132837 15.4375 45.197266 C 15.754974 45.096169 16.086404 45.027344 16.435547 45.027344 L 16.578125 45.027344 C 17.24129 45.027344 17.858323 45.230088 18.380859 45.568359 L 18.380859 45.25 C 18.380859 42.0475 18.639648 41.616836 16.869141 41.585938 z M 16.435547 46.027344 C 15.143818 46.027344 14.083984 47.085184 14.083984 48.376953 L 14.083984 49.607422 L 13.570312 49.607422 C 13.375448 49.607422 13.210603 49.704118 13.119141 49.791016 C 13.027691 49.877916 12.983569 49.958238 12.951172 50.03125 C 12.886382 50.177277 12.867187 50.304789 12.867188 50.441406 L 12.867188 51.523438 L 12.867188 52.119141 L 12.867188 53.677734 L 12.867188 54.509766 L 13.570312 54.509766 L 19.472656 54.509766 L 20.173828 54.509766 L 20.173828 53.677734 L 20.173828 51.523438 L 20.173828 50.441406 C 20.173828 50.304794 20.156597 50.177281 20.091797 50.03125 C 20.059397 49.95824 20.015299 49.877916 19.923828 49.791016 C 19.832368 49.704116 19.667509 49.607422 19.472656 49.607422 L 18.927734 49.607422 L 18.927734 48.376953 C 18.927734 47.085184 17.867902 46.027344 16.576172 46.027344 L 16.435547 46.027344 z M 16.435547 47.220703 L 16.576172 47.220703 C 17.22782 47.220703 17.734375 47.725101 17.734375 48.376953 L 17.734375 49.607422 L 15.277344 49.607422 L 15.277344 48.376953 C 15.277344 47.725101 15.7839 47.220703 16.435547 47.220703 z M 11.470703 47.490234 C 11.410703 47.510234 11.349063 47.539844 11.289062 47.589844 L 6.3496094 51.519531 C 6.1096094 51.699531 6.1096094 52.120781 6.3496094 52.300781 L 11.289062 56.220703 C 11.569064 56.440703 12.070312 56.199844 12.070312 55.839844 L 12.070312 55.509766 L 11.869141 55.509766 L 11.869141 53.677734 L 11.869141 52.119141 L 11.869141 51.523438 L 11.869141 50.441406 C 11.869141 50.217489 11.912641 49.907486 12.037109 49.626953 C 12.043809 49.611855 12.061451 49.584424 12.070312 49.566406 L 12.070312 47.960938 C 12.070312 47.660938 11.770703 47.430234 11.470703 47.490234 z M 16.435547 48.220703 C 16.301234 48.220703 16.277344 48.244432 16.277344 48.378906 L 16.277344 48.607422 L 16.734375 48.607422 L 16.734375 48.378906 C 16.734375 48.244433 16.712442 48.220703 16.578125 48.220703 L 16.435547 48.220703 z M 13.060547 57.650391 L 13.060547 57.660156 C 12.830547 57.690156 12.660156 57.920156 12.660156 58.160156 L 12.660156 60.630859 L 7.2792969 60.630859 C 6.1992969 60.630859 5.4208594 60.589844 4.8808594 60.589844 C 3.2408594 60.589844 3.6308594 61.020234 3.6308594 64.240234 L 3.6308594 69.109375 L 6.5605469 66.740234 L 6.5605469 64.240234 C 6.5605469 63.800234 6.8392969 63.519531 7.2792969 63.519531 L 12.660156 63.519531 L 12.660156 66.019531 C 12.660156 66.299799 12.960394 66.500006 13.226562 66.474609 C 13.625751 65.076914 14.904956 64.035678 16.421875 64.029297 L 18.380859 62.470703 C 18.620859 62.290703 18.620859 61.869453 18.380859 61.689453 L 13.439453 57.769531 C 13.339453 57.669531 13.200547 57.630391 13.060547 57.650391 z M 18.359375 63.810547 L 17.800781 64.269531 C 18.004793 64.350836 18.198411 64.450249 18.380859 64.568359 L 18.380859 64.25 L 18.380859 63.810547 L 18.359375 63.810547 z M 16.435547 65.027344 C 15.143818 65.027344 14.083984 66.085184 14.083984 67.376953 L 14.083984 68.607422 L 13.570312 68.607422 C 13.375448 68.607422 13.210603 68.704118 13.119141 68.791016 C 13.027691 68.877916 12.983569 68.958238 12.951172 69.03125 C 12.886382 69.177277 12.867187 69.304789 12.867188 69.441406 L 12.867188 70.523438 L 12.867188 71.119141 L 12.867188 72.677734 L 12.867188 73.509766 L 13.570312 73.509766 L 19.472656 73.509766 L 20.173828 73.509766 L 20.173828 72.677734 L 20.173828 70.523438 L 20.173828 69.441406 C 20.173828 69.304794 20.156597 69.177281 20.091797 69.03125 C 20.059397 68.95824 20.015299 68.877916 19.923828 68.791016 C 19.832368 68.704116 19.667509 68.607422 19.472656 68.607422 L 18.927734 68.607422 L 18.927734 67.376953 C 18.927734 66.085184 17.867902 65.027344 16.576172 65.027344 L 16.435547 65.027344 z M 16.435547 66.220703 L 16.576172 66.220703 C 17.22782 66.220703 17.734375 66.725101 17.734375 67.376953 L 17.734375 68.607422 L 15.277344 68.607422 L 15.277344 67.376953 C 15.277344 66.725101 15.7839 66.220703 16.435547 66.220703 z M 8.7207031 66.509766 C 8.6507031 66.529766 8.5895312 66.559375 8.5195312 66.609375 L 3.5996094 70.519531 C 3.3496094 70.699531 3.3496094 71.120781 3.5996094 71.300781 L 8.5292969 75.220703 C 8.8092969 75.440703 9.3105469 75.199844 9.3105469 74.839844 L 9.3105469 72.339844 L 11.869141 72.339844 L 11.869141 71.119141 L 11.869141 70.523438 L 11.869141 69.449219 L 9.3203125 69.449219 L 9.3203125 66.980469 C 9.3203125 66.680469 9.0007031 66.449766 8.7207031 66.509766 z M 16.435547 67.220703 C 16.301234 67.220703 16.277344 67.244432 16.277344 67.378906 L 16.277344 67.607422 L 16.734375 67.607422 L 16.734375 67.378906 C 16.734375 67.244433 16.712442 67.220703 16.578125 67.220703 L 16.435547 67.220703 z M 19.248047 78.800781 C 19.148558 78.831033 19.050295 78.90106 18.970703 78.970703 L 18.070312 79.869141 C 17.630312 79.569141 16.710703 79.619141 14.720703 79.619141 L 7.2792969 79.619141 C 6.1992969 79.619141 5.4208594 79.589844 4.8808594 79.589844 C 3.2408594 79.589844 3.6308594 80.020234 3.6308594 83.240234 L 3.6308594 83.939453 L 6.5605469 84.240234 L 6.5605469 83.240234 C 6.5605469 82.800234 6.8392969 82.519531 7.2792969 82.519531 L 14.720703 82.519531 C 14.920703 82.519531 15.090703 82.600703 15.220703 82.720703 L 13.419922 84.519531 C 13.279464 84.665607 13.281282 84.881022 13.363281 85.054688 C 13.880838 83.867655 15.067337 83.027344 16.435547 83.027344 L 16.578125 83.027344 C 18.290465 83.027344 19.703357 84.345788 19.890625 86.011719 L 19.960938 86.019531 C 20.240938 86.049531 20.520234 85.770234 20.490234 85.490234 L 19.789062 79.240234 C 19.789062 78.973661 19.498025 78.767523 19.25 78.800781 L 19.248047 78.800781 z M 16.435547 84.027344 C 15.143818 84.027344 14.083984 85.085184 14.083984 86.376953 L 14.083984 87.607422 L 13.570312 87.607422 C 13.375448 87.607422 13.210603 87.704118 13.119141 87.791016 C 13.027691 87.877916 12.983569 87.958238 12.951172 88.03125 C 12.886382 88.177277 12.867187 88.304789 12.867188 88.441406 L 12.867188 89.523438 L 12.867188 90.119141 L 12.867188 91.677734 L 12.867188 92.509766 L 13.570312 92.509766 L 19.472656 92.509766 L 20.173828 92.509766 L 20.173828 91.677734 L 20.173828 89.523438 L 20.173828 88.441406 C 20.173828 88.304794 20.156597 88.177281 20.091797 88.03125 C 20.059397 87.95824 20.015299 87.877916 19.923828 87.791016 C 19.832368 87.704116 19.667509 87.607422 19.472656 87.607422 L 18.927734 87.607422 L 18.927734 86.376953 C 18.927734 85.085184 17.867902 84.027344 16.576172 84.027344 L 16.435547 84.027344 z M 2.0507812 84.900391 C 1.8507824 84.970391 1.6907031 85.199453 1.7207031 85.439453 L 2.4199219 91.689453 C 2.4399219 92.049453 3 92.240929 3.25 91.960938 L 4.0507812 91.160156 C 4.0707812 91.160156 4.0898437 91.140156 4.0898438 91.160156 C 4.5498437 91.400156 5.4595313 91.330078 7.2695312 91.330078 L 11.869141 91.330078 L 11.869141 90.119141 L 11.869141 89.523438 L 11.869141 88.441406 C 11.869141 88.437991 11.871073 88.433136 11.871094 88.429688 L 7.2792969 88.429688 C 7.1292969 88.429688 6.9808594 88.400078 6.8808594 88.330078 L 8.8007812 86.400391 C 9.1007822 86.160391 8.8992969 85.600547 8.5292969 85.560547 L 2.25 84.910156 L 2.0507812 84.910156 L 2.0507812 84.900391 z M 16.435547 85.220703 L 16.576172 85.220703 C 17.22782 85.220703 17.734375 85.725101 17.734375 86.376953 L 17.734375 87.607422 L 15.277344 87.607422 L 15.277344 86.376953 C 15.277344 85.725101 15.7839 85.220703 16.435547 85.220703 z M 4.8808594 98.599609 C 3.5508594 98.599609 3.5400781 99.080402 3.5800781 100.90039 L 4.7207031 99.529297 C 4.8007031 99.429297 4.9405469 99.360078 5.0605469 99.330078 C 5.2205469 99.330078 5.4 99.409297 5.5 99.529297 L 7.1601562 101.56055 C 7.2001563 101.56055 7.2292969 101.5293 7.2792969 101.5293 L 14.720703 101.5293 C 15.060703 101.5293 15.289141 101.7293 15.369141 102.0293 L 12.939453 102.0293 C 12.599453 102.0793 12.410625 102.55055 12.640625 102.81055 L 13.470703 103.85742 C 14.029941 102.77899 15.146801 102.02734 16.435547 102.02734 L 16.578125 102.02734 C 18.158418 102.02734 19.491598 103.14879 19.835938 104.63086 L 21.279297 102.82031 C 21.499297 102.55031 21.260156 102.06078 20.910156 102.05078 L 18.400391 102.05078 C 18.420391 98.150792 19.000234 98.650391 14.740234 98.650391 L 7.2792969 98.650391 C 6.1992969 98.650391 5.4208594 98.609375 4.8808594 98.609375 L 4.8808594 98.599609 z M 5.0292969 101.06055 C 4.9292969 101.09055 4.83 101.15977 4.75 101.25977 L 0.81054688 106.16016 C 0.61054688 106.44016 0.8409375 106.92945 1.2109375 106.93945 L 3.5996094 106.93945 C 3.5796094 110.87945 3.1497656 110.33984 7.2597656 110.33984 L 11.869141 110.33984 L 11.869141 109.11914 L 11.869141 108.52344 L 11.869141 107.44141 L 11.869141 107.43945 L 7.2792969 107.43945 C 6.9292969 107.43945 6.7091406 107.23945 6.6191406 106.93945 L 9.0605469 106.93945 C 9.4305469 106.93945 9.6909375 106.44016 9.4609375 106.16016 L 5.5 101.25977 C 5.4 101.10977 5.1992969 101.03055 5.0292969 101.06055 z M 16.435547 103.02734 C 15.143818 103.02734 14.083984 104.08518 14.083984 105.37695 L 14.083984 106.60742 L 13.570312 106.60742 C 13.375448 106.60742 13.210603 106.70409 13.119141 106.79102 C 13.027691 106.87792 12.983569 106.95823 12.951172 107.03125 C 12.886382 107.17727 12.867187 107.30479 12.867188 107.44141 L 12.867188 108.52344 L 12.867188 109.11914 L 12.867188 110.67773 L 12.867188 111.50977 L 13.570312 111.50977 L 19.472656 111.50977 L 20.173828 111.50977 L 20.173828 110.67773 L 20.173828 108.52344 L 20.173828 107.44141 C 20.173828 107.3048 20.156597 107.17728 20.091797 107.03125 C 20.059397 106.95825 20.015299 106.87792 19.923828 106.79102 C 19.832368 106.70412 19.667509 106.60742 19.472656 106.60742 L 18.927734 106.60742 L 18.927734 105.37695 C 18.927734 104.08518 17.867902 103.02734 16.576172 103.02734 L 16.435547 103.02734 z M 16.435547 104.2207 L 16.576172 104.2207 C 17.22782 104.2207 17.734375 104.7251 17.734375 105.37695 L 17.734375 106.60742 L 15.277344 106.60742 L 15.277344 105.37695 C 15.277344 104.7251 15.7839 104.2207 16.435547 104.2207 z M 16.435547 105.2207 C 16.301234 105.2207 16.277344 105.24444 16.277344 105.37891 L 16.277344 105.60742 L 16.734375 105.60742 L 16.734375 105.37891 C 16.734375 105.24441 16.712442 105.2207 16.578125 105.2207 L 16.435547 105.2207 z M 4.8808594 117.58984 L 4.8808594 117.59961 C 3.7208594 117.59961 3.5800781 117.90016 3.5800781 119.16016 L 4.7207031 117.7793 C 4.8007031 117.6793 4.9405469 117.63914 5.0605469 117.61914 C 5.2205469 117.61914 5.4 117.6593 5.5 117.7793 L 7.7207031 120.5293 L 14.720703 120.5293 C 15.123595 120.5293 15.408576 120.79174 15.431641 121.20117 C 15.750992 121.09876 16.08404 121.02734 16.435547 121.02734 L 16.578125 121.02734 C 17.24903 121.02734 17.874081 121.23262 18.400391 121.57812 L 18.400391 121.25 C 18.400391 117.05 19.120234 117.61914 14.740234 117.61914 L 7.2792969 117.61914 C 6.1992969 117.61914 5.4208594 117.58984 4.8808594 117.58984 z M 4.9804688 119.33984 C 4.8804688 119.36984 4.81 119.44 4.75 119.5 L 0.80078125 124.43945 C 0.60078125 124.71945 0.8292182 125.2107 1.1992188 125.2207 L 3.5996094 125.2207 L 3.5996094 125.7207 C 3.5996094 129.9807 3.0497656 129.33984 7.2597656 129.33984 L 11.869141 129.33984 L 11.869141 128.11914 L 11.869141 127.52344 L 11.869141 126.44141 C 11.869141 126.43799 11.871073 126.43314 11.871094 126.42969 L 7.2792969 126.42969 C 6.8392969 126.42969 6.5605469 126.13094 6.5605469 125.71094 L 6.5605469 125.21094 L 9.0605469 125.21094 C 9.4305469 125.23094 9.6909375 124.70969 9.4609375 124.42969 L 5.5 119.5 C 5.3820133 119.35252 5.1682348 119.28513 4.9804688 119.33984 z M 12.839844 121.7793 C 12.539844 121.8793 12.410625 122.32055 12.640625 122.56055 L 13.267578 123.34375 C 13.473522 122.72168 13.852237 122.1828 14.353516 121.7793 L 12.839844 121.7793 z M 18.658203 121.7793 C 19.393958 122.37155 19.878978 123.25738 19.916016 124.25781 L 21.279297 122.56055 C 21.499297 122.28055 21.260156 121.7893 20.910156 121.7793 L 18.658203 121.7793 z M 16.435547 122.02734 C 15.143818 122.02734 14.083984 123.08518 14.083984 124.37695 L 14.083984 125.60742 L 13.570312 125.60742 C 13.375448 125.60742 13.210603 125.70409 13.119141 125.79102 C 13.027691 125.87792 12.983569 125.95823 12.951172 126.03125 C 12.886382 126.17727 12.867187 126.30479 12.867188 126.44141 L 12.867188 127.52344 L 12.867188 128.11914 L 12.867188 129.67773 L 12.867188 130.50977 L 13.570312 130.50977 L 19.472656 130.50977 L 20.173828 130.50977 L 20.173828 129.67773 L 20.173828 127.52344 L 20.173828 126.44141 C 20.173828 126.3048 20.156597 126.17728 20.091797 126.03125 C 20.059397 125.95825 20.015299 125.87792 19.923828 125.79102 C 19.832368 125.70412 19.667509 125.60742 19.472656 125.60742 L 18.927734 125.60742 L 18.927734 124.37695 C 18.927734 123.08518 17.867902 122.02734 16.576172 122.02734 L 16.435547 122.02734 z M 16.435547 123.2207 L 16.576172 123.2207 C 17.22782 123.2207 17.734375 123.7251 17.734375 124.37695 L 17.734375 125.60742 L 15.277344 125.60742 L 15.277344 124.37695 C 15.277344 123.7251 15.7839 123.2207 16.435547 123.2207 z M 16.435547 124.2207 C 16.301234 124.2207 16.277344 124.24444 16.277344 124.37891 L 16.277344 124.60742 L 16.734375 124.60742 L 16.734375 124.37891 C 16.734375 124.24441 16.712442 124.2207 16.578125 124.2207 L 16.435547 124.2207 z M 5.9394531 136.58984 L 5.9394531 136.59961 L 8.3105469 139.5293 L 14.730469 139.5293 C 15.131912 139.5293 15.414551 139.79039 15.439453 140.19727 C 15.756409 140.09653 16.087055 140.02734 16.435547 140.02734 L 16.578125 140.02734 C 17.24903 140.02734 17.874081 140.23261 18.400391 140.57812 L 18.400391 140.25 C 18.400391 136.05 19.120234 136.61914 14.740234 136.61914 L 7.2792969 136.61914 C 6.6792969 136.61914 6.3594531 136.59984 5.9394531 136.58984 z M 4.2207031 136.66016 C 3.8207031 136.74016 3.6791406 136.96016 3.6191406 137.41016 L 4.2207031 136.66992 L 4.2207031 136.66016 z M 5.0605469 137.57031 L 5.0605469 137.58984 C 4.9405469 137.58984 4.8197656 137.66953 4.7597656 137.76953 L 0.81054688 142.66992 C 0.57054688 142.96992 0.8109375 143.50023 1.2109375 143.49023 L 3.5996094 143.49023 L 3.5996094 144.71094 C 3.5996094 148.97094 3.0497656 148.33008 7.2597656 148.33008 L 11.869141 148.33008 L 11.869141 147.11914 L 11.869141 146.52344 L 11.869141 145.44141 C 11.869141 145.43799 11.871073 145.43314 11.871094 145.42969 L 7.2792969 145.42969 C 6.8392969 145.42969 6.5605469 145.13094 6.5605469 144.71094 L 6.5605469 143.49023 L 9.0605469 143.49023 C 9.4605469 143.53023 9.7309375 142.95945 9.4609375 142.68945 L 5.5 137.76953 C 5.4 137.63953 5.2305469 137.57031 5.0605469 137.57031 z M 16.435547 141.02734 C 15.143818 141.02734 14.083984 142.08518 14.083984 143.37695 L 14.083984 144.60742 L 13.570312 144.60742 C 13.375448 144.60742 13.210603 144.70409 13.119141 144.79102 C 13.027691 144.87792 12.983569 144.95823 12.951172 145.03125 C 12.886382 145.17727 12.867187 145.30479 12.867188 145.44141 L 12.867188 146.52344 L 12.867188 147.11914 L 12.867188 148.67773 L 12.867188 149.50977 L 13.570312 149.50977 L 19.472656 149.50977 L 20.173828 149.50977 L 20.173828 148.67773 L 20.173828 146.52344 L 20.173828 145.44141 C 20.173828 145.3048 20.156597 145.17728 20.091797 145.03125 C 20.059397 144.95825 20.015299 144.87792 19.923828 144.79102 C 19.832368 144.70412 19.667509 144.60742 19.472656 144.60742 L 18.927734 144.60742 L 18.927734 143.37695 C 18.927734 142.08518 17.867902 141.02734 16.576172 141.02734 L 16.435547 141.02734 z M 12.849609 141.5 C 12.549609 141.6 12.420391 142.0393 12.650391 142.2793 L 13.136719 142.88672 C 13.213026 142.38119 13.390056 141.90696 13.667969 141.5 L 12.849609 141.5 z M 19.34375 141.5 C 19.710704 142.03735 19.927734 142.68522 19.927734 143.37891 L 19.927734 143.79102 C 19.965561 143.80421 20.005506 143.81448 20.044922 143.82617 L 21.289062 142.2793 C 21.509062 141.9993 21.269922 141.51 20.919922 141.5 L 19.34375 141.5 z M 16.435547 142.2207 L 16.576172 142.2207 C 17.22782 142.2207 17.734375 142.7251 17.734375 143.37695 L 17.734375 144.60742 L 15.277344 144.60742 L 15.277344 143.37695 C 15.277344 142.7251 15.7839 142.2207 16.435547 142.2207 z M 16.435547 143.2207 C 16.301234 143.2207 16.277344 143.24444 16.277344 143.37891 L 16.277344 143.60742 L 16.734375 143.60742 L 16.734375 143.37891 C 16.734375 143.24441 16.712442 143.2207 16.578125 143.2207 L 16.435547 143.2207 z M 17.130859 155.59961 C 16.580859 155.57961 15.810469 155.63086 14.730469 155.63086 L 6.5292969 155.63086 L 8.9101562 158.5293 L 14.730469 158.5293 C 15.131912 158.5293 15.414551 158.79039 15.439453 159.19727 C 15.756409 159.09653 16.087055 159.02734 16.435547 159.02734 L 16.578125 159.02734 C 17.24903 159.02734 17.874081 159.23261 18.400391 159.57812 L 18.400391 159.25977 C 18.400391 156.10977 18.800391 155.63961 17.150391 155.59961 L 17.130859 155.59961 z M 5.0292969 155.86914 L 5.0292969 155.88086 C 4.9292969 155.90086 4.83 155.98055 4.75 156.06055 L 0.81054688 160.96094 C 0.61054688 161.26094 0.8409375 161.73977 1.2109375 161.75977 L 3.5996094 161.75977 L 3.5996094 163.7207 C 3.5996094 167.9807 3.0497656 167.33984 7.2597656 167.33984 L 11.869141 167.33984 L 11.869141 166.11914 L 11.869141 165.52344 L 11.869141 164.44141 L 11.869141 164.43945 L 7.2792969 164.43945 C 6.8392969 164.43945 6.5605469 164.1407 6.5605469 163.7207 L 6.5605469 161.75 L 9.0605469 161.75 C 9.4305469 161.77 9.6909375 161.2507 9.4609375 160.9707 L 5.5 156.07031 C 5.4 155.92031 5.1992969 155.84914 5.0292969 155.86914 z M 16.435547 160.02734 C 15.143818 160.02734 14.083984 161.08518 14.083984 162.37695 L 14.083984 163.60742 L 13.570312 163.60742 C 13.375448 163.60742 13.210603 163.70409 13.119141 163.79102 C 13.027691 163.87792 12.983569 163.95823 12.951172 164.03125 C 12.886382 164.17727 12.867187 164.30479 12.867188 164.44141 L 12.867188 165.52344 L 12.867188 166.11914 L 12.867188 167.67773 L 12.867188 168.50977 L 13.570312 168.50977 L 19.472656 168.50977 L 20.173828 168.50977 L 20.173828 167.67773 L 20.173828 165.52344 L 20.173828 164.44141 C 20.173828 164.3048 20.156597 164.17728 20.091797 164.03125 C 20.059397 163.95825 20.015299 163.87792 19.923828 163.79102 C 19.832368 163.70412 19.667509 163.60742 19.472656 163.60742 L 18.927734 163.60742 L 18.927734 162.37695 C 18.927734 161.08518 17.867902 160.02734 16.576172 160.02734 L 16.435547 160.02734 z M 12.900391 161.2207 C 12.580391 161.2807 12.419141 161.74 12.619141 162 L 13.085938 162.58594 L 13.085938 162.37891 C 13.085938 161.97087 13.170592 161.58376 13.306641 161.2207 L 12.900391 161.2207 z M 16.435547 161.2207 L 16.576172 161.2207 C 17.22782 161.2207 17.734375 161.7251 17.734375 162.37695 L 17.734375 163.60742 L 15.277344 163.60742 L 15.277344 162.37695 C 15.277344 161.7251 15.7839 161.2207 16.435547 161.2207 z M 19.708984 161.23047 C 19.842743 161.59081 19.927734 161.97449 19.927734 162.37891 L 19.927734 162.79102 C 20.119162 162.85779 20.322917 162.91147 20.484375 163 L 21.279297 162.00977 C 21.499297 161.72977 21.260156 161.24047 20.910156 161.23047 L 19.708984 161.23047 z M 16.435547 162.2207 C 16.301234 162.2207 16.277344 162.24444 16.277344 162.37891 L 16.277344 162.60742 L 16.734375 162.60742 L 16.734375 162.37891 C 16.734375 162.24441 16.712442 162.2207 16.578125 162.2207 L 16.435547 162.2207 z M 5.0996094 174.49023 L 5.1308594 174.5 C 4.9808594 174.5 4.83 174.56922 4.75 174.69922 L 0.80078125 179.59961 C 0.56078125 179.86961 0.7992182 180.42039 1.1992188 180.40039 L 3.5996094 180.40039 L 3.5996094 182.7207 C 3.5996094 186.9807 3.0497656 186.33984 7.2597656 186.33984 L 11.869141 186.33984 L 11.869141 185.11914 L 11.869141 184.52344 L 11.869141 183.44141 L 11.869141 183.43945 L 7.25 183.43945 C 6.82 183.43945 6.5507814 183.1407 6.5507812 182.7207 L 6.5507812 180.41992 L 9.0507812 180.41992 C 9.4307824 180.44992 9.7092187 179.87984 9.4492188 179.58984 L 5.4804688 174.68945 C 5.3804688 174.55945 5.2496094 174.49023 5.0996094 174.49023 z M 17.150391 174.58008 L 17.130859 174.59961 C 16.580859 174.57961 15.810469 174.63086 14.730469 174.63086 L 6.8300781 174.63086 L 9.1796875 177.5293 L 14.699219 177.5293 C 15.104107 177.5293 15.391475 177.79407 15.412109 178.20703 C 15.737096 178.1006 16.076913 178.02734 16.435547 178.02734 L 16.578125 178.02734 C 17.24903 178.02734 17.874081 178.2326 18.400391 178.57812 L 18.400391 178.24023 C 18.400391 175.09023 18.800391 174.62008 17.150391 174.58008 z M 16.435547 179.02734 C 15.143818 179.02734 14.083984 180.08518 14.083984 181.37695 L 14.083984 182.60742 L 13.570312 182.60742 C 13.375448 182.60742 13.210603 182.70409 13.119141 182.79102 C 13.027691 182.87792 12.983569 182.95823 12.951172 183.03125 C 12.886382 183.17727 12.867187 183.30479 12.867188 183.44141 L 12.867188 184.52344 L 12.867188 185.11914 L 12.867188 186.67773 L 12.867188 187.50977 L 13.570312 187.50977 L 19.472656 187.50977 L 20.173828 187.50977 L 20.173828 186.67773 L 20.173828 184.52344 L 20.173828 183.44141 C 20.173828 183.3048 20.156597 183.17728 20.091797 183.03125 C 20.059397 182.95825 20.015299 182.87792 19.923828 182.79102 C 19.832368 182.70412 19.667509 182.60742 19.472656 182.60742 L 18.927734 182.60742 L 18.927734 181.37695 C 18.927734 180.08518 17.867902 179.02734 16.576172 179.02734 L 16.435547 179.02734 z M 16.435547 180.2207 L 16.576172 180.2207 C 17.22782 180.2207 17.734375 180.7251 17.734375 181.37695 L 17.734375 182.60742 L 15.277344 182.60742 L 15.277344 181.37695 C 15.277344 180.7251 15.7839 180.2207 16.435547 180.2207 z M 19.816406 180.57031 C 19.882311 180.83091 19.927734 181.09907 19.927734 181.37891 L 19.927734 181.79102 C 20.168811 181.87511 20.455966 181.91694 20.613281 182.06641 C 20.630645 182.0829 20.639883 182.10199 20.65625 182.11914 L 21.259766 181.36914 C 21.479766 181.06914 21.240625 180.59031 20.890625 180.57031 L 19.816406 180.57031 z M 12.820312 180.58984 C 12.520316 180.68984 12.389141 181.11914 12.619141 181.36914 L 12.990234 181.83203 C 13.022029 181.82207 13.055579 181.81406 13.085938 181.80273 L 13.085938 181.37891 C 13.085938 181.10616 13.128698 180.84442 13.191406 180.58984 L 12.820312 180.58984 z M 16.435547 181.2207 C 16.301234 181.2207 16.277344 181.24444 16.277344 181.37891 L 16.277344 181.60742 L 16.734375 181.60742 L 16.734375 181.37891 C 16.734375 181.24441 16.712442 181.2207 16.578125 181.2207 L 16.435547 181.2207 z M 4.9609375 193.15039 L 4.9707031 193.16016 C 4.8707031 193.19016 4.8 193.25984 4.75 193.33984 L 0.81054688 198.24023 C 0.61054688 198.54023 0.8409375 199.01906 1.2109375 199.03906 L 3.5996094 199.03906 L 3.5996094 201.7207 C 3.5996094 205.9807 3.0497656 205.33984 7.2597656 205.33984 L 11.869141 205.33984 L 11.869141 204.11914 L 11.869141 203.52344 L 11.869141 202.44141 C 11.869141 202.44141 11.869141 202.43945 11.869141 202.43945 L 7.2695312 202.43945 C 6.8295312 202.43945 6.5507814 202.1407 6.5507812 201.7207 L 6.5507812 199.01953 L 9.0507812 199.01953 C 9.4207814 199.04953 9.6792188 198.54 9.4492188 198.25 L 5.4902344 193.34961 C 5.3702344 193.17961 5.1509375 193.10039 4.9609375 193.15039 z M 17.150391 193.58008 L 17.130859 193.58984 C 16.580859 193.56984 15.810469 193.61914 14.730469 193.61914 L 7.0996094 193.61914 L 9.4199219 196.46094 L 9.4492188 196.51953 L 14.699219 196.51953 C 15.106887 196.51953 15.397075 196.78718 15.414062 197.20508 C 15.738375 197.09913 16.077769 197.02734 16.435547 197.02734 L 16.578125 197.02734 C 17.24903 197.02734 17.874081 197.23259 18.400391 197.57812 L 18.400391 197.24023 C 18.400391 194.09023 18.800391 193.62008 17.150391 193.58008 z M 16.435547 198.02734 C 15.143818 198.02734 14.083984 199.08518 14.083984 200.37695 L 14.083984 201.60742 L 13.570312 201.60742 C 13.375448 201.60742 13.210603 201.70409 13.119141 201.79102 C 13.027691 201.87792 12.983569 201.95823 12.951172 202.03125 C 12.886382 202.17727 12.867187 202.30479 12.867188 202.44141 L 12.867188 203.52344 L 12.867188 204.11914 L 12.867188 205.67773 L 12.867188 206.50977 L 13.570312 206.50977 L 19.472656 206.50977 L 20.173828 206.50977 L 20.173828 205.67773 L 20.173828 203.52344 L 20.173828 202.44141 C 20.173828 202.3048 20.156597 202.17728 20.091797 202.03125 C 20.059397 201.95825 20.015299 201.87792 19.923828 201.79102 C 19.832368 201.70412 19.667509 201.60742 19.472656 201.60742 L 18.927734 201.60742 L 18.927734 200.37695 C 18.927734 199.08518 17.867902 198.02734 16.576172 198.02734 L 16.435547 198.02734 z M 16.435547 199.2207 L 16.576172 199.2207 C 17.22782 199.2207 17.734375 199.7251 17.734375 200.37695 L 17.734375 201.60742 L 15.277344 201.60742 L 15.277344 200.37695 C 15.277344 199.7251 15.7839 199.2207 16.435547 199.2207 z M 12.919922 199.93945 C 12.559922 199.95945 12.359141 200.48023 12.619141 200.74023 L 12.751953 200.9043 C 12.862211 200.87013 12.980058 200.84224 13.085938 200.80273 L 13.085938 200.37891 C 13.085938 200.22863 13.111295 200.08474 13.130859 199.93945 L 12.919922 199.93945 z M 19.882812 199.93945 C 19.902378 200.08474 19.927734 200.22863 19.927734 200.37891 L 19.927734 200.79102 C 20.168811 200.87511 20.455966 200.91694 20.613281 201.06641 C 20.691227 201.14046 20.749315 201.22305 20.806641 201.30273 L 21.259766 200.74023 C 21.519766 200.46023 21.260625 199.90945 20.890625 199.93945 L 19.882812 199.93945 z M 16.435547 200.2207 C 16.301234 200.2207 16.277344 200.24444 16.277344 200.37891 L 16.277344 200.60742 L 16.734375 200.60742 L 16.734375 200.37891 C 16.734375 200.24441 16.712442 200.2207 16.578125 200.2207 L 16.435547 200.2207 z ' fill='#{hex-color($highlight-text-color)}' stroke-width='0' /></svg>");
+      }
+    }
+
+    &.disabled {
+      i.fa-retweet {
+        background-image: url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' height='209' width='22'><path d='M 18.972656 1.2011719 C 18.829825 1.1881782 18.685932 1.2302188 18.572266 1.3300781 L 15.990234 3.5996094 C 15.58109 3.6070661 15.297269 3.609375 14.730469 3.609375 L 7.0996094 3.609375 L 9.4199219 6.4609375 L 9.4492188 6.5195312 L 12.664062 6.5195312 L 6.5761719 11.867188 C 6.5674697 11.818249 6.5507813 11.773891 6.5507812 11.720703 L 6.5507812 9.0195312 L 9.0507812 9.0195312 C 9.4207813 9.0495313 9.6792188 8.54 9.4492188 8.25 L 5.5 3.3496094 C 5.38 3.1796094 5.1607031 3.1003906 4.9707031 3.1503906 L 4.9707031 3.1601562 C 4.8707031 3.1901563 4.8 3.2598438 4.75 3.3398438 L 0.80078125 8.2402344 C 0.60078125 8.5402344 0.8292187 9.0190625 1.1992188 9.0390625 L 3.5996094 9.0390625 L 3.5996094 11.720703 C 3.5996094 13.045739 3.5690668 13.895038 3.6503906 14.4375 L 2.6152344 15.347656 C 2.3879011 15.547375 2.3754917 15.901081 2.5859375 16.140625 L 3.1464844 16.78125 C 3.3569308 17.020794 3.7101667 17.053234 3.9375 16.853516 L 19.892578 2.8359375 C 20.119911 2.6362188 20.134275 2.282513 19.923828 2.0429688 L 19.361328 1.4023438 C 19.256105 1.282572 19.115488 1.2141655 18.972656 1.2011719 z M 18.410156 6.7753906 L 15.419922 9.4042969 L 15.419922 9.9394531 L 14.810547 9.9394531 L 13.148438 11.400391 L 16.539062 15.640625 C 16.719062 15.890625 17.140313 15.890625 17.320312 15.640625 L 21.259766 10.740234 C 21.519766 10.460234 21.260625 9.9094531 20.890625 9.9394531 L 18.400391 9.9394531 L 18.400391 7.2402344 C 18.400391 7.0470074 18.407711 6.9489682 18.410156 6.7753906 z M 11.966797 12.439453 L 8.6679688 15.339844 L 14.919922 15.339844 L 12.619141 12.5 C 12.589141 12.48 12.590313 12.459453 12.570312 12.439453 L 11.966797 12.439453 z' fill='#{hex-color($white)}' stroke-width='0'/></svg>");
+      }
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/columns.scss b/app/javascript/flavours/blobfox/styles/components/columns.scss
new file mode 100644
index 00000000000000..ac1658a2de9933
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/columns.scss
@@ -0,0 +1,1364 @@
+.column__wrapper {
+  display: flex;
+  flex: 1 1 auto;
+  position: relative;
+}
+
+.columns-area {
+  display: flex;
+  flex: 1 1 auto;
+  flex-direction: row;
+  justify-content: flex-start;
+  overflow-x: auto;
+  position: relative;
+
+  &__panels {
+    display: flex;
+    justify-content: center;
+    width: 100%;
+    height: 100%;
+    min-height: 100vh;
+
+    &__pane {
+      height: 100%;
+      overflow: hidden;
+      pointer-events: none;
+      display: flex;
+      justify-content: flex-end;
+      min-width: 285px;
+
+      &--start {
+        justify-content: flex-start;
+      }
+
+      &__inner {
+        position: fixed;
+        width: 285px;
+        pointer-events: auto;
+        height: 100%;
+      }
+    }
+
+    &__main {
+      box-sizing: border-box;
+      width: 100%;
+      flex: 0 0 auto;
+      display: flex;
+      flex-direction: column;
+
+      @media screen and (min-width: $no-gap-breakpoint) {
+        padding: 0 10px;
+        max-width: 600px;
+      }
+    }
+  }
+}
+
+$ui-header-height: 55px;
+
+.ui__header {
+  display: none;
+  box-sizing: border-box;
+  height: $ui-header-height;
+  position: sticky;
+  top: 0;
+  z-index: 3;
+  justify-content: space-between;
+  align-items: center;
+
+  &__logo {
+    display: inline-flex;
+    padding: 15px;
+
+    .logo {
+      height: $ui-header-height - 30px;
+      width: auto;
+    }
+
+    .logo--wordmark {
+      display: none;
+    }
+
+    @media screen and (width >= 320px) {
+      .logo--wordmark {
+        display: block;
+      }
+
+      .logo--icon {
+        display: none;
+      }
+    }
+  }
+
+  &__links {
+    display: flex;
+    align-items: center;
+    gap: 10px;
+    padding: 0 10px;
+    overflow: hidden;
+
+    .button {
+      flex: 0 0 auto;
+    }
+
+    .button-tertiary {
+      flex-shrink: 1;
+    }
+  }
+}
+
+.tabs-bar__wrapper {
+  background: darken($ui-base-color, 8%);
+  position: sticky;
+  top: $ui-header-height;
+  z-index: 2;
+  padding-top: 0;
+
+  @media screen and (min-width: $no-gap-breakpoint) {
+    padding-top: 10px;
+    top: 0;
+  }
+
+  .tabs-bar {
+    margin-bottom: 0;
+
+    @media screen and (min-width: $no-gap-breakpoint) {
+      margin-bottom: 10px;
+    }
+  }
+}
+
+.react-swipeable-view-container {
+  &,
+  .columns-area,
+  .column {
+    height: 100%;
+  }
+}
+
+.react-swipeable-view-container > * {
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  height: 100%;
+}
+
+.column {
+  width: 330px;
+  position: relative;
+  box-sizing: border-box;
+  display: flex;
+  flex-direction: column;
+
+  > .scrollable {
+    background: $ui-base-color;
+  }
+}
+
+.ui {
+  flex: 0 0 auto;
+  display: flex;
+  flex-direction: column;
+  width: 100%;
+  height: 100%;
+}
+
+.column {
+  overflow: hidden;
+}
+
+.column-back-button {
+  box-sizing: border-box;
+  width: 100%;
+  background: lighten($ui-base-color, 4%);
+  border-radius: 4px 4px 0 0;
+  color: $highlight-text-color;
+  cursor: pointer;
+  flex: 0 0 auto;
+  font-size: 16px;
+  border: 0;
+  text-align: unset;
+  padding: 15px;
+  margin: 0;
+  z-index: 3;
+
+  &:hover {
+    text-decoration: underline;
+  }
+}
+
+.column-header__back-button {
+  background: lighten($ui-base-color, 4%);
+  border: 0;
+  font-family: inherit;
+  color: $highlight-text-color;
+  cursor: pointer;
+  flex: 0 0 auto;
+  font-size: 16px;
+  padding: 0;
+  padding-inline-end: 5px;
+  z-index: 3;
+
+  &:hover {
+    text-decoration: underline;
+  }
+
+  &:last-child {
+    padding: 0;
+    padding-inline-end: 15px;
+  }
+}
+
+.column-back-button__icon {
+  display: inline-block;
+  margin-inline-end: 5px;
+}
+
+.column-back-button--slim {
+  position: relative;
+}
+
+.column-back-button--slim-button {
+  cursor: pointer;
+  flex: 0 0 auto;
+  font-size: 16px;
+  padding: 15px;
+  position: absolute;
+  inset-inline-end: 0;
+  top: -48px;
+}
+
+.switch-to-advanced {
+  color: $light-text-color;
+  background-color: $ui-base-color;
+  padding: 15px;
+  border-radius: 4px;
+  margin-top: 4px;
+  margin-bottom: 12px;
+  font-size: 13px;
+  line-height: 18px;
+
+  .switch-to-advanced__toggle {
+    color: $ui-button-tertiary-color;
+    font-weight: bold;
+  }
+}
+
+.column-link {
+  background: lighten($ui-base-color, 8%);
+  color: $primary-text-color;
+  display: block;
+  font-size: 16px;
+  padding: 15px;
+  text-decoration: none;
+  overflow: hidden;
+  white-space: nowrap;
+
+  &:hover,
+  &:focus,
+  &:active {
+    background: lighten($ui-base-color, 11%);
+  }
+
+  &:focus {
+    outline: 0;
+  }
+
+  &--transparent {
+    background: transparent;
+    color: $ui-secondary-color;
+
+    &:hover,
+    &:focus,
+    &:active {
+      background: transparent;
+      color: $primary-text-color;
+    }
+
+    &.active {
+      color: $highlight-text-color;
+    }
+  }
+
+  &--logo {
+    background: transparent;
+    padding: 10px;
+
+    &:hover,
+    &:focus,
+    &:active {
+      background: transparent;
+    }
+  }
+}
+
+.column-link__icon {
+  display: inline-block;
+  margin-inline-end: 5px;
+}
+
+.column-subheading {
+  background: $ui-base-color;
+  color: $dark-text-color;
+  padding: 8px 20px;
+  font-size: 12px;
+  font-weight: 500;
+  text-transform: uppercase;
+  cursor: default;
+}
+
+.column-header__wrapper {
+  position: relative;
+  flex: 0 0 auto;
+  z-index: 1;
+
+  &.active {
+    box-shadow: 0 1px 0 rgba($highlight-text-color, 0.3);
+
+    &::before {
+      display: block;
+      content: '';
+      position: absolute;
+      bottom: -13px;
+      inset-inline-start: 0;
+      inset-inline-end: 0;
+      margin: 0 auto;
+      width: 60%;
+      pointer-events: none;
+      height: 28px;
+      z-index: 1;
+      background: radial-gradient(
+        ellipse,
+        rgba($ui-highlight-color, 0.23) 0%,
+        rgba($ui-highlight-color, 0) 60%
+      );
+    }
+  }
+
+  .announcements {
+    z-index: 1;
+    position: relative;
+  }
+}
+
+.column-header {
+  display: flex;
+  font-size: 16px;
+  background: lighten($ui-base-color, 4%);
+  border-radius: 4px 4px 0 0;
+  flex: 0 0 auto;
+  cursor: pointer;
+  position: relative;
+  z-index: 2;
+  outline: 0;
+  overflow: hidden;
+
+  & > button {
+    margin: 0;
+    border: 0;
+    padding: 15px;
+    color: inherit;
+    background: transparent;
+    font: inherit;
+    text-align: start;
+    text-overflow: ellipsis;
+    overflow: hidden;
+    white-space: nowrap;
+    flex: 1;
+  }
+
+  & > .column-header__back-button {
+    color: $highlight-text-color;
+  }
+
+  &.active {
+    .column-header__icon {
+      color: $highlight-text-color;
+      text-shadow: 0 0 10px rgba($ui-highlight-color, 0.4);
+    }
+  }
+
+  &:focus,
+  &:active {
+    outline: 0;
+  }
+}
+
+.column {
+  width: 330px;
+  position: relative;
+  box-sizing: border-box;
+  display: flex;
+  flex-direction: column;
+  overflow: hidden;
+
+  .wide .columns-area:not(.columns-area--mobile) & {
+    flex: auto;
+    min-width: 330px;
+    max-width: 600px;
+  }
+
+  > .scrollable {
+    background: $ui-base-color;
+    border-radius: 0 0 4px 4px;
+  }
+}
+
+.column-header__buttons {
+  height: 48px;
+  display: flex;
+  margin-inline-start: 0;
+}
+
+.column-header__links {
+  margin-bottom: 14px;
+}
+
+.column-header__links .text-btn {
+  margin-inline-end: 10px;
+}
+
+.column-header__button {
+  background: lighten($ui-base-color, 4%);
+  border: 0;
+  color: $darker-text-color;
+  cursor: pointer;
+  font-size: 16px;
+  padding: 0 15px;
+
+  &:hover {
+    color: lighten($darker-text-color, 7%);
+  }
+
+  &.active {
+    color: $primary-text-color;
+    background: lighten($ui-base-color, 8%);
+
+    &:hover {
+      color: $primary-text-color;
+      background: lighten($ui-base-color, 8%);
+    }
+  }
+
+  // blobfox - added focus ring for keyboard navigation
+  &:focus {
+    text-shadow: 0 0 4px darken($ui-highlight-color, 5%);
+  }
+
+  &:disabled {
+    color: $dark-text-color;
+    cursor: default;
+  }
+}
+
+.column-header__notif-cleaning-buttons {
+  display: flex;
+  align-items: stretch;
+  justify-content: space-around;
+
+  .column-header__button {
+    background: transparent;
+    text-align: center;
+    padding: 10px 5px;
+    font-size: 14px;
+  }
+
+  b {
+    font-weight: bold;
+  }
+}
+
+.layout-single-column .column-header__notif-cleaning-buttons {
+  @media screen and (min-width: $no-gap-breakpoint) {
+    b,
+    i {
+      margin-inline-end: 5px;
+    }
+
+    br {
+      display: none;
+    }
+
+    button {
+      padding: 15px 5px;
+    }
+  }
+}
+
+// The notifs drawer with no padding to have more space for the buttons
+.column-header__collapsible-inner.nopad-drawer {
+  padding: 0;
+}
+
+.column-header__collapsible {
+  max-height: 70vh;
+  overflow: hidden;
+  overflow-y: auto;
+  color: $darker-text-color;
+  transition:
+    max-height 150ms ease-in-out,
+    opacity 300ms linear;
+  opacity: 1;
+  z-index: 1;
+  position: relative;
+
+  &.collapsed {
+    max-height: 0;
+    opacity: 0.5;
+  }
+
+  &.animating {
+    overflow-y: hidden;
+  }
+
+  hr {
+    height: 0;
+    background: transparent;
+    border: 0;
+    border-top: 1px solid lighten($ui-base-color, 12%);
+    margin: 10px 0;
+  }
+
+  // notif cleaning drawer
+  &.ncd {
+    transition: none;
+
+    &.collapsed {
+      max-height: 0;
+      opacity: 0.7;
+    }
+  }
+}
+
+.column-header__collapsible-inner {
+  background: lighten($ui-base-color, 8%);
+  padding: 15px;
+}
+
+.column-header__setting-btn {
+  &:hover,
+  &:focus {
+    color: $darker-text-color;
+    text-decoration: underline;
+  }
+}
+
+.column-header__collapsible__extra + .column-header__setting-btn {
+  padding-top: 5px;
+}
+
+.column-header__permission-btn {
+  display: inline;
+  font-weight: inherit;
+  text-decoration: underline;
+}
+
+.column-header__setting-arrows {
+  float: right;
+
+  .column-header__setting-btn {
+    padding: 5px;
+
+    &:first-child {
+      padding-inline-end: 7px;
+    }
+
+    &:last-child {
+      padding-inline-start: 7px;
+      margin-inline-start: 5px;
+    }
+  }
+}
+
+.column-header__title {
+  display: inline-block;
+  text-overflow: ellipsis;
+  overflow: hidden;
+  white-space: nowrap;
+  flex: 1;
+}
+
+.column-header__issue-btn {
+  color: $warning-red;
+
+  &:hover {
+    color: $error-red;
+    text-decoration: underline;
+  }
+}
+
+.column-header__icon {
+  display: inline-block;
+  margin-inline-end: 5px;
+}
+
+.column-settings__pillbar {
+  display: flex;
+  overflow: hidden;
+  background-color: transparent;
+  border: 0;
+  border-radius: 4px;
+  margin-bottom: 10px;
+  align-items: stretch;
+  gap: 2px;
+}
+
+.pillbar-button {
+  border: 0;
+  color: #fafafa;
+  padding: 2px;
+  margin: 0;
+  font-size: inherit;
+  flex: auto;
+  background-color: $ui-base-color;
+  transition: all 0.2s ease;
+  transition-property: background-color, box-shadow;
+
+  &[disabled] {
+    cursor: not-allowed;
+    opacity: 0.5;
+  }
+
+  &:not([disabled]) {
+    &:hover,
+    &:focus {
+      background-color: darken($ui-base-color, 10%);
+    }
+
+    &.active {
+      background-color: darken($ui-highlight-color, 2%);
+
+      &:hover,
+      &:focus {
+        background-color: $ui-highlight-color;
+      }
+    }
+  }
+}
+
+.limited-account-hint {
+  p {
+    color: $secondary-text-color;
+    font-size: 15px;
+    font-weight: 500;
+    margin-bottom: 20px;
+  }
+}
+
+.empty-column-indicator,
+.follow_requests-unlocked_explanation {
+  color: $dark-text-color;
+  background: $ui-base-color;
+  text-align: center;
+  padding: 20px;
+  font-size: 15px;
+  font-weight: 400;
+  cursor: default;
+  display: flex;
+  flex: 1 1 auto;
+  align-items: center;
+  justify-content: center;
+
+  & > span {
+    max-width: 500px;
+  }
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+  }
+}
+
+.follow_requests-unlocked_explanation {
+  background: darken($ui-base-color, 4%);
+  contain: initial;
+  flex-grow: 0;
+}
+
+.error-column {
+  padding: 20px;
+  background: $ui-base-color;
+  border-radius: 4px;
+  display: flex;
+  flex: 1 1 auto;
+  align-items: center;
+  justify-content: center;
+  flex-direction: column;
+  cursor: default;
+
+  &__image {
+    width: 70%;
+    max-width: 350px;
+    margin-top: -50px;
+  }
+
+  &__message {
+    text-align: center;
+    color: $darker-text-color;
+    font-size: 15px;
+    line-height: 22px;
+
+    h1 {
+      font-size: 28px;
+      line-height: 33px;
+      font-weight: 700;
+      margin-bottom: 15px;
+      color: $primary-text-color;
+    }
+
+    p {
+      max-width: 48ch;
+    }
+
+    &__actions {
+      margin-top: 30px;
+      display: flex;
+      gap: 10px;
+      align-items: center;
+      justify-content: center;
+    }
+  }
+}
+
+.column-inline-form {
+  padding: 7px 15px;
+  padding-inline-end: 5px;
+  display: flex;
+  justify-content: flex-start;
+  align-items: center;
+  background: lighten($ui-base-color, 4%);
+
+  label {
+    flex: 1 1 auto;
+
+    input {
+      width: 100%;
+      margin-bottom: 6px;
+
+      &:focus {
+        outline: 0;
+      }
+    }
+  }
+
+  .icon-button {
+    flex: 0 0 auto;
+    margin: 0 5px;
+  }
+}
+
+.column-settings__outer {
+  background: lighten($ui-base-color, 8%);
+  padding: 15px;
+}
+
+.column-settings__section {
+  color: $darker-text-color;
+  cursor: default;
+  display: block;
+  font-weight: 500;
+  margin-bottom: 10px;
+}
+
+.column-settings__row--with-margin {
+  margin-bottom: 15px;
+}
+
+.column-settings__hashtags {
+  .column-settings__row {
+    margin-bottom: 15px;
+  }
+
+  .column-select {
+    &__control {
+      @include search-input;
+
+      &::placeholder {
+        color: lighten($darker-text-color, 4%);
+      }
+
+      &::-moz-focus-inner {
+        border: 0;
+      }
+
+      &::-moz-focus-inner,
+      &:focus,
+      &:active {
+        outline: 0 !important;
+      }
+
+      &:focus {
+        background: lighten($ui-base-color, 4%);
+      }
+
+      @media screen and (width <= 600px) {
+        font-size: 16px;
+      }
+    }
+
+    &__placeholder {
+      color: $dark-text-color;
+      padding-inline-start: 2px;
+      font-size: 12px;
+    }
+
+    &__value-container {
+      padding-inline-start: 6px;
+    }
+
+    &__multi-value {
+      background: lighten($ui-base-color, 8%);
+
+      &__remove {
+        cursor: pointer;
+
+        &:hover,
+        &:active,
+        &:focus {
+          background: lighten($ui-base-color, 12%);
+          color: lighten($darker-text-color, 4%);
+        }
+      }
+    }
+
+    &__multi-value__label,
+    &__input,
+    &__input-container {
+      color: $darker-text-color;
+    }
+
+    &__clear-indicator,
+    &__dropdown-indicator {
+      cursor: pointer;
+      transition: none;
+      color: $dark-text-color;
+
+      &:hover,
+      &:active,
+      &:focus {
+        color: lighten($dark-text-color, 4%);
+      }
+    }
+
+    &__indicator-separator {
+      background-color: lighten($ui-base-color, 8%);
+    }
+
+    &__menu {
+      @include search-popout;
+
+      padding: 0;
+      background: $ui-secondary-color;
+    }
+
+    &__menu-list {
+      padding: 6px;
+    }
+
+    &__option {
+      color: $inverted-text-color;
+      border-radius: 4px;
+      font-size: 14px;
+
+      &--is-focused,
+      &--is-selected {
+        background: darken($ui-secondary-color, 10%);
+      }
+    }
+  }
+}
+
+.column-settings__row {
+  .text-btn:not(.column-header__permission-btn) {
+    margin-bottom: 15px;
+  }
+}
+
+.notifications-permission-banner {
+  padding: 30px;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  display: flex;
+  flex-direction: column;
+  align-items: center;
+  justify-content: center;
+  position: relative;
+
+  &__close {
+    position: absolute;
+    top: 10px;
+    inset-inline-end: 10px;
+  }
+
+  h2 {
+    font-size: 16px;
+    font-weight: 500;
+    margin-bottom: 15px;
+    text-align: center;
+  }
+
+  p {
+    color: $darker-text-color;
+    margin-bottom: 15px;
+    text-align: center;
+  }
+}
+
+.column-title {
+  text-align: center;
+  padding-bottom: 40px;
+
+  h3 {
+    font-size: 24px;
+    line-height: 1.5;
+    font-weight: 700;
+    margin-bottom: 10px;
+  }
+
+  p {
+    font-size: 16px;
+    line-height: 24px;
+    font-weight: 400;
+    color: $darker-text-color;
+  }
+
+  @media screen and (width >= 600px) {
+    padding: 40px;
+  }
+}
+
+.onboarding__footer {
+  margin-top: 30px;
+  color: $dark-text-color;
+  text-align: center;
+  font-size: 14px;
+
+  .link-button {
+    display: inline-block;
+    color: inherit;
+    font-size: inherit;
+  }
+}
+
+.onboarding__link {
+  display: flex;
+  align-items: center;
+  justify-content: space-between;
+  gap: 10px;
+  color: $highlight-text-color;
+  background: lighten($ui-base-color, 4%);
+  border-radius: 8px;
+  padding: 10px 15px;
+  box-sizing: border-box;
+  font-size: 14px;
+  font-weight: 500;
+  height: 56px;
+  text-decoration: none;
+
+  svg {
+    height: 1.5em;
+  }
+
+  &:hover,
+  &:focus,
+  &:active {
+    background: lighten($ui-base-color, 8%);
+  }
+}
+
+.onboarding__illustration {
+  display: block;
+  margin: 0 auto;
+  margin-bottom: 10px;
+  max-height: 200px;
+  width: auto;
+}
+
+.onboarding__lead {
+  font-size: 16px;
+  line-height: 24px;
+  font-weight: 400;
+  color: $darker-text-color;
+  text-align: center;
+  margin-bottom: 30px;
+
+  strong {
+    font-weight: 700;
+    color: $secondary-text-color;
+  }
+}
+
+.onboarding__links {
+  margin-bottom: 30px;
+
+  & > * {
+    margin-bottom: 2px;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+}
+
+.onboarding__steps {
+  margin-bottom: 30px;
+
+  &__item {
+    background: lighten($ui-base-color, 4%);
+    border: 0;
+    border-radius: 8px;
+    display: flex;
+    width: 100%;
+    box-sizing: border-box;
+    align-items: center;
+    gap: 10px;
+    padding: 10px;
+    padding-inline-end: 15px;
+    margin-bottom: 2px;
+    text-decoration: none;
+    text-align: start;
+
+    &:hover,
+    &:focus,
+    &:active {
+      background: lighten($ui-base-color, 8%);
+    }
+
+    &__icon {
+      flex: 0 0 auto;
+      border-radius: 50%;
+      display: none;
+      align-items: center;
+      justify-content: center;
+      width: 36px;
+      height: 36px;
+      color: $highlight-text-color;
+      font-size: 1.2rem;
+
+      @media screen and (width >= 600px) {
+        display: flex;
+      }
+    }
+
+    &__progress {
+      flex: 0 0 auto;
+      background: $valid-value-color;
+      border-radius: 50%;
+      display: flex;
+      align-items: center;
+      justify-content: center;
+      width: 21px;
+      height: 21px;
+      color: $primary-text-color;
+
+      svg {
+        height: 14px;
+        width: auto;
+      }
+    }
+
+    &__go {
+      flex: 0 0 auto;
+      display: flex;
+      align-items: center;
+      justify-content: center;
+      width: 21px;
+      height: 21px;
+      color: $highlight-text-color;
+      font-size: 17px;
+
+      svg {
+        height: 1.5em;
+        width: auto;
+      }
+    }
+
+    &__description {
+      flex: 1 1 auto;
+      line-height: 20px;
+
+      h6 {
+        color: $highlight-text-color;
+        font-weight: 500;
+        font-size: 14px;
+      }
+
+      p {
+        color: $darker-text-color;
+        overflow: hidden;
+      }
+    }
+  }
+}
+
+.onboarding__progress-indicator {
+  display: flex;
+  align-items: center;
+  margin-bottom: 30px;
+  position: sticky;
+  background: $ui-base-color;
+
+  @media screen and (width >= 600) {
+    padding: 0 40px;
+  }
+
+  &__line {
+    height: 4px;
+    flex: 1 1 auto;
+    background: lighten($ui-base-color, 4%);
+  }
+
+  &__step {
+    flex: 0 0 auto;
+    width: 30px;
+    height: 30px;
+    background: lighten($ui-base-color, 4%);
+    border-radius: 50%;
+    color: $primary-text-color;
+    display: flex;
+    align-items: center;
+    justify-content: center;
+
+    svg {
+      width: 15px;
+      height: auto;
+    }
+
+    &.active {
+      background: $valid-value-color;
+    }
+  }
+
+  &__step.active,
+  &__line.active {
+    background: $valid-value-color;
+    background-image: linear-gradient(
+      90deg,
+      $valid-value-color,
+      lighten($valid-value-color, 8%),
+      $valid-value-color
+    );
+    background-size: 200px 100%;
+    animation: skeleton 1.2s ease-in-out infinite;
+  }
+}
+
+.follow-recommendations {
+  background: darken($ui-base-color, 4%);
+  border-radius: 8px;
+  margin-bottom: 30px;
+
+  .account:last-child {
+    border-bottom: 0;
+  }
+
+  &__empty {
+    text-align: center;
+    color: $darker-text-color;
+    font-weight: 500;
+    padding: 40px;
+  }
+}
+
+.tip-carousel {
+  border: 1px solid transparent;
+  border-radius: 8px;
+  padding: 16px;
+  margin-bottom: 30px;
+
+  &:focus {
+    outline: 0;
+    border-color: $highlight-text-color;
+  }
+
+  .media-modal__pagination {
+    margin-bottom: 0;
+  }
+}
+
+.copy-paste-text {
+  background: lighten($ui-base-color, 4%);
+  border-radius: 8px;
+  border: 1px solid lighten($ui-base-color, 8%);
+  padding: 16px;
+  color: $primary-text-color;
+  font-size: 15px;
+  line-height: 22px;
+  display: flex;
+  flex-direction: column;
+  align-items: flex-end;
+  transition: border-color 300ms linear;
+  margin-bottom: 30px;
+
+  &:focus,
+  &.focused {
+    transition: none;
+    outline: 0;
+    border-color: $highlight-text-color;
+  }
+
+  &.copied {
+    border-color: $valid-value-color;
+    transition: none;
+  }
+
+  textarea {
+    width: 100%;
+    height: auto;
+    background: transparent;
+    color: inherit;
+    font: inherit;
+    border: 0;
+    padding: 0;
+    margin-bottom: 30px;
+    resize: none;
+
+    &:focus {
+      outline: 0;
+    }
+  }
+}
+
+.compose-form__highlightable {
+  display: flex;
+  flex-direction: column;
+  flex: 0 1 auto;
+  border-radius: 4px;
+  transition: box-shadow 300ms linear;
+  min-height: 0;
+  position: relative;
+
+  &.active {
+    transition: none;
+    box-shadow: 0 0 0 6px rgba(lighten($highlight-text-color, 8%), 0.7);
+  }
+}
+
+.dismissable-banner,
+.warning-banner {
+  position: relative;
+  margin: 10px;
+  margin-bottom: 5px;
+  border-radius: 8px;
+  border: 1px solid $highlight-text-color;
+  background: rgba($highlight-text-color, 0.15);
+  overflow: hidden;
+
+  &__background-image {
+    width: 125%;
+    position: absolute;
+    bottom: -25%;
+    inset-inline-end: -25%;
+    z-index: -1;
+    opacity: 0.15;
+    mix-blend-mode: luminosity;
+  }
+
+  &__message {
+    flex: 1 1 auto;
+    padding: 15px;
+    font-size: 15px;
+    line-height: 22px;
+    font-weight: 500;
+    color: $primary-text-color;
+
+    p {
+      margin-bottom: 15px;
+
+      &:last-child {
+        margin-bottom: 0;
+      }
+    }
+
+    h1 {
+      color: $highlight-text-color;
+      font-size: 22px;
+      line-height: 33px;
+      font-weight: 700;
+      margin-bottom: 15px;
+    }
+
+    &__actions {
+      display: flex;
+      flex-wrap: wrap;
+      gap: 4px;
+
+      &__wrapper {
+        display: flex;
+        margin-top: 30px;
+      }
+
+      .button {
+        display: block;
+        flex-grow: 1;
+      }
+    }
+
+    .button-tertiary {
+      background: rgba($ui-base-color, 0.15);
+      backdrop-filter: blur(8px);
+    }
+  }
+
+  &__action {
+    float: right;
+    padding: 15px 10px;
+
+    .icon-button {
+      color: $highlight-text-color;
+    }
+  }
+}
+
+.warning-banner {
+  border: 1px solid $warning-red;
+  background: rgba($warning-red, 0.15);
+
+  &__message {
+    h1 {
+      color: $warning-red;
+    }
+
+    a {
+      color: $primary-text-color;
+    }
+  }
+}
+
+.hashtag-header {
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  padding: 15px;
+  font-size: 17px;
+  line-height: 22px;
+  color: $darker-text-color;
+
+  strong {
+    font-weight: 700;
+  }
+
+  &__header {
+    display: flex;
+    justify-content: space-between;
+    align-items: center;
+    margin-bottom: 15px;
+    gap: 15px;
+
+    h1 {
+      color: $primary-text-color;
+      white-space: nowrap;
+      text-overflow: ellipsis;
+      overflow: hidden;
+      font-size: 22px;
+      line-height: 33px;
+      font-weight: 700;
+    }
+  }
+
+  &:focus {
+    outline: 0;
+    background-color: $highlight-text-color;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/compose_form.scss b/app/javascript/flavours/blobfox/styles/components/compose_form.scss
new file mode 100644
index 00000000000000..0f64c0dcc10f5a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/compose_form.scss
@@ -0,0 +1,685 @@
+.compose-form {
+  padding: 10px;
+
+  .emoji-picker-dropdown {
+    position: absolute;
+    top: 0;
+    inset-inline-end: 0;
+
+    ::-webkit-scrollbar-track:hover,
+    ::-webkit-scrollbar-track:active {
+      background-color: rgba($base-overlay-background, 0.3);
+    }
+  }
+}
+
+.character-counter {
+  cursor: default;
+  font-family: $font-sans-serif, sans-serif;
+  font-size: 14px;
+  font-weight: 600;
+  color: $lighter-text-color;
+
+  &.character-counter--over {
+    color: $warning-red;
+  }
+}
+
+.no-reduce-motion .spoiler-input {
+  transition:
+    height 0.4s ease,
+    opacity 0.4s ease;
+}
+
+.spoiler-input {
+  height: 0;
+  transform-origin: bottom;
+  opacity: 0;
+
+  &.spoiler-input--visible {
+    height: 36px;
+    margin-bottom: 11px;
+    opacity: 1;
+  }
+
+  input {
+    display: block;
+    box-sizing: border-box;
+    margin: 0;
+    border: 0;
+    border-radius: 4px;
+    padding: 10px;
+    width: 100%;
+    outline: 0;
+    color: $inverted-text-color;
+    background: $simple-background-color;
+    font-size: 14px;
+    font-family: inherit;
+    resize: vertical;
+
+    &::placeholder {
+      color: $dark-text-color;
+    }
+
+    &:focus {
+      outline: 0;
+    }
+    @include single-column('screen and (max-width: 630px)') {
+      font-size: 16px;
+    }
+  }
+}
+
+.compose-form__warning {
+  color: $inverted-text-color;
+  margin-bottom: 15px;
+  background: $ui-primary-color;
+  box-shadow: 0 2px 6px rgba($base-shadow-color, 0.3);
+  padding: 8px 10px;
+  border-radius: 4px;
+  font-size: 13px;
+  font-weight: 400;
+
+  a {
+    color: $lighter-text-color;
+    font-weight: 500;
+    text-decoration: underline;
+
+    &:active,
+    &:focus,
+    &:hover {
+      text-decoration: none;
+    }
+  }
+}
+
+.compose-form__sensitive-button {
+  padding: 10px;
+  padding-top: 0;
+  font-size: 14px;
+  font-weight: 500;
+
+  &.active {
+    color: $highlight-text-color;
+  }
+
+  input[type='checkbox'] {
+    appearance: none;
+    display: inline-block;
+    position: relative;
+    border: 1px solid $ui-primary-color;
+    box-sizing: border-box;
+    width: 18px;
+    height: 18px;
+    flex: 0 0 auto;
+    margin-inline-start: 5px;
+    margin-inline-end: 10px;
+    top: -1px;
+    border-radius: 4px;
+    vertical-align: middle;
+    cursor: inherit;
+
+    &:checked {
+      border-color: $highlight-text-color;
+      background: $highlight-text-color
+        url("data:image/svg+xml;utf8,<svg width='18' height='18' fill='none' xmlns='http://www.w3.org/2000/svg'><path d='M4.5 8.5L8 12l6-6' stroke='white' stroke-width='1.5'/></svg>")
+        center center no-repeat;
+    }
+  }
+}
+
+.reply-indicator {
+  margin: 0 0 10px;
+  border-radius: 4px;
+  padding: 10px;
+  background: $ui-primary-color;
+  min-height: 23px;
+  overflow-y: auto;
+  flex: 0 2 auto;
+}
+
+.reply-indicator__header {
+  margin-bottom: 5px;
+  overflow: hidden;
+
+  & > .account.small {
+    color: $inverted-text-color;
+  }
+}
+
+.reply-indicator__cancel {
+  float: right;
+  line-height: 24px;
+}
+
+.reply-indicator__content {
+  position: relative;
+  font-size: 14px;
+  line-height: 20px;
+  word-wrap: break-word;
+  font-weight: 400;
+  overflow: hidden;
+  padding-top: 5px;
+  color: $inverted-text-color;
+  white-space: pre-wrap;
+
+  p,
+  pre {
+    margin-bottom: 20px;
+    white-space: pre-wrap;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  a {
+    color: $lighter-text-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+
+    &.mention {
+      &:hover {
+        text-decoration: none;
+
+        span {
+          text-decoration: underline;
+        }
+      }
+    }
+  }
+
+  .emojione {
+    width: 20px;
+    height: 20px;
+    margin: -5px 0 0;
+  }
+}
+
+.compose-form .compose-form__autosuggest-wrapper {
+  position: relative;
+}
+
+.compose-form .autosuggest-textarea,
+.compose-form .autosuggest-input {
+  position: relative;
+  width: 100%;
+
+  label {
+    .autosuggest-textarea__textarea {
+      display: block;
+      box-sizing: border-box;
+      margin: 0;
+      border: 0;
+      border-radius: 4px 4px 0 0;
+      padding: 10px 32px 0 10px;
+      width: 100%;
+      min-height: 100px;
+      outline: 0;
+      color: $inverted-text-color;
+      background: $simple-background-color;
+      font-size: 14px;
+      font-family: inherit;
+      resize: none;
+      scrollbar-color: initial;
+
+      &::placeholder {
+        color: $dark-text-color;
+      }
+
+      &::-webkit-scrollbar {
+        all: unset;
+      }
+
+      &:focus {
+        outline: 0;
+      }
+
+      @include single-column('screen and (max-width: 630px)') {
+        font-size: 16px;
+      }
+
+      @include limited-single-column('screen and (max-width: 600px)') {
+        height: 100px !important; // prevent auto-resize textarea
+        resize: vertical;
+      }
+    }
+  }
+}
+
+.compose-form__textarea-icons {
+  display: block;
+  position: absolute;
+  top: 29px;
+  inset-inline-end: 5px;
+  bottom: 5px;
+  overflow: hidden;
+
+  & > .textarea_icon {
+    display: block;
+    margin-top: 2px;
+    margin-inline-start: 2px;
+    width: 24px;
+    height: 24px;
+    color: $lighter-text-color;
+    font-size: 18px;
+    line-height: 24px;
+    text-align: center;
+    opacity: 0.8;
+  }
+}
+
+.autosuggest-textarea__suggestions-wrapper {
+  position: relative;
+  height: 0;
+}
+
+.autosuggest-textarea__suggestions {
+  box-sizing: border-box;
+  display: none;
+  position: absolute;
+  top: 100%;
+  width: 100%;
+  z-index: 99;
+  box-shadow: 4px 4px 6px rgba($base-shadow-color, 0.4);
+  background: $ui-secondary-color;
+  border-radius: 0 0 4px 4px;
+  color: $inverted-text-color;
+  font-size: 14px;
+  padding: 6px;
+}
+
+.autosuggest-textarea__suggestions--visible {
+  display: block;
+}
+
+.autosuggest-textarea__suggestions__item {
+  padding: 10px;
+  cursor: pointer;
+  border-radius: 4px;
+
+  &:hover,
+  &:focus,
+  &:active,
+  &.selected {
+    background: darken($ui-secondary-color, 10%);
+  }
+
+  .autosuggest-account,
+  .autosuggest-emoji,
+  .autosuggest-hashtag {
+    display: flex;
+    flex-direction: row;
+    align-items: center;
+    justify-content: flex-start;
+    line-height: 18px;
+    font-size: 14px;
+  }
+
+  .autosuggest-hashtag {
+    justify-content: space-between;
+
+    &__name {
+      flex: 1 1 auto;
+      overflow: hidden;
+      text-overflow: ellipsis;
+      white-space: nowrap;
+    }
+
+    strong {
+      font-weight: 500;
+    }
+
+    &__uses {
+      flex: 0 0 auto;
+      text-align: end;
+      overflow: hidden;
+      text-overflow: ellipsis;
+      white-space: nowrap;
+    }
+  }
+
+  .autosuggest-account-icon,
+  .autosuggest-emoji img {
+    margin-inline-end: 8px;
+  }
+
+  .autosuggest-account .display-name > span {
+    color: $lighter-text-color;
+  }
+}
+
+.compose-form__upload-wrapper {
+  overflow: hidden;
+}
+
+.compose-form__uploads-wrapper {
+  display: flex;
+  flex-direction: row;
+  flex-wrap: wrap;
+  font-family: inherit;
+  padding: 5px;
+  overflow: hidden;
+}
+
+.compose-form__upload {
+  flex: 1 1 0;
+  margin: 5px;
+  min-width: 40%;
+
+  .compose-form__upload-thumbnail {
+    position: relative;
+    border-radius: 4px;
+    height: 140px;
+    width: 100%;
+    background-color: $base-shadow-color;
+    background-position: center;
+    background-size: cover;
+    background-repeat: no-repeat;
+    overflow: hidden;
+
+    & > .close {
+      mix-blend-mode: difference;
+    }
+  }
+
+  .icon-button {
+    flex: 0 1 auto;
+    color: $secondary-text-color;
+    font-size: 14px;
+    font-weight: 500;
+    padding: 10px;
+    font-family: inherit;
+
+    &:hover,
+    &:focus,
+    &:active {
+      color: lighten($secondary-text-color, 7%);
+    }
+  }
+
+  &__warning {
+    position: absolute;
+    z-index: 2;
+    bottom: 0;
+    inset-inline-start: 0;
+    inset-inline-end: 0;
+    box-sizing: border-box;
+    background: linear-gradient(
+      0deg,
+      rgba($base-shadow-color, 0.8) 0,
+      rgba($base-shadow-color, 0.35) 80%,
+      transparent
+    );
+  }
+}
+
+.compose-form__upload__actions {
+  background: linear-gradient(
+    180deg,
+    rgba($base-shadow-color, 0.8) 0,
+    rgba($base-shadow-color, 0.35) 80%,
+    transparent
+  );
+  display: flex;
+  align-items: flex-start;
+  justify-content: space-between;
+}
+
+.upload-progress {
+  display: flex;
+  padding: 10px;
+  color: $darker-text-color;
+  overflow: hidden;
+
+  .fa {
+    font-size: 34px;
+    margin-inline-end: 10px;
+  }
+
+  span {
+    display: block;
+    font-size: 12px;
+    font-weight: 500;
+    text-transform: uppercase;
+  }
+}
+
+.upload-progress__message {
+  flex: 1 1 auto;
+}
+
+.upload-progress__backdrop {
+  position: relative;
+  margin-top: 5px;
+  border-radius: 6px;
+  width: 100%;
+  height: 6px;
+  background: darken($simple-background-color, 8%);
+}
+
+.upload-progress__tracker {
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  height: 6px;
+  border-radius: 6px;
+  background: $ui-highlight-color;
+}
+
+.compose-form__modifiers {
+  color: $inverted-text-color;
+  font-family: inherit;
+  font-size: 14px;
+  background: $simple-background-color;
+}
+
+.compose-form__buttons-wrapper {
+  padding: 10px;
+  background: darken($simple-background-color, 8%);
+  border-radius: 0 0 4px 4px;
+  height: 27px;
+  display: flex;
+  justify-content: space-between;
+  flex: 0 0 auto;
+}
+
+.compose-form__buttons {
+  display: flex;
+  flex: 0 0 auto;
+
+  & .icon-button,
+  & .text-icon-button {
+    display: inline-block;
+    box-sizing: content-box;
+    padding: 0 3px;
+    height: 27px;
+    line-height: 27px;
+    vertical-align: bottom;
+  }
+
+  & > hr {
+    display: inline-block;
+    margin: 0 3px;
+    border-width: 0 0 0 1px;
+    border-style: none none none solid;
+    border-color: transparent transparent transparent
+      darken($simple-background-color, 24%);
+    padding: 0;
+    width: 0;
+    height: 27px;
+    background: transparent;
+  }
+}
+
+.character-counter__wrapper {
+  align-self: center;
+  margin-inline-end: 4px;
+}
+
+.privacy-dropdown.active {
+  .privacy-dropdown__value {
+    background: $simple-background-color;
+    border-radius: 4px 4px 0 0;
+    box-shadow: 0 -4px 4px rgba($base-shadow-color, 0.1);
+
+    .icon-button {
+      transition: none;
+    }
+
+    &.active {
+      background: $ui-highlight-color;
+
+      .icon-button {
+        color: $primary-text-color;
+      }
+    }
+  }
+
+  &.top .privacy-dropdown__value {
+    border-radius: 0 0 4px 4px;
+  }
+
+  .privacy-dropdown__dropdown {
+    display: block;
+    box-shadow: 2px 4px 6px rgba($base-shadow-color, 0.1);
+  }
+}
+
+.privacy-dropdown__dropdown {
+  border-radius: 4px;
+  box-shadow: 2px 4px 15px rgba($base-shadow-color, 0.4);
+  background: $simple-background-color;
+  overflow: hidden;
+  transform-origin: 50% 0;
+}
+
+.privacy-dropdown__option {
+  display: flex;
+  align-items: center;
+  padding: 10px;
+  color: $inverted-text-color;
+  cursor: pointer;
+
+  .privacy-dropdown__option__content {
+    flex: 1 1 auto;
+    color: $lighter-text-color;
+
+    &:not(:first-child) {
+      margin-inline-start: 10px;
+    }
+
+    strong {
+      display: block;
+      color: $inverted-text-color;
+      font-weight: 500;
+    }
+  }
+
+  &:hover,
+  &.active {
+    background: $ui-highlight-color;
+    color: $primary-text-color;
+
+    .privacy-dropdown__option__content {
+      color: $primary-text-color;
+
+      strong {
+        color: $primary-text-color;
+      }
+    }
+  }
+
+  &.active:hover {
+    background: lighten($ui-highlight-color, 4%);
+  }
+}
+
+.compose-form__publish {
+  display: flex;
+  justify-content: flex-end;
+  min-width: 0;
+  flex: 0 0 auto;
+  column-gap: 5px;
+
+  .compose-form__publish-button-wrapper {
+    overflow: hidden;
+    padding-top: 10px;
+
+    button {
+      padding: 7px 10px;
+      text-align: center;
+    }
+
+    & > .side_arm {
+      width: 36px;
+    }
+  }
+}
+
+.language-dropdown {
+  &__dropdown {
+    background: $simple-background-color;
+    box-shadow: 2px 4px 15px rgba($base-shadow-color, 0.4);
+    border-radius: 4px;
+    overflow: hidden;
+    z-index: 2;
+
+    &.top {
+      transform-origin: 50% 100%;
+    }
+
+    &.bottom {
+      transform-origin: 50% 0;
+    }
+
+    .emoji-mart-search {
+      padding-inline-end: 10px;
+    }
+
+    .emoji-mart-search-icon {
+      inset-inline-end: 10px + 5px;
+    }
+
+    .emoji-mart-scroll {
+      padding: 0 10px 10px;
+    }
+
+    &__results {
+      &__item {
+        cursor: pointer;
+        color: $inverted-text-color;
+        font-weight: 500;
+        padding: 10px;
+        border-radius: 4px;
+
+        &:focus,
+        &:active,
+        &:hover {
+          background: $ui-secondary-color;
+        }
+
+        &__common-name {
+          color: $darker-text-color;
+        }
+
+        &.active {
+          background: $ui-highlight-color;
+          color: $primary-text-color;
+          outline: 0;
+
+          .language-dropdown__dropdown__results__item__common-name {
+            color: $secondary-text-color;
+          }
+
+          &:hover {
+            background: lighten($ui-highlight-color, 4%);
+          }
+        }
+      }
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/directory.scss b/app/javascript/flavours/blobfox/styles/components/directory.scss
new file mode 100644
index 00000000000000..db9a23bce2f550
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/directory.scss
@@ -0,0 +1,68 @@
+.scrollable .account-card {
+  margin: 10px;
+  background: lighten($ui-base-color, 8%);
+}
+
+.scrollable .account-card__title__avatar {
+  img,
+  .account__avatar {
+    border-color: lighten($ui-base-color, 8%);
+  }
+}
+
+.scrollable .account-card__bio::after {
+  background: linear-gradient(
+    to left,
+    lighten($ui-base-color, 8%),
+    transparent
+  );
+}
+
+.filter-form {
+  background: $ui-base-color;
+
+  &__column {
+    padding: 10px 15px;
+    padding-bottom: 0;
+  }
+
+  .radio-button {
+    display: block;
+  }
+}
+
+.radio-button {
+  font-size: 14px;
+  position: relative;
+  display: inline-block;
+  padding: 6px 0;
+  line-height: 18px;
+  white-space: nowrap;
+  overflow: hidden;
+  text-overflow: ellipsis;
+  cursor: pointer;
+
+  input[type='radio'],
+  input[type='checkbox'] {
+    display: none;
+  }
+
+  &__input {
+    display: inline-block;
+    position: relative;
+    border: 1px solid $ui-primary-color;
+    box-sizing: border-box;
+    width: 18px;
+    height: 18px;
+    flex: 0 0 auto;
+    margin-inline-end: 10px;
+    top: -1px;
+    border-radius: 50%;
+    vertical-align: middle;
+
+    &.checked {
+      border-color: lighten($ui-highlight-color, 4%);
+      background: lighten($ui-highlight-color, 4%);
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/domains.scss b/app/javascript/flavours/blobfox/styles/components/domains.scss
new file mode 100644
index 00000000000000..a99ccd02b60693
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/domains.scss
@@ -0,0 +1,23 @@
+.domain {
+  padding: 10px;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+
+  .domain__domain-name {
+    flex: 1 1 auto;
+    display: block;
+    color: $primary-text-color;
+    text-decoration: none;
+    font-size: 14px;
+    font-weight: 500;
+  }
+}
+
+.domain__wrapper {
+  display: flex;
+}
+
+.domain_buttons {
+  height: 18px;
+  padding: 10px;
+  white-space: nowrap;
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/doodle.scss b/app/javascript/flavours/blobfox/styles/components/doodle.scss
new file mode 100644
index 00000000000000..eb053c14db53f1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/doodle.scss
@@ -0,0 +1,86 @@
+$doodle-background: #d9e1e8;
+
+.doodle-modal {
+  width: unset;
+}
+
+.doodle-modal__container {
+  background: $doodle-background;
+  text-align: center;
+  line-height: 0; // remove weird gap under canvas
+  canvas {
+    border: 5px solid $doodle-background;
+  }
+}
+
+.doodle-modal__action-bar {
+  .filler {
+    flex-grow: 1;
+    margin: 0;
+    padding: 0;
+  }
+
+  .doodle-toolbar {
+    line-height: 1;
+    display: flex;
+    flex-direction: column;
+    flex-grow: 0;
+    justify-content: space-around;
+
+    &.with-inputs {
+      label {
+        display: inline-block;
+        width: 70px;
+        text-align: end;
+        margin-inline-end: 2px;
+      }
+
+      input[type='number'],
+      input[type='text'] {
+        width: 40px;
+      }
+
+      span.val {
+        display: inline-block;
+        text-align: start;
+        width: 50px;
+      }
+    }
+  }
+
+  .doodle-palette {
+    padding-inline-end: 0 !important;
+    border: 1px solid black;
+    line-height: 0.2rem;
+    flex-grow: 0;
+    background: white;
+
+    button {
+      appearance: none;
+      width: 1rem;
+      height: 1rem;
+      margin: 0;
+      padding: 0;
+      text-align: center;
+      color: black;
+      text-shadow: 0 0 1px white;
+      cursor: pointer;
+      box-shadow: inset 0 0 1px rgba(white, 0.5);
+      border: 1px solid black;
+      outline-offset: -1px;
+
+      &.foreground {
+        outline: 1px dashed white;
+      }
+
+      &.background {
+        outline: 1px dashed red;
+      }
+
+      &.foreground.background {
+        outline: 1px dashed red;
+        border-color: white;
+      }
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/drawer.scss b/app/javascript/flavours/blobfox/styles/components/drawer.scss
new file mode 100644
index 00000000000000..dcccc0acbb175e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/drawer.scss
@@ -0,0 +1,298 @@
+.drawer {
+  width: 300px;
+  box-sizing: border-box;
+  display: flex;
+  flex-direction: column;
+  overflow-y: hidden;
+  padding: 10px 5px;
+  flex: none;
+
+  &:first-child {
+    padding-inline-start: 10px;
+  }
+
+  &:last-child {
+    padding-inline-end: 10px;
+  }
+
+  @include single-column('screen and (max-width: 630px)') {
+    flex: auto;
+  }
+
+  @include limited-single-column('screen and (max-width: 630px)') {
+    &,
+    &:first-child,
+    &:last-child {
+      padding: 0;
+    }
+  }
+
+  .wide & {
+    min-width: 300px;
+    max-width: 400px;
+    flex: 1 1 200px;
+  }
+
+  @include single-column('screen and (max-width: 630px)') {
+    :root & {
+      //  Overrides `.wide` for single-column view
+      flex: auto;
+      width: 100%;
+      min-width: 0;
+      max-width: none;
+      padding: 0;
+    }
+  }
+
+  .react-swipeable-view-container & {
+    height: 100%;
+  }
+}
+
+.drawer__header {
+  flex: none;
+  font-size: 16px;
+  background: lighten($ui-base-color, 8%);
+  margin-bottom: 10px;
+  display: flex;
+  flex-direction: row;
+  border-radius: 4px;
+  overflow: hidden;
+
+  & > * {
+    display: block;
+    box-sizing: border-box;
+    border-bottom: 2px solid transparent;
+    padding: 15px 5px 13px;
+    height: 48px;
+    flex: 1 1 auto;
+    color: $darker-text-color;
+    text-align: center;
+    text-decoration: none;
+    cursor: pointer;
+  }
+
+  a {
+    transition: background 100ms ease-in;
+
+    &:focus,
+    &:hover {
+      outline: none;
+      background: lighten($ui-base-color, 3%);
+      transition: background 200ms ease-out;
+    }
+  }
+}
+
+.search {
+  position: relative;
+  margin-bottom: 10px;
+  flex: none;
+
+  @include limited-single-column(
+    'screen and (max-width: #{$no-gap-breakpoint})'
+  ) {
+    margin-bottom: 0;
+  }
+  @include single-column('screen and (max-width: 630px)') {
+    font-size: 16px;
+  }
+}
+
+.navigation-bar {
+  padding: 10px;
+  color: $darker-text-color;
+  display: flex;
+  align-items: center;
+
+  a {
+    color: inherit;
+    text-decoration: none;
+  }
+
+  .acct {
+    display: block;
+    color: $secondary-text-color;
+    font-weight: 500;
+    white-space: nowrap;
+    overflow: hidden;
+    text-overflow: ellipsis;
+  }
+}
+
+.navigation-bar__profile {
+  flex: 1 1 auto;
+  margin-inline-start: 8px;
+  overflow: hidden;
+}
+
+.drawer--results {
+  overflow-x: hidden;
+  overflow-y: scroll;
+}
+
+.search-results__section {
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+
+  &:last-child {
+    border-bottom: 0;
+  }
+
+  &__header {
+    background: darken($ui-base-color, 4%);
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+    padding: 15px;
+    font-weight: 500;
+    font-size: 14px;
+    color: $darker-text-color;
+    display: flex;
+    justify-content: space-between;
+
+    h3 .fa {
+      margin-inline-end: 5px;
+    }
+
+    button {
+      color: $highlight-text-color;
+      padding: 0;
+      border: 0;
+      background: 0;
+      font: inherit;
+
+      &:hover,
+      &:active,
+      &:focus {
+        text-decoration: underline;
+      }
+    }
+  }
+
+  .account:last-child,
+  & > div:last-child .status {
+    border-bottom: 0;
+  }
+
+  & > .hashtag {
+    display: block;
+    padding: 10px;
+    color: $secondary-text-color;
+    text-decoration: none;
+
+    &:hover,
+    &:active,
+    &:focus {
+      color: lighten($secondary-text-color, 4%);
+      text-decoration: underline;
+    }
+  }
+}
+
+.drawer__pager {
+  box-sizing: border-box;
+  padding: 0;
+  flex-grow: 1;
+  position: relative;
+  overflow: hidden;
+  display: flex;
+  border-radius: 4px;
+}
+
+.drawer__inner {
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  background: lighten($ui-base-color, 13%);
+  box-sizing: border-box;
+  padding: 0;
+  display: flex;
+  flex-direction: column;
+  overflow: hidden;
+  overflow-y: auto;
+  width: 100%;
+  height: 100%;
+
+  &.darker {
+    background: $ui-base-color;
+  }
+}
+
+.drawer__inner__mastodon {
+  background: lighten($ui-base-color, 13%)
+    url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 234.80078 31.757813" width="234.80078" height="31.757812"><path d="M19.599609 0c-1.05 0-2.10039.375-2.90039 1.125L0 16.925781v14.832031h234.80078V17.025391l-16.5-15.900391c-1.6-1.5-4.20078-1.5-5.80078 0l-13.80078 13.099609c-1.6 1.5-4.19883 1.5-5.79883 0L179.09961 1.125c-1.6-1.5-4.19883-1.5-5.79883 0L159.5 14.224609c-1.6 1.5-4.20078 1.5-5.80078 0L139.90039 1.125c-1.6-1.5-4.20078-1.5-5.80078 0l-13.79883 13.099609c-1.6 1.5-4.20078 1.5-5.80078 0L100.69922 1.125c-1.600001-1.5-4.198829-1.5-5.798829 0l-13.59961 13.099609c-1.6 1.5-4.200781 1.5-5.800781 0L61.699219 1.125c-1.6-1.5-4.198828-1.5-5.798828 0L42.099609 14.224609c-1.6 1.5-4.198828 1.5-5.798828 0L22.5 1.125C21.7.375 20.649609 0 19.599609 0z" fill="#{hex-color($ui-base-color)}"/></svg>')
+    no-repeat bottom / 100% auto;
+  flex: 1;
+  min-height: 47px;
+  display: none;
+
+  > img {
+    display: block;
+    object-fit: contain;
+    object-position: bottom left;
+    width: 85%;
+    height: 100%;
+    pointer-events: none;
+    user-select: none;
+  }
+
+  > .mastodon {
+    display: block;
+    width: 100%;
+    height: 100%;
+    border: 0;
+    cursor: inherit;
+  }
+
+  @media screen and (height >= 640px) {
+    display: block;
+  }
+}
+
+.pseudo-drawer {
+  background: lighten($ui-base-color, 13%);
+  font-size: 13px;
+  text-align: start;
+}
+
+.drawer__backdrop {
+  cursor: pointer;
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  width: 100%;
+  height: 100%;
+  background: rgba($base-overlay-background, 0.5);
+}
+
+@for $i from 0 through 3 {
+  .mbstobon-#{$i} .drawer__inner__mastodon {
+    @if $i == 3 {
+      background:
+        url('~flavours/blobfox/images/wave-drawer.png')
+          no-repeat
+          bottom /
+          100%
+          auto,
+        lighten($ui-base-color, 13%);
+    } @else {
+      background:
+        url('~flavours/blobfox/images/wave-drawer-blobfoxed.png')
+          no-repeat
+          bottom /
+          100%
+          auto,
+        lighten($ui-base-color, 13%);
+    }
+
+    & > .mastodon {
+      background: url('~flavours/blobfox/images/mbstobon-ui-#{$i}.png')
+        no-repeat
+        left
+        bottom /
+        contain;
+
+      @if $i != 3 {
+        filter: contrast(50%) brightness(50%);
+      }
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/emoji.scss b/app/javascript/flavours/blobfox/styles/components/emoji.scss
new file mode 100644
index 00000000000000..f76288978d630d
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/emoji.scss
@@ -0,0 +1,104 @@
+.emojione {
+  font-size: inherit;
+  vertical-align: middle;
+  object-fit: contain;
+  margin: -0.2ex 0.15em 0.2ex;
+  width: 16px;
+  height: 16px;
+
+  img {
+    width: auto;
+  }
+}
+
+.emoji-picker-dropdown__menu {
+  background: $simple-background-color;
+  position: relative;
+  box-shadow: 4px 4px 6px rgba($base-shadow-color, 0.4);
+  border-radius: 4px;
+  margin-top: 5px;
+  z-index: 2;
+
+  .emoji-mart-scroll {
+    transition: opacity 200ms ease;
+  }
+
+  &.selecting .emoji-mart-scroll {
+    opacity: 0.5;
+  }
+}
+
+.emoji-picker-dropdown__modifiers {
+  position: absolute;
+  top: 60px;
+  inset-inline-end: 11px;
+  cursor: pointer;
+}
+
+.emoji-picker-dropdown__modifiers__menu {
+  position: absolute;
+  z-index: 4;
+  top: -4px;
+  inset-inline-start: -8px;
+  background: $simple-background-color;
+  border-radius: 4px;
+  box-shadow: 1px 2px 6px rgba($base-shadow-color, 0.2);
+  overflow: hidden;
+
+  button {
+    display: block;
+    cursor: pointer;
+    border: 0;
+    padding: 4px 8px;
+    background: transparent;
+
+    &:hover,
+    &:focus,
+    &:active {
+      background: rgba($ui-secondary-color, 0.4);
+    }
+  }
+
+  .emoji-mart-emoji {
+    height: 22px;
+  }
+}
+
+.emoji-mart-emoji {
+  span {
+    background-repeat: no-repeat;
+  }
+}
+
+.emoji-button {
+  display: block;
+  padding-top: 5px;
+  padding-bottom: 2px;
+  padding-inline-start: 2px;
+  padding-inline-end: 5px;
+  outline: 0;
+  cursor: pointer;
+
+  &:active,
+  &:focus {
+    outline: 0 !important;
+  }
+
+  img {
+    filter: grayscale(100%);
+    opacity: 0.8;
+    display: block;
+    margin: 0;
+    width: 22px;
+    height: 22px;
+  }
+
+  &:hover,
+  &:active,
+  &:focus {
+    img {
+      opacity: 1;
+      filter: none;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/emoji_picker.scss b/app/javascript/flavours/blobfox/styles/components/emoji_picker.scss
new file mode 100644
index 00000000000000..e402838dbf9641
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/emoji_picker.scss
@@ -0,0 +1,261 @@
+.emoji-mart {
+  &,
+  * {
+    box-sizing: border-box;
+    line-height: 1.15;
+  }
+
+  font-size: 13px;
+  display: inline-block;
+  color: $inverted-text-color;
+
+  .emoji-mart-emoji {
+    padding: 6px;
+  }
+}
+
+.emoji-mart-bar {
+  border: 0 solid darken($ui-secondary-color, 8%);
+
+  &:first-child {
+    border-bottom-width: 1px;
+    border-top-left-radius: 5px;
+    border-top-right-radius: 5px;
+    background: $ui-secondary-color;
+  }
+
+  &:last-child {
+    border-top-width: 1px;
+    border-bottom-left-radius: 5px;
+    border-bottom-right-radius: 5px;
+    display: none;
+  }
+}
+
+.emoji-mart-anchors {
+  display: flex;
+  justify-content: space-between;
+  padding: 0 6px;
+  color: $lighter-text-color;
+  line-height: 0;
+}
+
+.emoji-mart-anchor {
+  position: relative;
+  flex: 1;
+  text-align: center;
+  padding: 12px 4px;
+  overflow: hidden;
+  transition: color 0.1s ease-out;
+  cursor: pointer;
+  background: transparent;
+  border: 0;
+
+  &:hover {
+    color: darken($lighter-text-color, 4%);
+  }
+}
+
+.emoji-mart-anchor-selected {
+  color: $highlight-text-color;
+
+  &:hover {
+    color: darken($highlight-text-color, 4%);
+  }
+
+  .emoji-mart-anchor-bar {
+    bottom: 0;
+  }
+}
+
+.emoji-mart-anchor-bar {
+  position: absolute;
+  bottom: -3px;
+  inset-inline-start: 0;
+  width: 100%;
+  height: 3px;
+  background-color: darken($ui-highlight-color, 3%);
+}
+
+.emoji-mart-anchors {
+  i {
+    display: inline-block;
+    width: 100%;
+    max-width: 22px;
+  }
+
+  svg {
+    fill: currentColor;
+    max-height: 18px;
+  }
+}
+
+.emoji-mart-scroll {
+  overflow-y: scroll;
+  height: 270px;
+  max-height: 35vh;
+  padding: 0 6px 6px;
+  background: $simple-background-color;
+  will-change: transform;
+
+  &::-webkit-scrollbar-track:hover,
+  &::-webkit-scrollbar-track:active {
+    background-color: rgba($base-overlay-background, 0.3);
+  }
+}
+
+.emoji-mart-search {
+  padding: 10px;
+  padding-inline-end: 45px;
+  background: $simple-background-color;
+  position: relative;
+
+  input {
+    font-size: 16px;
+    font-weight: 400;
+    padding: 7px 9px;
+    padding-inline-end: 25px;
+    font-family: inherit;
+    display: block;
+    width: 100%;
+    background: rgba($ui-secondary-color, 0.3);
+    color: $inverted-text-color;
+    border: 1px solid $ui-secondary-color;
+    border-radius: 4px;
+
+    &::-moz-focus-inner {
+      border: 0;
+    }
+
+    &::-moz-focus-inner,
+    &:focus,
+    &:active {
+      outline: 0 !important;
+    }
+
+    &::-webkit-search-cancel-button {
+      display: none;
+    }
+  }
+}
+
+.emoji-mart-search-icon {
+  position: absolute;
+  top: 18px;
+  inset-inline-end: 45px + 5px;
+  z-index: 2;
+  padding: 2px 5px 1px;
+  border: 0;
+  background: none;
+  transition: all 100ms linear;
+  transition-property: opacity;
+  pointer-events: auto;
+  opacity: 0.7;
+
+  &:disabled {
+    cursor: default;
+    pointer-events: none;
+    opacity: 0.3;
+  }
+
+  svg {
+    fill: $action-button-color;
+  }
+}
+
+.emoji-mart-category .emoji-mart-emoji {
+  cursor: pointer;
+
+  span {
+    z-index: 1;
+    position: relative;
+    text-align: center;
+  }
+
+  &:hover::before {
+    z-index: 0;
+    content: '';
+    position: absolute;
+    top: 0;
+    inset-inline-start: 0;
+    width: 100%;
+    height: 100%;
+    background-color: rgba($ui-secondary-color, 0.7);
+    border-radius: 100%;
+  }
+}
+
+.emoji-mart-category-label {
+  z-index: 2;
+  position: relative;
+  position: -webkit-sticky;
+  position: sticky;
+  top: 0;
+
+  span {
+    display: block;
+    width: 100%;
+    font-weight: 500;
+    padding: 5px 6px;
+    background: $simple-background-color;
+  }
+}
+
+/* For screenreaders only, via https://stackoverflow.com/a/19758620 */
+.emoji-mart-sr-only {
+  position: absolute;
+  width: 1px;
+  height: 1px;
+  padding: 0;
+  margin: -1px;
+  overflow: hidden;
+  clip: rect(0, 0, 0, 0);
+  border: 0;
+}
+
+.emoji-mart-category-list {
+  margin: 0;
+  padding: 0;
+}
+
+.emoji-mart-category-list li {
+  list-style: none;
+  margin: 0;
+  padding: 0;
+  display: inline-block;
+}
+
+.emoji-mart-emoji {
+  position: relative;
+  display: inline-block;
+  background: transparent;
+  border: 0;
+  padding: 0;
+  font-size: 0;
+
+  span {
+    width: 22px;
+    height: 22px;
+  }
+}
+
+.emoji-mart-no-results {
+  font-size: 14px;
+  color: $light-text-color;
+  text-align: center;
+  padding: 5px 6px;
+  padding-top: 70px;
+
+  .emoji-mart-no-results-label {
+    margin-top: 0.2em;
+  }
+
+  .emoji-mart-emoji:hover::before {
+    cursor: default;
+    content: none;
+  }
+}
+
+.emoji-mart-preview {
+  display: none;
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/explore.scss b/app/javascript/flavours/blobfox/styles/components/explore.scss
new file mode 100644
index 00000000000000..79da9f21668365
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/explore.scss
@@ -0,0 +1,147 @@
+.account-card__header {
+  position: relative;
+}
+
+.explore__search-header {
+  background: darken($ui-base-color, 4%);
+  justify-content: center;
+  align-items: center;
+  padding: 15px;
+
+  .search {
+    width: 100%;
+    margin-bottom: 0;
+  }
+
+  .search__input {
+    border: 1px solid lighten($ui-base-color, 8%);
+    padding: 10px;
+  }
+
+  .search__popout {
+    border: 1px solid lighten($ui-base-color, 8%);
+  }
+
+  .search .fa {
+    top: 10px;
+    inset-inline-end: 10px;
+    color: $dark-text-color;
+  }
+
+  .search .fa-times-circle {
+    top: 12px;
+  }
+}
+
+.explore__search-results {
+  flex: 1 1 auto;
+  display: flex;
+  flex-direction: column;
+  background: $ui-base-color;
+  border-bottom-left-radius: 4px;
+  border-bottom-right-radius: 4px;
+}
+
+.story {
+  display: flex;
+  align-items: center;
+  color: $primary-text-color;
+  text-decoration: none;
+  padding: 15px;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  gap: 15px;
+
+  &:last-child {
+    border-bottom: 0;
+  }
+
+  &:hover,
+  &:active,
+  &:focus {
+    color: $highlight-text-color;
+
+    .story__details__publisher,
+    .story__details__shared {
+      color: $highlight-text-color;
+    }
+  }
+
+  &__details {
+    flex: 1 1 auto;
+
+    &__publisher {
+      color: $darker-text-color;
+      margin-bottom: 8px;
+    }
+
+    &__title {
+      font-size: 19px;
+      line-height: 24px;
+      font-weight: 500;
+      margin-bottom: 8px;
+    }
+
+    &__shared {
+      color: $darker-text-color;
+    }
+
+    strong {
+      font-weight: 500;
+    }
+  }
+
+  &__thumbnail {
+    flex: 0 0 auto;
+    position: relative;
+    width: 120px;
+    height: 120px;
+
+    .skeleton {
+      width: 100%;
+      height: 100%;
+    }
+
+    img {
+      border-radius: 8px;
+      display: block;
+      margin: 0;
+      width: 100%;
+      height: 100%;
+      object-fit: cover;
+    }
+
+    &__preview {
+      border-radius: 8px;
+      display: block;
+      margin: 0;
+      width: 100%;
+      height: 100%;
+      object-fit: fill;
+      position: absolute;
+      top: 0;
+      inset-inline-start: 0;
+      z-index: 0;
+
+      &--hidden {
+        display: none;
+      }
+    }
+  }
+
+  &.expanded {
+    flex-direction: column;
+
+    .story__thumbnail {
+      order: 1;
+      width: 100%;
+      height: auto;
+      aspect-ratio: 1.91 / 1;
+    }
+
+    .story__details {
+      order: 2;
+      width: 100%;
+      flex: 0 0 auto;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/index.scss b/app/javascript/flavours/blobfox/styles/components/index.scss
new file mode 100644
index 00000000000000..d94f1236483ff8
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/index.scss
@@ -0,0 +1,25 @@
+@import 'misc';
+@import 'boost';
+@import 'accounts';
+@import 'domains';
+@import 'status';
+@import 'modal';
+@import 'compose_form';
+@import 'columns';
+@import 'regeneration_indicator';
+@import 'directory';
+@import 'search';
+@import 'emoji';
+@import 'doodle';
+@import 'drawer';
+@import 'media';
+@import 'sensitive';
+@import 'lists';
+@import 'emoji_picker';
+@import 'local_settings';
+@import 'single_column';
+@import 'announcements';
+@import 'explore';
+@import 'signed_out';
+@import 'privacy_policy';
+@import 'about';
diff --git a/app/javascript/flavours/blobfox/styles/components/lists.scss b/app/javascript/flavours/blobfox/styles/components/lists.scss
new file mode 100644
index 00000000000000..e173016b6784d7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/lists.scss
@@ -0,0 +1,94 @@
+.list-editor {
+  background: $ui-base-color;
+  flex-direction: column;
+  border-radius: 8px;
+  box-shadow: 2px 4px 15px rgba($base-shadow-color, 0.4);
+  width: 380px;
+  overflow: hidden;
+
+  @media screen and (width <= 420px) {
+    width: 90%;
+  }
+
+  h4 {
+    padding: 15px 0;
+    background: lighten($ui-base-color, 13%);
+    font-weight: 500;
+    font-size: 16px;
+    text-align: center;
+    border-radius: 8px 8px 0 0;
+  }
+
+  .drawer__pager {
+    height: 50vh;
+  }
+
+  .drawer__inner {
+    border-radius: 0 0 8px 8px;
+
+    &.backdrop {
+      width: calc(100% - 60px);
+      box-shadow: 2px 4px 15px rgba($base-shadow-color, 0.4);
+      border-radius: 0 0 0 8px;
+    }
+  }
+
+  &__accounts {
+    overflow-y: auto;
+  }
+
+  .account__display-name {
+    &:hover strong {
+      text-decoration: none;
+    }
+  }
+
+  .account__avatar {
+    cursor: default;
+  }
+
+  .search {
+    margin-bottom: 0;
+  }
+}
+
+.list-adder {
+  background: $ui-base-color;
+  flex-direction: column;
+  border-radius: 8px;
+  box-shadow: 2px 4px 15px rgba($base-shadow-color, 0.4);
+  width: 380px;
+  overflow: hidden;
+
+  @media screen and (width <= 420px) {
+    width: 90%;
+  }
+
+  &__account {
+    background: lighten($ui-base-color, 13%);
+  }
+
+  &__lists {
+    background: lighten($ui-base-color, 13%);
+    height: 50vh;
+    border-radius: 0 0 8px 8px;
+    overflow-y: auto;
+  }
+
+  .list {
+    padding: 10px;
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+  }
+
+  .list__wrapper {
+    display: flex;
+  }
+
+  .list__display-name {
+    flex: 1 1 auto;
+    overflow: hidden;
+    text-decoration: none;
+    font-size: 16px;
+    padding: 10px;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/local_settings.scss b/app/javascript/flavours/blobfox/styles/components/local_settings.scss
new file mode 100644
index 00000000000000..7fa1529cbaa760
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/local_settings.scss
@@ -0,0 +1,173 @@
+.blobfox.local-settings {
+  position: relative;
+  display: flex;
+  flex-direction: row;
+  background: $ui-secondary-color;
+  color: $inverted-text-color;
+  border-radius: 8px;
+  height: 80vh;
+  width: 80vw;
+  max-width: 740px;
+  max-height: 450px;
+  overflow: hidden;
+
+  label,
+  legend {
+    display: block;
+    font-size: 14px;
+  }
+
+  .boolean label,
+  .radio_buttons label {
+    position: relative;
+    padding-inline-start: 28px;
+    padding-top: 3px;
+
+    input {
+      position: absolute;
+      inset-inline-start: 0;
+      top: 0;
+    }
+  }
+
+  span.hint {
+    display: block;
+    color: $lighter-text-color;
+  }
+
+  h1 {
+    font-size: 18px;
+    font-weight: 500;
+    line-height: 24px;
+    margin-bottom: 20px;
+  }
+
+  h2 {
+    font-size: 15px;
+    font-weight: 500;
+    line-height: 20px;
+    margin-top: 20px;
+    margin-bottom: 10px;
+  }
+}
+
+.blobfox.local-settings__navigation__item {
+  display: block;
+  padding: 15px 20px;
+  color: inherit;
+  background: lighten($ui-secondary-color, 8%);
+  border: 0;
+  border-bottom: 1px $ui-secondary-color solid;
+  cursor: pointer;
+  text-decoration: none;
+  outline: none;
+  transition: background 0.3s;
+  box-sizing: border-box;
+  width: 100%;
+  text-align: start;
+  font-size: inherit;
+
+  .text-icon-button {
+    color: inherit;
+    transition: unset;
+    unicode-bidi: embed;
+  }
+
+  &:hover {
+    background: $ui-secondary-color;
+  }
+
+  &.active {
+    background: $ui-highlight-color;
+    color: $primary-text-color;
+  }
+
+  &.close,
+  &.close:hover {
+    background: $error-value-color;
+    color: $primary-text-color;
+  }
+}
+
+.blobfox.local-settings__navigation {
+  background: lighten($ui-secondary-color, 8%);
+  width: 212px;
+  font-size: 15px;
+  line-height: 20px;
+  overflow-y: auto;
+}
+
+.blobfox.local-settings__page {
+  display: block;
+  flex: auto;
+  padding: 15px 20px;
+  width: 360px;
+  overflow-y: auto;
+}
+
+.blobfox.local-settings__page__item {
+  margin-bottom: 2px;
+
+  .hint a {
+    color: $lighter-text-color;
+    font-weight: 500;
+    text-decoration: underline;
+
+    &:active,
+    &:focus,
+    &:hover {
+      text-decoration: none;
+    }
+  }
+
+  #mastodon-settings--collapsed-auto-height {
+    width: calc(4ch + 20px);
+  }
+}
+
+.blobfox.local-settings__page__item.string,
+.blobfox.local-settings__page__item.radio_buttons {
+  margin-top: 10px;
+  margin-bottom: 10px;
+}
+
+@media screen and (width <= 630px) {
+  .blobfox.local-settings__navigation {
+    width: 40px;
+    flex-shrink: 0;
+  }
+
+  .blobfox.local-settings__navigation__item {
+    padding: 10px;
+
+    span:last-of-type {
+      display: none;
+    }
+  }
+}
+
+.deprecated-settings-label {
+  white-space: nowrap;
+}
+
+.deprecated-settings-info {
+  text-align: start;
+
+  ul {
+    padding: 10px;
+    margin-inline-start: 12px;
+    list-style: disc inside;
+  }
+
+  a {
+    color: $lighter-text-color;
+    font-weight: 500;
+    text-decoration: underline;
+
+    &:active,
+    &:focus,
+    &:hover {
+      text-decoration: none;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/media.scss b/app/javascript/flavours/blobfox/styles/components/media.scss
new file mode 100644
index 00000000000000..535af9d0f25ce1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/media.scss
@@ -0,0 +1,795 @@
+.video-error-cover {
+  align-items: center;
+  background: $base-overlay-background;
+  color: $primary-text-color;
+  cursor: pointer;
+  display: flex;
+  flex-direction: column;
+  height: 100%;
+  justify-content: center;
+  margin-top: 8px;
+  position: relative;
+  text-align: center;
+  z-index: 100;
+}
+
+.media-spoiler {
+  background: $base-overlay-background;
+  color: $darker-text-color;
+  border: 0;
+  width: 100%;
+  height: 100%;
+
+  &:hover,
+  &:active,
+  &:focus {
+    color: lighten($darker-text-color, 8%);
+  }
+
+  .status__content > & {
+    margin-top: 15px; // Add margin when used bare for NSFW video player
+  }
+  @include fullwidth-gallery;
+}
+
+.media-spoiler__warning {
+  display: block;
+  font-size: 14px;
+}
+
+.media-spoiler__trigger {
+  display: block;
+  font-size: 11px;
+  font-weight: 500;
+}
+
+.media-gallery__item__badges {
+  position: absolute;
+  bottom: 6px;
+  inset-inline-start: 6px;
+  display: flex;
+  gap: 2px;
+}
+
+.media-gallery__gifv__label {
+  display: block;
+  color: $white;
+  background: rgba($black, 0.65);
+  padding: 2px 6px;
+  border-radius: 4px;
+  font-size: 11px;
+  font-weight: 700;
+  z-index: 1;
+  pointer-events: none;
+  line-height: 18px;
+}
+
+.media-gallery {
+  box-sizing: border-box;
+  margin-top: 8px;
+  overflow: hidden;
+  border-radius: 4px;
+  position: relative;
+  width: 100%;
+  min-height: 64px;
+  display: grid;
+  grid-template-columns: 50% 50%;
+  grid-template-rows: 50% 50%;
+  gap: 2px;
+
+  @include fullwidth-gallery;
+}
+
+.media-gallery__item {
+  border: 0;
+  box-sizing: border-box;
+  display: block;
+  position: relative;
+  border-radius: 4px;
+  overflow: hidden;
+
+  &--tall {
+    grid-row: span 2;
+  }
+
+  &--wide {
+    grid-column: span 2;
+  }
+
+  .full-width & {
+    border-radius: 0;
+  }
+
+  &.letterbox {
+    background: $base-shadow-color;
+  }
+}
+
+.media-gallery__item-thumbnail {
+  cursor: zoom-in;
+  display: block;
+  text-decoration: none;
+  color: $secondary-text-color;
+  position: relative;
+  z-index: 1;
+
+  &,
+  img {
+    height: 100%;
+    width: 100%;
+    object-fit: contain;
+
+    &:not(.letterbox) {
+      height: 100%;
+      object-fit: cover;
+    }
+  }
+}
+
+.media-gallery__preview {
+  width: 100%;
+  height: 100%;
+  object-fit: cover;
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  z-index: 0;
+  background: $base-overlay-background;
+
+  &--hidden {
+    display: none;
+  }
+}
+
+.media-gallery__gifv {
+  height: 100%;
+  overflow: hidden;
+  position: relative;
+  width: 100%;
+  display: flex;
+  justify-content: center;
+}
+
+.media-gallery__item-gifv-thumbnail {
+  cursor: zoom-in;
+  height: 100%;
+  width: 100%;
+  object-fit: contain;
+  user-select: none;
+
+  &:not(.letterbox) {
+    height: 100%;
+    object-fit: cover;
+  }
+}
+
+.media-gallery__item-thumbnail-label {
+  clip: rect(1px 1px 1px 1px); /* IE6, IE7 */
+  clip: rect(1px, 1px, 1px, 1px);
+  overflow: hidden;
+  position: absolute;
+}
+
+.video-modal__container {
+  max-width: 100vw;
+  max-height: 100vh;
+}
+
+.audio-modal__container {
+  width: 50vw;
+}
+
+.media-modal {
+  width: 100%;
+  height: 100%;
+  position: relative;
+
+  &__close,
+  &__zoom-button {
+    color: rgba($white, 0.7);
+
+    &:hover,
+    &:focus,
+    &:active {
+      color: $white;
+      background-color: rgba($white, 0.15);
+    }
+
+    &:focus {
+      background-color: rgba($white, 0.3);
+    }
+  }
+}
+
+.media-modal__closer {
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  inset-inline-end: 0;
+  bottom: 0;
+}
+
+.media-modal__navigation {
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  inset-inline-end: 0;
+  bottom: 0;
+  pointer-events: none;
+  transition: opacity 0.3s linear;
+  will-change: opacity;
+
+  * {
+    pointer-events: auto;
+  }
+
+  &.media-modal__navigation--hidden {
+    opacity: 0;
+
+    * {
+      pointer-events: none;
+    }
+  }
+}
+
+.media-modal__nav {
+  background: transparent;
+  box-sizing: border-box;
+  border: 0;
+  color: rgba($white, 0.7);
+  cursor: pointer;
+  display: flex;
+  align-items: center;
+  font-size: 24px;
+  height: 20vmax;
+  margin: auto 0;
+  padding: 30px 15px;
+  position: absolute;
+  top: 0;
+  bottom: 0;
+
+  &:hover,
+  &:focus,
+  &:active {
+    color: $white;
+  }
+}
+
+.media-modal__nav--left {
+  inset-inline-start: 0;
+}
+
+.media-modal__nav--right {
+  inset-inline-end: 0;
+}
+
+.media-modal__overlay {
+  max-width: 600px;
+  position: absolute;
+  inset-inline-start: 0;
+  inset-inline-end: 0;
+  bottom: 0;
+  margin: 0 auto;
+
+  .picture-in-picture__footer {
+    border-radius: 0;
+    background: transparent;
+    padding: 20px 0;
+
+    .icon-button {
+      color: $white;
+
+      &:hover,
+      &:focus,
+      &:active {
+        color: $white;
+        background-color: rgba($white, 0.15);
+      }
+
+      &:focus {
+        background-color: rgba($white, 0.3);
+      }
+
+      &.active {
+        color: $highlight-text-color;
+
+        &:hover,
+        &:focus,
+        &:active {
+          background: rgba($highlight-text-color, 0.15);
+        }
+
+        &:focus {
+          background: rgba($highlight-text-color, 0.3);
+        }
+      }
+
+      &.star-icon.active {
+        color: $gold-star;
+
+        &:hover,
+        &:focus,
+        &:active {
+          background: rgba($gold-star, 0.15);
+        }
+
+        &:focus {
+          background: rgba($gold-star, 0.3);
+        }
+      }
+
+      &.disabled {
+        color: $white;
+        background-color: transparent;
+        cursor: default;
+        opacity: 0.4;
+      }
+    }
+  }
+}
+
+.media-modal__pagination {
+  display: flex;
+  justify-content: center;
+  margin-bottom: 20px;
+}
+
+.media-modal__page-dot {
+  flex: 0 0 auto;
+  background-color: $white;
+  opacity: 0.4;
+  height: 6px;
+  width: 6px;
+  border-radius: 50%;
+  margin: 0 4px;
+  padding: 0;
+  border: 0;
+  font-size: 0;
+  transition: opacity 0.2s ease-in-out;
+
+  &.active {
+    opacity: 1;
+  }
+}
+
+.media-modal__close {
+  position: absolute;
+  inset-inline-end: 8px;
+  top: 8px;
+  z-index: 100;
+}
+
+.detailed,
+.fullscreen {
+  .video-player__volume__current,
+  .video-player__volume::before {
+    bottom: 27px;
+  }
+
+  .video-player__volume__handle {
+    bottom: 23px;
+  }
+}
+
+.audio-player {
+  overflow: hidden;
+  box-sizing: border-box;
+  position: relative;
+  background: darken($ui-base-color, 8%);
+  border-radius: 4px;
+  padding-bottom: 44px;
+  width: 100%;
+
+  &.editable {
+    border-radius: 0;
+    height: 100%;
+  }
+
+  &.inactive {
+    audio,
+    .video-player__controls {
+      visibility: hidden;
+    }
+  }
+
+  .video-player__volume::before,
+  .video-player__seek::before {
+    background: currentColor;
+    opacity: 0.15;
+  }
+
+  .video-player__seek__buffer {
+    background: currentColor;
+    opacity: 0.2;
+  }
+
+  .video-player__buttons button,
+  .video-player__buttons a {
+    color: currentColor;
+    opacity: 0.75;
+
+    &:active,
+    &:hover,
+    &:focus {
+      color: currentColor;
+      opacity: 1;
+    }
+  }
+
+  .video-player__time-sep,
+  .video-player__time-total,
+  .video-player__time-current {
+    color: currentColor;
+  }
+
+  .video-player__seek::before,
+  .video-player__seek__buffer,
+  .video-player__seek__progress {
+    top: 0;
+  }
+
+  .video-player__seek__handle {
+    top: -4px;
+  }
+
+  .video-player__controls {
+    padding-top: 10px;
+    background: transparent;
+  }
+}
+
+.video-player {
+  overflow: hidden;
+  position: relative;
+  background: $base-shadow-color;
+  max-width: 100%;
+  border-radius: 4px;
+  box-sizing: border-box;
+  color: $white;
+  display: flex;
+  align-items: center;
+
+  &.editable {
+    border-radius: 0;
+    height: 100% !important;
+  }
+
+  &:focus {
+    outline: 0;
+  }
+
+  .detailed-status & {
+    width: 100%;
+    height: 100%;
+  }
+
+  @include fullwidth-gallery;
+
+  video {
+    display: block;
+    max-width: 100vw;
+    max-height: 80vh;
+    z-index: 1;
+    position: relative;
+  }
+
+  &.fullscreen {
+    width: 100% !important;
+    height: 100% !important;
+    margin: 0;
+
+    video {
+      max-width: 100% !important;
+      max-height: 100% !important;
+      width: 100% !important;
+      height: 100% !important;
+      outline: 0;
+    }
+  }
+
+  &.inline {
+    video {
+      object-fit: contain;
+    }
+  }
+
+  &__controls {
+    position: absolute;
+    direction: ltr;
+    z-index: 2;
+    bottom: 0;
+    inset-inline-start: 0;
+    inset-inline-end: 0;
+    box-sizing: border-box;
+    background: linear-gradient(
+      0deg,
+      rgba($base-shadow-color, 0.85) 0,
+      rgba($base-shadow-color, 0.45) 60%,
+      transparent
+    );
+    padding: 0 15px;
+    opacity: 0;
+    transition: opacity 0.1s ease;
+
+    &.active {
+      opacity: 1;
+    }
+  }
+
+  &.inactive {
+    video,
+    .video-player__controls {
+      visibility: hidden;
+    }
+  }
+
+  &__spoiler {
+    display: none;
+    position: absolute;
+    top: 0;
+    inset-inline-start: 0;
+    width: 100%;
+    height: 100%;
+    z-index: 4;
+    border: 0;
+    background: $base-shadow-color;
+    color: $darker-text-color;
+    transition: none;
+    pointer-events: none;
+
+    &.active {
+      display: block;
+      pointer-events: auto;
+
+      &:hover,
+      &:active,
+      &:focus {
+        color: lighten($darker-text-color, 7%);
+      }
+    }
+
+    &__title {
+      display: block;
+      font-size: 14px;
+    }
+
+    &__subtitle {
+      display: block;
+      font-size: 11px;
+      font-weight: 500;
+    }
+  }
+
+  &__buttons-bar {
+    display: flex;
+    justify-content: space-between;
+    padding-bottom: 8px;
+    margin: 0 -5px;
+
+    .video-player__download__icon {
+      color: inherit;
+
+      .fa,
+      &:active .fa,
+      &:hover .fa,
+      &:focus .fa {
+        color: inherit;
+      }
+    }
+  }
+
+  &__buttons {
+    display: flex;
+    flex: 0 1 auto;
+    min-width: 30px;
+    align-items: center;
+    font-size: 16px;
+    white-space: nowrap;
+    overflow: hidden;
+    text-overflow: ellipsis;
+
+    .player-button {
+      display: inline-block;
+      outline: 0;
+      flex: 0 0 auto;
+      background: transparent;
+      padding: 5px;
+      font-size: 16px;
+      border: 0;
+      color: rgba($white, 0.75);
+
+      &:active,
+      &:hover,
+      &:focus {
+        color: $white;
+      }
+    }
+  }
+
+  &__time {
+    display: inline;
+    flex: 0 1 auto;
+    overflow: hidden;
+    text-overflow: ellipsis;
+    margin: 0 5px;
+  }
+
+  &__time-sep,
+  &__time-total,
+  &__time-current {
+    font-size: 14px;
+    font-weight: 500;
+  }
+
+  &__time-current {
+    color: $white;
+  }
+
+  &__time-sep {
+    display: inline-block;
+    margin: 0 6px;
+  }
+
+  &__time-sep,
+  &__time-total {
+    color: $white;
+  }
+
+  &__volume {
+    flex: 0 0 auto;
+    display: inline-flex;
+    cursor: pointer;
+    height: 24px;
+    position: relative;
+    overflow: hidden;
+
+    .no-reduce-motion & {
+      transition: all 100ms linear;
+    }
+
+    &.active {
+      overflow: visible;
+      width: 50px;
+      margin-inline-end: 16px;
+    }
+
+    &::before {
+      content: '';
+      width: 50px;
+      background: rgba($white, 0.35);
+      border-radius: 4px;
+      display: block;
+      position: absolute;
+      height: 4px;
+      inset-inline-start: 0;
+      top: 50%;
+      transform: translate(0, -50%);
+    }
+
+    &__current {
+      display: block;
+      position: absolute;
+      height: 4px;
+      border-radius: 4px;
+      inset-inline-start: 0;
+      top: 50%;
+      transform: translate(0, -50%);
+      background: lighten($ui-highlight-color, 8%);
+    }
+
+    &__handle {
+      position: absolute;
+      z-index: 3;
+      border-radius: 50%;
+      width: 12px;
+      height: 12px;
+      top: 50%;
+      inset-inline-start: 0;
+      margin-inline-start: -6px;
+      transform: translate(0, -50%);
+      background: lighten($ui-highlight-color, 8%);
+      box-shadow: 1px 2px 6px rgba($base-shadow-color, 0.2);
+      opacity: 0;
+
+      .no-reduce-motion & {
+        transition: opacity 100ms linear;
+      }
+    }
+
+    &.active &__handle {
+      opacity: 1;
+    }
+  }
+
+  &__link {
+    padding: 2px 10px;
+
+    a {
+      text-decoration: none;
+      font-size: 14px;
+      font-weight: 500;
+      color: $white;
+
+      &:hover,
+      &:active,
+      &:focus {
+        text-decoration: underline;
+      }
+    }
+  }
+
+  &__seek {
+    cursor: pointer;
+    height: 24px;
+    position: relative;
+
+    &::before {
+      content: '';
+      width: 100%;
+      background: rgba($white, 0.35);
+      border-radius: 4px;
+      display: block;
+      position: absolute;
+      height: 4px;
+      top: 14px;
+    }
+
+    &__progress,
+    &__buffer {
+      display: block;
+      position: absolute;
+      height: 4px;
+      border-radius: 4px;
+      top: 14px;
+      background: lighten($ui-highlight-color, 8%);
+    }
+
+    &__buffer {
+      background: rgba($white, 0.2);
+    }
+
+    &__handle {
+      position: absolute;
+      z-index: 3;
+      opacity: 0;
+      border-radius: 50%;
+      width: 12px;
+      height: 12px;
+      top: 10px;
+      margin-inline-start: -6px;
+      background: lighten($ui-highlight-color, 8%);
+      box-shadow: 1px 2px 6px rgba($base-shadow-color, 0.2);
+
+      .no-reduce-motion & {
+        transition: opacity 0.1s ease;
+      }
+
+      &.active {
+        opacity: 1;
+      }
+    }
+
+    &:hover {
+      .video-player__seek__handle {
+        opacity: 1;
+      }
+    }
+  }
+
+  &.detailed,
+  &.fullscreen {
+    .video-player__buttons {
+      .player-button {
+        padding-top: 10px;
+        padding-bottom: 10px;
+      }
+    }
+  }
+}
+
+.gifv {
+  video {
+    max-width: 100vw;
+    max-height: 80vh;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/misc.scss b/app/javascript/flavours/blobfox/styles/components/misc.scss
new file mode 100644
index 00000000000000..f6d0f5b0731907
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/misc.scss
@@ -0,0 +1,1740 @@
+.app-body {
+  -webkit-overflow-scrolling: touch;
+  -ms-overflow-style: -ms-autohiding-scrollbar;
+}
+
+.animated-number {
+  display: inline-flex;
+  flex-direction: column;
+  align-items: stretch;
+  overflow: hidden;
+  position: relative;
+}
+
+.inline-alert {
+  color: $valid-value-color;
+  font-weight: 400;
+
+  .no-reduce-motion & {
+    transition: opacity 200ms ease;
+  }
+}
+
+.link-button {
+  display: block;
+  font-size: 15px;
+  line-height: 20px;
+  color: $highlight-text-color;
+  border: 0;
+  background: transparent;
+  padding: 0;
+  cursor: pointer;
+  text-decoration: none;
+
+  &--destructive {
+    color: $error-value-color;
+  }
+
+  &:hover,
+  &:active {
+    text-decoration: underline;
+  }
+
+  &:disabled {
+    color: $ui-primary-color;
+    cursor: default;
+  }
+}
+
+.button {
+  background-color: $ui-button-background-color;
+  border: 10px none;
+  border-radius: 4px;
+  box-sizing: border-box;
+  color: $ui-button-color;
+  cursor: pointer;
+  display: inline-block;
+  font-family: inherit;
+  font-size: 15px;
+  font-weight: 500;
+  letter-spacing: 0;
+  line-height: 22px;
+  overflow: hidden;
+  padding: 7px 18px;
+  position: relative;
+  text-align: center;
+  text-decoration: none;
+  text-overflow: ellipsis;
+  white-space: nowrap;
+  width: auto;
+
+  &:active,
+  &:focus,
+  &:hover {
+    background-color: $ui-button-focus-background-color;
+  }
+
+  &--destructive {
+    &:active,
+    &:focus,
+    &:hover {
+      background-color: $ui-button-destructive-focus-background-color;
+      transition: none;
+    }
+  }
+
+  &:disabled {
+    background-color: $ui-primary-color;
+    cursor: default;
+  }
+
+  &.button-secondary {
+    color: $ui-button-secondary-color;
+    background: transparent;
+    padding: 6px 17px;
+    border: 1px solid $ui-button-secondary-border-color;
+
+    &:active,
+    &:focus,
+    &:hover {
+      border-color: $ui-button-secondary-focus-background-color;
+      color: $ui-button-secondary-focus-color;
+      background-color: $ui-button-secondary-focus-background-color;
+      text-decoration: none;
+    }
+
+    &:disabled {
+      opacity: 0.5;
+    }
+  }
+
+  &.button-tertiary {
+    background: transparent;
+    padding: 6px 17px;
+    color: $ui-button-tertiary-color;
+    border: 1px solid $ui-button-tertiary-border-color;
+
+    &:active,
+    &:focus,
+    &:hover {
+      background-color: $ui-button-tertiary-focus-background-color;
+      color: $ui-button-tertiary-focus-color;
+      border: 0;
+      padding: 7px 18px;
+    }
+
+    &:disabled {
+      opacity: 0.5;
+    }
+
+    &.button--confirmation {
+      color: $valid-value-color;
+      border-color: $valid-value-color;
+
+      &:active,
+      &:focus,
+      &:hover {
+        background: $valid-value-color;
+        color: $primary-text-color;
+      }
+    }
+
+    &.button--destructive {
+      color: $error-value-color;
+      border-color: $error-value-color;
+
+      &:active,
+      &:focus,
+      &:hover {
+        background: $error-value-color;
+        color: $primary-text-color;
+      }
+    }
+  }
+
+  &.button--block {
+    display: block;
+    width: 100%;
+  }
+}
+
+.icon-button {
+  display: inline-block;
+  padding: 0;
+  color: $action-button-color;
+  border: 0;
+  border-radius: 4px;
+  background: transparent;
+  cursor: pointer;
+  transition: all 100ms ease-in;
+  transition-property: background-color, color;
+  text-decoration: none;
+
+  a {
+    color: inherit;
+    text-decoration: none;
+  }
+
+  &:hover,
+  &:active,
+  &:focus {
+    color: lighten($action-button-color, 7%);
+    background-color: rgba($action-button-color, 0.15);
+    transition: all 200ms ease-out;
+    transition-property: background-color, color;
+  }
+
+  &:focus {
+    background-color: rgba($action-button-color, 0.3);
+  }
+
+  &.disabled {
+    color: darken($action-button-color, 13%);
+    background-color: transparent;
+    cursor: default;
+  }
+
+  &.active {
+    color: $highlight-text-color;
+  }
+
+  &.copyable {
+    transition: background 300ms linear;
+  }
+
+  &.copied {
+    background: $valid-value-color;
+    transition: none;
+  }
+
+  &::-moz-focus-inner {
+    border: 0;
+  }
+
+  &::-moz-focus-inner,
+  &:focus,
+  &:active {
+    outline: 0 !important;
+  }
+
+  &.inverted {
+    color: $lighter-text-color;
+
+    &:hover,
+    &:active,
+    &:focus {
+      color: darken($lighter-text-color, 7%);
+      background-color: rgba($lighter-text-color, 0.15);
+    }
+
+    &:focus {
+      background-color: rgba($lighter-text-color, 0.3);
+    }
+
+    &.disabled {
+      color: lighten($lighter-text-color, 7%);
+      background-color: transparent;
+    }
+
+    &.active {
+      color: $highlight-text-color;
+
+      &.disabled {
+        color: lighten($highlight-text-color, 13%);
+      }
+    }
+  }
+
+  &.overlayed {
+    box-sizing: content-box;
+    background: rgba($base-overlay-background, 0.6);
+    color: rgba($primary-text-color, 0.7);
+    border-radius: 4px;
+    padding: 2px;
+
+    &:hover {
+      background: rgba($base-overlay-background, 0.9);
+    }
+  }
+
+  &--with-counter {
+    display: inline-flex;
+    align-items: center;
+    width: auto !important;
+    padding: 0;
+    padding-inline-end: 4px;
+    padding-inline-start: 2px;
+  }
+
+  &__counter {
+    display: inline-block;
+    width: auto;
+    margin-inline-start: 4px;
+    font-size: 12px;
+    font-weight: 500;
+  }
+}
+
+.text-icon,
+.text-icon-button {
+  font-weight: 600;
+  font-size: 11px;
+  line-height: 27px;
+  cursor: default;
+}
+
+.text-icon-button {
+  color: $lighter-text-color;
+  border: 0;
+  border-radius: 4px;
+  background: transparent;
+  cursor: pointer;
+  padding: 0 3px;
+  white-space: nowrap;
+  outline: 0;
+  transition: all 100ms ease-in;
+  transition-property: background-color, color;
+
+  &:hover,
+  &:active,
+  &:focus {
+    color: darken($lighter-text-color, 7%);
+    background-color: rgba($lighter-text-color, 0.15);
+    transition: all 200ms ease-out;
+    transition-property: background-color, color;
+  }
+
+  &:focus {
+    background-color: rgba($lighter-text-color, 0.3);
+  }
+
+  &.disabled {
+    color: lighten($lighter-text-color, 20%);
+    background-color: transparent;
+    cursor: default;
+  }
+
+  &.active {
+    color: $highlight-text-color;
+  }
+
+  &::-moz-focus-inner {
+    border: 0;
+  }
+
+  &::-moz-focus-inner,
+  &:focus,
+  &:active {
+    outline: 0 !important;
+  }
+}
+
+body > [data-popper-placement] {
+  z-index: 3;
+}
+
+.invisible {
+  font-size: 0;
+  line-height: 0;
+  display: inline-block;
+  width: 0;
+  height: 0;
+  position: absolute;
+
+  img,
+  svg {
+    margin: 0 !important;
+    border: 0 !important;
+    padding: 0 !important;
+    width: 0 !important;
+    height: 0 !important;
+  }
+}
+
+.ellipsis {
+  &::after {
+    content: '…';
+  }
+}
+
+.notification__favourite-icon-wrapper {
+  inset-inline-start: 0;
+  position: absolute;
+
+  .fa.star-icon {
+    color: $gold-star;
+  }
+}
+
+.icon-button.star-icon.active {
+  color: $gold-star;
+}
+
+.icon-button.bookmark-icon.active {
+  color: $red-bookmark;
+}
+
+.no-reduce-motion .icon-button.star-icon {
+  &.activate {
+    & > .fa-star {
+      animation: spring-rotate-in 1s linear;
+    }
+  }
+
+  &.deactivate {
+    & > .fa-star {
+      animation: spring-rotate-out 1s linear;
+    }
+  }
+}
+
+.notification__display-name {
+  color: inherit;
+  font-weight: 500;
+  text-decoration: none;
+
+  &:hover {
+    color: $primary-text-color;
+    text-decoration: underline;
+  }
+}
+
+.display-name {
+  display: block;
+  max-width: 100%;
+  overflow: hidden;
+  text-overflow: ellipsis;
+  white-space: nowrap;
+
+  &__account {
+    text-overflow: ellipsis;
+    overflow: hidden;
+  }
+
+  a {
+    color: inherit;
+    text-decoration: inherit;
+  }
+
+  strong {
+    display: block;
+  }
+
+  > a:hover {
+    strong {
+      text-decoration: underline;
+    }
+  }
+
+  &.inline {
+    padding: 0;
+    height: 18px;
+    font-size: 15px;
+    line-height: 18px;
+    text-overflow: ellipsis;
+    white-space: nowrap;
+    overflow: hidden;
+
+    strong {
+      display: inline;
+      height: auto;
+      font-size: inherit;
+      line-height: inherit;
+    }
+
+    span {
+      display: inline;
+      height: auto;
+      font-size: inherit;
+      line-height: inherit;
+    }
+  }
+}
+
+.display-name__html {
+  font-weight: 500;
+}
+
+.display-name__account {
+  font-size: 14px;
+}
+
+.image-loader {
+  position: relative;
+  width: 100%;
+  height: 100%;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  flex-direction: column;
+  scrollbar-width: none; /* Firefox */
+  -ms-overflow-style: none; /* IE 10+ */
+
+  * {
+    scrollbar-width: none; /* Firefox */
+    -ms-overflow-style: none; /* IE 10+ */
+  }
+
+  &::-webkit-scrollbar,
+  *::-webkit-scrollbar {
+    width: 0;
+    height: 0;
+    background: transparent; /* Chrome/Safari/Webkit */
+  }
+
+  .image-loader__preview-canvas {
+    max-width: $media-modal-media-max-width;
+    max-height: $media-modal-media-max-height;
+    background: url('~images/void.png') repeat;
+    object-fit: contain;
+  }
+
+  .loading-bar__container {
+    position: relative;
+  }
+
+  .loading-bar {
+    position: absolute;
+  }
+
+  &.image-loader--amorphous .image-loader__preview-canvas {
+    display: none;
+  }
+}
+
+.zoomable-image {
+  position: relative;
+  width: 100%;
+  height: 100%;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+
+  img {
+    max-width: $media-modal-media-max-width;
+    max-height: $media-modal-media-max-height;
+    width: auto;
+    height: auto;
+    object-fit: contain;
+  }
+}
+
+.dropdown-animation {
+  animation: dropdown 150ms cubic-bezier(0.1, 0.7, 0.1, 1);
+
+  @keyframes dropdown {
+    from {
+      opacity: 0;
+    }
+
+    to {
+      opacity: 1;
+    }
+  }
+
+  .reduce-motion & {
+    animation: none;
+  }
+}
+
+.dropdown {
+  display: inline-block;
+}
+
+.dropdown__content {
+  display: none;
+  position: absolute;
+}
+
+.dropdown-menu__separator {
+  border-bottom: 1px solid var(--dropdown-border-color);
+  margin: 2px 0;
+  height: 0;
+}
+
+.dropdown-menu {
+  background: var(--dropdown-background-color);
+  border: 1px solid var(--dropdown-border-color);
+  padding: 2px;
+  border-radius: 4px;
+  box-shadow: var(--dropdown-shadow);
+  z-index: 9999;
+
+  &__text-button {
+    display: inline;
+    color: inherit;
+    background: transparent;
+    border: 0;
+    margin: 0;
+    padding: 0;
+    font-family: inherit;
+    font-size: inherit;
+    line-height: inherit;
+
+    &:focus {
+      outline: 1px dotted;
+    }
+  }
+
+  &__container {
+    &__header {
+      border-bottom: 1px solid var(--dropdown-border-color);
+      padding: 6px 14px;
+      padding-bottom: 12px;
+      margin-bottom: 4px;
+      font-size: 13px;
+      line-height: 18px;
+      color: $darker-text-color;
+    }
+
+    &__list {
+      list-style: none;
+
+      &--scrollable {
+        max-height: 300px;
+        overflow-y: scroll;
+      }
+    }
+
+    &--loading {
+      display: flex;
+      align-items: center;
+      justify-content: center;
+      padding: 30px 45px;
+    }
+  }
+}
+
+.dropdown-menu__item {
+  font-size: 13px;
+  line-height: 18px;
+  font-weight: 500;
+  display: block;
+
+  &--dangerous {
+    color: $error-value-color;
+  }
+
+  a,
+  button {
+    font: inherit;
+    display: block;
+    width: 100%;
+    padding: 6px 14px;
+    border: 0;
+    margin: 0;
+    background: transparent;
+    box-sizing: border-box;
+    text-decoration: none;
+    color: inherit;
+    overflow: hidden;
+    text-overflow: ellipsis;
+    white-space: nowrap;
+    text-align: inherit;
+    border-radius: 4px;
+
+    &:focus,
+    &:hover,
+    &:active {
+      background: var(--dropdown-border-color);
+      outline: 0;
+    }
+  }
+}
+
+.inline-account {
+  display: inline-flex;
+  align-items: center;
+  vertical-align: top;
+
+  .account__avatar {
+    margin-inline-end: 5px;
+    border-radius: 50%;
+  }
+
+  strong {
+    font-weight: 600;
+  }
+}
+
+.static-content {
+  padding: 10px;
+  padding-top: 20px;
+  color: $dark-text-color;
+
+  h1 {
+    font-size: 16px;
+    font-weight: 500;
+    margin-bottom: 40px;
+    text-align: center;
+  }
+
+  p {
+    font-size: 13px;
+    margin-bottom: 20px;
+  }
+}
+
+.column,
+.drawer {
+  flex: 1 1 100%;
+  overflow: hidden;
+}
+
+@media screen and (width >= 631px) {
+  .columns-area {
+    padding: 0;
+  }
+
+  .column,
+  .drawer {
+    flex: 0 0 auto;
+    padding: 10px;
+    padding-inline-start: 5px;
+    padding-inline-end: 5px;
+
+    &:first-child {
+      padding-inline-start: 10px;
+    }
+
+    &:last-child {
+      padding-inline-end: 10px;
+    }
+  }
+
+  .columns-area > div {
+    .column,
+    .drawer {
+      padding-inline-start: 5px;
+      padding-inline-end: 5px;
+    }
+  }
+}
+
+.tabs-bar {
+  box-sizing: border-box;
+  display: flex;
+  background: lighten($ui-base-color, 8%);
+  flex: 0 0 auto;
+  overflow-y: auto;
+}
+
+.tabs-bar__link {
+  display: block;
+  flex: 1 1 auto;
+  padding: 15px 10px;
+  padding-bottom: 13px;
+  color: $primary-text-color;
+  text-decoration: none;
+  text-align: center;
+  font-size: 14px;
+  font-weight: 500;
+  border-bottom: 2px solid lighten($ui-base-color, 8%);
+  transition: all 50ms linear;
+  transition-property: border-bottom, background, color;
+
+  .fa {
+    font-weight: 400;
+    font-size: 16px;
+  }
+
+  &:hover,
+  &:focus,
+  &:active {
+    @include multi-columns('screen and (min-width: 631px)') {
+      background: lighten($ui-base-color, 14%);
+      border-bottom-color: lighten($ui-base-color, 14%);
+    }
+  }
+
+  &.active {
+    border-bottom: 2px solid $ui-highlight-color;
+    color: $highlight-text-color;
+  }
+
+  span {
+    margin-inline-start: 5px;
+    display: none;
+  }
+
+  span.icon {
+    margin-inline-start: 0;
+    display: inline;
+  }
+}
+
+.icon-with-badge {
+  position: relative;
+
+  &__badge {
+    position: absolute;
+    inset-inline-start: 9px;
+    top: -13px;
+    background: $ui-highlight-color;
+    border: 2px solid lighten($ui-base-color, 8%);
+    padding: 1px 6px;
+    border-radius: 6px;
+    font-size: 10px;
+    font-weight: 500;
+    line-height: 14px;
+    color: $primary-text-color;
+  }
+
+  &__issue-badge {
+    position: absolute;
+    inset-inline-start: 11px;
+    bottom: 1px;
+    display: block;
+    background: $error-red;
+    border-radius: 50%;
+    width: 0.625rem;
+    height: 0.625rem;
+  }
+}
+
+.column-link--transparent .icon-with-badge__badge {
+  border-color: darken($ui-base-color, 8%);
+}
+
+.scrollable {
+  overflow-y: scroll;
+  overflow-x: hidden;
+  flex: 1 1 auto;
+  -webkit-overflow-scrolling: touch;
+
+  &.optionally-scrollable {
+    overflow-y: auto;
+  }
+
+  @supports (display: grid) {
+    // hack to fix Chrome <57
+    contain: strict;
+  }
+
+  &--flex {
+    display: flex;
+    flex-direction: column;
+  }
+
+  &__append {
+    flex: 1 1 auto;
+    position: relative;
+    min-height: 120px;
+  }
+
+  .scrollable {
+    flex: 1 1 auto;
+  }
+}
+
+.scrollable.fullscreen {
+  @supports (display: grid) {
+    // hack to fix Chrome <57
+    contain: none;
+  }
+}
+
+.react-toggle {
+  display: inline-block;
+  position: relative;
+  cursor: pointer;
+  background-color: transparent;
+  border: 0;
+  padding: 0;
+  user-select: none;
+  -webkit-tap-highlight-color: rgba($base-overlay-background, 0);
+  -webkit-tap-highlight-color: transparent;
+}
+
+.react-toggle-screenreader-only {
+  border: 0;
+  clip: rect(0 0 0 0);
+  height: 1px;
+  margin: -1px;
+  overflow: hidden;
+  padding: 0;
+  position: absolute;
+  width: 1px;
+}
+
+.react-toggle--disabled {
+  cursor: not-allowed;
+  opacity: 0.5;
+  transition: opacity 0.25s;
+}
+
+.react-toggle-track {
+  width: 50px;
+  height: 24px;
+  padding: 0;
+  border-radius: 30px;
+  background-color: $ui-base-color;
+  transition: background-color 0.2s ease;
+}
+
+.react-toggle:is(:hover, :focus-within):not(.react-toggle--disabled)
+  .react-toggle-track {
+  background-color: darken($ui-base-color, 10%);
+}
+
+.react-toggle--checked .react-toggle-track {
+  background-color: darken($ui-highlight-color, 2%);
+}
+
+.react-toggle--checked:is(:hover, :focus-within):not(.react-toggle--disabled)
+  .react-toggle-track {
+  background-color: $ui-highlight-color;
+}
+
+.react-toggle-track-check {
+  position: absolute;
+  width: 14px;
+  height: 10px;
+  top: 0;
+  bottom: 0;
+  margin-top: auto;
+  margin-bottom: auto;
+  line-height: 0;
+  inset-inline-start: 8px;
+  opacity: 0;
+  transition: opacity 0.25s ease;
+}
+
+.react-toggle--checked .react-toggle-track-check {
+  opacity: 1;
+  transition: opacity 0.25s ease;
+}
+
+.react-toggle-track-x {
+  position: absolute;
+  width: 10px;
+  height: 10px;
+  top: 0;
+  bottom: 0;
+  margin-top: auto;
+  margin-bottom: auto;
+  line-height: 0;
+  inset-inline-end: 10px;
+  opacity: 1;
+  transition: opacity 0.25s ease;
+}
+
+.react-toggle--checked .react-toggle-track-x {
+  opacity: 0;
+}
+
+.react-toggle-thumb {
+  position: absolute;
+  top: 1px;
+  inset-inline-start: 1px;
+  width: 22px;
+  height: 22px;
+  border: 1px solid $ui-base-color;
+  border-radius: 50%;
+  background-color: darken($simple-background-color, 2%);
+  box-sizing: border-box;
+  transition: all 0.25s ease;
+  transition-property: border-color, left;
+}
+
+.react-toggle--checked .react-toggle-thumb {
+  inset-inline-start: 27px;
+  border-color: $ui-highlight-color;
+}
+
+.getting-started__wrapper,
+.getting_started,
+.flex-spacer {
+  background: $ui-base-color;
+}
+
+.getting-started__wrapper {
+  position: relative;
+  overflow-y: auto;
+}
+
+.flex-spacer {
+  flex: 1 1 auto;
+}
+
+.getting-started {
+  background: $ui-base-color;
+  flex: 1 0 auto;
+
+  p {
+    color: $secondary-text-color;
+  }
+
+  a {
+    color: $dark-text-color;
+  }
+
+  &__trends {
+    flex: 0 1 auto;
+    opacity: 1;
+    animation: fade 150ms linear;
+    margin-top: 10px;
+
+    h4 {
+      border-bottom: 1px solid lighten($ui-base-color, 8%);
+      padding: 10px;
+      font-size: 12px;
+      text-transform: uppercase;
+      font-weight: 500;
+
+      a {
+        color: $darker-text-color;
+        text-decoration: none;
+      }
+    }
+
+    @media screen and (height <= 810px) {
+      .trends__item:nth-of-type(3) {
+        display: none;
+      }
+    }
+
+    @media screen and (height <= 720px) {
+      .trends__item:nth-of-type(2) {
+        display: none;
+      }
+    }
+
+    @media screen and (height <= 670px) {
+      display: none;
+    }
+
+    .trends__item {
+      border-bottom: 0;
+      padding: 10px;
+
+      &__current {
+        color: $darker-text-color;
+      }
+    }
+  }
+}
+
+.column-link__badge {
+  display: inline-block;
+  border-radius: 4px;
+  font-size: 12px;
+  line-height: 19px;
+  font-weight: 500;
+  background: $ui-base-color;
+  padding: 4px 8px;
+  margin: -6px 10px;
+}
+
+.keyboard-shortcuts {
+  padding: 8px 0 0;
+  overflow: hidden;
+
+  thead {
+    position: absolute;
+    inset-inline-start: -9999px;
+  }
+
+  td {
+    padding: 0 10px 8px;
+  }
+
+  kbd {
+    display: inline-block;
+    padding: 3px 5px;
+    background-color: lighten($ui-base-color, 8%);
+    border: 1px solid darken($ui-base-color, 4%);
+  }
+}
+
+.setting-text {
+  color: $darker-text-color;
+  background: transparent;
+  border: 0;
+  border-bottom: 2px solid $ui-primary-color;
+  outline: 0;
+  box-sizing: border-box;
+  display: block;
+  font-family: inherit;
+  margin-bottom: 10px;
+  padding: 7px 0;
+  width: 100%;
+
+  &:focus,
+  &:active {
+    color: $primary-text-color;
+    border-bottom-color: $ui-highlight-color;
+  }
+
+  @include limited-single-column('screen and (max-width: 600px)') {
+    font-size: 16px;
+  }
+
+  &.light {
+    color: $inverted-text-color;
+    border-bottom: 2px solid lighten($ui-base-color, 27%);
+
+    &:focus,
+    &:active {
+      color: $inverted-text-color;
+      border-bottom-color: $ui-highlight-color;
+    }
+  }
+}
+
+button.icon-button i.fa-retweet {
+  background-position: 0 0;
+  height: 19px;
+  transition: background-position 0.9s steps(10);
+  transition-duration: 0s;
+  vertical-align: middle;
+  width: 22px;
+
+  &::before {
+    display: none !important;
+  }
+}
+
+button.icon-button.active i.fa-retweet {
+  transition-duration: 0.9s;
+  background-position: 0 100%;
+}
+
+.reduce-motion button.icon-button i.fa-retweet,
+.reduce-motion button.icon-button.active i.fa-retweet {
+  transition: none;
+}
+
+.reduce-motion button.icon-button.disabled i.fa-retweet {
+  color: darken($action-button-color, 13%);
+}
+
+.load-more {
+  display: block;
+  color: $dark-text-color;
+  background-color: transparent;
+  border: 0;
+  font-size: inherit;
+  text-align: center;
+  line-height: inherit;
+  margin: 0;
+  padding: 15px;
+  box-sizing: border-box;
+  width: 100%;
+  clear: both;
+  text-decoration: none;
+
+  &:hover {
+    background: lighten($ui-base-color, 2%);
+  }
+}
+
+.load-gap {
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+}
+
+.timeline-hint {
+  text-align: center;
+  color: $darker-text-color;
+  padding: 15px;
+  box-sizing: border-box;
+  width: 100%;
+  cursor: default;
+
+  strong {
+    font-weight: 500;
+  }
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: none;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: underline;
+      color: lighten($highlight-text-color, 4%);
+    }
+  }
+}
+
+.missing-indicator {
+  padding-top: 20px + 48px;
+
+  .regeneration-indicator__figure {
+    background-image: url('~flavours/blobfox/images/elephant_ui_disappointed.svg');
+  }
+}
+
+.scrollable > div > :first-child .notification__dismiss-overlay > .wrappy {
+  border-top: 1px solid $ui-base-color;
+}
+
+.notification__dismiss-overlay {
+  overflow: hidden;
+  position: absolute;
+  top: 0;
+  inset-inline-end: 0;
+  bottom: -1px;
+  padding-inline-start: 15px; // space for the box shadow to be visible
+  z-index: 999;
+  align-items: center;
+  justify-content: flex-end;
+  cursor: pointer;
+  display: flex;
+
+  .wrappy {
+    width: $dismiss-overlay-width;
+    align-self: stretch;
+    display: flex;
+    flex-direction: column;
+    align-items: center;
+    justify-content: center;
+    background: lighten($ui-base-color, 8%);
+    border-inline-start: 1px solid lighten($ui-base-color, 20%);
+    box-shadow: 0 0 5px black;
+    border-bottom: 1px solid $ui-base-color;
+  }
+
+  .ckbox {
+    border: 2px solid $ui-primary-color;
+    border-radius: 2px;
+    width: 30px;
+    height: 30px;
+    font-size: 20px;
+    color: $darker-text-color;
+    text-shadow: 0 0 5px black;
+    display: flex;
+    justify-content: center;
+    align-items: center;
+  }
+
+  &:focus {
+    outline: 0 !important;
+
+    .ckbox {
+      box-shadow: 0 0 1px 1px $ui-highlight-color;
+    }
+  }
+}
+
+.text-btn {
+  display: inline-block;
+  padding: 0;
+  font-family: inherit;
+  font-size: inherit;
+  color: inherit;
+  border: 0;
+  background: transparent;
+  cursor: pointer;
+}
+
+.loading-indicator {
+  color: $dark-text-color;
+  font-size: 12px;
+  font-weight: 400;
+  text-transform: uppercase;
+  overflow: visible;
+  position: absolute;
+  top: 50%;
+  inset-inline-start: 50%;
+  transform: translate(-50%, -50%);
+  display: flex;
+  align-items: center;
+  justify-content: center;
+}
+
+.circular-progress {
+  color: lighten($ui-base-color, 26%);
+  animation: 1.4s linear 0s infinite normal none running simple-rotate;
+
+  circle {
+    stroke: currentColor;
+    stroke-dasharray: 80px, 200px;
+    stroke-dashoffset: 0;
+    animation: circular-progress 1.4s ease-in-out infinite;
+  }
+}
+
+@keyframes circular-progress {
+  0% {
+    stroke-dasharray: 1px, 200px;
+    stroke-dashoffset: 0;
+  }
+
+  50% {
+    stroke-dasharray: 100px, 200px;
+    stroke-dashoffset: -15px;
+  }
+
+  100% {
+    stroke-dasharray: 100px, 200px;
+    stroke-dashoffset: -125px;
+  }
+}
+
+@keyframes simple-rotate {
+  0% {
+    transform: rotate(0deg);
+  }
+
+  100% {
+    transform: rotate(360deg);
+  }
+}
+
+@keyframes spring-rotate-in {
+  0% {
+    transform: rotate(0deg);
+  }
+
+  30% {
+    transform: rotate(-484.8deg);
+  }
+
+  60% {
+    transform: rotate(-316.7deg);
+  }
+
+  90% {
+    transform: rotate(-375deg);
+  }
+
+  100% {
+    transform: rotate(-360deg);
+  }
+}
+
+@keyframes spring-rotate-out {
+  0% {
+    transform: rotate(-360deg);
+  }
+
+  30% {
+    transform: rotate(124.8deg);
+  }
+
+  60% {
+    transform: rotate(-43.27deg);
+  }
+
+  90% {
+    transform: rotate(15deg);
+  }
+
+  100% {
+    transform: rotate(0deg);
+  }
+}
+
+.spoiler-button {
+  top: 0;
+  inset-inline-start: 0;
+  width: 100%;
+  height: 100%;
+  position: absolute;
+  z-index: 100;
+
+  &--minified {
+    display: flex;
+    inset-inline-start: 4px;
+    top: 4px;
+    width: auto;
+    height: auto;
+    align-items: center;
+  }
+
+  &--click-thru {
+    pointer-events: none;
+  }
+
+  &--hidden {
+    display: none;
+  }
+
+  &__overlay {
+    display: flex;
+    align-items: center;
+    justify-content: center;
+    background: transparent;
+    width: 100%;
+    height: 100%;
+    padding: 0;
+    margin: 0;
+    border: 0;
+    color: $white;
+
+    &__label {
+      background-color: rgba($black, 0.45);
+      backdrop-filter: blur(10px) saturate(180%) contrast(75%) brightness(70%);
+      border-radius: 6px;
+      padding: 10px 15px;
+      display: flex;
+      align-items: center;
+      justify-content: center;
+      gap: 8px;
+      flex-direction: column;
+      font-weight: 500;
+      font-size: 14px;
+    }
+
+    &__action {
+      font-weight: 400;
+      font-size: 13px;
+    }
+
+    &:hover,
+    &:focus {
+      .spoiler-button__overlay__label {
+        background-color: rgba($black, 0.9);
+      }
+    }
+  }
+}
+
+.setting-toggle {
+  display: block;
+  line-height: 24px;
+}
+
+.setting-toggle__label,
+.setting-meta__label {
+  color: $darker-text-color;
+  display: inline-block;
+  margin-bottom: 14px;
+  margin-inline-start: 8px;
+  vertical-align: middle;
+}
+
+.column-settings__row .radio-button {
+  display: block;
+}
+
+.setting-meta__label {
+  float: right;
+}
+
+@keyframes heartbeat {
+  0% {
+    transform: scale(1);
+    transform-origin: center center;
+    animation-timing-function: ease-out;
+  }
+
+  10% {
+    transform: scale(0.91);
+    animation-timing-function: ease-in;
+  }
+
+  17% {
+    transform: scale(0.98);
+    animation-timing-function: ease-out;
+  }
+
+  33% {
+    transform: scale(0.87);
+    animation-timing-function: ease-in;
+  }
+
+  45% {
+    transform: scale(1);
+    animation-timing-function: ease-out;
+  }
+}
+
+.pulse-loading {
+  animation: heartbeat 1.5s ease-in-out infinite both;
+}
+
+.upload-area {
+  align-items: center;
+  background: rgba($base-overlay-background, 0.8);
+  display: flex;
+  height: 100vh;
+  justify-content: center;
+  inset-inline-start: 0;
+  opacity: 0;
+  position: fixed;
+  top: 0;
+  visibility: hidden;
+  width: 100vw;
+  z-index: 2000;
+
+  * {
+    pointer-events: none;
+  }
+}
+
+.upload-area__drop {
+  width: 320px;
+  height: 160px;
+  display: flex;
+  box-sizing: border-box;
+  position: relative;
+  padding: 8px;
+}
+
+.upload-area__background {
+  position: absolute;
+  top: 0;
+  inset-inline-end: 0;
+  bottom: 0;
+  inset-inline-start: 0;
+  z-index: -1;
+  border-radius: 4px;
+  background: $ui-base-color;
+  box-shadow: 0 0 5px rgba($base-shadow-color, 0.2);
+}
+
+.upload-area__content {
+  flex: 1;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  text-align: center;
+  color: $secondary-text-color;
+  font-size: 18px;
+  font-weight: 500;
+  border: 2px dashed $ui-base-lighter-color;
+  border-radius: 4px;
+}
+
+.dropdown--active .emoji-button img {
+  opacity: 1;
+  filter: none;
+}
+
+.loading-bar {
+  background-color: $ui-highlight-color;
+  height: 3px;
+  position: fixed;
+  top: 0;
+  inset-inline-start: 0;
+  z-index: 9999;
+}
+
+.icon-badge-wrapper {
+  position: relative;
+}
+
+.icon-badge {
+  position: absolute;
+  display: block;
+  inset-inline-end: -0.25em;
+  top: -0.25em;
+  background-color: $ui-highlight-color;
+  border-radius: 50%;
+  font-size: 75%;
+  width: 1em;
+  height: 1em;
+}
+
+.conversation {
+  display: flex;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  padding: 5px;
+  padding-bottom: 0;
+
+  &:focus {
+    background: lighten($ui-base-color, 2%);
+    outline: 0;
+  }
+
+  &__avatar {
+    flex: 0 0 auto;
+    padding: 10px;
+    padding-top: 12px;
+    position: relative;
+    cursor: pointer;
+  }
+
+  &__unread {
+    display: inline-block;
+    background: $highlight-text-color;
+    border-radius: 50%;
+    width: 0.625rem;
+    height: 0.625rem;
+    margin: -0.1ex 0.15em 0.1ex;
+  }
+
+  &__content {
+    flex: 1 1 auto;
+    padding: 10px 5px;
+    padding-inline-end: 15px;
+    overflow: hidden;
+
+    &__info {
+      overflow: hidden;
+      display: flex;
+      flex-direction: row-reverse;
+      justify-content: space-between;
+    }
+
+    &__relative-time {
+      font-size: 15px;
+      color: $darker-text-color;
+      padding-inline-start: 15px;
+    }
+
+    &__names {
+      color: $darker-text-color;
+      font-size: 15px;
+      white-space: nowrap;
+      overflow: hidden;
+      text-overflow: ellipsis;
+      margin-bottom: 4px;
+      flex-basis: 90px;
+      flex-grow: 1;
+
+      a {
+        color: $primary-text-color;
+        text-decoration: none;
+
+        &:hover,
+        &:focus,
+        &:active {
+          text-decoration: underline;
+        }
+      }
+    }
+
+    .status__content {
+      margin: 0;
+    }
+  }
+
+  &--unread {
+    background: lighten($ui-base-color, 2%);
+
+    &:focus {
+      background: lighten($ui-base-color, 4%);
+    }
+
+    .conversation__content__info {
+      font-weight: 700;
+    }
+
+    .conversation__content__relative-time {
+      color: $primary-text-color;
+    }
+  }
+}
+
+.ui .flash-message {
+  margin-top: 10px;
+  margin-inline-start: auto;
+  margin-inline-end: auto;
+  margin-bottom: 0;
+  min-width: 75%;
+}
+
+::-webkit-scrollbar-thumb {
+  border-radius: 0;
+}
+
+noscript {
+  text-align: center;
+
+  img {
+    width: 200px;
+    opacity: 0.5;
+    animation: flicker 4s infinite;
+  }
+
+  div {
+    font-size: 14px;
+    margin: 30px auto;
+    color: $secondary-text-color;
+    max-width: 400px;
+
+    a {
+      color: $highlight-text-color;
+      text-decoration: underline;
+
+      &:hover {
+        text-decoration: none;
+      }
+    }
+
+    a {
+      word-break: break-word;
+    }
+  }
+}
+
+@keyframes flicker {
+  0% {
+    opacity: 1;
+  }
+
+  30% {
+    opacity: 0.75;
+  }
+
+  100% {
+    opacity: 1;
+  }
+}
+
+.notification-list {
+  position: fixed;
+  bottom: 2rem;
+  inset-inline-start: 0;
+  z-index: 999;
+  display: flex;
+  flex-direction: column;
+  gap: 4px;
+}
+
+.notification-bar {
+  flex: 0 0 auto;
+  position: relative;
+  inset-inline-start: -100%;
+  width: auto;
+  padding: 15px;
+  margin: 0;
+  color: $white;
+  background: rgba($black, 0.85);
+  backdrop-filter: blur(8px);
+  border: 1px solid rgba(lighten($classic-base-color, 4%), 0.85);
+  border-radius: 8px;
+  box-shadow:
+    0 10px 15px -3px rgba($base-shadow-color, 0.25),
+    0 4px 6px -4px rgba($base-shadow-color, 0.25);
+  cursor: default;
+  font-size: 15px;
+  line-height: 21px;
+
+  &.notification-bar-active {
+    inset-inline-start: 1rem;
+  }
+
+  .no-reduce-motion & {
+    transition: 0.5s cubic-bezier(0.89, 0.01, 0.5, 1.1);
+    transform: translateZ(0);
+  }
+}
+
+.notification-bar-title {
+  margin-inline-end: 5px;
+}
+
+.notification-bar-title,
+.notification-bar-action {
+  font-weight: 700;
+}
+
+.notification-bar-action {
+  text-transform: uppercase;
+  margin-inline-start: 10px;
+  cursor: pointer;
+  color: $blurple-300;
+  border-radius: 4px;
+  padding: 0 4px;
+
+  &:hover,
+  &:focus,
+  &:active {
+    background: rgba($ui-base-color, 0.85);
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/modal.scss b/app/javascript/flavours/blobfox/styles/components/modal.scss
new file mode 100644
index 00000000000000..5cbe405e0ea7ac
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/modal.scss
@@ -0,0 +1,1502 @@
+.modal-container--preloader {
+  background: lighten($ui-base-color, 8%);
+}
+
+.modal-root {
+  position: relative;
+  z-index: 9999;
+}
+
+.modal-root__overlay {
+  position: fixed;
+  top: 0;
+  inset-inline-start: 0;
+  inset-inline-end: 0;
+  bottom: 0;
+  background: rgba($base-overlay-background, 0.7);
+  transition: background 0.5s;
+}
+
+.modal-root__container {
+  position: fixed;
+  top: 0;
+  inset-inline-start: 0;
+  width: 100%;
+  height: 100%;
+  box-sizing: border-box;
+  display: flex;
+  flex-direction: column;
+  align-items: center;
+  justify-content: center;
+  align-content: space-around;
+  z-index: 9999;
+  pointer-events: none;
+  user-select: none;
+}
+
+.modal-root__modal {
+  pointer-events: auto;
+  user-select: text;
+  display: flex;
+}
+
+.media-modal__zoom-button {
+  position: absolute;
+  inset-inline-end: 64px;
+  top: 8px;
+  z-index: 100;
+  pointer-events: auto;
+  transition: opacity 0.3s linear;
+  will-change: opacity;
+}
+
+.media-modal__zoom-button--hidden {
+  pointer-events: none;
+  opacity: 0;
+}
+
+.onboarding-modal,
+.error-modal,
+.embed-modal {
+  background: $ui-secondary-color;
+  color: $inverted-text-color;
+  border-radius: 8px;
+  overflow: hidden;
+  display: flex;
+  flex-direction: column;
+}
+
+.onboarding-modal__pager {
+  height: 80vh;
+  width: 80vw;
+  max-width: 520px;
+  max-height: 470px;
+
+  .react-swipeable-view-container > div {
+    width: 100%;
+    height: 100%;
+    box-sizing: border-box;
+    flex-direction: column;
+    align-items: center;
+    justify-content: center;
+    display: flex;
+    user-select: text;
+  }
+}
+
+.error-modal__body {
+  height: 80vh;
+  width: 80vw;
+  max-width: 520px;
+  max-height: 420px;
+  position: relative;
+
+  & > div {
+    position: absolute;
+    top: 0;
+    inset-inline-start: 0;
+    width: 100%;
+    height: 100%;
+    box-sizing: border-box;
+    padding: 25px;
+    flex-direction: column;
+    align-items: center;
+    justify-content: center;
+    display: flex;
+    opacity: 0;
+    user-select: text;
+  }
+}
+
+.error-modal__body {
+  display: flex;
+  flex-direction: column;
+  justify-content: center;
+  align-items: center;
+  text-align: center;
+}
+
+@media screen and (width <= 550px) {
+  .onboarding-modal {
+    width: 100%;
+    height: 100%;
+    border-radius: 0;
+  }
+
+  .onboarding-modal__pager {
+    width: 100%;
+    height: auto;
+    max-width: none;
+    max-height: none;
+    flex: 1 1 auto;
+  }
+}
+
+.onboarding-modal__paginator,
+.error-modal__footer {
+  flex: 0 0 auto;
+  background: darken($ui-secondary-color, 8%);
+  display: flex;
+  padding: 25px;
+
+  & > div {
+    min-width: 33px;
+  }
+
+  .onboarding-modal__nav,
+  .error-modal__nav {
+    color: $lighter-text-color;
+    border: 0;
+    font-size: 14px;
+    font-weight: 500;
+    padding: 10px 25px;
+    line-height: inherit;
+    height: auto;
+    margin: -10px;
+    border-radius: 4px;
+    background-color: transparent;
+
+    &:hover,
+    &:focus,
+    &:active {
+      color: darken($lighter-text-color, 4%);
+      background-color: darken($ui-secondary-color, 16%);
+    }
+
+    &.onboarding-modal__done,
+    &.onboarding-modal__next {
+      color: $inverted-text-color;
+
+      &:hover,
+      &:focus,
+      &:active {
+        color: lighten($inverted-text-color, 4%);
+      }
+    }
+  }
+}
+
+.error-modal__footer {
+  justify-content: center;
+}
+
+.onboarding-modal__dots {
+  flex: 1 1 auto;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+}
+
+.onboarding-modal__dot {
+  width: 14px;
+  height: 14px;
+  border-radius: 14px;
+  background: darken($ui-secondary-color, 16%);
+  margin: 0 3px;
+  cursor: pointer;
+
+  &:hover {
+    background: darken($ui-secondary-color, 18%);
+  }
+
+  &.active {
+    cursor: default;
+    background: darken($ui-secondary-color, 24%);
+  }
+}
+
+.onboarding-modal__page__wrapper {
+  pointer-events: none;
+  padding: 25px;
+  padding-bottom: 0;
+
+  &.onboarding-modal__page__wrapper--active {
+    pointer-events: auto;
+  }
+}
+
+.onboarding-modal__page {
+  cursor: default;
+  line-height: 21px;
+
+  h1 {
+    font-size: 18px;
+    font-weight: 500;
+    color: $inverted-text-color;
+    margin-bottom: 20px;
+  }
+
+  a {
+    color: $highlight-text-color;
+
+    &:hover,
+    &:focus,
+    &:active {
+      color: lighten($highlight-text-color, 4%);
+    }
+  }
+
+  .navigation-bar a {
+    color: inherit;
+  }
+
+  p {
+    font-size: 16px;
+    color: $lighter-text-color;
+    margin-top: 10px;
+    margin-bottom: 10px;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+
+    strong {
+      font-weight: 500;
+      background: $ui-base-color;
+      color: $secondary-text-color;
+      border-radius: 4px;
+      font-size: 14px;
+      padding: 3px 6px;
+
+      @each $lang in $cjk-langs {
+        &:lang(#{$lang}) {
+          font-weight: 700;
+        }
+      }
+    }
+  }
+}
+
+.onboarding-modal__page__wrapper-0 {
+  background: url('~images/elephant_ui_greeting.svg') no-repeat left bottom /
+    auto 250px;
+  height: 100%;
+  padding: 0;
+}
+
+.onboarding-modal__page-one {
+  &__lead {
+    padding: 65px;
+    padding-top: 45px;
+    padding-bottom: 0;
+    margin-bottom: 10px;
+
+    h1 {
+      font-size: 26px;
+      line-height: 36px;
+      margin-bottom: 8px;
+    }
+
+    p {
+      margin-bottom: 0;
+    }
+  }
+
+  &__extra {
+    padding-inline-end: 65px;
+    padding-inline-start: 185px;
+    text-align: center;
+  }
+}
+
+.display-case {
+  text-align: center;
+  font-size: 15px;
+  margin-bottom: 15px;
+
+  &__label {
+    font-weight: 500;
+    color: $inverted-text-color;
+    margin-bottom: 5px;
+    text-transform: uppercase;
+    font-size: 12px;
+  }
+
+  &__case {
+    background: $ui-base-color;
+    color: $secondary-text-color;
+    font-weight: 500;
+    padding: 10px;
+    border-radius: 4px;
+  }
+}
+
+.onboarding-modal__page-two,
+.onboarding-modal__page-three,
+.onboarding-modal__page-four,
+.onboarding-modal__page-five {
+  p {
+    text-align: start;
+  }
+
+  .figure {
+    background: darken($ui-base-color, 8%);
+    color: $secondary-text-color;
+    margin-bottom: 20px;
+    border-radius: 4px;
+    padding: 10px;
+    text-align: center;
+    font-size: 14px;
+    box-shadow: 1px 2px 6px rgba($base-shadow-color, 0.3);
+
+    .onboarding-modal__image {
+      border-radius: 4px;
+      margin-bottom: 10px;
+    }
+
+    &.non-interactive {
+      pointer-events: none;
+      text-align: start;
+    }
+  }
+}
+
+.onboarding-modal__page-four__columns {
+  .row {
+    display: flex;
+    margin-bottom: 20px;
+
+    & > div {
+      flex: 1 1 0;
+      margin: 0 10px;
+
+      &:first-child {
+        margin-inline-start: 0;
+      }
+
+      &:last-child {
+        margin-inline-end: 0;
+      }
+
+      p {
+        text-align: center;
+      }
+    }
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  .column-header {
+    color: $primary-text-color;
+  }
+}
+
+@media screen and (width <= 320px) and (height <= 600px) {
+  .onboarding-modal__page p {
+    font-size: 14px;
+    line-height: 20px;
+  }
+
+  .onboarding-modal__page-two .figure,
+  .onboarding-modal__page-three .figure,
+  .onboarding-modal__page-four .figure,
+  .onboarding-modal__page-five .figure {
+    font-size: 12px;
+    margin-bottom: 10px;
+  }
+
+  .onboarding-modal__page-four__columns .row {
+    margin-bottom: 10px;
+  }
+
+  .onboarding-modal__page-four__columns .column-header {
+    padding: 5px;
+    font-size: 12px;
+  }
+}
+
+.onboard-sliders {
+  display: inline-block;
+  max-width: 30px;
+  max-height: auto;
+  margin-inline-start: 10px;
+}
+
+.doodle-modal,
+.boost-modal,
+.confirmation-modal,
+.report-modal,
+.actions-modal,
+.mute-modal,
+.block-modal,
+.compare-history-modal {
+  background: lighten($ui-secondary-color, 8%);
+  color: $inverted-text-color;
+  border-radius: 8px;
+  overflow: hidden;
+  max-width: 90vw;
+  width: 480px;
+  position: relative;
+  flex-direction: column;
+
+  .status__relative-time {
+    color: $dark-text-color;
+    float: right;
+    font-size: 14px;
+    width: auto;
+    margin: initial;
+    padding: initial;
+  }
+
+  .status__visibility-icon {
+    color: $dark-text-color;
+    font-size: 14px;
+    padding: 0 4px;
+  }
+
+  .status__display-name {
+    display: flex;
+  }
+
+  .status__avatar {
+    height: 48px;
+    width: 48px;
+  }
+
+  .status__content__spoiler-link {
+    color: lighten($secondary-text-color, 8%);
+  }
+}
+
+.boost-modal .status-direct {
+  background-color: inherit;
+}
+
+.actions-modal {
+  .status {
+    background: $white;
+    border-bottom-color: $ui-secondary-color;
+    padding-top: 10px;
+    padding-bottom: 10px;
+  }
+
+  .dropdown-menu__separator {
+    border-bottom-color: $ui-secondary-color;
+  }
+}
+
+.boost-modal__container {
+  overflow-x: scroll;
+  padding: 10px;
+
+  .status {
+    user-select: text;
+    border-bottom: 0;
+  }
+}
+
+.doodle-modal__action-bar,
+.boost-modal__action-bar,
+.confirmation-modal__action-bar,
+.mute-modal__action-bar,
+.block-modal__action-bar {
+  display: flex;
+  justify-content: space-between;
+  background: $ui-secondary-color;
+  padding: 10px;
+  line-height: 36px;
+
+  & > div {
+    flex: 1 1 auto;
+    text-align: end;
+    color: $lighter-text-color;
+    padding-inline-end: 10px;
+  }
+
+  .button {
+    flex: 0 0 auto;
+  }
+}
+
+.boost-modal__status-header {
+  font-size: 15px;
+}
+
+.boost-modal__status-time {
+  float: right;
+  font-size: 14px;
+}
+
+.mute-modal,
+.block-modal {
+  line-height: 24px;
+}
+
+.mute-modal .react-toggle,
+.block-modal .react-toggle {
+  vertical-align: middle;
+}
+
+.report-modal {
+  width: 90vw;
+  max-width: 700px;
+}
+
+.report-dialog-modal {
+  max-width: 90vw;
+  width: 480px;
+  height: 80vh;
+  background: lighten($ui-secondary-color, 8%);
+  color: $inverted-text-color;
+  border-radius: 8px;
+  overflow: hidden;
+  position: relative;
+  flex-direction: column;
+  display: flex;
+
+  &__container {
+    box-sizing: border-box;
+    border-top: 1px solid $ui-secondary-color;
+    padding: 20px;
+    flex-grow: 1;
+    display: flex;
+    flex-direction: column;
+    min-height: 0;
+    overflow: auto;
+  }
+
+  &__title {
+    font-size: 28px;
+    line-height: 33px;
+    font-weight: 700;
+    margin-bottom: 15px;
+
+    @media screen and (height <= 800px) {
+      font-size: 22px;
+    }
+  }
+
+  &__subtitle {
+    font-size: 17px;
+    font-weight: 600;
+    line-height: 22px;
+    margin-bottom: 4px;
+  }
+
+  &__lead {
+    font-size: 17px;
+    line-height: 22px;
+    color: lighten($inverted-text-color, 16%);
+    margin-bottom: 30px;
+
+    a {
+      text-decoration: none;
+      color: $inverted-text-color;
+      font-weight: 500;
+
+      &:hover {
+        text-decoration: underline;
+      }
+    }
+  }
+
+  &__actions {
+    margin-top: 30px;
+    display: flex;
+
+    .button {
+      flex: 1 1 auto;
+    }
+  }
+
+  &__statuses {
+    flex-grow: 1;
+    min-height: 0;
+    overflow: auto;
+  }
+
+  .status__content a {
+    color: $highlight-text-color;
+  }
+
+  .status__content,
+  .status__content p {
+    color: $inverted-text-color;
+  }
+
+  .status__content__spoiler-link {
+    color: $primary-text-color;
+    background: $ui-primary-color;
+
+    &:hover {
+      background: lighten($ui-primary-color, 8%);
+    }
+  }
+
+  .dialog-option .poll__input {
+    border-color: $inverted-text-color;
+    color: $ui-secondary-color;
+    display: inline-flex;
+    align-items: center;
+    justify-content: center;
+
+    svg {
+      width: 8px;
+      height: auto;
+    }
+
+    &:active,
+    &:focus,
+    &:hover {
+      border-color: lighten($inverted-text-color, 15%);
+      border-width: 4px;
+    }
+
+    &.active {
+      border-color: $inverted-text-color;
+      background: $inverted-text-color;
+    }
+  }
+
+  .poll__option.dialog-option {
+    padding: 15px 0;
+    flex: 0 0 auto;
+    border-bottom: 1px solid $ui-secondary-color;
+
+    &:last-child {
+      border-bottom: 0;
+    }
+
+    & > .poll__option__text {
+      font-size: 13px;
+      color: lighten($inverted-text-color, 16%);
+
+      strong {
+        font-size: 17px;
+        font-weight: 500;
+        line-height: 22px;
+        color: $inverted-text-color;
+        display: block;
+        margin-bottom: 4px;
+
+        &:last-child {
+          margin-bottom: 0;
+        }
+      }
+    }
+  }
+
+  .flex-spacer {
+    background: transparent;
+  }
+
+  &__textarea {
+    display: block;
+    box-sizing: border-box;
+    width: 100%;
+    color: $inverted-text-color;
+    background: $simple-background-color;
+    padding: 10px;
+    font-family: inherit;
+    font-size: 17px;
+    line-height: 22px;
+    resize: vertical;
+    border: 0;
+    outline: 0;
+    border-radius: 4px;
+    margin: 20px 0;
+
+    &::placeholder {
+      color: $dark-text-color;
+    }
+
+    &:focus {
+      outline: 0;
+    }
+  }
+
+  &__toggle {
+    display: flex;
+    align-items: center;
+    margin-bottom: 10px;
+
+    & > span {
+      font-size: 17px;
+      font-weight: 500;
+      margin-inline-start: 10px;
+    }
+  }
+
+  .button.button-secondary {
+    border-color: $inverted-text-color;
+    color: $inverted-text-color;
+    flex: 0 0 auto;
+
+    &:hover,
+    &:focus,
+    &:active {
+      background: transparent;
+      border-color: $ui-button-background-color;
+      color: $ui-button-background-color;
+    }
+  }
+
+  hr {
+    border: 0;
+    background: transparent;
+    margin: 15px 0;
+  }
+
+  .emoji-mart-search {
+    padding-inline-end: 10px;
+  }
+
+  .emoji-mart-search-icon {
+    inset-inline-end: 10px + 5px;
+  }
+}
+
+.report-modal__container {
+  display: flex;
+  border-top: 1px solid $ui-secondary-color;
+
+  @media screen and (width <= 480px) {
+    flex-wrap: wrap;
+    overflow-y: auto;
+  }
+}
+
+.report-modal__statuses,
+.report-modal__comment {
+  box-sizing: border-box;
+  width: 50%;
+
+  @media screen and (width <= 480px) {
+    width: 100%;
+  }
+}
+
+.report-modal__statuses,
+.focal-point-modal__content {
+  flex: 1 1 auto;
+  min-height: 20vh;
+  max-height: 80vh;
+  overflow-y: auto;
+  overflow-x: hidden;
+
+  .status__content a {
+    color: $highlight-text-color;
+  }
+
+  @media screen and (width <= 480px) {
+    max-height: 10vh;
+  }
+}
+
+.focal-point-modal__content {
+  @media screen and (width <= 480px) {
+    max-height: 40vh;
+  }
+}
+
+.setting-divider {
+  background: transparent;
+  border: 0;
+  margin: 0;
+  width: 100%;
+  height: 1px;
+  margin-bottom: 29px;
+}
+
+.report-modal__comment {
+  padding: 20px;
+  border-inline-end: 1px solid $ui-secondary-color;
+  max-width: 320px;
+
+  p {
+    font-size: 14px;
+    line-height: 20px;
+    margin-bottom: 20px;
+  }
+
+  .setting-text {
+    display: block;
+    box-sizing: border-box;
+    width: 100%;
+    margin: 0;
+    color: $inverted-text-color;
+    background: $white;
+    padding: 10px;
+    font-family: inherit;
+    font-size: 14px;
+    resize: none;
+    outline: 0;
+    border-radius: 4px;
+    border: 1px solid $ui-secondary-color;
+    min-height: 100px;
+    max-height: 50vh;
+    margin-bottom: 10px;
+
+    &:focus {
+      border: 1px solid darken($ui-secondary-color, 8%);
+    }
+
+    &__wrapper {
+      background: $white;
+      border: 1px solid $ui-secondary-color;
+      margin-bottom: 10px;
+      border-radius: 4px;
+
+      .setting-text {
+        border: 0;
+        margin-bottom: 0;
+        border-radius: 0;
+
+        &:focus {
+          border: 0;
+        }
+      }
+
+      &__modifiers {
+        color: $inverted-text-color;
+        font-family: inherit;
+        font-size: 14px;
+        background: $white;
+      }
+    }
+
+    &__toolbar {
+      display: flex;
+      justify-content: space-between;
+      margin-bottom: 20px;
+    }
+  }
+
+  .setting-text-label {
+    display: block;
+    color: $inverted-text-color;
+    font-size: 14px;
+    font-weight: 500;
+    margin-bottom: 10px;
+  }
+
+  .setting-toggle {
+    margin-top: 20px;
+    margin-bottom: 24px;
+
+    &__label {
+      color: $inverted-text-color;
+      font-size: 14px;
+    }
+  }
+
+  @media screen and (width <= 480px) {
+    padding: 10px;
+    max-width: 100%;
+    order: 2;
+
+    .setting-toggle {
+      margin-bottom: 4px;
+    }
+  }
+}
+
+.actions-modal {
+  .status {
+    overflow-y: auto;
+    max-height: 300px;
+  }
+
+  strong {
+    display: block;
+    font-weight: 500;
+  }
+
+  max-height: 80vh;
+  max-width: 80vw;
+
+  .actions-modal__item-label {
+    font-weight: 500;
+  }
+
+  ul {
+    overflow-y: auto;
+    flex-shrink: 0;
+    max-height: 80vh;
+
+    &.with-status {
+      max-height: calc(80vh - 75px);
+    }
+
+    li:empty {
+      margin: 0;
+    }
+
+    li:not(:empty) {
+      a {
+        color: $inverted-text-color;
+        display: flex;
+        padding: 12px 16px;
+        font-size: 15px;
+        align-items: center;
+        text-decoration: none;
+
+        &,
+        button {
+          transition: none;
+        }
+
+        &.active,
+        &:hover,
+        &:active,
+        &:focus {
+          &,
+          button {
+            background: $ui-highlight-color;
+            color: $primary-text-color;
+          }
+        }
+
+        & > .react-toggle,
+        & > .icon,
+        button:first-child {
+          margin-inline-end: 10px;
+        }
+      }
+    }
+  }
+}
+
+.confirmation-modal__action-bar,
+.mute-modal__action-bar,
+.block-modal__action-bar {
+  .confirmation-modal__secondary-button {
+    flex-shrink: 1;
+  }
+}
+
+.confirmation-modal__secondary-button,
+.confirmation-modal__cancel-button,
+.mute-modal__cancel-button,
+.block-modal__cancel-button {
+  background-color: transparent;
+  color: $lighter-text-color;
+  font-size: 14px;
+  font-weight: 500;
+
+  &:hover,
+  &:focus,
+  &:active {
+    color: darken($lighter-text-color, 4%);
+    background-color: transparent;
+  }
+}
+
+.confirmation-modal__do_not_ask_again {
+  padding-inline-start: 20px;
+  padding-inline-end: 20px;
+  padding-bottom: 10px;
+  font-size: 14px;
+
+  label,
+  input {
+    vertical-align: middle;
+  }
+}
+
+.confirmation-modal__container,
+.mute-modal__container,
+.block-modal__container,
+.report-modal__target {
+  padding: 30px;
+  font-size: 16px;
+
+  strong {
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  select {
+    appearance: none;
+    box-sizing: border-box;
+    font-size: 14px;
+    color: $inverted-text-color;
+    display: inline-block;
+    width: auto;
+    outline: 0;
+    font-family: inherit;
+    background: $simple-background-color
+      url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 14.933 18.467' height='19.698' width='15.929'><path d='M3.467 14.967l-3.393-3.5H14.86l-3.392 3.5c-1.866 1.925-3.666 3.5-4 3.5-.335 0-2.135-1.575-4-3.5zm.266-11.234L7.467 0 11.2 3.733l3.733 3.734H0l3.733-3.734z' fill='#{hex-color(darken($simple-background-color, 14%))}'/></svg>")
+      no-repeat right 8px center / auto 16px;
+    border: 1px solid darken($simple-background-color, 14%);
+    border-radius: 4px;
+    padding: 6px 10px;
+    padding-inline-end: 30px;
+  }
+}
+
+.confirmation-modal__container,
+.report-modal__target {
+  text-align: center;
+}
+
+.block-modal,
+.mute-modal {
+  &__explanation {
+    margin-top: 20px;
+  }
+
+  .setting-toggle {
+    margin-top: 20px;
+    margin-bottom: 24px;
+    display: flex;
+    align-items: center;
+
+    &__label {
+      color: $inverted-text-color;
+      margin: 0;
+      margin-inline-start: 8px;
+    }
+  }
+}
+
+.report-modal__target {
+  padding: 15px;
+
+  .report-modal__close {
+    position: absolute;
+    top: 10px;
+    inset-inline-end: 10px;
+  }
+}
+
+.compare-history-modal {
+  .report-modal__target {
+    border-bottom: 1px solid $ui-secondary-color;
+  }
+
+  &__container {
+    padding: 30px;
+    pointer-events: all;
+    overflow-y: auto;
+  }
+
+  .status__content {
+    color: $inverted-text-color;
+    font-size: 19px;
+    line-height: 24px;
+
+    .emojione {
+      width: 24px;
+      height: 24px;
+      margin: -1px 0 0;
+    }
+
+    a {
+      color: $highlight-text-color;
+    }
+
+    hr {
+      height: 0.25rem;
+      padding: 0;
+      background-color: $ui-secondary-color;
+      border: 0;
+      margin: 20px 0;
+    }
+  }
+
+  .media-gallery,
+  .audio-player,
+  .video-player {
+    margin-top: 15px;
+  }
+}
+
+.embed-modal {
+  width: auto;
+  max-width: 80vw;
+  max-height: 80vh;
+
+  h4 {
+    padding: 30px;
+    font-weight: 500;
+    font-size: 16px;
+    text-align: center;
+  }
+
+  .embed-modal__container {
+    padding: 10px;
+
+    .hint {
+      margin-bottom: 15px;
+    }
+
+    .embed-modal__html {
+      outline: 0;
+      box-sizing: border-box;
+      display: block;
+      width: 100%;
+      border: 0;
+      padding: 10px;
+      font-family: mastodon-font-monospace, monospace;
+      background: $ui-base-color;
+      color: $primary-text-color;
+      font-size: 14px;
+      margin: 0;
+      margin-bottom: 15px;
+      border-radius: 4px;
+
+      &::-moz-focus-inner {
+        border: 0;
+      }
+
+      &::-moz-focus-inner,
+      &:focus,
+      &:active {
+        outline: 0 !important;
+      }
+
+      &:focus {
+        background: lighten($ui-base-color, 4%);
+      }
+
+      @media screen and (width <= 600px) {
+        font-size: 16px;
+      }
+    }
+
+    .embed-modal__iframe {
+      width: 400px;
+      max-width: 100%;
+      overflow: hidden;
+      border: 0;
+      border-radius: 4px;
+    }
+  }
+}
+
+.focal-point {
+  position: relative;
+  cursor: move;
+  overflow: hidden;
+  height: 100%;
+  display: flex;
+  justify-content: center;
+  align-items: center;
+  background: $base-shadow-color;
+
+  img,
+  video,
+  canvas {
+    display: block;
+    max-height: 80vh;
+    width: 100%;
+    height: auto;
+    margin: 0;
+    object-fit: contain;
+    background: $base-shadow-color;
+  }
+
+  &__reticle {
+    position: absolute;
+    width: 100px;
+    height: 100px;
+    transform: translate(-50%, -50%);
+    background: url('~images/reticle.png') no-repeat 0 0;
+    border-radius: 50%;
+    box-shadow: 0 0 0 9999em rgba($base-shadow-color, 0.35);
+  }
+
+  &__overlay {
+    position: absolute;
+    width: 100%;
+    height: 100%;
+    top: 0;
+    inset-inline-start: 0;
+  }
+
+  &__preview {
+    position: absolute;
+    bottom: 10px;
+    inset-inline-end: 10px;
+    z-index: 2;
+    cursor: move;
+    transition: opacity 0.1s ease;
+
+    &:hover {
+      opacity: 0.5;
+    }
+
+    strong {
+      color: $primary-text-color;
+      font-size: 14px;
+      font-weight: 500;
+      display: block;
+      margin-bottom: 5px;
+    }
+
+    div {
+      border-radius: 4px;
+      box-shadow: 0 0 14px rgba($base-shadow-color, 0.2);
+    }
+  }
+
+  @media screen and (width <= 480px) {
+    img,
+    video {
+      max-height: 100%;
+    }
+
+    &__preview {
+      display: none;
+    }
+  }
+}
+
+.filtered-status-info {
+  text-align: start;
+
+  .spoiler__text {
+    margin-top: 20px;
+  }
+
+  .account {
+    border-bottom: 0;
+  }
+
+  .account__display-name strong {
+    color: $inverted-text-color;
+  }
+
+  .status__content__spoiler {
+    display: none;
+
+    &--visible {
+      display: flex;
+    }
+  }
+
+  ul {
+    padding: 10px;
+    margin-inline-start: 12px;
+    list-style: disc inside;
+  }
+
+  .filtered-status-edit-link {
+    color: $action-button-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+  }
+}
+
+.modal-root__container .privacy-dropdown {
+  flex-grow: 0;
+}
+
+.modal-root__container .privacy-dropdown__dropdown {
+  pointer-events: auto;
+  z-index: 9999;
+}
+
+img.modal-warning {
+  display: block;
+  margin: auto;
+  margin-bottom: 15px;
+  width: 60px;
+}
+
+.interaction-modal {
+  max-width: 90vw;
+  width: 600px;
+  background: var(--modal-background-color);
+  border: 1px solid var(--modal-border-color);
+  border-radius: 8px;
+  overflow: visible;
+  position: relative;
+  display: block;
+  padding: 40px;
+
+  h3 {
+    font-size: 22px;
+    line-height: 33px;
+    font-weight: 700;
+    text-align: center;
+  }
+
+  p {
+    font-size: 17px;
+    line-height: 22px;
+    color: $darker-text-color;
+
+    strong {
+      color: $primary-text-color;
+      font-weight: 700;
+    }
+  }
+
+  p.hint {
+    margin-bottom: 14px;
+    font-size: 14px;
+  }
+
+  &__icon {
+    color: $highlight-text-color;
+    margin: 0 5px;
+  }
+
+  &__lead {
+    margin-bottom: 20px;
+
+    h3 {
+      margin-bottom: 15px;
+    }
+  }
+
+  &__login {
+    position: relative;
+    margin-bottom: 20px;
+
+    &__input {
+      @include search-input;
+
+      border: 1px solid lighten($ui-base-color, 8%);
+      padding: 4px 6px;
+      color: $primary-text-color;
+      font-size: 16px;
+      line-height: 18px;
+      display: flex;
+      align-items: center;
+
+      input {
+        background: transparent;
+        color: inherit;
+        font: inherit;
+        border: 0;
+        padding: 15px - 4px 15px - 6px;
+        flex: 1 1 auto;
+
+        &::placeholder {
+          color: lighten($darker-text-color, 4%);
+        }
+
+        &:focus {
+          outline: 0;
+        }
+      }
+
+      .button {
+        flex: 0 0 auto;
+      }
+    }
+
+    .search__popout {
+      margin-top: -1px;
+      padding-top: 5px;
+      padding-bottom: 5px;
+      border: 1px solid lighten($ui-base-color, 8%);
+    }
+
+    &.focused &__input {
+      border-color: $highlight-text-color;
+      background: lighten($ui-base-color, 4%);
+    }
+
+    &.invalid &__input {
+      border-color: $error-red;
+    }
+
+    &.expanded .search__popout {
+      display: block;
+    }
+
+    &.expanded &__input {
+      border-radius: 4px 4px 0 0;
+    }
+  }
+
+  &__choices {
+    display: flex;
+    gap: 40px;
+
+    &__choice {
+      flex: 1;
+      box-sizing: border-box;
+
+      h3 {
+        margin-bottom: 20px;
+      }
+
+      p {
+        color: $darker-text-color;
+        margin-bottom: 20px;
+        font-size: 15px;
+      }
+
+      .button {
+        margin-bottom: 10px;
+
+        &:last-child {
+          margin-bottom: 0;
+        }
+      }
+    }
+  }
+
+  @media screen and (max-width: $no-gap-breakpoint - 1px) {
+    &__choices {
+      flex-direction: column;
+
+      &__choice {
+        margin-top: 40px;
+      }
+    }
+  }
+
+  .link-button {
+    font-size: inherit;
+    display: inline;
+  }
+}
+
+.copypaste {
+  display: flex;
+  align-items: center;
+  gap: 10px;
+
+  input {
+    display: block;
+    font-family: inherit;
+    background: darken($ui-base-color, 8%);
+    border: 1px solid $highlight-text-color;
+    color: $darker-text-color;
+    border-radius: 4px;
+    padding: 6px 9px;
+    line-height: 22px;
+    font-size: 14px;
+    transition: border-color 300ms linear;
+    flex: 1 1 auto;
+    overflow: hidden;
+
+    &:focus {
+      outline: 0;
+      background: darken($ui-base-color, 4%);
+    }
+  }
+
+  .button {
+    flex: 0 0 auto;
+    transition: background 300ms linear;
+  }
+
+  &.copied {
+    input {
+      border: 1px solid $valid-value-color;
+      transition: none;
+    }
+
+    .button {
+      background: $valid-value-color;
+      transition: none;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/privacy_policy.scss b/app/javascript/flavours/blobfox/styles/components/privacy_policy.scss
new file mode 100644
index 00000000000000..cab78402b214da
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/privacy_policy.scss
@@ -0,0 +1,209 @@
+.privacy-policy {
+  background: $ui-base-color;
+  padding: 20px;
+
+  @media screen and (min-width: $no-gap-breakpoint) {
+    border-radius: 4px;
+  }
+
+  &__body {
+    margin-top: 20px;
+  }
+}
+
+.prose {
+  color: $secondary-text-color;
+  font-size: 15px;
+  line-height: 22px;
+
+  p,
+  ul,
+  ol {
+    margin-top: 1.25em;
+    margin-bottom: 1.25em;
+  }
+
+  img {
+    margin-top: 2em;
+    margin-bottom: 2em;
+    max-width: 100%;
+  }
+
+  video {
+    margin-top: 2em;
+    margin-bottom: 2em;
+    max-width: 100%;
+  }
+
+  figure {
+    margin-top: 2em;
+    margin-bottom: 2em;
+
+    figcaption {
+      font-size: 0.875em;
+      line-height: 1.4285714;
+      margin-top: 0.8571429em;
+    }
+  }
+
+  figure > * {
+    margin-top: 0;
+    margin-bottom: 0;
+  }
+
+  h1 {
+    font-size: 1.5em;
+    margin-top: 0;
+    margin-bottom: 1em;
+    line-height: 1.33;
+  }
+
+  h2 {
+    font-size: 1.25em;
+    margin-top: 1.6em;
+    margin-bottom: 0.6em;
+    line-height: 1.6;
+  }
+
+  h3,
+  h4,
+  h5,
+  h6 {
+    margin-top: 1.5em;
+    margin-bottom: 0.5em;
+    line-height: 1.5;
+  }
+
+  ol {
+    counter-reset: list-counter;
+  }
+
+  li {
+    margin-top: 0.5em;
+    margin-bottom: 0.5em;
+  }
+
+  ol > li {
+    counter-increment: list-counter;
+
+    &::before {
+      content: counter(list-counter) '.';
+      position: absolute;
+      inset-inline-start: 0;
+    }
+  }
+
+  ul > li::before {
+    content: '';
+    position: absolute;
+    background-color: $darker-text-color;
+    border-radius: 50%;
+    width: 0.375em;
+    height: 0.375em;
+    top: 0.5em;
+    inset-inline-start: 0.25em;
+  }
+
+  ul > li,
+  ol > li {
+    position: relative;
+    padding-inline-start: 1.75em;
+  }
+
+  & > ul > li p {
+    margin-top: 0.75em;
+    margin-bottom: 0.75em;
+  }
+
+  & > ul > li > *:first-child {
+    margin-top: 1.25em;
+  }
+
+  & > ul > li > *:last-child {
+    margin-bottom: 1.25em;
+  }
+
+  & > ol > li > *:first-child {
+    margin-top: 1.25em;
+  }
+
+  & > ol > li > *:last-child {
+    margin-bottom: 1.25em;
+  }
+
+  ul ul,
+  ul ol,
+  ol ul,
+  ol ol {
+    margin-top: 0.75em;
+    margin-bottom: 0.75em;
+  }
+
+  h1,
+  h2,
+  h3,
+  h4,
+  h5,
+  h6,
+  strong,
+  b {
+    color: $primary-text-color;
+    font-weight: 700;
+  }
+
+  em,
+  i {
+    font-style: italic;
+  }
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: underline;
+
+    &:focus,
+    &:hover,
+    &:active {
+      text-decoration: none;
+    }
+  }
+
+  code {
+    font-size: 0.875em;
+    background: darken($ui-base-color, 8%);
+    border-radius: 4px;
+    padding: 0.2em 0.3em;
+  }
+
+  hr {
+    border: 0;
+    border-top: 1px solid lighten($ui-base-color, 4%);
+    margin-top: 3em;
+    margin-bottom: 3em;
+  }
+
+  hr + * {
+    margin-top: 0;
+  }
+
+  h2 + * {
+    margin-top: 0;
+  }
+
+  h3 + * {
+    margin-top: 0;
+  }
+
+  h4 + *,
+  h5 + *,
+  h6 + * {
+    margin-top: 0;
+  }
+
+  & > :first-child {
+    margin-top: 0;
+  }
+
+  & > :last-child {
+    margin-bottom: 0;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/regeneration_indicator.scss b/app/javascript/flavours/blobfox/styles/components/regeneration_indicator.scss
new file mode 100644
index 00000000000000..c65e6a9afcda8c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/regeneration_indicator.scss
@@ -0,0 +1,43 @@
+.regeneration-indicator {
+  text-align: center;
+  font-size: 16px;
+  font-weight: 500;
+  color: $dark-text-color;
+  background: $ui-base-color;
+  cursor: default;
+  display: flex;
+  flex: 1 1 auto;
+  flex-direction: column;
+  align-items: center;
+  justify-content: center;
+  padding: 20px;
+
+  &__figure {
+    &,
+    img {
+      display: block;
+      width: auto;
+      height: 160px;
+      margin: 0;
+    }
+  }
+
+  &--without-header {
+    padding-top: 20px + 48px;
+  }
+
+  &__label {
+    margin-top: 30px;
+
+    strong {
+      display: block;
+      margin-bottom: 10px;
+      color: $dark-text-color;
+    }
+
+    span {
+      font-size: 15px;
+      font-weight: 400;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/search.scss b/app/javascript/flavours/blobfox/styles/components/search.scss
new file mode 100644
index 00000000000000..aa54fc26db3a71
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/search.scss
@@ -0,0 +1,337 @@
+.search {
+  margin-bottom: 10px;
+  position: relative;
+
+  &__popout {
+    box-sizing: border-box;
+    display: none;
+    position: absolute;
+    inset-inline-start: 0;
+    margin-top: -2px;
+    width: 100%;
+    background: $ui-base-color;
+    border-radius: 0 0 4px 4px;
+    box-shadow: 4px 4px 6px rgba($base-shadow-color, 0.4);
+    z-index: 99;
+    font-size: 13px;
+    padding: 15px 5px;
+
+    h4 {
+      text-transform: uppercase;
+      color: $dark-text-color;
+      font-weight: 500;
+      padding: 0 10px;
+      margin-bottom: 10px;
+    }
+
+    &__menu {
+      margin-bottom: 20px;
+
+      &:last-child {
+        margin-bottom: 0;
+      }
+
+      &__message {
+        color: $dark-text-color;
+        padding: 0 10px;
+      }
+
+      &__item {
+        display: block;
+        box-sizing: border-box;
+        width: 100%;
+        border: 0;
+        font: inherit;
+        background: transparent;
+        color: $darker-text-color;
+        padding: 10px;
+        cursor: pointer;
+        border-radius: 4px;
+        text-align: start;
+        text-overflow: ellipsis;
+        overflow: hidden;
+        white-space: nowrap;
+
+        &--flex {
+          display: flex;
+          justify-content: space-between;
+        }
+
+        .icon-button {
+          transition: none;
+        }
+
+        &:hover,
+        &:focus,
+        &:active,
+        &.selected {
+          background: $ui-highlight-color;
+          color: $primary-text-color;
+
+          .icon-button {
+            color: $primary-text-color;
+          }
+        }
+
+        mark {
+          background: transparent;
+          font-weight: 700;
+          color: $primary-text-color;
+        }
+
+        span {
+          overflow: inherit;
+          text-overflow: inherit;
+        }
+      }
+    }
+  }
+
+  &.active {
+    .search__popout {
+      display: block;
+    }
+  }
+}
+
+.search__input {
+  @include search-input;
+
+  display: block;
+  padding: 15px;
+  padding-inline-end: 30px;
+  line-height: 18px;
+  font-size: 16px;
+
+  &::placeholder {
+    color: lighten($darker-text-color, 4%);
+  }
+
+  &::-moz-focus-inner {
+    border: 0;
+  }
+
+  &::-moz-focus-inner,
+  &:focus,
+  &:active {
+    outline: 0 !important;
+  }
+
+  &:focus {
+    background: lighten($ui-base-color, 4%);
+  }
+}
+
+.search__icon {
+  &::-moz-focus-inner {
+    border: 0;
+  }
+
+  &::-moz-focus-inner,
+  &:focus {
+    outline: 0 !important;
+  }
+
+  .fa {
+    position: absolute;
+    top: 16px;
+    inset-inline-end: 10px;
+    display: inline-block;
+    opacity: 0;
+    transition: all 100ms linear;
+    transition-property: color, transform, opacity;
+    font-size: 18px;
+    width: 18px;
+    height: 18px;
+    color: $secondary-text-color;
+    cursor: default;
+    pointer-events: none;
+
+    &.active {
+      pointer-events: auto;
+      opacity: 0.3;
+    }
+  }
+
+  .fa-search {
+    transform: rotate(0deg);
+
+    &.active {
+      pointer-events: auto;
+      opacity: 0.3;
+    }
+  }
+
+  .fa-times-circle {
+    top: 17px;
+    transform: rotate(0deg);
+    color: $action-button-color;
+    cursor: pointer;
+
+    &.active {
+      transform: rotate(90deg);
+      opacity: 1;
+    }
+
+    &:hover {
+      color: lighten($action-button-color, 7%);
+    }
+  }
+}
+
+.search-results__header {
+  color: $dark-text-color;
+  background: lighten($ui-base-color, 2%);
+  padding: 15px;
+  font-weight: 500;
+  font-size: 16px;
+  cursor: default;
+
+  .fa {
+    display: inline-block;
+    margin-inline-end: 5px;
+  }
+}
+
+.search-results__info {
+  padding: 20px;
+  color: $darker-text-color;
+  text-align: center;
+}
+
+.trends {
+  &__header {
+    color: $dark-text-color;
+    background: lighten($ui-base-color, 2%);
+    border-bottom: 1px solid darken($ui-base-color, 4%);
+    font-weight: 500;
+    padding: 15px;
+    font-size: 16px;
+    cursor: default;
+
+    .fa {
+      display: inline-block;
+      margin-inline-end: 5px;
+    }
+  }
+
+  &__item {
+    display: flex;
+    align-items: center;
+    padding: 15px;
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+    gap: 15px;
+
+    &:last-child {
+      border-bottom: 0;
+    }
+
+    &__name {
+      flex: 1 1 auto;
+      color: $dark-text-color;
+      overflow: hidden;
+      text-overflow: ellipsis;
+      white-space: nowrap;
+
+      strong {
+        font-weight: 500;
+      }
+
+      a {
+        color: $darker-text-color;
+        text-decoration: none;
+        font-size: 14px;
+        font-weight: 500;
+        display: block;
+        overflow: hidden;
+        text-overflow: ellipsis;
+        white-space: nowrap;
+
+        &:hover,
+        &:focus,
+        &:active {
+          span {
+            text-decoration: underline;
+          }
+        }
+      }
+    }
+
+    &__current {
+      flex: 0 0 auto;
+      font-size: 24px;
+      font-weight: 500;
+      text-align: end;
+      color: $secondary-text-color;
+      text-decoration: none;
+    }
+
+    &__sparkline {
+      flex: 0 0 auto;
+      width: 50px;
+
+      path:first-child {
+        fill: rgba($highlight-text-color, 0.25) !important;
+        fill-opacity: 1 !important;
+      }
+
+      path:last-child {
+        stroke: lighten($highlight-text-color, 6%) !important;
+        fill: none !important;
+      }
+    }
+
+    &--requires-review {
+      .trends__item__name {
+        color: $gold-star;
+
+        a {
+          color: $gold-star;
+        }
+      }
+
+      .trends__item__current {
+        color: $gold-star;
+      }
+
+      .trends__item__sparkline {
+        path:first-child {
+          fill: rgba($gold-star, 0.25) !important;
+        }
+
+        path:last-child {
+          stroke: lighten($gold-star, 6%) !important;
+        }
+      }
+    }
+
+    &--disabled {
+      .trends__item__name {
+        color: lighten($ui-base-color, 12%);
+
+        a {
+          color: lighten($ui-base-color, 12%);
+        }
+      }
+
+      .trends__item__current {
+        color: lighten($ui-base-color, 12%);
+      }
+
+      .trends__item__sparkline {
+        path:first-child {
+          fill: rgba(lighten($ui-base-color, 12%), 0.25) !important;
+        }
+
+        path:last-child {
+          stroke: lighten(lighten($ui-base-color, 12%), 6%) !important;
+        }
+      }
+    }
+  }
+
+  &--compact &__item {
+    padding: 10px;
+    padding-inline-end: 28px;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/sensitive.scss b/app/javascript/flavours/blobfox/styles/components/sensitive.scss
new file mode 100644
index 00000000000000..c77515eb70fd18
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/sensitive.scss
@@ -0,0 +1,26 @@
+.sensitive-info {
+  display: flex;
+  flex-direction: row;
+  align-items: center;
+  position: absolute;
+  top: 4px;
+  inset-inline-start: 4px;
+  z-index: 100;
+}
+
+.sensitive-marker {
+  margin: 0 3px;
+  border-radius: 2px;
+  padding: 2px 6px;
+  color: rgba($primary-text-color, 0.8);
+  background: rgba($base-overlay-background, 0.5);
+  font-size: 12px;
+  line-height: 18px;
+  text-transform: uppercase;
+  opacity: 0.9;
+  transition: opacity 0.1s ease;
+
+  .media-gallery:hover & {
+    opacity: 1;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/signed_out.scss b/app/javascript/flavours/blobfox/styles/components/signed_out.scss
new file mode 100644
index 00000000000000..18492983e5887b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/signed_out.scss
@@ -0,0 +1,110 @@
+.sign-in-banner {
+  padding: 10px;
+
+  p {
+    color: $darker-text-color;
+    margin-bottom: 20px;
+
+    a {
+      color: $secondary-text-color;
+      text-decoration: none;
+      unicode-bidi: isolate;
+
+      &:hover {
+        text-decoration: underline;
+
+        .fa {
+          color: lighten($dark-text-color, 7%);
+        }
+      }
+    }
+  }
+
+  .button {
+    margin-bottom: 10px;
+  }
+}
+
+.server-banner {
+  padding: 20px 0;
+
+  &__introduction {
+    color: $darker-text-color;
+    margin-bottom: 20px;
+
+    strong {
+      font-weight: 600;
+    }
+
+    a {
+      color: inherit;
+      text-decoration: underline;
+
+      &:hover,
+      &:active,
+      &:focus {
+        text-decoration: none;
+      }
+    }
+  }
+
+  &__hero {
+    display: block;
+    border-radius: 4px;
+    width: 100%;
+    height: auto;
+    margin-bottom: 20px;
+    aspect-ratio: 1.9;
+    border: 0;
+    background: $ui-base-color;
+    object-fit: cover;
+  }
+
+  &__description {
+    margin-bottom: 20px;
+  }
+
+  &__meta {
+    display: flex;
+    gap: 10px;
+    max-width: 100%;
+
+    &__column {
+      flex: 0 0 auto;
+      width: calc(50% - 5px);
+      overflow: hidden;
+    }
+  }
+
+  &__number {
+    font-weight: 600;
+    color: $primary-text-color;
+    font-size: 14px;
+  }
+
+  &__number-label {
+    color: $darker-text-color;
+    font-weight: 500;
+    font-size: 14px;
+  }
+
+  h4 {
+    text-transform: uppercase;
+    color: $darker-text-color;
+    margin-bottom: 10px;
+    font-weight: 600;
+  }
+
+  .account {
+    padding: 0;
+    border: 0;
+  }
+
+  .account__avatar-wrapper {
+    margin-inline-start: 0;
+  }
+
+  .spacer {
+    margin: 10px 0;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/single_column.scss b/app/javascript/flavours/blobfox/styles/components/single_column.scss
new file mode 100644
index 00000000000000..87fdd170d3b7e7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/single_column.scss
@@ -0,0 +1,335 @@
+.compose-panel {
+  width: 285px;
+  margin-top: 10px;
+  display: flex;
+  flex-direction: column;
+  height: calc(100% - 10px);
+  overflow-y: hidden;
+
+  .hero-widget {
+    box-shadow: none;
+
+    &__text,
+    &__img,
+    &__img img {
+      border-radius: 0;
+    }
+
+    &__text {
+      padding: 15px;
+      color: $secondary-text-color;
+
+      strong {
+        font-weight: 700;
+        color: $primary-text-color;
+      }
+    }
+  }
+
+  .search__input {
+    line-height: 18px;
+    font-size: 16px;
+    padding: 15px;
+    padding-inline-end: 30px;
+  }
+
+  .search__icon .fa {
+    top: 15px;
+  }
+
+  .navigation-bar {
+    flex: 0 1 48px;
+  }
+
+  .compose-form {
+    flex: 1;
+    display: flex;
+    flex-direction: column;
+    min-height: 310px;
+  }
+
+  .compose-form__autosuggest-wrapper {
+    overflow-y: auto;
+    background-color: $white;
+    border-radius: 4px 4px 0 0;
+    flex: 0 1 auto;
+  }
+
+  .autosuggest-textarea__textarea {
+    overflow-y: hidden;
+  }
+}
+
+.navigation-panel {
+  margin-top: 10px;
+  margin-bottom: 10px;
+  height: calc(100% - 20px);
+  overflow-y: auto;
+  display: flex;
+  flex-direction: column;
+
+  & > a {
+    flex: 0 0 auto;
+  }
+
+  .logo {
+    height: 30px;
+    width: auto;
+  }
+}
+
+.navigation-panel,
+.compose-panel {
+  hr {
+    flex: 0 0 auto;
+    border: 0;
+    background: transparent;
+    border-top: 1px solid lighten($ui-base-color, 4%);
+    margin: 10px 0;
+  }
+
+  .flex-spacer {
+    background: transparent;
+  }
+}
+
+@media screen and (width >= 600px) {
+  .tabs-bar__link {
+    span {
+      display: inline;
+    }
+  }
+}
+
+.columns-area--mobile {
+  flex-direction: column;
+  width: 100%;
+  margin: 0 auto;
+
+  .column,
+  .drawer {
+    width: 100%;
+    height: 100%;
+    padding: 0;
+  }
+
+  .account-card {
+    margin-bottom: 0;
+  }
+
+  .filter-form {
+    display: flex;
+    flex-wrap: wrap;
+  }
+
+  .autosuggest-textarea__textarea {
+    font-size: 16px;
+  }
+
+  .search__input {
+    line-height: 18px;
+    font-size: 16px;
+    padding: 15px;
+    padding-inline-end: 30px;
+  }
+
+  .search__icon .fa {
+    top: 15px;
+  }
+
+  .scrollable {
+    overflow: visible;
+
+    @supports (display: grid) {
+      contain: content;
+    }
+  }
+
+  @media screen and (min-width: $no-gap-breakpoint) {
+    padding: 10px 0;
+    padding-top: 0;
+  }
+
+  .detailed-status {
+    padding: 15px;
+
+    .media-gallery,
+    .video-player,
+    .audio-player {
+      margin-top: 15px;
+    }
+  }
+
+  .account__header__bar {
+    padding: 5px 10px;
+  }
+
+  .navigation-bar,
+  .compose-form {
+    padding: 15px;
+  }
+
+  .compose-form .compose-form__publish .compose-form__publish-button-wrapper {
+    padding-top: 15px;
+  }
+
+  .notification__report {
+    padding: 15px;
+    padding-inline-start: (48px + 15px * 2);
+    min-height: 48px + 2px;
+
+    &__avatar {
+      inset-inline-start: 15px;
+      top: 17px;
+    }
+  }
+
+  .status {
+    padding: 15px;
+    min-height: 48px + 2px;
+
+    .media-gallery,
+    &__action-bar,
+    .video-player,
+    .audio-player {
+      margin-top: 10px;
+    }
+  }
+
+  .account {
+    padding: 15px 10px;
+
+    &__header__bio {
+      margin: 0 -10px;
+    }
+  }
+
+  .notification {
+    &__message {
+      padding-top: 15px;
+    }
+
+    .status {
+      padding-top: 8px;
+    }
+
+    .account {
+      padding-top: 8px;
+    }
+  }
+}
+
+@media screen and (min-width: $no-gap-breakpoint) {
+  .tabs-bar {
+    width: 100%;
+  }
+
+  .react-swipeable-view-container .columns-area--mobile {
+    height: calc(100% - 10px) !important;
+  }
+
+  .getting-started__wrapper {
+    margin-bottom: 10px;
+  }
+
+  .tabs-bar__link.optional {
+    display: none;
+  }
+
+  .search-page .search {
+    display: none;
+  }
+
+  .navigation-panel__legal {
+    display: none;
+  }
+}
+
+@media screen and (max-width: $no-gap-breakpoint - 1px) {
+  $sidebar-width: 285px;
+
+  .columns-area__panels__main {
+    width: calc(100% - $sidebar-width);
+  }
+
+  .columns-area__panels {
+    min-height: calc(100vh - $ui-header-height);
+  }
+
+  .columns-area__panels__pane--navigational {
+    min-width: $sidebar-width;
+
+    .columns-area__panels__pane__inner {
+      width: $sidebar-width;
+    }
+
+    .navigation-panel {
+      margin: 0;
+      background: $ui-base-color;
+      border-inline-start: 1px solid lighten($ui-base-color, 8%);
+      height: 100vh;
+    }
+
+    .navigation-panel__sign-in-banner,
+    .navigation-panel__logo,
+    .navigation-panel__banner,
+    .getting-started__trends {
+      display: none;
+    }
+
+    .column-link__icon {
+      font-size: 18px;
+    }
+  }
+
+  .layout-single-column .ui__header {
+    display: flex;
+    background: $ui-base-color;
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+  }
+
+  .column-header,
+  .column-back-button,
+  .scrollable,
+  .error-column {
+    border-radius: 0 !important;
+  }
+}
+
+@media screen and (max-width: $no-gap-breakpoint - 285px - 1px) {
+  $sidebar-width: 55px;
+
+  .columns-area__panels__main {
+    width: calc(100% - $sidebar-width);
+  }
+
+  .columns-area__panels__pane--navigational {
+    min-width: $sidebar-width;
+
+    .columns-area__panels__pane__inner {
+      width: $sidebar-width;
+    }
+
+    .column-link span {
+      display: none;
+    }
+
+    .list-panel {
+      display: none;
+    }
+  }
+}
+
+.explore__search-header {
+  display: none;
+}
+
+@media screen and (max-width: $no-gap-breakpoint - 1px) {
+  .columns-area__panels__pane--compositional {
+    display: none;
+  }
+
+  .explore__search-header {
+    display: flex;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/components/status.scss b/app/javascript/flavours/blobfox/styles/components/status.scss
new file mode 100644
index 00000000000000..cb7d72e5ccf42a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/components/status.scss
@@ -0,0 +1,1212 @@
+@keyframes spring-flip-in {
+  0% {
+    transform: rotate(0deg);
+  }
+
+  30% {
+    transform: rotate(-242.4deg);
+  }
+
+  60% {
+    transform: rotate(-158.35deg);
+  }
+
+  90% {
+    transform: rotate(-187.5deg);
+  }
+
+  100% {
+    transform: rotate(-180deg);
+  }
+}
+
+@keyframes spring-flip-out {
+  0% {
+    transform: rotate(-180deg);
+  }
+
+  30% {
+    transform: rotate(62.4deg);
+  }
+
+  60% {
+    transform: rotate(-21.635deg);
+  }
+
+  90% {
+    transform: rotate(7.5deg);
+  }
+
+  100% {
+    transform: rotate(0deg);
+  }
+}
+
+.status__content--with-action {
+  cursor: pointer;
+}
+
+.status__content {
+  position: relative;
+  margin: 10px 0;
+  font-size: 15px;
+  line-height: 20px;
+  word-wrap: break-word;
+  font-weight: 400;
+  overflow: visible;
+  padding-top: 5px;
+  clear: both;
+
+  &:focus {
+    outline: 0;
+  }
+
+  .emojione {
+    width: 20px;
+    height: 20px;
+    margin: -3px 0 0;
+  }
+
+  p,
+  pre {
+    margin-bottom: 20px;
+    white-space: pre-wrap;
+    unicode-bidi: plaintext;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  a {
+    color: $secondary-text-color;
+    text-decoration: none;
+    unicode-bidi: isolate;
+
+    &:hover {
+      text-decoration: underline;
+
+      .fa {
+        color: lighten($dark-text-color, 7%);
+      }
+    }
+
+    &.mention {
+      &:hover {
+        text-decoration: none;
+
+        span {
+          text-decoration: underline;
+        }
+      }
+    }
+
+    .fa {
+      color: $dark-text-color;
+    }
+  }
+
+  .status__content__spoiler {
+    display: none;
+
+    &.status__content__spoiler--visible {
+      display: block;
+    }
+  }
+
+  a.unhandled-link {
+    color: $highlight-text-color;
+
+    .link-origin-tag {
+      color: $gold-star;
+      font-size: 0.8em;
+    }
+  }
+
+  .status__content__spoiler-link {
+    background: lighten($ui-base-color, 30%);
+
+    &:hover,
+    &:focus {
+      background: lighten($ui-base-color, 33%);
+      text-decoration: none;
+    }
+  }
+}
+
+.translate-button {
+  margin-top: 16px;
+  font-size: 15px;
+  line-height: 20px;
+  display: flex;
+  justify-content: space-between;
+  color: $dark-text-color;
+}
+
+.status__content__spoiler-link {
+  display: inline-block;
+  border-radius: 2px;
+  background: lighten($ui-base-color, 30%);
+  border: 0;
+  color: $inverted-text-color;
+  font-weight: 700;
+  font-size: 11px;
+  padding: 0 5px;
+  text-transform: uppercase;
+  line-height: inherit;
+  cursor: pointer;
+  vertical-align: top;
+
+  &:hover {
+    background: lighten($ui-base-color, 33%);
+    text-decoration: none;
+  }
+
+  .status__content__spoiler-icon {
+    display: inline-block;
+    margin-inline-start: 5px;
+    border-inline-start: 1px solid currentColor;
+    padding: 0;
+    padding-inline-start: 4px;
+    font-size: 16px;
+    vertical-align: -2px;
+  }
+}
+
+.notif-cleaning {
+  .status,
+  .notification {
+    padding-inline-end: ($dismiss-overlay-width + 0.5rem);
+  }
+}
+
+.status__wrapper--filtered {
+  color: $dark-text-color;
+  border: 0;
+  font-size: inherit;
+  text-align: center;
+  line-height: inherit;
+  margin: 0;
+  padding: 15px;
+  box-sizing: border-box;
+  width: 100%;
+  clear: both;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+}
+
+.status__prepend-icon-wrapper {
+  inset-inline-start: -26px;
+  position: absolute;
+}
+
+.notification-follow,
+.notification-follow-request {
+  position: relative;
+
+  // same like Status
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+
+  .account {
+    border-bottom: 0 none;
+  }
+}
+
+.focusable {
+  &:focus {
+    outline: 0;
+    background: lighten($ui-base-color, 4%);
+
+    &.status.status-direct {
+      background: mix(lighten($ui-base-color, 4%), $ui-highlight-color, 95%);
+
+      &.muted {
+        background: transparent;
+      }
+    }
+
+    .detailed-status,
+    .detailed-status__action-bar {
+      background: lighten($ui-base-color, 8%);
+    }
+  }
+}
+
+.status {
+  padding: 10px 14px;
+  position: relative;
+  height: auto;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  cursor: auto;
+
+  @supports (-ms-overflow-style: -ms-autohiding-scrollbar) {
+    // Add margin to avoid Edge auto-hiding scrollbar appearing over content.
+    // On Edge 16 this is 16px and Edge <=15 it's 12px, so aim for 16px.
+    padding-inline-end: 28px; // 12px + 16px
+  }
+
+  @keyframes fade {
+    0% {
+      opacity: 0;
+    }
+
+    100% {
+      opacity: 1;
+    }
+  }
+
+  opacity: 1;
+  animation: fade 150ms linear;
+
+  .video-player,
+  .audio-player {
+    margin-top: 8px;
+  }
+
+  &.status-direct {
+    background: mix($ui-base-color, $ui-highlight-color, 95%);
+    border-bottom-color: lighten($ui-base-color, 12%);
+  }
+
+  &.light {
+    .status__relative-time {
+      color: $lighter-text-color;
+    }
+
+    .status__display-name {
+      color: $inverted-text-color;
+    }
+
+    .display-name {
+      color: $light-text-color;
+
+      strong {
+        color: $inverted-text-color;
+      }
+    }
+
+    .status__content {
+      color: $inverted-text-color;
+
+      a {
+        color: $highlight-text-color;
+      }
+
+      a.status__content__spoiler-link {
+        color: $primary-text-color;
+        background: $ui-primary-color;
+
+        &:hover {
+          background: lighten($ui-primary-color, 8%);
+        }
+      }
+    }
+  }
+
+  &.collapsed {
+    background-position: center;
+    background-size: cover;
+    user-select: none;
+
+    &.has-background::before {
+      display: block;
+      position: absolute;
+      inset-inline-start: 0;
+      inset-inline-end: 0;
+      top: 0;
+      bottom: 0;
+      background-image: linear-gradient(
+        to bottom,
+        rgba($base-shadow-color, 0.75),
+        rgba($base-shadow-color, 0.65) 24px,
+        rgba($base-shadow-color, 0.8)
+      );
+      pointer-events: none;
+      content: '';
+    }
+
+    .display-name:hover .display-name__html {
+      text-decoration: none;
+    }
+
+    .status__content {
+      height: 20px;
+      overflow: hidden;
+      text-overflow: ellipsis;
+      padding-top: 0;
+
+      &::after {
+        content: '';
+        position: absolute;
+        top: 0;
+        bottom: 0;
+        inset-inline-start: 0;
+        inset-inline-end: 0;
+        background: linear-gradient(
+          rgba($ui-base-color, 0),
+          rgba($ui-base-color, 1)
+        );
+        pointer-events: none;
+      }
+
+      a:hover {
+        text-decoration: none;
+      }
+    }
+
+    &:focus > .status__content::after {
+      background: linear-gradient(
+        rgba(lighten($ui-base-color, 4%), 0),
+        rgba(lighten($ui-base-color, 4%), 1)
+      );
+    }
+
+    &.status-direct > .status__content::after {
+      background: linear-gradient(
+        rgba(mix($ui-base-color, $ui-highlight-color, 95%), 0),
+        rgba(mix($ui-base-color, $ui-highlight-color, 95%), 1)
+      );
+    }
+
+    .notification__message {
+      margin-bottom: 0;
+    }
+
+    .status__info .notification__message > span {
+      white-space: nowrap;
+    }
+  }
+
+  .notification__message {
+    margin: -10px 0 10px;
+  }
+
+  .reactions-bar--empty {
+    display: none;
+  }
+}
+
+.notification-favourite {
+  .status.status-direct {
+    background: transparent;
+
+    .icon-button.disabled {
+      color: lighten($action-button-color, 13%);
+    }
+  }
+}
+
+.status__relative-time {
+  display: inline-block;
+  color: $dark-text-color;
+  font-size: 14px;
+  text-align: end;
+  white-space: nowrap;
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
+
+.status__display-name {
+  color: $dark-text-color;
+  overflow: hidden;
+}
+
+.status__info__account .status__display-name {
+  display: block;
+  max-width: 100%;
+}
+
+.status__info {
+  display: flex;
+  justify-content: space-between;
+  font-size: 15px;
+
+  > span {
+    text-overflow: ellipsis;
+    overflow: hidden;
+  }
+
+  .notification__message > span {
+    word-wrap: break-word;
+  }
+}
+
+.status__info__icons {
+  display: flex;
+  align-items: center;
+  height: 1em;
+  color: $action-button-color;
+
+  .status__media-icon,
+  .status__visibility-icon,
+  .status__reply-icon,
+  .text-icon {
+    padding-inline-start: 2px;
+    padding-inline-end: 2px;
+  }
+
+  .status__collapse-button.active > .fa-angle-double-up {
+    transform: rotate(-180deg);
+  }
+}
+
+.no-reduce-motion .status__collapse-button {
+  &.activate {
+    & > .fa-angle-double-up {
+      animation: spring-flip-in 1s linear;
+    }
+  }
+
+  &.deactivate {
+    & > .fa-angle-double-up {
+      animation: spring-flip-out 1s linear;
+    }
+  }
+}
+
+.status__info__account {
+  display: flex;
+  align-items: center;
+  justify-content: flex-start;
+}
+
+.status-check-box__status {
+  display: block;
+  box-sizing: border-box;
+  width: 100%;
+  padding: 0 10px;
+
+  .detailed-status__display-name {
+    color: lighten($inverted-text-color, 16%);
+
+    span {
+      display: inline;
+    }
+
+    &:hover strong {
+      text-decoration: none;
+    }
+  }
+
+  .media-gallery,
+  .audio-player,
+  .video-player {
+    margin-top: 15px;
+    max-width: 250px;
+  }
+
+  .status__content {
+    padding: 0;
+    white-space: normal;
+  }
+
+  .media-gallery__item-thumbnail {
+    cursor: default;
+  }
+}
+
+.status__prepend {
+  margin-top: -2px;
+  margin-bottom: 8px;
+  margin-inline-start: 58px;
+  color: $dark-text-color;
+  font-size: 14px;
+  position: relative;
+
+  .status__display-name strong {
+    color: $dark-text-color;
+  }
+
+  > span {
+    display: block;
+    overflow: hidden;
+    text-overflow: ellipsis;
+  }
+}
+
+.status__action-bar {
+  align-items: center;
+  display: flex;
+  margin-top: 8px;
+
+  & > .emoji-picker-dropdown > .emoji-button {
+    padding: 0;
+  }
+}
+
+.status__action-bar-button {
+  margin-inline-end: 18px;
+
+  &.icon-button--with-counter {
+    margin-inline-end: 14px;
+  }
+
+  .fa-plus {
+    padding-top: 1px;
+  }
+}
+
+.status__action-bar-dropdown {
+  height: 23.15px;
+  width: 23.15px;
+}
+
+.status__action-bar-spacer {
+  flex-grow: 1;
+}
+
+.detailed-status__action-bar-dropdown {
+  flex: 1 1 auto;
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  position: relative;
+}
+
+.detailed-status {
+  background: lighten($ui-base-color, 4%);
+  padding: 14px 10px;
+  border-top: 1px solid lighten($ui-base-color, 8%);
+
+  &--flex {
+    display: flex;
+    flex-wrap: wrap;
+    justify-content: space-between;
+    align-items: flex-start;
+
+    .status__content,
+    .detailed-status__meta {
+      flex: 100%;
+    }
+  }
+
+  .status__content {
+    font-size: 19px;
+    line-height: 24px;
+
+    .emojione {
+      width: 24px;
+      height: 24px;
+      margin: -1px 0 0;
+    }
+  }
+
+  .video-player,
+  .audio-player {
+    margin-top: 8px;
+  }
+}
+
+.detailed-status__meta {
+  margin-top: 15px;
+  color: $dark-text-color;
+  font-size: 14px;
+  line-height: 18px;
+}
+
+.detailed-status__action-bar {
+  background: lighten($ui-base-color, 4%);
+  border-top: 1px solid lighten($ui-base-color, 8%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+  display: flex;
+  flex-direction: row;
+  padding: 10px 0;
+
+  .fa-plus {
+    padding-top: 2px;
+  }
+}
+
+.detailed-status__link {
+  color: inherit;
+  text-decoration: none;
+  white-space: nowrap;
+}
+
+.detailed-status__favorites,
+.detailed-status__reblogs {
+  display: inline-block;
+  font-weight: 500;
+  font-size: 12px;
+  line-height: 17px;
+  margin-inline-start: 6px;
+}
+
+.status__display-name,
+.status__relative-time,
+.detailed-status__display-name,
+.detailed-status__datetime,
+.detailed-status__application,
+.account__display-name {
+  text-decoration: none;
+}
+
+.status__display-name,
+.account__display-name {
+  .display-name strong {
+    color: $primary-text-color;
+  }
+}
+
+.muted {
+  .emojione {
+    opacity: 0.5;
+  }
+}
+
+a.status__display-name,
+.reply-indicator__display-name,
+.detailed-status__display-name,
+.account__display-name {
+  &:hover .display-name strong {
+    text-decoration: underline;
+  }
+}
+
+.account__display-name .display-name strong {
+  display: block;
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
+
+.detailed-status__application,
+.detailed-status__datetime {
+  color: inherit;
+}
+
+.detailed-status__display-name {
+  color: $secondary-text-color;
+  display: block;
+  line-height: 24px;
+  margin-bottom: 15px;
+  overflow: hidden;
+
+  strong,
+  span {
+    display: block;
+    text-overflow: ellipsis;
+    overflow: hidden;
+  }
+
+  strong {
+    font-size: 16px;
+    color: $primary-text-color;
+  }
+}
+
+.detailed-status__display-avatar {
+  float: left;
+  margin-inline-end: 10px;
+}
+
+.status__avatar {
+  flex: none;
+  margin-inline-end: 10px;
+}
+
+.muted {
+  .status__content,
+  .status__content p,
+  .status__content a,
+  .status__content__text {
+    color: $dark-text-color;
+  }
+
+  .status__display-name strong {
+    color: $dark-text-color;
+  }
+
+  .status__avatar {
+    opacity: 0.5;
+  }
+
+  a.status__content__spoiler-link {
+    background: $ui-base-lighter-color;
+    color: $inverted-text-color;
+
+    &:hover,
+    &:focus {
+      background: lighten($ui-base-color, 29%);
+      text-decoration: none;
+    }
+  }
+}
+
+.status__relative-time,
+.detailed-status__datetime {
+  &:hover {
+    text-decoration: underline;
+  }
+}
+
+.status-card {
+  position: relative;
+  display: flex;
+  font-size: 14px;
+  border: 1px solid lighten($ui-base-color, 8%);
+  border-radius: 4px;
+  color: $dark-text-color;
+  margin-top: 14px;
+  text-decoration: none;
+  overflow: hidden;
+
+  &__actions {
+    bottom: 0;
+    inset-inline-start: 0;
+    position: absolute;
+    inset-inline-end: 0;
+    top: 0;
+    display: flex;
+    justify-content: center;
+    align-items: center;
+
+    & > div {
+      background: rgba($base-shadow-color, 0.6);
+      border-radius: 8px;
+      padding: 12px 9px;
+      flex: 0 0 auto;
+      display: flex;
+      justify-content: center;
+      align-items: center;
+    }
+
+    button,
+    a {
+      display: inline;
+      color: $secondary-text-color;
+      background: transparent;
+      border: 0;
+      padding: 0 8px;
+      text-decoration: none;
+      font-size: 18px;
+      line-height: 18px;
+
+      &:hover,
+      &:active,
+      &:focus {
+        color: $primary-text-color;
+      }
+    }
+
+    a {
+      font-size: 19px;
+      position: relative;
+      bottom: -1px;
+    }
+
+    a .fa,
+    a:hover .fa {
+      color: inherit;
+    }
+  }
+}
+
+a.status-card {
+  cursor: pointer;
+
+  &:hover {
+    background: lighten($ui-base-color, 8%);
+  }
+}
+
+.status-card-photo {
+  cursor: zoom-in;
+  display: block;
+  text-decoration: none;
+  width: 100%;
+  height: auto;
+  margin: 0;
+}
+
+.status-card-video {
+  // Firefox has a bug where frameborder=0 iframes add some extra blank space
+  // see https://bugzilla.mozilla.org/show_bug.cgi?id=155174
+  overflow: hidden;
+
+  iframe {
+    width: 100%;
+    height: 100%;
+  }
+}
+
+.status-card__title {
+  display: block;
+  font-weight: 500;
+  margin-bottom: 5px;
+  color: $darker-text-color;
+  overflow: hidden;
+  text-overflow: ellipsis;
+  white-space: nowrap;
+  text-decoration: none;
+}
+
+.status-card__content {
+  flex: 1 1 auto;
+  overflow: hidden;
+  padding: 14px;
+  padding-inline-start: 8px;
+}
+
+.status-card__description {
+  color: $darker-text-color;
+  overflow: hidden;
+  display: -webkit-box;
+  -webkit-box-orient: vertical;
+  -webkit-line-clamp: 2;
+}
+
+.status-card__host {
+  display: block;
+  margin-top: 5px;
+  font-size: 13px;
+  overflow: hidden;
+  text-overflow: ellipsis;
+  white-space: nowrap;
+}
+
+.status-card__image {
+  flex: 0 0 100px;
+  background: lighten($ui-base-color, 8%);
+  position: relative;
+
+  & > .fa {
+    font-size: 21px;
+    position: absolute;
+    transform-origin: 50% 50%;
+    top: 50%;
+    inset-inline-start: 50%;
+    transform: translate(-50%, -50%);
+  }
+}
+
+.status-card.horizontal {
+  display: block;
+
+  .status-card__image {
+    width: 100%;
+  }
+
+  .status-card__image-image,
+  .status-card__image-preview {
+    border-radius: 4px 4px 0 0;
+  }
+
+  .status-card__title {
+    white-space: inherit;
+  }
+}
+
+.status-card.compact {
+  border-color: lighten($ui-base-color, 4%);
+
+  &.interactive {
+    border: 0;
+  }
+
+  .status-card__content {
+    padding: 8px;
+    padding-top: 10px;
+  }
+
+  .status-card__title {
+    white-space: nowrap;
+  }
+
+  .status-card__image {
+    flex: 0 0 60px;
+  }
+}
+
+a.status-card.compact:hover {
+  background-color: lighten($ui-base-color, 4%);
+}
+
+.status-card__image-image {
+  border-radius: 4px 0 0 4px;
+  display: block;
+  margin: 0;
+  width: 100%;
+  height: 100%;
+  object-fit: cover;
+  background-size: cover;
+  background-position: center center;
+}
+
+.status-card__image-preview {
+  border-radius: 4px 0 0 4px;
+  display: block;
+  margin: 0;
+  width: 100%;
+  height: 100%;
+  object-fit: fill;
+  position: absolute;
+  top: 0;
+  inset-inline-start: 0;
+  z-index: 0;
+  background: $base-overlay-background;
+
+  &--hidden {
+    display: none;
+  }
+}
+
+.attachment-list {
+  display: flex;
+  font-size: 14px;
+  border: 1px solid lighten($ui-base-color, 8%);
+  border-radius: 4px;
+  margin-top: 14px;
+  overflow: hidden;
+
+  &__icon {
+    flex: 0 0 auto;
+    color: $dark-text-color;
+    padding: 8px 18px;
+    cursor: default;
+    border-inline-end: 1px solid lighten($ui-base-color, 8%);
+    display: flex;
+    flex-direction: column;
+    align-items: center;
+    justify-content: center;
+    font-size: 26px;
+
+    .fa {
+      display: block;
+    }
+  }
+
+  &__list {
+    list-style: none;
+    padding: 4px 0;
+    padding-inline-start: 8px;
+    display: flex;
+    flex-direction: column;
+    justify-content: center;
+
+    li {
+      display: block;
+      padding: 4px 0;
+    }
+
+    a {
+      text-decoration: none;
+      color: $dark-text-color;
+      font-weight: 500;
+
+      &:hover {
+        text-decoration: underline;
+      }
+    }
+  }
+
+  &.compact {
+    border: 0;
+    margin-top: 4px;
+
+    .attachment-list__list {
+      padding: 0;
+      display: block;
+    }
+
+    .fa {
+      color: $dark-text-color;
+    }
+  }
+}
+
+.status__wrapper--filtered__button {
+  display: inline;
+  color: lighten($ui-highlight-color, 8%);
+  border: 0;
+  background: transparent;
+  padding: 0;
+  font-size: inherit;
+  line-height: inherit;
+
+  &:hover,
+  &:active {
+    text-decoration: underline;
+  }
+}
+
+.notification,
+.status {
+  position: relative;
+
+  &.unread {
+    &::before {
+      content: '';
+      position: absolute;
+      top: 0;
+      inset-inline-start: 0;
+      width: 100%;
+      height: 100%;
+      border-inline-start: 4px solid $highlight-text-color;
+      pointer-events: none;
+    }
+  }
+
+  &--in-thread {
+    border-bottom: 0;
+
+    .status__content,
+    .status__action-bar,
+    .reactions-bar {
+      margin-inline-start: 46px + 10px;
+      width: calc(100% - (46px + 10px));
+    }
+  }
+
+  &--first-in-thread {
+    border-top: 1px solid lighten($ui-base-color, 8%);
+  }
+
+  &__line {
+    height: 10px - 4px;
+    border-inline-start: 2px solid lighten($ui-base-color, 8%);
+    width: 0;
+    position: absolute;
+    top: 0;
+    inset-inline-start: 14px + ((46px - 2px) * 0.5);
+
+    &--full {
+      top: 0;
+      height: 100%;
+
+      &::before {
+        content: '';
+        display: block;
+        position: absolute;
+        top: 10px - 4px;
+        height: 46px + 4px + 4px;
+        width: 2px;
+        background: $ui-base-color;
+        inset-inline-start: -2px;
+      }
+    }
+
+    &--first {
+      top: 10px + 46px + 4px;
+      height: calc(100% - (10px + 46px + 4px));
+
+      &::before {
+        display: none;
+      }
+    }
+  }
+}
+
+.picture-in-picture {
+  position: fixed;
+  bottom: 20px;
+  inset-inline-end: 20px;
+  width: 300px;
+
+  &.left {
+    inset-inline-end: unset;
+    inset-inline-start: 20px;
+  }
+
+  &__footer {
+    border-radius: 0 0 4px 4px;
+    background: lighten($ui-base-color, 4%);
+    padding: 10px;
+    padding-top: 12px;
+    display: flex;
+    justify-content: space-between;
+  }
+
+  &__header {
+    border-radius: 4px 4px 0 0;
+    background: lighten($ui-base-color, 4%);
+    padding: 10px;
+    display: flex;
+    justify-content: space-between;
+
+    &__account {
+      display: flex;
+      text-decoration: none;
+      overflow: hidden;
+    }
+
+    .account__avatar {
+      margin-inline-end: 10px;
+    }
+
+    .display-name {
+      color: $primary-text-color;
+      text-decoration: none;
+
+      strong,
+      span {
+        display: block;
+        text-overflow: ellipsis;
+        overflow: hidden;
+      }
+
+      span {
+        color: $darker-text-color;
+      }
+    }
+  }
+
+  .video-player,
+  .audio-player {
+    border-radius: 0;
+  }
+}
+
+.picture-in-picture-placeholder {
+  box-sizing: border-box;
+  border: 2px dashed lighten($ui-base-color, 8%);
+  background: $base-shadow-color;
+  display: flex;
+  flex-direction: column;
+  align-items: center;
+  justify-content: center;
+  margin-top: 10px;
+  font-size: 16px;
+  font-weight: 500;
+  cursor: pointer;
+  color: $darker-text-color;
+  aspect-ratio: 16 / 9;
+
+  i {
+    display: block;
+    font-size: 24px;
+    font-weight: 400;
+    margin-bottom: 10px;
+  }
+
+  &:hover,
+  &:focus,
+  &:active {
+    border-color: lighten($ui-base-color, 12%);
+  }
+}
+
+.hashtag-bar {
+  margin-top: 16px;
+  display: flex;
+  flex-wrap: wrap;
+  font-size: 14px;
+  line-height: 18px;
+  gap: 4px;
+  color: $darker-text-color;
+
+  a {
+    display: inline-flex;
+    color: inherit;
+    text-decoration: none;
+
+    &:hover span {
+      text-decoration: underline;
+    }
+  }
+
+  .link-button {
+    color: inherit;
+    font-size: inherit;
+    line-height: inherit;
+    padding: 0;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/containers.scss b/app/javascript/flavours/blobfox/styles/containers.scss
new file mode 100644
index 00000000000000..4d3d4c546c1ffe
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/containers.scss
@@ -0,0 +1,109 @@
+.container-alt {
+  width: 700px;
+  margin: 0 auto;
+
+  @media screen and (width <= 740px) {
+    width: 100%;
+    margin: 0;
+  }
+}
+
+.logo-container {
+  margin: 50px auto;
+
+  h1 {
+    display: flex;
+    justify-content: center;
+    align-items: center;
+
+    .logo {
+      height: 42px;
+      margin-inline-end: 10px;
+    }
+
+    a {
+      display: flex;
+      justify-content: center;
+      align-items: center;
+      color: $primary-text-color;
+      text-decoration: none;
+      outline: 0;
+      padding: 12px 16px;
+      line-height: 32px;
+      font-weight: 500;
+      font-size: 14px;
+    }
+  }
+}
+
+.compose-standalone {
+  .compose-form {
+    width: 400px;
+    margin: 0 auto;
+    padding: 20px 0;
+    margin-top: 40px;
+    box-sizing: border-box;
+
+    @media screen and (width <= 400px) {
+      width: 100%;
+      margin-top: 0;
+      padding: 20px;
+    }
+  }
+}
+
+.account-header {
+  width: 400px;
+  margin: 0 auto;
+  display: flex;
+  font-size: 13px;
+  line-height: 18px;
+  box-sizing: border-box;
+  padding: 20px 0;
+  margin-top: 40px;
+  margin-bottom: 10px;
+  border-bottom: 1px solid $ui-base-color;
+
+  @media screen and (width <= 440px) {
+    width: 100%;
+    margin: 0;
+    padding: 20px;
+  }
+
+  .avatar {
+    width: 40px;
+    height: 40px;
+    @include avatar-size(40px);
+
+    margin-inline-end: 10px;
+
+    img {
+      width: 100%;
+      height: 100%;
+      display: block;
+      margin: 0;
+      border-radius: 4px;
+      @include avatar-radius;
+    }
+  }
+
+  .name {
+    flex: 1 1 auto;
+    color: $secondary-text-color;
+    width: calc(100% - 90px);
+
+    .username {
+      display: block;
+      font-weight: 500;
+      text-overflow: ellipsis;
+      overflow: hidden;
+    }
+  }
+
+  .logout-link {
+    display: block;
+    font-size: 32px;
+    line-height: 40px;
+    margin-inline-start: 10px;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/contrast.scss b/app/javascript/flavours/blobfox/styles/contrast.scss
new file mode 100644
index 00000000000000..4de31db9aec577
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/contrast.scss
@@ -0,0 +1,3 @@
+@import 'contrast/variables';
+@import 'index';
+@import 'contrast/diff';
diff --git a/app/javascript/flavours/blobfox/styles/contrast/diff.scss b/app/javascript/flavours/blobfox/styles/contrast/diff.scss
new file mode 100644
index 00000000000000..1c2386f02d2df9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/contrast/diff.scss
@@ -0,0 +1,79 @@
+.compose-form {
+  .compose-form__modifiers {
+    .compose-form__upload {
+      &-description {
+        input {
+          &::placeholder {
+            opacity: 1;
+          }
+        }
+      }
+    }
+  }
+}
+
+.status__content a,
+.link-footer a,
+.reply-indicator__content a,
+.status__content__read-more-button,
+.status__content__translate-button {
+  text-decoration: underline;
+
+  &:hover,
+  &:focus,
+  &:active {
+    text-decoration: none;
+  }
+
+  &.mention {
+    text-decoration: none;
+
+    span {
+      text-decoration: underline;
+    }
+
+    &:hover,
+    &:focus,
+    &:active {
+      span {
+        text-decoration: none;
+      }
+    }
+  }
+}
+
+.status__content a {
+  color: $highlight-text-color;
+}
+
+.nothing-here {
+  color: $darker-text-color;
+}
+
+.compose-form__poll-wrapper .button.button-secondary,
+.compose-form .autosuggest-textarea__textarea::placeholder,
+.compose-form .spoiler-input__input::placeholder,
+.report-dialog-modal__textarea::placeholder,
+.language-dropdown__dropdown__results__item__common-name,
+.compose-form .icon-button {
+  color: $inverted-text-color;
+}
+
+.text-icon-button.active {
+  color: $ui-highlight-color;
+}
+
+.language-dropdown__dropdown__results__item.active {
+  background: $ui-highlight-color;
+  font-weight: 500;
+}
+
+.link-button:disabled {
+  cursor: not-allowed;
+
+  &:hover,
+  &:focus,
+  &:active {
+    text-decoration: none !important;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/contrast/variables.scss b/app/javascript/flavours/blobfox/styles/contrast/variables.scss
new file mode 100644
index 00000000000000..e38d24b271cf8e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/contrast/variables.scss
@@ -0,0 +1,22 @@
+// Dependent colors
+$black: #000000;
+
+$classic-base-color: #282c37;
+$classic-primary-color: #9baec8;
+$classic-secondary-color: #d9e1e8;
+$classic-highlight-color: #6364ff;
+
+$ui-base-color: $classic-base-color !default;
+$ui-primary-color: $classic-primary-color !default;
+$ui-secondary-color: $classic-secondary-color !default;
+$ui-highlight-color: $classic-highlight-color !default;
+
+$darker-text-color: lighten($ui-primary-color, 20%) !default;
+$dark-text-color: lighten($ui-primary-color, 12%) !default;
+$secondary-text-color: lighten($ui-secondary-color, 6%) !default;
+$highlight-text-color: lighten($ui-highlight-color, 10%) !default;
+$action-button-color: lighten($ui-base-color, 50%);
+
+$inverted-text-color: $black !default;
+$lighter-text-color: darken($ui-base-color, 6%) !default;
+$light-text-color: darken($ui-primary-color, 40%) !default;
diff --git a/app/javascript/flavours/blobfox/styles/dashboard.scss b/app/javascript/flavours/blobfox/styles/dashboard.scss
new file mode 100644
index 00000000000000..36a7f44253f423
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/dashboard.scss
@@ -0,0 +1,123 @@
+.dashboard__counters {
+  display: flex;
+  flex-wrap: wrap;
+  margin: 0 -5px;
+  margin-bottom: 20px;
+
+  & > div {
+    box-sizing: border-box;
+    flex: 0 0 33.333%;
+    padding: 0 5px;
+    margin-bottom: 10px;
+
+    & > div,
+    & > a {
+      padding: 20px;
+      background: lighten($ui-base-color, 4%);
+      border-radius: 4px;
+      box-sizing: border-box;
+      height: 100%;
+    }
+
+    & > a {
+      text-decoration: none;
+      color: inherit;
+      display: block;
+
+      &:hover,
+      &:focus,
+      &:active {
+        background: lighten($ui-base-color, 8%);
+      }
+    }
+  }
+
+  &__num,
+  &__text {
+    text-align: center;
+    font-weight: 500;
+    font-size: 24px;
+    color: $primary-text-color;
+    margin-bottom: 20px;
+    line-height: 30px;
+  }
+
+  &__text {
+    font-size: 18px;
+  }
+
+  &__label {
+    font-size: 14px;
+    color: $darker-text-color;
+    text-align: center;
+    font-weight: 500;
+  }
+}
+
+.dashboard {
+  display: grid;
+  grid-template-columns: minmax(0, 1fr) minmax(0, 1fr) minmax(0, 1fr);
+  grid-gap: 10px;
+
+  @media screen and (width <= 1350px) {
+    grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
+  }
+
+  &__item {
+    &--span-double-column {
+      grid-column: span 2;
+    }
+
+    &--span-double-row {
+      grid-row: span 2;
+    }
+
+    h4 {
+      padding-top: 20px;
+    }
+  }
+
+  &__quick-access {
+    display: flex;
+    align-items: baseline;
+    border-radius: 4px;
+    background: $ui-button-background-color;
+    color: $primary-text-color;
+    transition: all 100ms ease-in;
+    font-size: 14px;
+    padding: 0 16px;
+    line-height: 36px;
+    height: 36px;
+    text-decoration: none;
+    margin-bottom: 4px;
+
+    &:active,
+    &:focus,
+    &:hover {
+      background-color: $ui-button-focus-background-color;
+      transition: all 200ms ease-out;
+    }
+
+    &.positive {
+      background: lighten($ui-base-color, 4%);
+      color: $valid-value-color;
+    }
+
+    &.negative {
+      background: lighten($ui-base-color, 4%);
+      color: $error-value-color;
+    }
+
+    span {
+      flex: 1 1 auto;
+    }
+
+    .fa {
+      flex: 0 0 auto;
+    }
+
+    strong {
+      font-weight: 700;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/forms.scss b/app/javascript/flavours/blobfox/styles/forms.scss
new file mode 100644
index 00000000000000..614f7fc6a08df2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/forms.scss
@@ -0,0 +1,1224 @@
+$no-columns-breakpoint: 600px;
+
+code {
+  font-family: $font-monospace, monospace;
+  font-weight: 400;
+}
+
+.form-container {
+  max-width: 450px;
+  padding: 20px;
+  padding-bottom: 50px;
+  margin: 50px auto;
+}
+
+.indicator-icon {
+  display: flex;
+  align-items: center;
+  justify-content: center;
+  width: 40px;
+  height: 40px;
+  border-radius: 50%;
+  color: $primary-text-color;
+
+  &.success {
+    background: $success-green;
+  }
+
+  &.failure {
+    background: $error-red;
+  }
+}
+
+.simple_form {
+  &.hidden {
+    display: none;
+  }
+
+  .input {
+    margin-bottom: 15px;
+    overflow: hidden;
+
+    &.hidden {
+      margin: 0;
+    }
+
+    &.radio_buttons {
+      .radio {
+        margin-bottom: 15px;
+
+        &:last-child {
+          margin-bottom: 0;
+        }
+      }
+
+      .radio > label {
+        position: relative;
+        padding-inline-start: 28px;
+
+        input {
+          position: absolute;
+          top: -2px;
+          inset-inline-start: 0;
+        }
+      }
+    }
+
+    &.boolean {
+      position: relative;
+      margin-bottom: 0;
+
+      .label_input > label {
+        font-family: inherit;
+        font-size: 14px;
+        padding-top: 5px;
+        color: $primary-text-color;
+        display: block;
+        width: auto;
+      }
+
+      .label_input,
+      .hint {
+        padding-inline-start: 28px;
+      }
+
+      .label_input__wrapper {
+        position: static;
+      }
+
+      label.checkbox {
+        position: absolute;
+        top: 2px;
+        inset-inline-start: 0;
+      }
+
+      label a {
+        color: $highlight-text-color;
+        text-decoration: underline;
+
+        &:hover,
+        &:active,
+        &:focus {
+          text-decoration: none;
+        }
+      }
+
+      .overridden,
+      .recommended,
+      .not_recommended,
+      .blobfox_only {
+        position: absolute;
+        margin: 0 4px;
+        margin-top: -2px;
+      }
+    }
+  }
+
+  .row {
+    display: flex;
+    margin: 0 -5px;
+
+    .input {
+      box-sizing: border-box;
+      flex: 1 1 auto;
+      width: 50%;
+      padding: 0 5px;
+    }
+  }
+
+  .title {
+    font-size: 28px;
+    line-height: 33px;
+    font-weight: 700;
+    margin-bottom: 15px;
+  }
+
+  .lead {
+    font-size: 17px;
+    line-height: 22px;
+    color: $secondary-text-color;
+    margin-bottom: 30px;
+
+    &.invited-by {
+      margin-bottom: 15px;
+    }
+
+    a {
+      color: $highlight-text-color;
+    }
+  }
+
+  .rules-list {
+    font-size: 17px;
+    line-height: 22px;
+    margin-bottom: 30px;
+  }
+
+  .hint {
+    color: $darker-text-color;
+
+    a {
+      color: $highlight-text-color;
+    }
+
+    code {
+      border-radius: 3px;
+      padding: 0.2em 0.4em;
+      background: darken($ui-base-color, 12%);
+    }
+
+    li {
+      list-style: disc;
+      margin-inline-start: 18px;
+    }
+  }
+
+  ul.hint {
+    margin-bottom: 15px;
+  }
+
+  span.hint {
+    display: block;
+    font-size: 12px;
+    margin-top: 4px;
+  }
+
+  p.hint {
+    margin-bottom: 15px;
+    color: $darker-text-color;
+
+    &.subtle-hint {
+      text-align: center;
+      font-size: 12px;
+      line-height: 18px;
+      margin-top: 15px;
+      margin-bottom: 0;
+    }
+  }
+
+  .authentication-hint {
+    margin-bottom: 25px;
+  }
+
+  .card {
+    margin-bottom: 15px;
+  }
+
+  strong {
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  .input.with_floating_label {
+    .label_input {
+      display: flex;
+
+      & > label {
+        font-family: inherit;
+        font-size: 14px;
+        color: $primary-text-color;
+        font-weight: 500;
+        min-width: 150px;
+        flex: 0 0 auto;
+      }
+
+      input,
+      select {
+        flex: 1 1 auto;
+      }
+    }
+
+    &.select .hint {
+      margin-top: 6px;
+      margin-inline-start: 150px;
+    }
+  }
+
+  .input.with_label {
+    .label_input > label {
+      font-family: inherit;
+      font-size: 14px;
+      color: $primary-text-color;
+      display: block;
+      margin-bottom: 8px;
+      word-wrap: break-word;
+      font-weight: 500;
+    }
+
+    .hint {
+      margin-top: 6px;
+    }
+
+    ul {
+      flex: 390px;
+    }
+  }
+
+  .input.with_block_label {
+    max-width: none;
+
+    & > label {
+      font-family: inherit;
+      font-size: 14px;
+      color: $primary-text-color;
+      display: block;
+      font-weight: 500;
+      padding-top: 5px;
+    }
+
+    .hint {
+      margin-bottom: 15px;
+    }
+
+    ul {
+      columns: 2;
+    }
+  }
+
+  .input.with_block_label.user_role_permissions_as_keys ul {
+    columns: unset;
+  }
+
+  .input.datetime .label_input select {
+    display: inline-block;
+    width: auto;
+    flex: 0;
+  }
+
+  .required abbr {
+    text-decoration: none;
+    color: lighten($error-value-color, 12%);
+  }
+
+  .fields-group {
+    margin-bottom: 25px;
+
+    .input:last-child {
+      margin-bottom: 0;
+    }
+
+    &__thumbnail {
+      display: block;
+      margin: 0;
+      margin-bottom: 10px;
+      max-width: 100%;
+      height: auto;
+      border-radius: 4px;
+      background: url('images/void.png');
+
+      &[src$='missing.png'] {
+        visibility: hidden;
+      }
+
+      &:last-child {
+        margin-bottom: 0;
+      }
+
+      &#account_avatar-preview {
+        width: 90px;
+        height: 90px;
+        object-fit: cover;
+      }
+    }
+  }
+
+  .fields-row {
+    display: flex;
+    margin: 0 -10px;
+    padding-top: 5px;
+    margin-bottom: 25px;
+
+    .input {
+      max-width: none;
+    }
+
+    &__column {
+      box-sizing: border-box;
+      padding: 0 10px;
+      flex: 1 1 auto;
+      min-height: 1px;
+
+      &-6 {
+        max-width: 50%;
+      }
+
+      .actions {
+        margin-top: 27px;
+      }
+    }
+
+    .fields-group:last-child,
+    .fields-row__column.fields-group {
+      margin-bottom: 0;
+    }
+
+    @media screen and (max-width: $no-columns-breakpoint) {
+      display: block;
+      margin-bottom: 0;
+
+      &__column {
+        max-width: none;
+      }
+
+      .fields-group:last-child,
+      .fields-row__column.fields-group,
+      .fields-row__column {
+        margin-bottom: 25px;
+      }
+    }
+
+    .fields-group.invited-by {
+      margin-bottom: 30px;
+
+      .hint {
+        text-align: center;
+      }
+    }
+  }
+
+  .input.radio_buttons .radio label {
+    margin-bottom: 5px;
+    font-family: inherit;
+    font-size: 14px;
+    color: $primary-text-color;
+    display: block;
+    width: auto;
+  }
+
+  .check_boxes {
+    .checkbox {
+      label {
+        font-family: inherit;
+        font-size: 14px;
+        color: $primary-text-color;
+        display: inline-block;
+        width: auto;
+        position: relative;
+        padding-top: 5px;
+        padding-inline-start: 25px;
+        flex: 1 1 auto;
+      }
+
+      input[type='checkbox'] {
+        position: absolute;
+        inset-inline-start: 0;
+        top: 5px;
+        margin: 0;
+      }
+    }
+  }
+
+  .input.static .label_input__wrapper {
+    font-size: 16px;
+    padding: 10px;
+    border: 1px solid $dark-text-color;
+    border-radius: 4px;
+  }
+
+  input[type='text'],
+  input[type='number'],
+  input[type='email'],
+  input[type='password'],
+  input[type='url'],
+  input[type='datetime-local'],
+  textarea {
+    box-sizing: border-box;
+    font-size: 16px;
+    color: $primary-text-color;
+    display: block;
+    width: 100%;
+    outline: 0;
+    font-family: inherit;
+    resize: vertical;
+    background: darken($ui-base-color, 10%);
+    border: 1px solid darken($ui-base-color, 14%);
+    border-radius: 4px;
+    padding: 10px;
+
+    &::placeholder {
+      color: lighten($darker-text-color, 4%);
+    }
+
+    &:invalid {
+      box-shadow: none;
+    }
+
+    &:required:valid {
+      border-color: $valid-value-color;
+    }
+
+    &:hover {
+      border-color: darken($ui-base-color, 20%);
+    }
+
+    &:active,
+    &:focus {
+      border-color: $highlight-text-color;
+      background: darken($ui-base-color, 8%);
+    }
+  }
+
+  input[type='text'],
+  input[type='number'],
+  input[type='email'],
+  input[type='password'],
+  input[type='datetime-local'] {
+    &:focus:invalid:not(:placeholder-shown),
+    &:required:invalid:not(:placeholder-shown) {
+      border-color: lighten($error-red, 12%);
+    }
+  }
+
+  .input.field_with_errors {
+    label {
+      color: lighten($error-red, 12%);
+    }
+
+    input[type='text'],
+    input[type='number'],
+    input[type='email'],
+    input[type='password'],
+    input[type='datetime-local'],
+    textarea,
+    select {
+      border-color: lighten($error-red, 12%);
+    }
+
+    .error {
+      display: block;
+      font-weight: 500;
+      color: lighten($error-red, 12%);
+      margin-top: 4px;
+    }
+  }
+
+  .input.disabled {
+    opacity: 0.5;
+  }
+
+  .actions {
+    margin-top: 30px;
+    display: flex;
+
+    &.actions--top {
+      margin-top: 0;
+      margin-bottom: 30px;
+    }
+  }
+
+  .stacked-actions {
+    margin-top: 30px;
+    margin-bottom: 15px;
+  }
+
+  button,
+  .button,
+  .block-button {
+    display: block;
+    width: 100%;
+    border: 0;
+    border-radius: 4px;
+    background: $ui-button-background-color;
+    color: $ui-button-color;
+    font-size: 18px;
+    line-height: inherit;
+    height: auto;
+    padding: 10px;
+    text-decoration: none;
+    text-transform: uppercase;
+    text-align: center;
+    box-sizing: border-box;
+    cursor: pointer;
+    font-weight: 500;
+    outline: 0;
+    margin-bottom: 10px;
+    margin-inline-end: 10px;
+
+    &:last-child {
+      margin-inline-end: 0;
+    }
+
+    &:active,
+    &:focus,
+    &:hover {
+      background-color: $ui-button-focus-background-color;
+    }
+
+    &:disabled:hover {
+      background-color: $ui-primary-color;
+    }
+
+    &.negative {
+      background: $ui-button-destructive-background-color;
+
+      &:hover,
+      &:active,
+      &:focus {
+        background-color: $ui-button-destructive-focus-background-color;
+      }
+    }
+  }
+
+  .button.button-tertiary {
+    padding: 9px;
+
+    &:hover,
+    &:focus,
+    &:active {
+      padding: 10px;
+    }
+  }
+
+  select {
+    appearance: none;
+    box-sizing: border-box;
+    font-size: 16px;
+    color: $primary-text-color;
+    display: block;
+    width: 100%;
+    outline: 0;
+    font-family: inherit;
+    resize: vertical;
+    background: darken($ui-base-color, 10%)
+      url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 14.933 18.467' height='19.698' width='15.929'><path d='M3.467 14.967l-3.393-3.5H14.86l-3.392 3.5c-1.866 1.925-3.666 3.5-4 3.5-.335 0-2.135-1.575-4-3.5zm.266-11.234L7.467 0 11.2 3.733l3.733 3.734H0l3.733-3.734z' fill='#{hex-color(lighten($ui-base-color, 12%))}'/></svg>")
+      no-repeat right 8px center / auto 16px;
+    border: 1px solid darken($ui-base-color, 14%);
+    border-radius: 4px;
+    padding-inline-start: 10px;
+    padding-inline-end: 30px;
+    height: 41px;
+  }
+
+  h4 {
+    margin-bottom: 15px !important;
+  }
+
+  .label_input {
+    &__wrapper {
+      position: relative;
+    }
+
+    &__append {
+      position: absolute;
+      inset-inline-end: 3px;
+      top: 1px;
+      padding: 10px;
+      padding-bottom: 9px;
+      font-size: 16px;
+      color: $dark-text-color;
+      font-family: inherit;
+      pointer-events: none;
+      cursor: default;
+      max-width: 140px;
+      white-space: nowrap;
+      overflow: hidden;
+
+      &::after {
+        content: '';
+        display: block;
+        position: absolute;
+        top: 0;
+        inset-inline-end: 0;
+        bottom: 1px;
+        width: 5px;
+        background-image: linear-gradient(
+          to right,
+          rgba(darken($ui-base-color, 10%), 0),
+          darken($ui-base-color, 10%)
+        );
+      }
+    }
+  }
+}
+
+.block-icon {
+  display: block;
+  margin: 0 auto;
+  margin-bottom: 10px;
+  font-size: 24px;
+}
+
+.flash-message {
+  background: lighten($ui-base-color, 8%);
+  color: $darker-text-color;
+  border-radius: 4px;
+  padding: 15px 10px;
+  margin-bottom: 30px;
+  text-align: center;
+
+  &.notice {
+    border: 1px solid rgba($valid-value-color, 0.5);
+    background: rgba($valid-value-color, 0.25);
+    color: $valid-value-color;
+  }
+
+  &.warning {
+    border: 1px solid rgba($gold-star, 0.5);
+    background: rgba($gold-star, 0.25);
+    color: $gold-star;
+  }
+
+  &.alert {
+    border: 1px solid rgba($error-value-color, 0.5);
+    background: rgba($error-value-color, 0.1);
+    color: $error-value-color;
+  }
+
+  &.hidden {
+    display: none;
+  }
+
+  a {
+    display: inline-block;
+    color: $darker-text-color;
+    text-decoration: none;
+
+    &:hover {
+      color: $primary-text-color;
+      text-decoration: underline;
+    }
+  }
+
+  &.warning a {
+    font-weight: 700;
+    color: inherit;
+    text-decoration: underline;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: none;
+      color: inherit;
+    }
+  }
+
+  p {
+    margin-bottom: 15px;
+  }
+
+  .oauth-code {
+    outline: 0;
+    box-sizing: border-box;
+    display: block;
+    width: 100%;
+    border: 0;
+    padding: 10px;
+    font-family: $font-monospace, monospace;
+    background: $ui-base-color;
+    color: $primary-text-color;
+    font-size: 14px;
+    margin: 0;
+
+    &::-moz-focus-inner {
+      border: 0;
+    }
+
+    &::-moz-focus-inner,
+    &:focus,
+    &:active {
+      outline: 0 !important;
+    }
+
+    &:focus {
+      background: lighten($ui-base-color, 4%);
+    }
+  }
+
+  strong {
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  @media screen and (width <= 740px) and (width >= 441px) {
+    margin-top: 40px;
+  }
+
+  &.translation-prompt {
+    text-align: unset;
+    color: unset;
+
+    a {
+      text-decoration: underline;
+    }
+  }
+}
+
+.flash-message-stack {
+  margin-bottom: 30px;
+
+  .flash-message {
+    border-radius: 0;
+    margin-bottom: 0;
+    border-top-width: 0;
+
+    &:first-child {
+      border-radius: 4px 4px 0 0;
+      border-top-width: 1px;
+    }
+
+    &:last-child {
+      border-radius: 0 0 4px 4px;
+
+      &:first-child {
+        border-radius: 4px;
+      }
+    }
+  }
+}
+
+.form-footer {
+  margin-top: 30px;
+  text-align: center;
+
+  a {
+    color: $darker-text-color;
+    text-decoration: none;
+
+    &:hover {
+      text-decoration: underline;
+    }
+  }
+}
+
+.quick-nav {
+  list-style: none;
+  margin-bottom: 25px;
+  font-size: 14px;
+
+  li {
+    display: inline-block;
+    margin-inline-end: 10px;
+  }
+
+  a {
+    color: $highlight-text-color;
+    text-transform: uppercase;
+    text-decoration: none;
+    font-weight: 700;
+
+    &:hover,
+    &:focus,
+    &:active {
+      color: lighten($highlight-text-color, 8%);
+    }
+  }
+}
+
+.oauth-prompt,
+.follow-prompt {
+  margin-bottom: 30px;
+  color: $darker-text-color;
+
+  h2 {
+    font-size: 16px;
+    margin-bottom: 30px;
+    text-align: center;
+  }
+
+  strong {
+    color: $secondary-text-color;
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+}
+
+.oauth-prompt {
+  h3 {
+    color: $ui-secondary-color;
+    font-size: 17px;
+    line-height: 22px;
+    font-weight: 500;
+    margin-bottom: 30px;
+  }
+
+  p {
+    font-size: 14px;
+    line-height: 18px;
+    margin-bottom: 30px;
+  }
+
+  .permissions-list {
+    border: 1px solid $ui-base-color;
+    border-radius: 4px;
+    background: darken($ui-base-color, 4%);
+    margin-bottom: 30px;
+  }
+
+  .actions {
+    margin: 0 -10px;
+    display: flex;
+
+    form {
+      box-sizing: border-box;
+      padding: 0 10px;
+      flex: 1 1 auto;
+      min-height: 1px;
+      width: 50%;
+    }
+  }
+}
+
+.qr-wrapper {
+  display: flex;
+  flex-wrap: wrap;
+  align-items: flex-start;
+}
+
+.qr-code {
+  flex: 0 0 auto;
+  background: $simple-background-color;
+  padding: 4px;
+  margin-inline-end: 10px;
+  margin-bottom: 20px;
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+  display: inline-block;
+
+  svg {
+    display: block;
+    margin: 0;
+  }
+}
+
+.qr-alternative {
+  margin-bottom: 20px;
+  color: $secondary-text-color;
+  flex: 150px;
+
+  samp {
+    display: block;
+    font-size: 14px;
+  }
+}
+
+.simple_form {
+  .warning {
+    box-sizing: border-box;
+    background: rgba($error-value-color, 0.5);
+    color: $primary-text-color;
+    text-shadow: 1px 1px 0 rgba($base-shadow-color, 0.3);
+    box-shadow: 0 2px 6px rgba($base-shadow-color, 0.4);
+    border-radius: 4px;
+    padding: 10px;
+    margin-bottom: 15px;
+
+    a {
+      color: $primary-text-color;
+      text-decoration: underline;
+
+      &:hover,
+      &:focus,
+      &:active {
+        text-decoration: none;
+      }
+    }
+
+    strong {
+      font-weight: 600;
+      display: block;
+      margin-bottom: 5px;
+
+      @each $lang in $cjk-langs {
+        &:lang(#{$lang}) {
+          font-weight: 700;
+        }
+      }
+
+      .fa {
+        font-weight: 400;
+      }
+    }
+  }
+}
+
+.action-pagination {
+  display: flex;
+  flex-wrap: wrap;
+  align-items: center;
+
+  .actions,
+  .pagination {
+    flex: 1 1 auto;
+  }
+
+  .actions {
+    padding: 30px 0;
+    padding-inline-end: 20px;
+    flex: 0 0 auto;
+  }
+}
+
+.post-follow-actions {
+  text-align: center;
+  color: $darker-text-color;
+
+  div {
+    margin-bottom: 4px;
+  }
+}
+
+.alternative-login {
+  margin-top: 20px;
+  margin-bottom: 20px;
+
+  h4 {
+    font-size: 16px;
+    color: $primary-text-color;
+    text-align: center;
+    margin-bottom: 20px;
+    border: 0;
+    padding: 0;
+  }
+
+  .button {
+    display: block;
+  }
+}
+
+.scope-danger {
+  color: $warning-red;
+}
+
+.form_admin_settings_site_short_description,
+.form_admin_settings_site_description,
+.form_admin_settings_site_extended_description,
+.form_admin_settings_site_terms,
+.form_admin_settings_custom_css,
+.form_admin_settings_closed_registrations_message {
+  textarea {
+    font-family: $font-monospace, monospace;
+  }
+}
+
+.input-copy {
+  background: darken($ui-base-color, 10%);
+  border: 1px solid darken($ui-base-color, 14%);
+  border-radius: 4px;
+  display: flex;
+  align-items: center;
+  padding-inline-end: 4px;
+  position: relative;
+  top: 1px;
+  transition: border-color 300ms linear;
+
+  &__wrapper {
+    flex: 1 1 auto;
+  }
+
+  input[type='text'] {
+    background: transparent;
+    border: 0;
+    padding: 10px;
+    font-size: 14px;
+    font-family: $font-monospace, monospace;
+  }
+
+  button {
+    flex: 0 0 auto;
+    margin: 4px;
+    text-transform: none;
+    font-weight: 400;
+    font-size: 14px;
+    padding: 7px 18px;
+    padding-bottom: 6px;
+    width: auto;
+    transition: background 300ms linear;
+  }
+
+  &.copied {
+    border-color: $valid-value-color;
+    transition: none;
+
+    button {
+      background: $valid-value-color;
+      transition: none;
+    }
+  }
+}
+
+.input.user_confirm_password,
+.input.user_website {
+  &:not(.field_with_errors) {
+    display: none;
+  }
+}
+
+.simple_form .h-captcha {
+  display: flex;
+  justify-content: center;
+  margin-bottom: 30px;
+}
+
+.permissions-list {
+  &__item {
+    padding: 15px;
+    color: $ui-secondary-color;
+    border-bottom: 1px solid lighten($ui-base-color, 4%);
+    display: flex;
+    align-items: center;
+
+    &__text {
+      flex: 1 1 auto;
+
+      &__title {
+        font-weight: 500;
+      }
+
+      &__type {
+        color: $darker-text-color;
+      }
+    }
+
+    &__icon {
+      flex: 0 0 auto;
+      font-size: 18px;
+      width: 30px;
+      color: $valid-value-color;
+      display: flex;
+      align-items: center;
+    }
+
+    &:last-child {
+      border-bottom: 0;
+    }
+  }
+}
+
+// Only remove padding when listing applications, to prevent styling issues on
+// the Authorization page.
+.applications-list {
+  .permissions-list__item:last-child {
+    padding-bottom: 0;
+  }
+}
+
+.keywords-table {
+  thead {
+    th {
+      white-space: nowrap;
+    }
+
+    th:first-child {
+      width: 100%;
+    }
+  }
+
+  tfoot {
+    td {
+      border: 0;
+    }
+  }
+
+  .input.string {
+    margin-bottom: 0;
+  }
+
+  .label_input__wrapper {
+    margin-top: 10px;
+  }
+
+  .table-action-link {
+    margin-top: 10px;
+    white-space: nowrap;
+  }
+}
+
+.progress-tracker {
+  display: flex;
+  align-items: center;
+  padding-bottom: 30px;
+  margin-bottom: 30px;
+
+  li {
+    flex: 0 0 auto;
+    position: relative;
+  }
+
+  .separator {
+    height: 2px;
+    background: $ui-base-lighter-color;
+    flex: 1 1 auto;
+
+    &.completed {
+      background: $highlight-text-color;
+    }
+  }
+
+  .circle {
+    box-sizing: border-box;
+    position: relative;
+    width: 30px;
+    height: 30px;
+    border-radius: 50%;
+    border: 2px solid $ui-base-lighter-color;
+    flex: 0 0 auto;
+    display: flex;
+    align-items: center;
+    justify-content: center;
+
+    svg {
+      width: 16px;
+    }
+  }
+
+  .label {
+    position: absolute;
+    font-size: 14px;
+    font-weight: 500;
+    color: $secondary-text-color;
+    padding-top: 10px;
+    text-align: center;
+    width: 100px;
+    left: 50%;
+    transform: translateX(-50%);
+  }
+
+  li:first-child .label {
+    inset-inline-start: 0;
+    inset-inline-end: auto;
+    text-align: start;
+    transform: none;
+  }
+
+  li:last-child .label {
+    inset-inline-start: auto;
+    inset-inline-end: 0;
+    text-align: end;
+    transform: none;
+  }
+
+  .active .circle {
+    border-color: $highlight-text-color;
+
+    &::before {
+      content: '';
+      width: 10px;
+      height: 10px;
+      border-radius: 50%;
+      background: $highlight-text-color;
+      position: absolute;
+      left: 50%;
+      top: 50%;
+      transform: translate(-50%, -50%);
+    }
+  }
+
+  .completed .circle {
+    border-color: $highlight-text-color;
+    background: $highlight-text-color;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/index.scss b/app/javascript/flavours/blobfox/styles/index.scss
new file mode 100644
index 00000000000000..1cb913c8b832ec
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/index.scss
@@ -0,0 +1,24 @@
+@import 'mixins';
+@import 'variables';
+@import 'styles/fonts/roboto';
+@import 'styles/fonts/roboto-mono';
+
+@import 'reset';
+@import 'basics';
+@import 'branding';
+@import 'containers';
+@import 'lists';
+@import 'modal';
+@import 'widgets';
+@import 'forms';
+@import 'accounts';
+@import 'statuses';
+@import 'components/index';
+@import 'polls';
+@import 'about';
+@import 'tables';
+@import 'admin';
+@import 'accessibility';
+@import 'rtl';
+@import 'dashboard';
+@import 'rich_text';
diff --git a/app/javascript/flavours/blobfox/styles/lists.scss b/app/javascript/flavours/blobfox/styles/lists.scss
new file mode 100644
index 00000000000000..6019cd800283d7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/lists.scss
@@ -0,0 +1,19 @@
+.no-list {
+  list-style: none;
+
+  li {
+    display: inline-block;
+    margin: 0 5px;
+  }
+}
+
+.recovery-codes {
+  list-style: none;
+  margin: 0 auto;
+
+  li {
+    font-size: 125%;
+    line-height: 1.5;
+    letter-spacing: 1px;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/mastodon-light.scss b/app/javascript/flavours/blobfox/styles/mastodon-light.scss
new file mode 100644
index 00000000000000..8fc132651bdf67
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/mastodon-light.scss
@@ -0,0 +1,3 @@
+@import 'mastodon-light/variables';
+@import 'index';
+@import 'mastodon-light/diff';
diff --git a/app/javascript/flavours/blobfox/styles/mastodon-light/diff.scss b/app/javascript/flavours/blobfox/styles/mastodon-light/diff.scss
new file mode 100644
index 00000000000000..30425afba022ce
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/mastodon-light/diff.scss
@@ -0,0 +1,743 @@
+// Notes!
+// Sass color functions, "darken" and "lighten" are automatically replaced.
+
+html {
+  scrollbar-color: $ui-base-color rgba($ui-base-color, 0.25);
+}
+
+.simple_form .button.button-tertiary {
+  color: $highlight-text-color;
+}
+
+.status-card__actions button,
+.status-card__actions a {
+  color: rgba($white, 0.8);
+
+  &:hover,
+  &:active,
+  &:focus {
+    color: $white;
+  }
+}
+
+// Change default background colors of columns
+.column > .scrollable,
+.getting-started,
+.column-inline-form,
+.regeneration-indicator {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+  border-top: 0;
+}
+
+.error-column {
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.column > .scrollable.about {
+  border-top: 1px solid lighten($ui-base-color, 8%);
+}
+
+.about__meta,
+.about__section__title,
+.interaction-modal {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.rules-list li::before {
+  background: $ui-highlight-color;
+}
+
+.directory__card__img {
+  background: lighten($ui-base-color, 12%);
+}
+
+.filter-form {
+  background: $white;
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+}
+
+.column-back-button,
+.column-header {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    border-top: 0;
+  }
+
+  &--slim-button {
+    top: -50px;
+    right: 0;
+  }
+}
+
+.column-header__back-button,
+.column-header__button,
+.column-header__button.active,
+.account__header {
+  background: $white;
+}
+
+.column-header__button.active {
+  color: $ui-highlight-color;
+
+  &:hover,
+  &:active,
+  &:focus {
+    color: $ui-highlight-color;
+    background: $white;
+  }
+}
+
+.account__header__bar .avatar .account__avatar {
+  border-color: $white;
+}
+
+.getting-started__footer a {
+  color: $ui-secondary-color;
+  text-decoration: underline;
+}
+
+.confirmation-modal__secondary-button,
+.confirmation-modal__cancel-button,
+.mute-modal__cancel-button,
+.block-modal__cancel-button {
+  color: lighten($ui-base-color, 26%);
+
+  &:hover,
+  &:focus,
+  &:active {
+    color: $primary-text-color;
+  }
+}
+
+.column-subheading {
+  background: darken($ui-base-color, 4%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+}
+
+.getting-started,
+.scrollable {
+  .column-link {
+    background: $white;
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+
+    &:hover,
+    &:active,
+    &:focus {
+      background: $ui-base-color;
+    }
+  }
+}
+
+.getting-started .navigation-bar {
+  border-top: 1px solid lighten($ui-base-color, 8%);
+  border-bottom: 1px solid lighten($ui-base-color, 8%);
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    border-top: 0;
+  }
+}
+
+.compose-form__autosuggest-wrapper,
+.poll__option input[type='text'],
+.compose-form .spoiler-input__input,
+.compose-form__poll-wrapper select,
+.search__input,
+.setting-text,
+.report-dialog-modal__textarea,
+.audio-player {
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.report-dialog-modal .dialog-option .poll__input {
+  color: $white;
+}
+
+.search__input {
+  @media screen and (max-width: $no-gap-breakpoint) {
+    border-top: 0;
+    border-bottom: 0;
+  }
+}
+
+.list-editor .search .search__input {
+  border-top: 0;
+  border-bottom: 0;
+}
+
+.compose-form__poll-wrapper select {
+  background: $simple-background-color
+    url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 14.933 18.467' height='19.698' width='15.929'><path d='M3.467 14.967l-3.393-3.5H14.86l-3.392 3.5c-1.866 1.925-3.666 3.5-4 3.5-.335 0-2.135-1.575-4-3.5zm.266-11.234L7.467 0 11.2 3.733l3.733 3.734H0l3.733-3.734z' fill='#{hex-color(lighten($ui-base-color, 8%))}'/></svg>")
+    no-repeat right 8px center / auto 16px;
+}
+
+.compose-form__poll-wrapper,
+.compose-form__poll-wrapper .poll__footer {
+  border-top-color: lighten($ui-base-color, 8%);
+}
+
+.notification__filter-bar {
+  border: 1px solid lighten($ui-base-color, 8%);
+  border-top: 0;
+}
+
+.compose-form .compose-form__buttons-wrapper {
+  background: $ui-base-color;
+  border: 1px solid lighten($ui-base-color, 8%);
+  border-top: 0;
+}
+
+.drawer__header,
+.drawer__inner {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.drawer__inner__mastodon {
+  background: $white
+    url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 234.80078 31.757813" width="234.80078" height="31.757812"><path d="M19.599609 0c-1.05 0-2.10039.375-2.90039 1.125L0 16.925781v14.832031h234.80078V17.025391l-16.5-15.900391c-1.6-1.5-4.20078-1.5-5.80078 0l-13.80078 13.099609c-1.6 1.5-4.19883 1.5-5.79883 0L179.09961 1.125c-1.6-1.5-4.19883-1.5-5.79883 0L159.5 14.224609c-1.6 1.5-4.20078 1.5-5.80078 0L139.90039 1.125c-1.6-1.5-4.20078-1.5-5.80078 0l-13.79883 13.099609c-1.6 1.5-4.20078 1.5-5.80078 0L100.69922 1.125c-1.600001-1.5-4.198829-1.5-5.798829 0l-13.59961 13.099609c-1.6 1.5-4.200781 1.5-5.800781 0L61.699219 1.125c-1.6-1.5-4.198828-1.5-5.798828 0L42.099609 14.224609c-1.6 1.5-4.198828 1.5-5.798828 0L22.5 1.125C21.7.375 20.649609 0 19.599609 0z" fill="#{hex-color($ui-base-color)}"/></svg>')
+    no-repeat bottom / 100% auto;
+}
+
+// Change the colors used in compose-form
+.compose-form {
+  .compose-form__modifiers {
+    .compose-form__upload__actions .icon-button,
+    .compose-form__upload__warning .icon-button {
+      color: lighten($white, 7%);
+
+      &:active,
+      &:focus,
+      &:hover {
+        color: $white;
+      }
+    }
+  }
+
+  .compose-form__buttons-wrapper {
+    background: darken($ui-base-color, 6%);
+  }
+
+  .autosuggest-textarea__suggestions {
+    background: darken($ui-base-color, 6%);
+  }
+
+  .autosuggest-textarea__suggestions__item {
+    &:hover,
+    &:focus,
+    &:active,
+    &.selected {
+      background: lighten($ui-base-color, 4%);
+    }
+  }
+}
+
+.emoji-mart-bar {
+  border-color: lighten($ui-base-color, 4%);
+
+  &:first-child {
+    background: darken($ui-base-color, 6%);
+  }
+}
+
+.emoji-mart-search input {
+  background: rgba($ui-base-color, 0.3);
+  border-color: $ui-base-color;
+}
+
+.upload-progress__backdrop {
+  background: $ui-base-color;
+}
+
+// Change the background colors of statuses
+.focusable:focus {
+  background: lighten($white, 4%);
+}
+
+.detailed-status,
+.detailed-status__action-bar {
+  background: $white;
+}
+
+// Change the background colors of status__content__spoiler-link
+.reply-indicator__content .status__content__spoiler-link,
+.status__content .status__content__spoiler-link {
+  background: $ui-base-color;
+
+  &:hover,
+  &:focus {
+    background: lighten($ui-base-color, 4%);
+  }
+}
+
+// Change the background colors of media and video spoilers
+.media-spoiler,
+.video-player__spoiler {
+  background: $ui-base-color;
+}
+
+.privacy-dropdown.active .privacy-dropdown__value.active .icon-button {
+  color: $white;
+}
+
+.account-gallery__item a {
+  background-color: $ui-base-color;
+}
+
+// Change the colors used in the dropdown menu
+.dropdown-menu {
+  background: $white;
+
+  &__arrow::before {
+    background-color: $white;
+  }
+
+  &__item {
+    color: $darker-text-color;
+
+    &--dangerous {
+      color: $error-value-color;
+    }
+
+    a,
+    button {
+      background: $white;
+    }
+  }
+}
+
+// Change the text colors on inverted background
+.privacy-dropdown__option.active,
+.privacy-dropdown__option:hover,
+.privacy-dropdown__option.active .privacy-dropdown__option__content,
+.privacy-dropdown__option.active .privacy-dropdown__option__content strong,
+.privacy-dropdown__option:hover .privacy-dropdown__option__content,
+.privacy-dropdown__option:hover .privacy-dropdown__option__content strong,
+.dropdown-menu__item:not(.dropdown-menu__item--dangerous) a:active,
+.dropdown-menu__item:not(.dropdown-menu__item--dangerous) a:focus,
+.dropdown-menu__item:not(.dropdown-menu__item--dangerous) a:hover,
+.actions-modal ul li:not(:empty) a.active,
+.actions-modal ul li:not(:empty) a.active button,
+.actions-modal ul li:not(:empty) a:active,
+.actions-modal ul li:not(:empty) a:active button,
+.actions-modal ul li:not(:empty) a:focus,
+.actions-modal ul li:not(:empty) a:focus button,
+.actions-modal ul li:not(:empty) a:hover,
+.actions-modal ul li:not(:empty) a:hover button,
+.language-dropdown__dropdown__results__item.active,
+.admin-wrapper .sidebar ul .simple-navigation-active-leaf a,
+.simple_form .block-button,
+.simple_form .button,
+.simple_form button {
+  color: $white;
+}
+
+.language-dropdown__dropdown__results__item
+  .language-dropdown__dropdown__results__item__common-name {
+  color: lighten($ui-base-color, 8%);
+}
+
+.language-dropdown__dropdown__results__item.active
+  .language-dropdown__dropdown__results__item__common-name {
+  color: darken($ui-base-color, 12%);
+}
+
+.dropdown-menu__separator,
+.dropdown-menu__item.edited-timestamp__history__item,
+.dropdown-menu__container__header,
+.compare-history-modal .report-modal__target,
+.report-dialog-modal .poll__option.dialog-option {
+  border-bottom-color: lighten($ui-base-color, 4%);
+}
+
+.report-dialog-modal__container {
+  border-top-color: lighten($ui-base-color, 4%);
+}
+
+// Change the background colors of modals
+.actions-modal,
+.boost-modal,
+.confirmation-modal,
+.mute-modal,
+.block-modal,
+.report-modal,
+.report-dialog-modal,
+.embed-modal,
+.error-modal,
+.onboarding-modal,
+.compare-history-modal,
+.report-modal__comment .setting-text__wrapper,
+.report-modal__comment .setting-text,
+.announcements,
+.picture-in-picture__header,
+.picture-in-picture__footer,
+.reactions-bar__item {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.reactions-bar__item:hover,
+.reactions-bar__item:focus,
+.reactions-bar__item:active,
+.language-dropdown__dropdown__results__item:hover,
+.language-dropdown__dropdown__results__item:focus,
+.language-dropdown__dropdown__results__item:active {
+  background-color: $ui-base-color;
+}
+
+.reactions-bar__item.active {
+  background-color: mix($white, $ui-highlight-color, 80%);
+  border-color: mix(lighten($ui-base-color, 8%), $ui-highlight-color, 80%);
+}
+
+.media-modal__overlay .picture-in-picture__footer {
+  border: 0;
+}
+
+.picture-in-picture__header {
+  border-bottom: 0;
+}
+
+.announcements,
+.picture-in-picture__footer {
+  border-top: 0;
+}
+
+.icon-with-badge__badge {
+  border-color: $white;
+  color: $white;
+}
+
+.report-modal__comment {
+  border-right-color: lighten($ui-base-color, 8%);
+}
+
+.report-modal__container {
+  border-top-color: lighten($ui-base-color, 8%);
+}
+
+.column-header__collapsible-inner {
+  background: darken($ui-base-color, 4%);
+  border: 1px solid lighten($ui-base-color, 8%);
+  border-top: 0;
+}
+
+.column-settings__hashtags .column-select__option {
+  color: $white;
+}
+
+.dashboard__quick-access,
+.focal-point__preview strong,
+.admin-wrapper .content__heading__tabs a.selected {
+  color: $white;
+}
+
+.flash-message.warning {
+  color: lighten($gold-star, 16%);
+}
+
+.boost-modal__action-bar,
+.confirmation-modal__action-bar,
+.mute-modal__action-bar,
+.block-modal__action-bar,
+.onboarding-modal__paginator,
+.error-modal__footer {
+  background: darken($ui-base-color, 6%);
+
+  .onboarding-modal__nav,
+  .error-modal__nav {
+    &:hover,
+    &:focus,
+    &:active {
+      background-color: darken($ui-base-color, 12%);
+    }
+  }
+}
+
+.display-case__case {
+  background: $white;
+}
+
+.embed-modal .embed-modal__container .embed-modal__html {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+
+  &:focus {
+    border-color: lighten($ui-base-color, 12%);
+    background: $white;
+  }
+}
+
+.react-toggle-track {
+  background: $ui-secondary-color;
+}
+
+.react-toggle:hover:not(.react-toggle--disabled) .react-toggle-track {
+  background: darken($ui-secondary-color, 10%);
+}
+
+.react-toggle.react-toggle--checked:hover:not(.react-toggle--disabled)
+  .react-toggle-track {
+  background: lighten($ui-highlight-color, 10%);
+}
+
+// Change the default color used for the text in an empty column or on the error column
+.empty-column-indicator,
+.error-column {
+  color: $primary-text-color;
+  background: $white;
+}
+
+// Change the default colors used on some parts of the profile pages
+.activity-stream-tabs {
+  background: $account-background-color;
+  border-bottom-color: lighten($ui-base-color, 8%);
+}
+
+.nothing-here,
+.page-header,
+.directory__tag > a,
+.directory__tag > div {
+  background: $white;
+  border: 1px solid lighten($ui-base-color, 8%);
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    border-left: 0;
+    border-right: 0;
+    border-top: 0;
+  }
+}
+
+.simple_form {
+  input[type='text'],
+  input[type='number'],
+  input[type='email'],
+  input[type='password'],
+  textarea {
+    &:hover {
+      border-color: lighten($ui-base-color, 12%);
+    }
+  }
+}
+
+.picture-in-picture-placeholder {
+  background: $white;
+  border-color: lighten($ui-base-color, 8%);
+  color: lighten($ui-base-color, 8%);
+}
+
+.directory__tag > a {
+  &:hover,
+  &:active,
+  &:focus {
+    background: $ui-base-color;
+  }
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    border: 0;
+  }
+}
+
+.batch-table {
+  &__toolbar,
+  &__row,
+  .nothing-here {
+    border-color: lighten($ui-base-color, 8%);
+  }
+}
+
+.activity-stream {
+  border: 1px solid lighten($ui-base-color, 8%);
+
+  &--under-tabs {
+    border-top: 0;
+  }
+
+  .entry {
+    background: $account-background-color;
+
+    .detailed-status.light,
+    .more.light,
+    .status.light {
+      border-bottom-color: lighten($ui-base-color, 8%);
+    }
+  }
+
+  .status.light {
+    .status__content {
+      color: $primary-text-color;
+    }
+
+    .display-name {
+      strong {
+        color: $primary-text-color;
+      }
+    }
+  }
+}
+
+.accounts-grid {
+  .account-grid-card {
+    .controls {
+      .icon-button {
+        color: $darker-text-color;
+      }
+    }
+
+    .name {
+      a {
+        color: $primary-text-color;
+      }
+    }
+
+    .username {
+      color: $darker-text-color;
+    }
+
+    .account__header__content {
+      color: $primary-text-color;
+    }
+  }
+}
+
+.simple_form {
+  .warning {
+    box-shadow: none;
+    background: rgba($error-red, 0.5);
+    text-shadow: none;
+  }
+
+  .recommended {
+    border-color: $ui-highlight-color;
+    color: $ui-highlight-color;
+    background-color: rgba($ui-highlight-color, 0.1);
+  }
+}
+
+.compose-form .compose-form__warning {
+  border-color: $ui-highlight-color;
+  background-color: rgba($ui-highlight-color, 0.1);
+
+  &,
+  a {
+    color: $ui-highlight-color;
+  }
+}
+
+.reply-indicator {
+  background: transparent;
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.status__content,
+.reply-indicator__content {
+  a {
+    color: $highlight-text-color;
+  }
+}
+
+.notification__filter-bar button.active::after,
+.account__section-headline a.active::after {
+  border-color: transparent transparent $white;
+}
+
+.hero-widget,
+.moved-account-widget,
+.memoriam-widget,
+.activity-stream,
+.nothing-here,
+.directory__tag > a,
+.directory__tag > div,
+.card > a,
+.page-header,
+.compose-form .compose-form__warning {
+  box-shadow: none;
+}
+
+.mute-modal select {
+  border: 1px solid lighten($ui-base-color, 8%);
+  background: $simple-background-color
+    url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 14.933 18.467' height='19.698' width='15.929'><path d='M3.467 14.967l-3.393-3.5H14.86l-3.392 3.5c-1.866 1.925-3.666 3.5-4 3.5-.335 0-2.135-1.575-4-3.5zm.266-11.234L7.467 0 11.2 3.733l3.733 3.734H0l3.733-3.734z' fill='#{hex-color(lighten($ui-base-color, 8%))}'/></svg>")
+    no-repeat right 8px center / auto 16px;
+}
+
+// blobfox-soc-specific changes
+
+.pillbar-button {
+  background: $ui-secondary-color;
+
+  &:not([disabled]) {
+    &:hover,
+    &:focus {
+      background: darken($ui-secondary-color, 10%);
+    }
+
+    &.active {
+      background-color: darken($ui-highlight-color, 2%);
+
+      &:hover,
+      &:focus {
+        background: lighten($ui-highlight-color, 10%);
+      }
+    }
+  }
+}
+
+.blobfox.local-settings {
+  background: $ui-base-color;
+  border: 1px solid lighten($ui-base-color, 8%);
+}
+
+.blobfox.local-settings__navigation {
+  background: darken($ui-base-color, 8%);
+}
+
+.blobfox.local-settings__navigation__item {
+  background: darken($ui-base-color, 8%);
+  border-bottom: 1px lighten($ui-base-color, 8%) solid;
+
+  &:hover {
+    background: $ui-base-color;
+  }
+
+  &.active {
+    background: $ui-highlight-color;
+    color: $white;
+  }
+
+  &.close,
+  &.close:hover {
+    background: $error-value-color;
+    color: $primary-text-color;
+  }
+}
+
+.notification__dismiss-overlay {
+  .wrappy {
+    box-shadow: unset;
+
+    .ckbox {
+      text-shadow: unset;
+    }
+  }
+}
+
+.status.collapsed .status__content::after {
+  background: linear-gradient(
+    rgba(darken($ui-base-color, 13%), 0),
+    rgba(darken($ui-base-color, 13%), 1)
+  );
+}
+
+.drawer__inner__mastodon {
+  background: $white
+    url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 234.80078 31.757813" width="234.80078" height="31.757812"><path d="M19.599609 0c-1.05 0-2.10039.375-2.90039 1.125L0 16.925781v14.832031h234.80078V17.025391l-16.5-15.900391c-1.6-1.5-4.20078-1.5-5.80078 0l-13.80078 13.099609c-1.6 1.5-4.19883 1.5-5.79883 0L179.09961 1.125c-1.6-1.5-4.19883-1.5-5.79883 0L159.5 14.224609c-1.6 1.5-4.20078 1.5-5.80078 0L139.90039 1.125c-1.6-1.5-4.20078-1.5-5.80078 0l-13.79883 13.099609c-1.6 1.5-4.20078 1.5-5.80078 0L100.69922 1.125c-1.600001-1.5-4.198829-1.5-5.798829 0l-13.59961 13.099609c-1.6 1.5-4.200781 1.5-5.800781 0L61.699219 1.125c-1.6-1.5-4.198828-1.5-5.798828 0L42.099609 14.224609c-1.6 1.5-4.198828 1.5-5.798828 0L22.5 1.125C21.7.375 20.649609 0 19.599609 0z" fill="#{hex-color($ui-base-color)}"/></svg>')
+    no-repeat bottom / 100% auto !important;
+
+  .mastodon {
+    filter: contrast(75%) brightness(75%) !important;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/mastodon-light/variables.scss b/app/javascript/flavours/blobfox/styles/mastodon-light/variables.scss
new file mode 100644
index 00000000000000..250e200fc6d258
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/mastodon-light/variables.scss
@@ -0,0 +1,57 @@
+// Dependent colors
+$black: #000000;
+$white: #ffffff;
+
+$classic-base-color: #282c37;
+$classic-primary-color: #9baec8;
+$classic-secondary-color: #d9e1e8;
+$classic-highlight-color: #6364ff;
+
+$blurple-600: #563acc; // Iris
+$blurple-500: #6364ff; // Brand purple
+$blurple-300: #858afa; // Faded Blue
+$grey-600: #4e4c5a; // Trout
+$grey-100: #dadaf3; // Topaz
+
+// Differences
+$success-green: lighten(#3c754d, 8%);
+
+$base-overlay-background: $white !default;
+$valid-value-color: $success-green !default;
+
+$ui-base-color: $classic-secondary-color !default;
+$ui-base-lighter-color: #b0c0cf;
+$ui-primary-color: #9bcbed;
+$ui-secondary-color: $classic-base-color !default;
+$ui-highlight-color: $classic-highlight-color !default;
+
+$ui-button-secondary-color: $grey-600 !default;
+$ui-button-secondary-border-color: $grey-600 !default;
+$ui-button-secondary-focus-color: $white !default;
+
+$ui-button-tertiary-color: $blurple-500 !default;
+$ui-button-tertiary-border-color: $blurple-500 !default;
+
+$primary-text-color: $black !default;
+$darker-text-color: $classic-base-color !default;
+$highlight-text-color: darken($ui-highlight-color, 8%) !default;
+$dark-text-color: #444b5d;
+$action-button-color: #606984;
+
+$inverted-text-color: $black !default;
+$lighter-text-color: $classic-base-color !default;
+$light-text-color: #444b5d;
+
+// Newly added colors
+$account-background-color: $white !default;
+
+// Invert darkened and lightened colors
+@function darken($color, $amount) {
+  @return hsl(hue($color), saturation($color), lightness($color) + $amount);
+}
+
+@function lighten($color, $amount) {
+  @return hsl(hue($color), saturation($color), lightness($color) - $amount);
+}
+
+$emojis-requiring-inversion: 'chains';
diff --git a/app/javascript/flavours/blobfox/styles/modal.scss b/app/javascript/flavours/blobfox/styles/modal.scss
new file mode 100644
index 00000000000000..0b7220b21d1115
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/modal.scss
@@ -0,0 +1,37 @@
+.modal-layout {
+  background: $ui-base-color
+    url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 234.80078 31.757813" width="234.80078" height="31.757812"><path d="M19.599609 0c-1.05 0-2.10039.375-2.90039 1.125L0 16.925781v14.832031h234.80078V17.025391l-16.5-15.900391c-1.6-1.5-4.20078-1.5-5.80078 0l-13.80078 13.099609c-1.6 1.5-4.19883 1.5-5.79883 0L179.09961 1.125c-1.6-1.5-4.19883-1.5-5.79883 0L159.5 14.224609c-1.6 1.5-4.20078 1.5-5.80078 0L139.90039 1.125c-1.6-1.5-4.20078-1.5-5.80078 0l-13.79883 13.099609c-1.6 1.5-4.20078 1.5-5.80078 0L100.69922 1.125c-1.600001-1.5-4.198829-1.5-5.798829 0l-13.59961 13.099609c-1.6 1.5-4.200781 1.5-5.800781 0L61.699219 1.125c-1.6-1.5-4.198828-1.5-5.798828 0L42.099609 14.224609c-1.6 1.5-4.198828 1.5-5.798828 0L22.5 1.125C21.7.375 20.649609 0 19.599609 0z" fill="#{hex-color($ui-base-lighter-color)}33"/></svg>')
+    repeat-x bottom fixed;
+  display: flex;
+  flex-direction: column;
+  height: 100vh;
+  padding: 0;
+}
+
+.modal-layout__mastodon {
+  display: flex;
+  flex: 1;
+  flex-direction: column;
+  justify-content: flex-end;
+
+  > div {
+    flex: 1;
+    max-height: 235px;
+    position: relative;
+
+    img {
+      max-height: 100%;
+      max-width: 100%;
+      height: 100%;
+      position: absolute;
+      bottom: 0;
+      inset-inline-start: 0;
+    }
+  }
+}
+
+@media screen and (width <= 600px) {
+  .account-header {
+    margin-top: 0;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/polls.scss b/app/javascript/flavours/blobfox/styles/polls.scss
new file mode 100644
index 00000000000000..4566a013a630fb
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/polls.scss
@@ -0,0 +1,314 @@
+.poll {
+  margin-top: 16px;
+  font-size: 14px;
+
+  ul,
+  .e-content & ul {
+    margin: 0;
+    list-style: none;
+  }
+
+  li {
+    margin-bottom: 10px;
+    position: relative;
+  }
+
+  &__chart {
+    border-radius: 4px;
+    display: block;
+    background: darken($ui-primary-color, 5%);
+    height: 5px;
+    min-width: 1%;
+
+    &.leading {
+      background: $ui-highlight-color;
+    }
+  }
+
+  progress {
+    border: 0;
+    display: block;
+    width: 100%;
+    height: 5px;
+    appearance: none;
+    background: transparent;
+
+    &::-webkit-progress-bar {
+      background: transparent;
+    }
+
+    // Those rules need to be entirely separate or they won't work, hence the
+    // duplication
+    &::-moz-progress-bar {
+      border-radius: 4px;
+      background: darken($ui-primary-color, 5%);
+    }
+
+    &::-ms-fill {
+      border-radius: 4px;
+      background: darken($ui-primary-color, 5%);
+    }
+
+    &::-webkit-progress-value {
+      border-radius: 4px;
+      background: darken($ui-primary-color, 5%);
+    }
+  }
+
+  &__option {
+    position: relative;
+    display: flex;
+    padding: 6px 0;
+    line-height: 18px;
+    cursor: default;
+    overflow: hidden;
+
+    &__text {
+      display: inline-block;
+      word-wrap: break-word;
+      overflow-wrap: break-word;
+      max-width: calc(100% - 45px - 25px);
+    }
+
+    input[type='radio'],
+    input[type='checkbox'] {
+      display: none;
+    }
+
+    .autosuggest-input {
+      flex: 1 1 auto;
+    }
+
+    input[type='text'] {
+      display: block;
+      box-sizing: border-box;
+      width: 100%;
+      font-size: 14px;
+      color: $inverted-text-color;
+      outline: 0;
+      font-family: inherit;
+      background: $simple-background-color;
+      border: 1px solid darken($simple-background-color, 14%);
+      border-radius: 4px;
+      padding: 6px 10px;
+
+      &:focus {
+        border-color: $highlight-text-color;
+      }
+    }
+
+    &.selectable {
+      cursor: pointer;
+    }
+
+    &.editable {
+      display: flex;
+      align-items: center;
+      overflow: visible;
+    }
+  }
+
+  &__input {
+    display: inline-block;
+    position: relative;
+    border: 1px solid $ui-primary-color;
+    box-sizing: border-box;
+    width: 18px;
+    height: 18px;
+    margin-inline-end: 10px;
+    top: -1px;
+    border-radius: 50%;
+    vertical-align: middle;
+    margin-top: auto;
+    margin-bottom: auto;
+    flex: 0 0 18px;
+
+    &.checkbox {
+      border-radius: 4px;
+    }
+
+    &:active,
+    &:focus,
+    &:hover {
+      border-color: lighten($valid-value-color, 15%);
+      border-width: 4px;
+    }
+
+    &.active {
+      background-color: $valid-value-color;
+      border-color: $valid-value-color;
+    }
+
+    &::-moz-focus-inner {
+      outline: 0 !important;
+      border: 0;
+    }
+
+    &:focus,
+    &:active {
+      outline: 0 !important;
+    }
+
+    &.disabled {
+      border-color: $dark-text-color;
+
+      &.active {
+        background: $dark-text-color;
+      }
+
+      &:active,
+      &:focus,
+      &:hover {
+        border-color: $dark-text-color;
+        border-width: 1px;
+      }
+    }
+  }
+
+  &__number {
+    display: inline-block;
+    width: 45px;
+    font-weight: 700;
+    flex: 0 0 45px;
+  }
+
+  &__voted {
+    padding: 0 5px;
+    display: inline-block;
+
+    &__mark {
+      font-size: 18px;
+    }
+  }
+
+  &__footer {
+    padding-top: 6px;
+    padding-bottom: 5px;
+    color: $dark-text-color;
+  }
+
+  &__link {
+    display: inline;
+    background: transparent;
+    padding: 0;
+    margin: 0;
+    border: 0;
+    color: $dark-text-color;
+    text-decoration: underline;
+    font-size: inherit;
+
+    &:hover {
+      text-decoration: none;
+    }
+
+    &:active,
+    &:focus {
+      background-color: rgba($dark-text-color, 0.1);
+    }
+  }
+
+  .button {
+    height: 36px;
+    padding: 0 16px;
+    margin-inline-end: 10px;
+    font-size: 14px;
+  }
+}
+
+.compose-form__poll-wrapper {
+  border-top: 1px solid darken($simple-background-color, 8%);
+  overflow-x: hidden;
+
+  ul {
+    padding: 10px;
+  }
+
+  .poll__input {
+    &:active,
+    &:focus,
+    &:hover {
+      border-color: $ui-button-focus-background-color;
+    }
+  }
+
+  .poll__footer {
+    border-top: 1px solid darken($simple-background-color, 8%);
+    padding: 10px;
+    display: flex;
+    align-items: center;
+
+    button,
+    select {
+      width: 100%;
+      flex: 1 1 50%;
+
+      &:focus {
+        border-color: $highlight-text-color;
+      }
+    }
+  }
+
+  .button.button-secondary {
+    font-size: 14px;
+    font-weight: 400;
+    padding: 6px 10px;
+    height: auto;
+    line-height: inherit;
+    color: $action-button-color;
+    border-color: $action-button-color;
+    margin-inline-end: 5px;
+
+    &:hover,
+    &:focus,
+    &.active {
+      border-color: $action-button-color;
+      background-color: $action-button-color;
+      color: $ui-button-color;
+    }
+  }
+
+  li {
+    display: flex;
+    align-items: center;
+
+    .poll__option {
+      flex: 0 0 auto;
+      width: calc(100% - (23px + 6px));
+      margin-inline-end: 6px;
+    }
+  }
+
+  select {
+    appearance: none;
+    box-sizing: border-box;
+    font-size: 14px;
+    color: $inverted-text-color;
+    display: inline-block;
+    width: auto;
+    outline: 0;
+    font-family: inherit;
+    background: $simple-background-color
+      url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 14.933 18.467' height='19.698' width='15.929'><path d='M3.467 14.967l-3.393-3.5H14.86l-3.392 3.5c-1.866 1.925-3.666 3.5-4 3.5-.335 0-2.135-1.575-4-3.5zm.266-11.234L7.467 0 11.2 3.733l3.733 3.734H0l3.733-3.734z' fill='#{hex-color(darken($simple-background-color, 14%))}'/></svg>")
+      no-repeat right 8px center / auto 16px;
+    border: 1px solid darken($simple-background-color, 14%);
+    border-radius: 4px;
+    padding: 6px 10px;
+    padding-inline-end: 30px;
+  }
+
+  .icon-button.disabled {
+    color: darken($simple-background-color, 14%);
+  }
+}
+
+.muted .poll {
+  color: $dark-text-color;
+
+  &__chart {
+    background: rgba(darken($ui-primary-color, 14%), 0.7);
+
+    &.leading {
+      background: rgba($ui-highlight-color, 0.5);
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/reset.scss b/app/javascript/flavours/blobfox/styles/reset.scss
new file mode 100644
index 00000000000000..f54ed5bc79b1a7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/reset.scss
@@ -0,0 +1,95 @@
+/* http://meyerweb.com/eric/tools/css/reset/
+   v2.0 | 20110126
+   License: none (public domain)
+*/
+
+html, body, div, span, applet, object, iframe,
+h1, h2, h3, h4, h5, h6, p, blockquote, pre,
+a, abbr, acronym, address, big, cite, code,
+del, dfn, em, img, ins, kbd, q, s, samp,
+small, strike, strong, sub, sup, tt, var,
+b, u, i, center,
+dl, dt, dd, ol, ul, li,
+fieldset, form, label, legend,
+table, caption, tbody, tfoot, thead, tr, th, td,
+article, aside, canvas, details, embed,
+figure, figcaption, footer, header, hgroup,
+menu, nav, output, ruby, section, summary,
+time, mark, audio, video {
+  margin: 0;
+  padding: 0;
+  border: 0;
+  font-size: 100%;
+  font: inherit;
+  vertical-align: baseline;
+}
+
+/* HTML5 display-role reset for older browsers */
+article, aside, details, figcaption, figure,
+footer, header, hgroup, menu, nav, section {
+  display: block;
+}
+
+body {
+  line-height: 1;
+}
+
+ol, ul {
+  list-style: none;
+}
+
+blockquote, q {
+  quotes: none;
+}
+
+blockquote:before, blockquote:after,
+q:before, q:after {
+  content: '';
+  content: none;
+}
+
+table {
+  border-collapse: collapse;
+  border-spacing: 0;
+}
+
+html {
+  scrollbar-color: lighten($ui-base-color, 4%) rgba($base-overlay-background, 0.1);
+}
+
+::-webkit-scrollbar {
+  width: 12px;
+  height: 12px;
+}
+
+::-webkit-scrollbar-thumb {
+  background: lighten($ui-base-color, 4%);
+  border: 0px none $base-border-color;
+  border-radius: 50px;
+}
+
+::-webkit-scrollbar-thumb:hover {
+  background: lighten($ui-base-color, 6%);
+}
+
+::-webkit-scrollbar-thumb:active {
+  background: lighten($ui-base-color, 4%);
+}
+
+::-webkit-scrollbar-track {
+  border: 0px none $base-border-color;
+  border-radius: 0;
+  background: rgba($base-overlay-background, 0.1);
+}
+
+::-webkit-scrollbar-track:hover {
+  background: $ui-base-color;
+}
+
+::-webkit-scrollbar-track:active {
+  background: $ui-base-color;
+}
+
+::-webkit-scrollbar-corner {
+  background: transparent;
+}
diff --git a/app/javascript/flavours/blobfox/styles/rich_text.scss b/app/javascript/flavours/blobfox/styles/rich_text.scss
new file mode 100644
index 00000000000000..6224302ee3f5f9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/rich_text.scss
@@ -0,0 +1,99 @@
+.status__content__text,
+.e-content,
+.reply-indicator__content {
+  pre,
+  blockquote {
+    margin-bottom: 20px;
+    white-space: pre-wrap;
+    unicode-bidi: plaintext;
+
+    &:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  blockquote {
+    padding-inline-start: 10px;
+    border-inline-start: 3px solid $darker-text-color;
+    color: $darker-text-color;
+    white-space: normal;
+
+    p:last-child {
+      margin-bottom: 0;
+    }
+  }
+
+  & > ul,
+  & > ol {
+    margin-bottom: 20px;
+  }
+
+  h1,
+  h2,
+  h3,
+  h4,
+  h5 {
+    margin-top: 20px;
+    margin-bottom: 20px;
+  }
+
+  h1,
+  h2 {
+    font-weight: 700;
+    font-size: 1.2em;
+  }
+
+  h2 {
+    font-size: 1.1em;
+  }
+
+  h3,
+  h4,
+  h5 {
+    font-weight: 500;
+  }
+
+  b,
+  strong {
+    font-weight: 700;
+  }
+
+  em,
+  i {
+    font-style: italic;
+  }
+
+  sub {
+    font-size: smaller;
+    vertical-align: sub;
+  }
+
+  sup {
+    font-size: smaller;
+    vertical-align: super;
+  }
+
+  ul,
+  ol {
+    margin-inline-start: 2em;
+
+    p {
+      margin: 0;
+    }
+  }
+
+  ul {
+    list-style-type: disc;
+  }
+
+  ol {
+    list-style-type: decimal;
+  }
+}
+
+.reply-indicator__content {
+  blockquote {
+    border-inline-start-color: $inverted-text-color;
+    color: $inverted-text-color;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/rtl.scss b/app/javascript/flavours/blobfox/styles/rtl.scss
new file mode 100644
index 00000000000000..e69d5d789118cf
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/rtl.scss
@@ -0,0 +1,123 @@
+body.rtl {
+  direction: rtl;
+
+  .reactions-bar {
+    direction: rtl;
+  }
+
+  .drawer__inner__mastodon > img {
+    transform: scaleX(-1);
+  }
+
+  .boost-modal__status-time {
+    float: left;
+  }
+
+  .compose-form .autosuggest-textarea__textarea {
+    padding-right: 10px;
+    padding-left: 10px + 22px;
+  }
+
+  .columns-area {
+    direction: rtl;
+  }
+
+  .react-swipeable-view-container > * {
+    direction: rtl;
+  }
+
+  .account__avatar-wrapper {
+    float: right;
+  }
+
+  .column-header__setting-arrows {
+    float: left;
+  }
+
+  .setting-meta__label {
+    float: left;
+  }
+
+  .activity-stream .status.light {
+    padding-left: 10px;
+    padding-right: 68px;
+  }
+
+  .status__info .status__display-name,
+  .activity-stream .status.light .status__display-name {
+    padding-left: 25px;
+    padding-right: 0;
+  }
+
+  .activity-stream .pre-header {
+    padding-right: 68px;
+    padding-left: 0;
+  }
+
+  .activity-stream .pre-header .pre-header__icon {
+    left: auto;
+    right: 42px;
+  }
+
+  .account__header__tabs__buttons > .icon-button {
+    margin-right: 0;
+    margin-left: 8px;
+  }
+
+  .status__relative-time,
+  .activity-stream .status.light .status__header .status__meta {
+    float: left;
+    text-align: left;
+  }
+
+  .status__action-bar-button {
+    float: right;
+  }
+
+  .status__action-bar-dropdown {
+    float: right;
+  }
+
+  .detailed-status__display-name .display-name {
+    text-align: right;
+  }
+
+  .detailed-status__display-avatar {
+    float: right;
+  }
+
+  .admin-wrapper {
+    direction: rtl;
+  }
+
+  .simple_form .label_input__append {
+    &::after {
+      background-image: linear-gradient(
+        to left,
+        rgba(darken($ui-base-color, 10%), 0),
+        darken($ui-base-color, 10%)
+      );
+    }
+  }
+
+  .simple_form select {
+    background: darken($ui-base-color, 10%)
+      url("data:image/svg+xml;utf8,<svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 14.933 18.467' height='19.698' width='15.929'><path d='M3.467 14.967l-3.393-3.5H14.86l-3.392 3.5c-1.866 1.925-3.666 3.5-4 3.5-.335 0-2.135-1.575-4-3.5zm.266-11.234L7.467 0 11.2 3.733l3.733 3.734H0l3.733-3.734z' fill='#{hex-color(lighten($ui-base-color, 12%))}'/></svg>")
+      no-repeat left 8px center / auto 16px;
+  }
+
+  .fa-chevron-left::before {
+    content: '\F054';
+  }
+
+  .fa-chevron-right::before {
+    content: '\F053';
+  }
+
+  .dismissable-banner,
+  .warning-banner {
+    &__action {
+      float: left;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/statuses.scss b/app/javascript/flavours/blobfox/styles/statuses.scss
new file mode 100644
index 00000000000000..0a46cf855fb74b
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/statuses.scss
@@ -0,0 +1,232 @@
+.activity-stream {
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+  border-radius: 4px;
+  overflow: hidden;
+  margin-bottom: 10px;
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    margin-bottom: 0;
+    border-radius: 0;
+    box-shadow: none;
+  }
+
+  &--headless {
+    border-radius: 0;
+    margin: 0;
+    box-shadow: none;
+
+    .detailed-status,
+    .status {
+      border-radius: 0 !important;
+    }
+  }
+
+  div[data-component] {
+    width: 100%;
+  }
+
+  .entry {
+    background: $ui-base-color;
+
+    .detailed-status,
+    .status,
+    .load-more {
+      animation: none;
+    }
+
+    &:last-child {
+      .detailed-status,
+      .status,
+      .load-more {
+        border-bottom: 0;
+        border-radius: 0 0 4px 4px;
+      }
+    }
+
+    &:first-child {
+      .detailed-status,
+      .status,
+      .load-more {
+        border-radius: 4px 4px 0 0;
+      }
+
+      &:last-child {
+        .detailed-status,
+        .status,
+        .load-more {
+          border-radius: 4px;
+        }
+      }
+    }
+
+    @media screen and (width <= 740px) {
+      .detailed-status,
+      .status,
+      .load-more {
+        border-radius: 0 !important;
+      }
+    }
+  }
+
+  &--highlighted .entry {
+    background: lighten($ui-base-color, 8%);
+  }
+}
+
+.button.logo-button svg {
+  width: 20px;
+  height: auto;
+  vertical-align: middle;
+  margin-inline-end: 5px;
+  fill: $primary-text-color;
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    display: none;
+  }
+}
+
+.embed {
+  .status__content[data-spoiler='folded'] {
+    .e-content {
+      display: none;
+    }
+
+    p:first-child {
+      margin-bottom: 0;
+    }
+  }
+
+  .detailed-status {
+    padding: 15px;
+
+    .detailed-status__display-avatar .account__avatar {
+      width: 48px;
+      height: 48px;
+    }
+  }
+
+  .status {
+    padding: 15px;
+    padding-inline-start: (48px + 15px * 2);
+    min-height: 48px + 2px;
+
+    &__avatar {
+      inset-inline-start: 15px;
+      top: 17px;
+
+      .account__avatar {
+        width: 48px;
+        height: 48px;
+      }
+    }
+
+    &__content {
+      padding-top: 5px;
+    }
+
+    &__prepend {
+      padding: 8px 0;
+      padding-bottom: 2px;
+      margin: initial;
+      margin-inline-start: 48px + 15px * 2;
+      padding-top: 15px;
+    }
+
+    &__prepend-icon-wrapper {
+      position: absolute;
+      margin: initial;
+      float: initial;
+      width: auto;
+      inset-inline-start: -32px;
+    }
+
+    .media-gallery,
+    &__action-bar,
+    .video-player {
+      margin-top: 10px;
+    }
+
+    &__action-bar-button {
+      font-size: 18px;
+      width: 23.1429px;
+      height: 23.1429px;
+      line-height: 23.15px;
+    }
+  }
+}
+
+// Styling from upstream's WebUI, as public pages use the same layout
+.embed {
+  .status {
+    .status__info {
+      font-size: 15px;
+      display: initial;
+    }
+
+    .status__relative-time {
+      color: $dark-text-color;
+      float: right;
+      font-size: 14px;
+      width: auto;
+      margin: initial;
+      padding: initial;
+      padding-bottom: 1px;
+    }
+
+    .status__visibility-icon {
+      padding: 0 4px;
+    }
+
+    .status__info .status__display-name {
+      display: block;
+      max-width: 100%;
+      padding: 6px 0;
+      padding-right: 25px;
+      margin: initial;
+    }
+
+    .status__avatar {
+      height: 48px;
+      position: absolute;
+      width: 48px;
+      margin: initial;
+    }
+  }
+}
+
+.rtl {
+  .embed {
+    .status {
+      padding-left: 10px;
+      padding-right: 68px;
+
+      .status__info .status__display-name {
+        padding-left: 25px;
+        padding-right: 0;
+      }
+
+      .status__relative-time,
+      .status__visibility-icon {
+        float: left;
+      }
+    }
+  }
+}
+
+.status__content__read-more-button,
+.status__content__translate-button {
+  display: block;
+  font-size: 15px;
+  line-height: 20px;
+  color: $highlight-text-color;
+  border: 0;
+  background: transparent;
+  padding: 0;
+  padding-top: 16px;
+  text-decoration: none;
+
+  &:hover,
+  &:active {
+    text-decoration: underline;
+  }
+}
diff --git a/app/javascript/flavours/blobfox/styles/tables.scss b/app/javascript/flavours/blobfox/styles/tables.scss
new file mode 100644
index 00000000000000..44ef00ba7378ed
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/tables.scss
@@ -0,0 +1,376 @@
+.table {
+  width: 100%;
+  max-width: 100%;
+  border-spacing: 0;
+  border-collapse: collapse;
+
+  th,
+  td {
+    padding: 8px;
+    line-height: 18px;
+    vertical-align: top;
+    border-top: 1px solid $ui-base-color;
+    text-align: start;
+    background: darken($ui-base-color, 4%);
+
+    &.critical {
+      font-weight: 700;
+      color: $gold-star;
+    }
+  }
+
+  & > thead > tr > th {
+    vertical-align: bottom;
+    border-bottom: 2px solid $ui-base-color;
+    border-top: 0;
+    font-weight: 500;
+  }
+
+  & > tbody > tr > th {
+    font-weight: 500;
+  }
+
+  & > tbody > tr:nth-child(odd) > td,
+  & > tbody > tr:nth-child(odd) > th {
+    background: $ui-base-color;
+  }
+
+  a {
+    color: $highlight-text-color;
+    text-decoration: underline;
+
+    &:hover {
+      text-decoration: none;
+    }
+  }
+
+  strong {
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  &.inline-table {
+    & > tbody > tr:nth-child(odd) {
+      & > td,
+      & > th {
+        background: transparent;
+      }
+    }
+
+    & > tbody > tr:first-child {
+      & > td,
+      & > th {
+        border-top: 0;
+      }
+    }
+  }
+
+  &.horizontal-table {
+    border-collapse: collapse;
+    border-style: hidden;
+
+    & > tbody > tr > th,
+    & > tbody > tr > td {
+      padding: 11px 10px;
+      background: transparent;
+      border: 1px solid lighten($ui-base-color, 8%);
+      color: $secondary-text-color;
+    }
+
+    & > tbody > tr > th {
+      color: $darker-text-color;
+      font-weight: 600;
+    }
+  }
+
+  &.batch-table {
+    & > thead > tr > th {
+      background: $ui-base-color;
+      border-top: 1px solid darken($ui-base-color, 8%);
+      border-bottom: 1px solid darken($ui-base-color, 8%);
+
+      &:first-child {
+        border-radius: 4px 0 0;
+        border-inline-start: 1px solid darken($ui-base-color, 8%);
+      }
+
+      &:last-child {
+        border-radius: 0 4px 0 0;
+        border-inline-end: 1px solid darken($ui-base-color, 8%);
+      }
+    }
+  }
+
+  &--invites tbody td {
+    vertical-align: middle;
+  }
+}
+
+.table-wrapper {
+  overflow: auto;
+  margin-bottom: 20px;
+}
+
+samp {
+  font-family: $font-monospace, monospace;
+}
+
+button.table-action-link {
+  background: transparent;
+  border: 0;
+  font: inherit;
+}
+
+button.table-action-link,
+a.table-action-link {
+  text-decoration: none;
+  display: inline-block;
+  margin-inline-end: 5px;
+  padding: 0 10px;
+  color: $darker-text-color;
+  font-weight: 500;
+
+  &:hover {
+    color: $primary-text-color;
+  }
+
+  i.fa {
+    font-weight: 400;
+    margin-inline-end: 5px;
+  }
+
+  &:first-child {
+    padding-inline-start: 0;
+  }
+}
+
+.batch-table {
+  &__toolbar,
+  &__row {
+    display: flex;
+
+    &__select {
+      box-sizing: border-box;
+      padding: 8px 16px;
+      cursor: pointer;
+      min-height: 100%;
+
+      input {
+        margin-top: 8px;
+      }
+
+      &--aligned {
+        display: flex;
+        align-items: center;
+
+        input {
+          margin-top: 0;
+        }
+      }
+    }
+
+    &__actions,
+    &__content {
+      padding: 8px 0;
+      padding-inline-end: 16px;
+      flex: 1 1 auto;
+    }
+  }
+
+  &__toolbar {
+    position: sticky;
+    top: 0;
+    z-index: 1;
+    border: 1px solid darken($ui-base-color, 8%);
+    background: $ui-base-color;
+    border-radius: 4px 0 0;
+    height: 47px;
+    align-items: center;
+
+    &__actions {
+      text-align: end;
+      padding-inline-end: 16px - 5px;
+    }
+  }
+
+  &__select-all {
+    background: $ui-base-color;
+    height: 47px;
+    align-items: center;
+    justify-content: center;
+    border: 1px solid darken($ui-base-color, 8%);
+    border-top: 0;
+    color: $secondary-text-color;
+    display: none;
+
+    &.active {
+      display: flex;
+    }
+
+    .selected,
+    .not-selected {
+      display: none;
+
+      &.active {
+        display: block;
+      }
+    }
+
+    strong {
+      font-weight: 700;
+    }
+
+    span {
+      padding: 8px;
+      display: inline-block;
+    }
+
+    button {
+      background: transparent;
+      border: 0;
+      font: inherit;
+      color: $highlight-text-color;
+      border-radius: 4px;
+      font-weight: 700;
+      padding: 8px;
+
+      &:hover,
+      &:focus,
+      &:active {
+        background: lighten($ui-base-color, 8%);
+      }
+    }
+  }
+
+  &__form {
+    padding: 16px;
+    border: 1px solid darken($ui-base-color, 8%);
+    border-top: 0;
+    background: $ui-base-color;
+
+    .fields-row {
+      padding-top: 0;
+      margin-bottom: 0;
+    }
+  }
+
+  &__row {
+    border: 1px solid darken($ui-base-color, 8%);
+    border-top: 0;
+    background: darken($ui-base-color, 4%);
+
+    @media screen and (max-width: $no-gap-breakpoint) {
+      .optional &:first-child {
+        border-top: 1px solid darken($ui-base-color, 8%);
+      }
+    }
+
+    &:hover {
+      background: darken($ui-base-color, 2%);
+    }
+
+    &:nth-child(even) {
+      background: $ui-base-color;
+
+      &:hover {
+        background: lighten($ui-base-color, 2%);
+      }
+    }
+
+    &__content {
+      padding-top: 12px;
+      padding-bottom: 16px;
+      overflow: hidden;
+
+      &--unpadded {
+        padding: 0;
+      }
+
+      &--with-image {
+        display: flex;
+        align-items: center;
+      }
+
+      &__image {
+        flex: 0 0 auto;
+        display: flex;
+        justify-content: center;
+        align-items: center;
+        margin-inline-end: 10px;
+
+        .emojione {
+          width: 32px;
+          height: 32px;
+        }
+      }
+
+      &__text {
+        flex: 1 1 auto;
+      }
+
+      &__quote {
+        padding: 12px;
+        padding-top: 0;
+      }
+
+      &__extra {
+        flex: 0 0 auto;
+        text-align: end;
+        color: $darker-text-color;
+        font-weight: 500;
+      }
+    }
+
+    .directory__tag {
+      margin: 0;
+      width: 100%;
+
+      a {
+        background: transparent;
+        border-radius: 0;
+      }
+    }
+  }
+
+  &.optional .batch-table__toolbar,
+  &.optional .batch-table__row__select {
+    @media screen and (max-width: $no-gap-breakpoint) {
+      display: none;
+    }
+  }
+
+  .status__content {
+    padding-top: 0;
+
+    strong {
+      font-weight: 700;
+    }
+  }
+
+  .nothing-here {
+    border: 1px solid darken($ui-base-color, 8%);
+    border-top: 0;
+    box-shadow: none;
+
+    @media screen and (max-width: $no-gap-breakpoint) {
+      border-top: 1px solid darken($ui-base-color, 8%);
+    }
+  }
+
+  @media screen and (width <= 870px) {
+    .accounts-table tbody td.optional {
+      display: none;
+    }
+  }
+}
+
+.one-liner {
+  white-space: nowrap;
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
diff --git a/app/javascript/flavours/blobfox/styles/variables.scss b/app/javascript/flavours/blobfox/styles/variables.scss
new file mode 100644
index 00000000000000..0b5d6f4067b4d6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/variables.scss
@@ -0,0 +1,103 @@
+// Commonly used web colors
+$black: #000000; // Black
+$white: #ffffff; // White
+$red-600: #b7253d !default; // Deep Carmine
+$red-500: #df405a !default; // Cerise
+$blurple-600: #563acc; // Iris
+$blurple-500: #6364ff; // Brand purple
+$blurple-300: #858afa; // Faded Blue
+$grey-600: #4e4c5a; // Trout
+$grey-100: #dadaf3; // Topaz
+
+$success-green: #79bd9a !default; // Padua
+$error-red: $red-500 !default; // Cerise
+$warning-red: #ff5050 !default; // Sunset Orange
+$gold-star: #ca8f04 !default; // Dark Goldenrod
+
+$red-bookmark: $warning-red;
+
+// Values from the classic Mastodon UI
+$classic-base-color: #282c37; // Midnight Express
+$classic-primary-color: #9baec8; // Echo Blue
+$classic-secondary-color: #d9e1e8; // Pattens Blue
+$classic-highlight-color: #6364ff; // Brand purple
+
+// Variables for defaults in UI
+$base-shadow-color: $black !default;
+$base-overlay-background: $black !default;
+$base-border-color: $white !default;
+$simple-background-color: $white !default;
+$valid-value-color: $success-green !default;
+$error-value-color: $error-red !default;
+
+// Tell UI to use selected colors
+$ui-base-color: $classic-base-color !default; // Darkest
+$ui-base-lighter-color: lighten(
+  $ui-base-color,
+  26%
+) !default; // Lighter darkest
+$ui-primary-color: $classic-primary-color !default; // Lighter
+$ui-secondary-color: $classic-secondary-color !default; // Lightest
+$ui-highlight-color: $classic-highlight-color !default;
+$ui-button-color: $white !default;
+$ui-button-background-color: $blurple-500 !default;
+$ui-button-focus-background-color: $blurple-600 !default;
+
+$ui-button-secondary-color: $grey-100 !default;
+$ui-button-secondary-border-color: $grey-100 !default;
+$ui-button-secondary-focus-background-color: $grey-600 !default;
+$ui-button-secondary-focus-color: $white !default;
+
+$ui-button-tertiary-color: $blurple-300 !default;
+$ui-button-tertiary-border-color: $blurple-300 !default;
+$ui-button-tertiary-focus-background-color: $blurple-600 !default;
+$ui-button-tertiary-focus-color: $white !default;
+
+$ui-button-destructive-background-color: $red-500 !default;
+$ui-button-destructive-focus-background-color: $red-600 !default;
+
+// Variables for texts
+$primary-text-color: $white !default;
+$darker-text-color: $ui-primary-color !default;
+$dark-text-color: $ui-base-lighter-color !default;
+$secondary-text-color: $ui-secondary-color !default;
+$highlight-text-color: lighten($ui-highlight-color, 8%) !default;
+$action-button-color: $ui-base-lighter-color !default;
+$action-button-focus-color: lighten($ui-base-lighter-color, 4%) !default;
+$passive-text-color: $gold-star !default;
+$active-passive-text-color: $success-green !default;
+
+// For texts on inverted backgrounds
+$inverted-text-color: $ui-base-color !default;
+$lighter-text-color: $ui-base-lighter-color !default;
+$light-text-color: $ui-primary-color !default;
+
+// Language codes that uses CJK fonts
+$cjk-langs: ja, ko, zh-CN, zh-HK, zh-TW;
+
+// Variables for components
+$media-modal-media-max-width: 100%;
+
+// put margins on top and bottom of image to avoid the screen covered by image.
+$media-modal-media-max-height: 80%;
+
+$no-gap-breakpoint: 1175px;
+
+$font-sans-serif: 'mastodon-font-sans-serif' !default;
+$font-display: 'mastodon-font-display' !default;
+$font-monospace: 'mastodon-font-monospace' !default;
+
+// Avatar border size (8% default, 100% for rounded avatars)
+$ui-avatar-border-size: 8%;
+
+// More variables
+$dismiss-overlay-width: 4rem;
+
+:root {
+  --dropdown-border-color: #{lighten($ui-base-color, 12%)};
+  --dropdown-background-color: #{lighten($ui-base-color, 4%)};
+  --dropdown-shadow: 0 20px 25px -5px #{rgba($base-shadow-color, 0.25)},
+    0 8px 10px -6px #{rgba($base-shadow-color, 0.25)};
+  --modal-background-color: #{darken($ui-base-color, 4%)};
+  --modal-border-color: #{lighten($ui-base-color, 4%)};
+}
diff --git a/app/javascript/flavours/blobfox/styles/widgets.scss b/app/javascript/flavours/blobfox/styles/widgets.scss
new file mode 100644
index 00000000000000..f54d2f2e587620
--- /dev/null
+++ b/app/javascript/flavours/blobfox/styles/widgets.scss
@@ -0,0 +1,402 @@
+@use 'sass:math';
+
+.hero-widget {
+  margin-bottom: 10px;
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+
+  &__img {
+    width: 100%;
+    position: relative;
+    overflow: hidden;
+    border-radius: 4px 4px 0 0;
+    background: $base-shadow-color;
+
+    img {
+      object-fit: cover;
+      display: block;
+      width: 100%;
+      height: 100%;
+      margin: 0;
+      border-radius: 4px 4px 0 0;
+    }
+  }
+
+  &__text {
+    background: $ui-base-color;
+    padding: 20px;
+    border-radius: 0 0 4px 4px;
+    font-size: 15px;
+    color: $darker-text-color;
+    line-height: 20px;
+    word-wrap: break-word;
+    font-weight: 400;
+
+    .emojione {
+      width: 20px;
+      height: 20px;
+      margin: -3px 0 0;
+    }
+
+    p {
+      margin-bottom: 20px;
+
+      &:last-child {
+        margin-bottom: 0;
+      }
+    }
+
+    em {
+      display: inline;
+      margin: 0;
+      padding: 0;
+      font-weight: 700;
+      background: transparent;
+      font-family: inherit;
+      font-size: inherit;
+      line-height: inherit;
+      color: lighten($darker-text-color, 10%);
+    }
+
+    a {
+      color: $secondary-text-color;
+      text-decoration: none;
+
+      &:hover {
+        text-decoration: underline;
+      }
+    }
+  }
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    display: none;
+  }
+}
+
+.endorsements-widget {
+  margin-bottom: 10px;
+  padding-bottom: 10px;
+
+  h4 {
+    padding: 10px;
+    text-transform: uppercase;
+    font-weight: 700;
+    font-size: 13px;
+    color: $darker-text-color;
+  }
+
+  .account {
+    padding: 10px 0;
+
+    &:last-child {
+      border-bottom: 0;
+    }
+
+    .account__display-name {
+      display: flex;
+      align-items: center;
+    }
+  }
+
+  .trends__item {
+    padding: 10px;
+  }
+}
+
+.trends-widget {
+  h4 {
+    color: $darker-text-color;
+  }
+}
+
+.placeholder-widget {
+  padding: 16px;
+  border-radius: 4px;
+  border: 2px dashed $dark-text-color;
+  text-align: center;
+  color: $darker-text-color;
+  margin-bottom: 10px;
+}
+
+.moved-account-widget {
+  padding: 15px;
+  padding-bottom: 20px;
+  border-radius: 4px;
+  background: $ui-base-color;
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+  color: $secondary-text-color;
+  font-weight: 400;
+  margin-bottom: 10px;
+
+  strong,
+  a {
+    font-weight: 500;
+
+    @each $lang in $cjk-langs {
+      &:lang(#{$lang}) {
+        font-weight: 700;
+      }
+    }
+  }
+
+  a {
+    color: inherit;
+    text-decoration: underline;
+
+    &.mention {
+      text-decoration: none;
+
+      span {
+        text-decoration: none;
+      }
+
+      &:focus,
+      &:hover,
+      &:active {
+        text-decoration: none;
+
+        span {
+          text-decoration: underline;
+        }
+      }
+    }
+  }
+
+  &__message {
+    margin-bottom: 15px;
+
+    .fa {
+      margin-inline-end: 5px;
+      color: $darker-text-color;
+    }
+  }
+
+  &__card {
+    .detailed-status__display-avatar {
+      position: relative;
+      cursor: pointer;
+    }
+
+    .detailed-status__display-name {
+      margin-bottom: 0;
+      text-decoration: none;
+
+      span {
+        font-weight: 400;
+      }
+    }
+  }
+}
+
+.memoriam-widget {
+  padding: 20px;
+  border-radius: 4px;
+  background: $base-shadow-color;
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+  font-size: 14px;
+  color: $darker-text-color;
+  margin-bottom: 10px;
+}
+
+.directory {
+  background: $ui-base-color;
+  border-radius: 4px;
+  box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+
+  &__tag {
+    box-sizing: border-box;
+    margin-bottom: 10px;
+
+    & > a,
+    & > div {
+      display: flex;
+      align-items: center;
+      justify-content: space-between;
+      background: $ui-base-color;
+      border-radius: 4px;
+      padding: 15px;
+      text-decoration: none;
+      color: inherit;
+      box-shadow: 0 0 15px rgba($base-shadow-color, 0.2);
+    }
+
+    & > a {
+      &:hover,
+      &:active,
+      &:focus {
+        background: lighten($ui-base-color, 8%);
+      }
+    }
+
+    &.active > a {
+      background: $ui-highlight-color;
+      cursor: default;
+    }
+
+    &.disabled > div {
+      opacity: 0.5;
+      cursor: default;
+    }
+
+    h4 {
+      flex: 1 1 auto;
+      font-size: 18px;
+      font-weight: 700;
+      color: $primary-text-color;
+      white-space: nowrap;
+      overflow: hidden;
+      text-overflow: ellipsis;
+
+      .fa {
+        color: $darker-text-color;
+      }
+
+      small {
+        display: block;
+        font-weight: 400;
+        font-size: 15px;
+        margin-top: 8px;
+        color: $darker-text-color;
+      }
+    }
+
+    &.active h4 {
+      &,
+      .fa,
+      small {
+        color: $primary-text-color;
+      }
+    }
+
+    .avatar-stack {
+      flex: 0 0 auto;
+      width: (36px + 4px) * 3;
+    }
+
+    &.active .avatar-stack .account__avatar {
+      border-color: $ui-highlight-color;
+    }
+  }
+}
+
+.accounts-table {
+  width: 100%;
+
+  .account {
+    padding: 0;
+    border: 0;
+  }
+
+  strong {
+    font-weight: 700;
+  }
+
+  thead th {
+    text-align: center;
+    text-transform: uppercase;
+    color: $darker-text-color;
+    font-weight: 700;
+    padding: 10px;
+
+    &:first-child {
+      text-align: start;
+    }
+  }
+
+  tbody td {
+    padding: 15px 0;
+    vertical-align: middle;
+    border-bottom: 1px solid lighten($ui-base-color, 8%);
+  }
+
+  tbody tr:last-child td {
+    border-bottom: 0;
+  }
+
+  &__count {
+    width: 120px;
+    text-align: center;
+    font-size: 15px;
+    font-weight: 500;
+    color: $primary-text-color;
+
+    small {
+      display: block;
+      color: $darker-text-color;
+      font-weight: 400;
+      font-size: 14px;
+    }
+  }
+
+  tbody td.accounts-table__extra {
+    width: 120px;
+    text-align: end;
+    color: $darker-text-color;
+    padding-inline-end: 16px;
+
+    a {
+      text-decoration: none;
+      color: inherit;
+
+      &:focus,
+      &:hover,
+      &:active {
+        text-decoration: underline;
+      }
+    }
+  }
+
+  &__comment {
+    width: 50%;
+    vertical-align: initial !important;
+  }
+
+  &__interrelationships {
+    width: 21px;
+  }
+
+  .fa {
+    font-size: 16px;
+
+    &.active {
+      color: $highlight-text-color;
+    }
+
+    &.passive {
+      color: $passive-text-color;
+    }
+
+    &.active.passive {
+      color: $active-passive-text-color;
+    }
+  }
+
+  @media screen and (max-width: $no-gap-breakpoint) {
+    tbody td.optional {
+      display: none;
+    }
+  }
+}
+
+.moved-account-widget,
+.memoriam-widget,
+.directory {
+  @media screen and (max-width: $no-gap-breakpoint) {
+    margin-bottom: 0;
+    box-shadow: none;
+    border-radius: 0;
+  }
+}
+
+.placeholder-widget {
+  a {
+    text-decoration: none;
+    font-weight: 500;
+    color: $ui-highlight-color;
+
+    &:hover,
+    &:focus,
+    &:active {
+      text-decoration: underline;
+    }
+  }
+}
diff --git a/app/javascript/flavours/blobfox/theme.yml b/app/javascript/flavours/blobfox/theme.yml
new file mode 100644
index 00000000000000..fc7a21dcc63e03
--- /dev/null
+++ b/app/javascript/flavours/blobfox/theme.yml
@@ -0,0 +1,48 @@
+#  (REQUIRED) The location of the pack files.
+pack:
+  admin:
+    - packs/admin.jsx
+    - packs/public.jsx
+  auth: packs/public.jsx
+  common:
+    filename: packs/common.js
+    stylesheet: true
+  embed: packs/public.jsx
+  error: packs/error.js
+  home:
+    filename: packs/home.js
+    preload:
+      - flavours/blobfox/async/compose
+      - flavours/blobfox/async/home_timeline
+      - flavours/blobfox/async/notifications
+  mailer:
+  modal:
+  public: packs/public.jsx
+  settings: packs/settings.js
+  sign_up: packs/sign_up.js
+  share: packs/share.jsx
+
+#  (OPTIONAL) The directory which contains localization files for
+#  the flavour, relative to this directory. The contents of this
+#  directory must be `.json` files whose names correspond to
+#  language tags and whose default exports are a messages object.
+locales: locales
+
+#  (OPTIONAL) Which flavour to inherit locales from
+inherit_locales: vanilla
+
+#  (OPTIONAL) A file to use as the preview screenshot for the flavour,
+#  or an array thereof. These are the full path from `app/javascript/`.
+screenshot: flavours/blobfox/images/blobfox-preview.jpg
+
+#  (OPTIONAL) The directory which contains the pack files.
+#  Defaults to the theme directory (`app/javascript/themes/[theme]`),
+#  which should be sufficient for like 99% of use-cases lol.
+
+#      pack_directory: app/javascript/packs
+
+#  (OPTIONAL) By default the theme will fallback to the default theme
+#  if a particular pack is not provided. You can specify different
+#  fallbacks here, or disable fallback behaviours altogether by
+#  specifying a `null` value.
+fallback:
diff --git a/app/javascript/flavours/blobfox/types/util.ts b/app/javascript/flavours/blobfox/types/util.ts
new file mode 100644
index 00000000000000..5f2cf2cf07f3f9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/types/util.ts
@@ -0,0 +1 @@
+export type ValueOf<T> = T[keyof T];
diff --git a/app/javascript/flavours/blobfox/utils/backend_links.js b/app/javascript/flavours/blobfox/utils/backend_links.js
new file mode 100644
index 00000000000000..2028a1e60852b7
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/backend_links.js
@@ -0,0 +1,18 @@
+export const preferencesLink = '/settings/preferences';
+export const profileLink = '/settings/profile';
+export const signOutLink = '/auth/sign_out';
+export const privacyPolicyLink = '/privacy-policy';
+export const accountAdminLink = (id) => `/admin/accounts/${id}`;
+export const statusAdminLink = (account_id, status_id) => `/admin/accounts/${account_id}/statuses/${status_id}`;
+export const filterEditLink = (id) => `/filters/${id}/edit`;
+export const relationshipsLink = '/relationships';
+export const securityLink = '/auth/edit';
+export const preferenceLink = (setting_name) => {
+  switch (setting_name) {
+  case 'user_setting_expand_spoilers':
+  case 'user_setting_disable_swiping':
+    return `/settings/preferences/appearance#${setting_name}`;
+  default:
+    return preferencesLink;
+  }
+};
diff --git a/app/javascript/flavours/blobfox/utils/base64.ts b/app/javascript/flavours/blobfox/utils/base64.ts
new file mode 100644
index 00000000000000..5a595ee12b5098
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/base64.ts
@@ -0,0 +1,10 @@
+export const decode = (base64: string): Uint8Array => {
+  const rawData = window.atob(base64);
+  const outputArray = new Uint8Array(rawData.length);
+
+  for (let i = 0; i < rawData.length; ++i) {
+    outputArray[i] = rawData.charCodeAt(i);
+  }
+
+  return outputArray;
+};
diff --git a/app/javascript/flavours/blobfox/utils/config.js b/app/javascript/flavours/blobfox/utils/config.js
new file mode 100644
index 00000000000000..932cd0cbf543e1
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/config.js
@@ -0,0 +1,10 @@
+import ready from '../ready';
+
+export let assetHost = '';
+
+ready(() => {
+  const cdnHost = document.querySelector('meta[name=cdn-host]');
+  if (cdnHost) {
+    assetHost = cdnHost.content || '';
+  }
+});
diff --git a/app/javascript/flavours/blobfox/utils/content_warning.js b/app/javascript/flavours/blobfox/utils/content_warning.js
new file mode 100644
index 00000000000000..18c21e795f819c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/content_warning.js
@@ -0,0 +1,31 @@
+import { expandSpoilers } from 'flavours/blobfox/initial_state';
+
+function _autoUnfoldCW(spoiler_text, skip_unfold_regex) {
+  if (!expandSpoilers)
+    return false;
+
+  if (!skip_unfold_regex)
+    return true;
+
+  let regex = null;
+
+  try {
+    regex = new RegExp(skip_unfold_regex.trim(), 'i');
+  } catch (e) {
+    // Bad regex, skip filters
+    return true;
+  }
+
+  return !regex.test(spoiler_text);
+}
+
+export function autoHideCW(settings, spoiler_text) {
+  return !_autoUnfoldCW(spoiler_text, settings.getIn(['content_warnings', 'filter']));
+}
+
+export function autoUnfoldCW(settings, status) {
+  if (!status)
+    return false;
+
+  return _autoUnfoldCW(status.get('spoiler_text'), settings.getIn(['content_warnings', 'filter']));
+}
diff --git a/app/javascript/flavours/blobfox/utils/filters.ts b/app/javascript/flavours/blobfox/utils/filters.ts
new file mode 100644
index 00000000000000..d299e80c40a91e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/filters.ts
@@ -0,0 +1,16 @@
+export const toServerSideType = (columnType: string) => {
+  switch (columnType) {
+    case 'home':
+    case 'notifications':
+    case 'public':
+    case 'thread':
+    case 'account':
+      return columnType;
+    default:
+      if (columnType.includes('list:')) {
+        return 'home';
+      } else {
+        return 'public'; // community, account, hashtag
+      }
+  }
+};
diff --git a/app/javascript/flavours/blobfox/utils/hashtag.js b/app/javascript/flavours/blobfox/utils/hashtag.js
new file mode 100644
index 00000000000000..6c529dce8e5d72
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/hashtag.js
@@ -0,0 +1,8 @@
+export function recoverHashtags (recognizedTags, text) {
+  return recognizedTags.map(tag => {
+    const re = new RegExp(`(?:^|[^/)\\w])#(${tag.name})`, 'i');
+    const matched_hashtag = text.match(re);
+    return matched_hashtag ? matched_hashtag[1] : null;
+  },
+  ).filter(x => x !== null);
+}
diff --git a/app/javascript/flavours/blobfox/utils/hashtags.ts b/app/javascript/flavours/blobfox/utils/hashtags.ts
new file mode 100644
index 00000000000000..0c5505c6c9a088
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/hashtags.ts
@@ -0,0 +1,29 @@
+const HASHTAG_SEPARATORS = '_\\u00b7\\u200c';
+const ALPHA = '\\p{L}\\p{M}';
+const WORD = '\\p{L}\\p{M}\\p{N}\\p{Pc}';
+
+const buildHashtagPatternRegex = () => {
+  try {
+    return new RegExp(
+      `(?:^|[^\\/\\)\\w])#(([${WORD}_][${WORD}${HASHTAG_SEPARATORS}]*[${ALPHA}${HASHTAG_SEPARATORS}][${WORD}${HASHTAG_SEPARATORS}]*[${WORD}_])|([${WORD}_]*[${ALPHA}][${WORD}_]*))`,
+      'iu',
+    );
+  } catch {
+    return /(?:^|[^/)\w])#(\w*[a-zA-Z·]\w*)/i;
+  }
+};
+
+const buildHashtagRegex = () => {
+  try {
+    return new RegExp(
+      `^(([${WORD}_][${WORD}${HASHTAG_SEPARATORS}]*[${ALPHA}${HASHTAG_SEPARATORS}][${WORD}${HASHTAG_SEPARATORS}]*[${WORD}_])|([${WORD}_]*[${ALPHA}][${WORD}_]*))$`,
+      'iu',
+    );
+  } catch {
+    return /^(\w*[a-zA-Z·]\w*)$/i;
+  }
+};
+
+export const HASHTAG_PATTERN_REGEX = buildHashtagPatternRegex();
+
+export const HASHTAG_REGEX = buildHashtagRegex();
diff --git a/app/javascript/flavours/blobfox/utils/html.js b/app/javascript/flavours/blobfox/utils/html.js
new file mode 100644
index 00000000000000..247e98c88a7f31
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/html.js
@@ -0,0 +1,6 @@
+// NB: This function can still return unsafe HTML
+export const unescapeHTML = (html) => {
+  const wrapper = document.createElement('div');
+  wrapper.innerHTML = html.replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n').replace(/<[^>]*>/g, '');
+  return wrapper.textContent;
+};
diff --git a/app/javascript/flavours/blobfox/utils/icons.jsx b/app/javascript/flavours/blobfox/utils/icons.jsx
new file mode 100644
index 00000000000000..be566032e06445
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/icons.jsx
@@ -0,0 +1,13 @@
+// Copied from emoji-mart for consistency with emoji picker and since
+// they don't export the icons in the package
+export const loupeIcon = (
+  <svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20' width='13' height='13'>
+    <path d='M12.9 14.32a8 8 0 1 1 1.41-1.41l5.35 5.33-1.42 1.42-5.33-5.34zM8 14A6 6 0 1 0 8 2a6 6 0 0 0 0 12z' />
+  </svg>
+);
+
+export const deleteIcon = (
+  <svg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 20 20' width='13' height='13'>
+    <path d='M10 8.586L2.929 1.515 1.515 2.929 8.586 10l-7.071 7.071 1.414 1.414L10 11.414l7.071 7.071 1.414-1.414L11.414 10l7.071-7.071-1.414-1.414L10 8.586z' />
+  </svg>
+);
diff --git a/app/javascript/flavours/blobfox/utils/idna.js b/app/javascript/flavours/blobfox/utils/idna.js
new file mode 100644
index 00000000000000..efab5bacf77a0c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/idna.js
@@ -0,0 +1,10 @@
+import punycode from 'punycode';
+
+const IDNA_PREFIX = 'xn--';
+
+export const decode = domain => {
+  return domain
+    .split('.')
+    .map(part => part.indexOf(IDNA_PREFIX) === 0 ? punycode.decode(part.slice(IDNA_PREFIX.length)) : part)
+    .join('.');
+};
diff --git a/app/javascript/flavours/blobfox/utils/js_helpers.js b/app/javascript/flavours/blobfox/utils/js_helpers.js
new file mode 100644
index 00000000000000..2ebd5b6c55ea68
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/js_helpers.js
@@ -0,0 +1,5 @@
+//  This function returns the new value unless it is `null` or
+//  `undefined`, in which case it returns the old one.
+export function overwrite (oldVal, newVal) {
+  return newVal === null || typeof newVal === 'undefined' ? oldVal : newVal;
+}
diff --git a/app/javascript/flavours/blobfox/utils/log_out.js b/app/javascript/flavours/blobfox/utils/log_out.js
new file mode 100644
index 00000000000000..28123890f1480a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/log_out.js
@@ -0,0 +1,35 @@
+import Rails from '@rails/ujs';
+
+import { signOutLink } from 'flavours/blobfox/utils/backend_links';
+
+export const logOut = () => {
+  const form = document.createElement('form');
+
+  const methodInput = document.createElement('input');
+  methodInput.setAttribute('name', '_method');
+  methodInput.setAttribute('value', 'delete');
+  methodInput.setAttribute('type', 'hidden');
+  form.appendChild(methodInput);
+
+  const csrfToken = Rails.csrfToken();
+  const csrfParam = Rails.csrfParam();
+
+  if (csrfParam && csrfToken) {
+    const csrfInput = document.createElement('input');
+    csrfInput.setAttribute('name', csrfParam);
+    csrfInput.setAttribute('value', csrfToken);
+    csrfInput.setAttribute('type', 'hidden');
+    form.appendChild(csrfInput);
+  }
+
+  const submitButton = document.createElement('input');
+  submitButton.setAttribute('type', 'submit');
+  form.appendChild(submitButton);
+
+  form.method = 'post';
+  form.action = signOutLink;
+  form.style.display = 'none';
+
+  document.body.appendChild(form);
+  submitButton.click();
+};
diff --git a/app/javascript/flavours/blobfox/utils/notifications.js b/app/javascript/flavours/blobfox/utils/notifications.js
new file mode 100644
index 00000000000000..42623ac7c6898c
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/notifications.js
@@ -0,0 +1,30 @@
+// Handles browser quirks, based on
+// https://developer.mozilla.org/en-US/docs/Web/API/Notifications_API/Using_the_Notifications_API
+
+const checkNotificationPromise = () => {
+  try {
+    // eslint-disable-next-line promise/valid-params, promise/catch-or-return
+    Notification.requestPermission().then();
+  } catch(e) {
+    return false;
+  }
+
+  return true;
+};
+
+const handlePermission = (permission, callback) => {
+  // Whatever the user answers, we make sure Chrome stores the information
+  if(!('permission' in Notification)) {
+    Notification.permission = permission;
+  }
+
+  callback(Notification.permission);
+};
+
+export const requestNotificationPermission = (callback) => {
+  if (checkNotificationPromise()) {
+    Notification.requestPermission().then((permission) => handlePermission(permission, callback)).catch(console.warn);
+  } else {
+    Notification.requestPermission((permission) => handlePermission(permission, callback));
+  }
+};
diff --git a/app/javascript/flavours/blobfox/utils/numbers.ts b/app/javascript/flavours/blobfox/utils/numbers.ts
new file mode 100644
index 00000000000000..35bcde83e2491a
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/numbers.ts
@@ -0,0 +1,71 @@
+import type { ValueOf } from '../types/util';
+
+export const DECIMAL_UNITS = Object.freeze({
+  ONE: 1,
+  TEN: 10,
+  HUNDRED: 100,
+  THOUSAND: 1_000,
+  MILLION: 1_000_000,
+  BILLION: 1_000_000_000,
+  TRILLION: 1_000_000_000_000,
+});
+export type DecimalUnits = ValueOf<typeof DECIMAL_UNITS>;
+
+const TEN_THOUSAND = DECIMAL_UNITS.THOUSAND * 10;
+const TEN_MILLIONS = DECIMAL_UNITS.MILLION * 10;
+
+export type ShortNumber = [number, DecimalUnits, 0 | 1]; // Array of: shorten number, unit of shorten number and maximum fraction digits
+
+/**
+ * @param sourceNumber Number to convert to short number
+ * @returns Calculated short number
+ * @example
+ * shortNumber(5936);
+ * // => [5.936, 1000, 1]
+ */
+export function toShortNumber(sourceNumber: number): ShortNumber {
+  if (sourceNumber < DECIMAL_UNITS.THOUSAND) {
+    return [sourceNumber, DECIMAL_UNITS.ONE, 0];
+  } else if (sourceNumber < DECIMAL_UNITS.MILLION) {
+    return [
+      sourceNumber / DECIMAL_UNITS.THOUSAND,
+      DECIMAL_UNITS.THOUSAND,
+      sourceNumber < TEN_THOUSAND ? 1 : 0,
+    ];
+  } else if (sourceNumber < DECIMAL_UNITS.BILLION) {
+    return [
+      sourceNumber / DECIMAL_UNITS.MILLION,
+      DECIMAL_UNITS.MILLION,
+      sourceNumber < TEN_MILLIONS ? 1 : 0,
+    ];
+  } else if (sourceNumber < DECIMAL_UNITS.TRILLION) {
+    return [sourceNumber / DECIMAL_UNITS.BILLION, DECIMAL_UNITS.BILLION, 0];
+  }
+
+  return [sourceNumber, DECIMAL_UNITS.ONE, 0];
+}
+
+/**
+ * @param sourceNumber Original number that is shortened
+ * @param division The scale in which short number is displayed
+ * @returns Number that can be used for plurals when short form used
+ * @example
+ * pluralReady(1793, DECIMAL_UNITS.THOUSAND)
+ * // => 1790
+ */
+export function pluralReady(
+  sourceNumber: number,
+  division: DecimalUnits | null,
+): number {
+  if (division == null || division < DECIMAL_UNITS.HUNDRED) {
+    return sourceNumber;
+  }
+
+  const closestScale = division / DECIMAL_UNITS.TEN;
+
+  return Math.trunc(sourceNumber / closestScale) * closestScale;
+}
+
+export function roundTo10(num: number): number {
+  return Math.round(num * 0.1) / 0.1;
+}
diff --git a/app/javascript/flavours/blobfox/utils/privacy_preference.js b/app/javascript/flavours/blobfox/utils/privacy_preference.js
new file mode 100644
index 00000000000000..51bdf072d78e84
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/privacy_preference.js
@@ -0,0 +1,5 @@
+export const order = ['public', 'unlisted', 'private', 'direct'];
+
+export function privacyPreference (a, b) {
+  return order[Math.max(order.indexOf(a), order.indexOf(b), 0)];
+}
diff --git a/app/javascript/flavours/blobfox/utils/react_helpers.js b/app/javascript/flavours/blobfox/utils/react_helpers.js
new file mode 100644
index 00000000000000..ea11acdb6188e2
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/react_helpers.js
@@ -0,0 +1,21 @@
+//  This function binds the given `handlers` to the `target`.
+export function assignHandlers (target, handlers) {
+  if (!target || !handlers) {
+    return;
+  }
+
+  //  We just bind each handler to the `target`.
+  const handle = target.handlers = {};
+  Object.keys(handlers).forEach(
+    key => handle[key] = handlers[key].bind(target),
+  );
+}
+
+//  This function only returns the component if the result of calling
+//  `test` with `data` is `true`.  Useful with funciton binding.
+export function conditionalRender (test, data, component) {
+  return test(data) ? component : null;
+}
+
+//  This object provides props to make the component not visible.
+export const hiddenComponent = { style: { display: 'none' } };
diff --git a/app/javascript/flavours/blobfox/utils/react_router.jsx b/app/javascript/flavours/blobfox/utils/react_router.jsx
new file mode 100644
index 00000000000000..fa8f0db2b5cbd6
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/react_router.jsx
@@ -0,0 +1,61 @@
+import PropTypes from "prop-types";
+
+import { __RouterContext } from "react-router";
+
+import hoistStatics from "hoist-non-react-statics";
+
+export const WithRouterPropTypes = {
+  match: PropTypes.object.isRequired,
+  location: PropTypes.object.isRequired,
+  history: PropTypes.object.isRequired,
+};
+
+export const WithOptionalRouterPropTypes = {
+  match: PropTypes.object,
+  location: PropTypes.object,
+  history: PropTypes.object,
+};
+
+// This is copied from https://github.com/remix-run/react-router/blob/v5.3.4/packages/react-router/modules/withRouter.js
+// but does not fail if called outside of a React Router context
+export function withOptionalRouter(Component) {
+  const displayName = `withRouter(${Component.displayName || Component.name})`;
+  const C = props => {
+    const { wrappedComponentRef, ...remainingProps } = props;
+
+    return (
+      <__RouterContext.Consumer>
+        {context => {
+          if(context)
+            return (
+              <Component
+                {...remainingProps}
+                {...context}
+                ref={wrappedComponentRef}
+              />
+            );
+          else
+            return (
+              <Component
+                {...remainingProps}
+                ref={wrappedComponentRef}
+              />
+            );
+        }}
+      </__RouterContext.Consumer>
+    );
+  };
+
+  C.displayName = displayName;
+  C.WrappedComponent = Component;
+  C.propTypes = {
+    ...Component.propTypes,
+    wrappedComponentRef: PropTypes.oneOfType([
+      PropTypes.string,
+      PropTypes.func,
+      PropTypes.object
+    ])
+  };
+
+  return hoistStatics(C, Component);
+}
diff --git a/app/javascript/flavours/blobfox/utils/resize_image.js b/app/javascript/flavours/blobfox/utils/resize_image.js
new file mode 100644
index 00000000000000..e3d4e6a354b9e9
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/resize_image.js
@@ -0,0 +1,191 @@
+import EXIF from 'exif-js';
+
+const MAX_IMAGE_PIXELS = 2073600; // 1920x1080px
+
+const _browser_quirks = {};
+
+// Some browsers will automatically draw images respecting their EXIF orientation
+// while others won't, and the safest way to detect that is to examine how it
+// is done on a known image.
+// See https://github.com/w3c/csswg-drafts/issues/4666
+// and https://github.com/blueimp/JavaScript-Load-Image/commit/1e4df707821a0afcc11ea0720ee403b8759f3881
+const dropOrientationIfNeeded = (orientation) => new Promise(resolve => {
+  switch (_browser_quirks['image-orientation-automatic']) {
+  case true:
+    resolve(1);
+    break;
+  case false:
+    resolve(orientation);
+    break;
+  default:
+    // black 2x1 JPEG, with the following meta information set:
+    // - EXIF Orientation: 6 (Rotated 90° CCW)
+    const testImageURL =
+      '' +
+      'AAAD/2wCEAAEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBA' +
+      'QEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQE' +
+      'BAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAf/AABEIAAEAAgMBEQACEQEDEQH/x' +
+      'ABKAAEAAAAAAAAAAAAAAAAAAAALEAEAAAAAAAAAAAAAAAAAAAAAAQEAAAAAAAAAAAAAAAA' +
+      'AAAAAEQEAAAAAAAAAAAAAAAAAAAAA/9oADAMBAAIRAxEAPwA/8H//2Q==';
+    const img = new Image();
+    img.onload = () => {
+      const automatic = (img.width === 1 && img.height === 2);
+      _browser_quirks['image-orientation-automatic'] = automatic;
+      resolve(automatic ? 1 : orientation);
+    };
+    img.onerror = () => {
+      _browser_quirks['image-orientation-automatic'] = false;
+      resolve(orientation);
+    };
+    img.src = testImageURL;
+  }
+});
+
+// Some browsers don't allow reading from a canvas and instead return all-white
+// or randomized data. Use a pre-defined image to check if reading the canvas
+// works.
+const checkCanvasReliability = () => new Promise((resolve, reject) => {
+  switch(_browser_quirks['canvas-read-unreliable']) {
+  case true:
+    reject('Canvas reading unreliable');
+    break;
+  case false:
+    resolve();
+    break;
+  default:
+    // 2×2 GIF with white, red, green and blue pixels
+    const testImageURL =
+      '';
+    const refData =
+      [255, 255, 255, 255,  255, 0, 0, 255,  0, 255, 0, 255,  0, 0, 255, 255];
+    const img = new Image();
+    img.onload = () => {
+      const canvas  = document.createElement('canvas');
+      const context = canvas.getContext('2d');
+      context.drawImage(img, 0, 0, 2, 2);
+      const imageData = context.getImageData(0, 0, 2, 2);
+      if (imageData.data.every((x, i) => refData[i] === x)) {
+        _browser_quirks['canvas-read-unreliable'] = false;
+        resolve();
+      } else {
+        _browser_quirks['canvas-read-unreliable'] = true;
+        reject('Canvas reading unreliable');
+      }
+    };
+    img.onerror = () => {
+      _browser_quirks['canvas-read-unreliable'] = true;
+      reject('Failed to load test image');
+    };
+    img.src = testImageURL;
+  }
+});
+
+const getImageUrl = inputFile => new Promise((resolve, reject) => {
+  if (window.URL && URL.createObjectURL) {
+    try {
+      resolve(URL.createObjectURL(inputFile));
+    } catch (error) {
+      reject(error);
+    }
+    return;
+  }
+
+  const reader = new FileReader();
+  reader.onerror = (...args) => reject(...args);
+  reader.onload  = ({ target }) => resolve(target.result);
+
+  reader.readAsDataURL(inputFile);
+});
+
+const loadImage = inputFile => new Promise((resolve, reject) => {
+  getImageUrl(inputFile).then(url => {
+    const img = new Image();
+
+    img.onerror = (...args) => reject(...args);
+    img.onload  = () => resolve(img);
+
+    img.src = url;
+  }).catch(reject);
+});
+
+const getOrientation = (img, type = 'image/png') => new Promise(resolve => {
+  if (!['image/jpeg', 'image/webp'].includes(type)) {
+    resolve(1);
+    return;
+  }
+
+  EXIF.getData(img, () => {
+    const orientation = EXIF.getTag(img, 'Orientation');
+    if (orientation !== 1) {
+      dropOrientationIfNeeded(orientation).then(resolve).catch(() => resolve(orientation));
+    } else {
+      resolve(orientation);
+    }
+  });
+});
+
+const processImage = (img, { width, height, orientation, type = 'image/png' }) => new Promise(resolve => {
+  const canvas  = document.createElement('canvas');
+
+  if (4 < orientation && orientation < 9) {
+    canvas.width  = height;
+    canvas.height = width;
+  } else {
+    canvas.width  = width;
+    canvas.height = height;
+  }
+
+  const context = canvas.getContext('2d');
+
+  switch (orientation) {
+  case 2: context.transform(-1, 0, 0, 1, width, 0); break;
+  case 3: context.transform(-1, 0, 0, -1, width, height); break;
+  case 4: context.transform(1, 0, 0, -1, 0, height); break;
+  case 5: context.transform(0, 1, 1, 0, 0, 0); break;
+  case 6: context.transform(0, 1, -1, 0, height, 0); break;
+  case 7: context.transform(0, -1, -1, 0, height, width); break;
+  case 8: context.transform(0, -1, 1, 0, 0, width); break;
+  }
+
+  context.drawImage(img, 0, 0, width, height);
+
+  canvas.toBlob(resolve, type);
+});
+
+const resizeImage = (img, type = 'image/png') => new Promise((resolve, reject) => {
+  const { width, height } = img;
+
+  const newWidth  = Math.round(Math.sqrt(MAX_IMAGE_PIXELS * (width / height)));
+  const newHeight = Math.round(Math.sqrt(MAX_IMAGE_PIXELS * (height / width)));
+
+  checkCanvasReliability()
+    .then(getOrientation(img, type))
+    .then(orientation => processImage(img, {
+      width: newWidth,
+      height: newHeight,
+      orientation,
+      type,
+    }))
+    .then(resolve)
+    .catch(reject);
+});
+
+const resizeFile = (inputFile) => new Promise((resolve) => {
+  if (!inputFile.type.match(/image.*/) || inputFile.type === 'image/gif') {
+    resolve(inputFile);
+    return;
+  }
+
+  loadImage(inputFile).then(img => {
+    if (img.width * img.height < MAX_IMAGE_PIXELS) {
+      resolve(inputFile);
+      return;
+    }
+
+    resizeImage(img, inputFile.type)
+      .then(resolve)
+      .catch(() => resolve(inputFile));
+  }).catch(() => resolve(inputFile));
+});
+
+export default resizeFile;
diff --git a/app/javascript/flavours/blobfox/utils/scrollbar.js b/app/javascript/flavours/blobfox/utils/scrollbar.js
new file mode 100644
index 00000000000000..b3f543ffb38a2e
--- /dev/null
+++ b/app/javascript/flavours/blobfox/utils/scrollbar.js
@@ -0,0 +1,34 @@
+/** @type {number | null} */
+let cachedScrollbarWidth = null;
+
+/**
+ * @returns {number}
+ */
+const getActualScrollbarWidth = () => {
+  const outer = document.createElement('div');
+  outer.style.visibility = 'hidden';
+  outer.style.overflow = 'scroll';
+  document.body.appendChild(outer);
+
+  const inner = document.createElement('div');
+  outer.appendChild(inner);
+
+  const scrollbarWidth = outer.offsetWidth - inner.offsetWidth;
+  outer.parentNode.removeChild(outer);
+
+  return scrollbarWidth;
+};
+
+/**
+ * @returns {number}
+ */
+export const getScrollbarWidth = () => {
+  if (cachedScrollbarWidth !== null) {
+    return cachedScrollbarWidth;
+  }
+
+  const scrollbarWidth = getActualScrollbarWidth();
+  cachedScrollbarWidth = scrollbarWidth;
+
+  return scrollbarWidth;
+};
diff --git a/app/javascript/flavours/blobfox/uuid.ts b/app/javascript/flavours/blobfox/uuid.ts
new file mode 100644
index 00000000000000..4d0a8a803637ba
--- /dev/null
+++ b/app/javascript/flavours/blobfox/uuid.ts
@@ -0,0 +1,8 @@
+export function uuid(a?: string): string {
+  return a
+    ? (
+        (a as unknown as number) ^
+        ((Math.random() * 16) >> ((a as unknown as number) / 4))
+      ).toString(16)
+    : ('' + 1e7 + -1e3 + -4e3 + -8e3 + -1e11).replace(/[018]/g, uuid);
+}
diff --git a/app/javascript/skins/blobfox/contrast/common.scss b/app/javascript/skins/blobfox/contrast/common.scss
new file mode 100644
index 00000000000000..cc090db08813e1
--- /dev/null
+++ b/app/javascript/skins/blobfox/contrast/common.scss
@@ -0,0 +1 @@
+@import 'flavours/blobfox/styles/contrast';
diff --git a/app/javascript/skins/blobfox/contrast/names.yml b/app/javascript/skins/blobfox/contrast/names.yml
new file mode 100644
index 00000000000000..b78cc72b9c935a
--- /dev/null
+++ b/app/javascript/skins/blobfox/contrast/names.yml
@@ -0,0 +1,12 @@
+en:
+  skins:
+    blobfox:
+      contrast: High contrast
+cs:
+  skins:
+    blobfox:
+      contrast: Vysoký kontrast
+es:
+  skins:
+    blobfox:
+      contrast: Alto contraste
diff --git a/app/javascript/skins/blobfox/mastodon-light/common.scss b/app/javascript/skins/blobfox/mastodon-light/common.scss
new file mode 100644
index 00000000000000..be496e0c4213d0
--- /dev/null
+++ b/app/javascript/skins/blobfox/mastodon-light/common.scss
@@ -0,0 +1 @@
+@import 'flavours/blobfox/styles/mastodon-light';
diff --git a/app/javascript/skins/blobfox/mastodon-light/names.yml b/app/javascript/skins/blobfox/mastodon-light/names.yml
new file mode 100644
index 00000000000000..8b9b81374c12b1
--- /dev/null
+++ b/app/javascript/skins/blobfox/mastodon-light/names.yml
@@ -0,0 +1,12 @@
+en:
+  skins:
+    blobfox:
+      mastodon-light: Mastodon (light)
+cs:
+  skins:
+    blobfox:
+      mastodon-light: Mastodon (světlý)
+es:
+  skins:
+    blobfox:
+      mastodon-light: Mastodon (claro)
diff --git a/app/javascript/skins/blobfox/queens-pink-contrast/common.scss b/app/javascript/skins/blobfox/queens-pink-contrast/common.scss
new file mode 100644
index 00000000000000..c1475586fa558b
--- /dev/null
+++ b/app/javascript/skins/blobfox/queens-pink-contrast/common.scss
@@ -0,0 +1,3 @@
+@import 'variables'; // var data
+@import 'flavours/blobfox/styles/index'; // vanilla style
+@import 'diff'; // modifications
diff --git a/app/javascript/skins/blobfox/queens-pink-contrast/diff.scss b/app/javascript/skins/blobfox/queens-pink-contrast/diff.scss
new file mode 100644
index 00000000000000..08812be8035c3d
--- /dev/null
+++ b/app/javascript/skins/blobfox/queens-pink-contrast/diff.scss
@@ -0,0 +1,381 @@
+@import 'variables';
+
+// This file houses all of the modifications to the base style. Ergo, anything omitted will use the default mastodon styling. This is useful if we just want to change a few things.
+
+body {
+  background-color: darken($ui-base-color, 5%);
+}
+
+@mixin pane-style {
+  @media screen and (max-width: 600px) {
+  // small screen
+    margin: .1rem;
+    padding: .1rem;
+  }
+  @media screen and (min-width: 600px) {
+    margin: 1rem;
+    padding: 1rem;
+  }
+}
+
+.icon-with-badge {
+  .icon-with-badge__badge {
+    border-radius: $queen-radius;
+    transform: scale(1.0);
+    transition: transform .5s ease-in-out;
+  }
+  .icon-with-badge__badge:hover {
+    transform: scale(1.5);
+  }
+}
+
+
+.drawer__inner__mastodon {
+  background-color: transparent !important;
+}
+
+.single-column,
+.auto-columns {
+  background: darken($ui-base-color, 5%);
+  margin: .1rem;
+
+  .column-header {
+    background: $ui-base-color;
+    border-radius: $queen-radius;
+    margin: .1rem;
+  }
+
+  .tabs-bar__wrapper {
+    background-color: transparent;
+  }
+
+
+  .columns-area__panels__pane__inner {
+    // left & right panels
+    @include pane-style;
+
+    background-color: $black;
+    border-radius: $queen-radius;
+    color: $queen-highlight-color;
+  }
+  .columns-area {
+    background-color: transparent;
+  }
+
+  .search {
+    margin: 1rem;
+    .search__input {
+      border-radius: $queen-radius;
+    }
+  }
+
+  .report-modal {
+    background-color: $ui-base-color;
+    color: $white;
+    border-radius: $queen-radius;
+  }
+
+  .column-header__button,
+  .column-header__back-button,
+  .button,
+  .content__heading__actions.button {
+    background-color: $ui-base-color;
+    border-radius: $queen-radius;
+    padding: .7rem;
+    border-width: 2px;
+    border-style: outset;
+    border-color: lighten($queen-highlight-color, 5%);
+    color: $white;
+
+    box-shadow: inset 0 0 0 0 $queen-highlight-color;
+    transition: box-shadow 1s ease-in-out;
+    // transition: background-color .3s;
+    // transition: color .3s;
+  }
+
+  .column-header__button:hover,
+  .column-header__back-button:hover,
+  .button:hover,
+  .content__heading__actions.button:hover {
+    box-shadow: inset 100px 0 0 0 $queen-highlight-color;
+    // background-color: $queen-highlight-color;
+    // color: $black;
+  }
+
+  .drawer__header {
+    background-color: $ui-base-color;
+    border-radius: $queen-radius;
+    i.fa {
+      color: $white;
+    }
+  }
+
+  .drawer__inner {
+    background-color: transparent;
+  }
+
+  .scrollable {
+    background-color: transparent;
+  }
+  .columns-area__panels__main {
+    background-color: transparent;
+    border-bottom-left-radius: $queen-radius;
+    border-bottom-right-radius: $queen-radius;
+    padding: 0%;
+  }
+
+  .status__info {
+    padding-bottom: .5rem;
+  }
+
+  // .account__avatar{
+  //   border-radius: $queen-radius-pfp;
+  //   border-style: solid;
+  //   border-color: $queen-highlight-color;
+  //   // border-width: 3px;
+  //   border-top-width: $queen-border-thickness;
+  //   border-left-width: $queen-border-thickness;
+  //   transform: none;
+  // }
+
+  .status__avatar {
+    // margin: 2rem;
+    border-radius: $queen-radius-pfp;
+    background-image: linear-gradient(45deg, $queen-highlight-color, $ui-base-color);
+    padding: .5rem;
+    border-width: $queen-border-thickness;
+    border-style: solid;
+    border-color: $queen-highlight-color;
+  }
+
+  .status__display-name {
+    .display-name {
+      background-image: linear-gradient(to right, $queen-highlight-color, $ui-base-color);
+      padding: .5rem;
+      padding-left: 1rem;
+      padding-right: 1rem;
+      border-radius: $queen-radius;
+      .display-name__account {
+        color: $white;
+      }
+    }
+  }
+
+  .account__avatar,
+  .account__avatar-overlay-base,
+  .account__avatar-overlay-overlay {
+    border-radius: $queen-radius-pfp;
+  }
+
+  .tabs-bar__wrapper {
+    background-color: transparent;
+  }
+
+  .status,
+  .detailed-status {
+    background-color: $black;
+    border-radius: $queen-radius;
+    border-style: solid;
+    border-color: desaturate($queen-highlight-color, 25%);
+    // border-width: 3px;
+    border-bottom-width: $queen-border-thickness;
+    border-right-width: $queen-border-thickness;
+    padding: 1rem;
+    margin: 1rem;
+
+  }
+
+  .status__content {
+    text-align: center;
+    align-content: center;
+    vertical-align: middle;
+    span {
+      padding: 1rem;
+    }
+  }
+  .status__content__text {
+    text-align: left;
+  }
+
+  .status__content--with-spoiler {
+    background-color: $ui-base-color;
+    padding: 1rem;
+    border-radius: $queen-radius;
+  }
+
+  .status__content__spoiler-link {
+    // dear god why isn't there a class for the spoiler text!?!?
+    padding: .5rem;
+    border-radius: $queen-radius;
+    background-color: $queen-highlight-color;
+    color: $white;
+    transition: background-color 1s;
+  }
+  .status__content__spoiler-link:hover {
+    background-color: $black;
+  }
+
+  .dropdown-animation {
+    padding: .5rem;
+    border-radius: $queen-radius;
+    background-color: $queen-highlight-color;
+    color: $white;
+  }
+  .privacy-dropdown__option {
+    border-radius: $queen-radius;
+    background-color: $queen-highlight-color;
+    color: $white;
+  }
+
+  .detailed-status-direct,
+  .status-direct,
+  .status-direct::selection,
+  .detailed-status-direct::selection {
+    padding-left: 2rem;
+    border-radius: $queen-radius;
+    border: .3rem solid $queen-highlight-color;
+    border-top-right-radius: 0%;
+    background-color: transparent;
+    background-image: linear-gradient(140deg, darken($queen-highlight-color, 15%), $black, $black);
+  }
+
+  .compose-form {
+    background-color: transparent;
+
+    .compose-form__autosuggest-wrapper {
+      background-color: transparent;
+
+      .autosuggest-textarea {
+        background-color: transparent;
+
+        .autosuggest-textarea__textarea {
+          background-color: $ui-base-color;
+          color: lighten($queen-highlight-color, 25%);
+          border-radius: $queen-radius;
+          padding: 1rem;
+        }
+      }
+    }
+  }
+  .spoiler-input__input {
+    background-color: $ui-base-color;
+    color: lighten($queen-highlight-color, 25%);
+    border-radius: $queen-radius;
+    // padding: 1rem;
+  }
+
+  .navigation-bar {
+    padding: .5rem;
+    background-color: $ui-base-color;
+    border-radius: $queen-radius;
+  }
+.autosuggest-textarea__suggestions-wrapper {
+  background-color: lighten($ui-base-color, 5%);
+}
+.compose-form__buttons-wrapper {
+  background-color: lighten($ui-base-color, 5%);
+  padding: .5rem;
+  margin-top: .5rem;
+  border-radius: $queen-radius;
+}
+
+  .hashtag,
+  .status-link,
+  .mention {
+    background-color: $queen-highlight-color;
+    border-radius: $queen-radius;
+
+    color: $white;
+    text-decoration-color: $white;
+    text-emphasis-color: $white;
+    transition: background-color 1s;
+    // clear values from the spoi
+    padding: .2rem;
+    margin: .3rem;
+    border: 2px outset $white;
+  }
+  .hashtag:hover,
+  .mention:hover {
+    background-color: $black;
+  }
+
+  .unhandled-link {
+    background-color: $black;
+    border-color: $queen-highlight-color;
+  }
+  .mention {
+    border-color: lighten($queen-highlight-color, 25%);
+  }
+  .hashtag {
+    border-color: darken($queen-highlight-color, 25%);
+  }
+  .status-card {
+    background-color: $ui-base-color;
+    border-radius: $queen-radius;
+    border-color: $queen-highlight-color;
+    margin: .5rem;
+    .status-card__image {
+      border-radius: $queen-radius;
+      border-color: $queen-highlight-color;
+      border-width: .1rem;
+      border-style: solid;
+      // padding: .5rem;
+    }
+  }
+
+  .reactions-bar__item,
+  .status__content__text,
+  .display-name {
+    .emojione {
+      transition: transform .3s;
+      transform: scale(1.0);
+    }
+    .emojione:hover {
+      transform: scale(3);
+    }
+  }
+
+  .display-name {
+    clip-path: none;
+  }
+
+  .status__action-bar,
+  .status__info__icons,
+  .columns-area__panels__pane__inner,
+  .notification__message {
+    i.fa,
+    .text-icon {
+      transform: scale(1);
+      transition: transform .3s ease-in-out;
+    }
+
+    i.fa:hover,
+    .text-icon:hover,
+    i.fa:active {
+      transform: scale(1.3);
+    }
+  }
+
+  .column-link {
+    background-color: transparent;
+    box-shadow: inset 0 0 0 0 $queen-highlight-color;
+    transition: box-shadow .3s;
+    border-radius: $queen-radius;
+  }
+  .column-link:hover {
+    box-shadow: inset 10px 0px 30px 0px $queen-highlight-color;
+  }
+
+  .getting-started__wrapper {
+    border-radius: $queen-radius;
+    margin: .5rem;
+  }
+
+  .reply-indicator,
+  .status__content__text {
+    p {
+      line-height: 2rem;
+    }
+  }
+
+}
diff --git a/app/javascript/skins/blobfox/queens-pink-contrast/names.yml b/app/javascript/skins/blobfox/queens-pink-contrast/names.yml
new file mode 100644
index 00000000000000..729cf44e755adc
--- /dev/null
+++ b/app/javascript/skins/blobfox/queens-pink-contrast/names.yml
@@ -0,0 +1,4 @@
+en:
+  skins:
+    blobfox:
+      queens-pink-contrast: Queen's Pink Contrast
diff --git a/app/javascript/skins/blobfox/queens-pink-contrast/screenshot.jpg b/app/javascript/skins/blobfox/queens-pink-contrast/screenshot.jpg
new file mode 100644
index 0000000000000000000000000000000000000000..a31cfac43068b76d45a03a6609ba11464fee19e7
GIT binary patch
literal 173565
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
J=>y~cn*d?ZOQ!$;

literal 0
HcmV?d00001

diff --git a/app/javascript/skins/blobfox/queens-pink-contrast/variables.scss b/app/javascript/skins/blobfox/queens-pink-contrast/variables.scss
new file mode 100644
index 00000000000000..c1de386ce702a8
--- /dev/null
+++ b/app/javascript/skins/blobfox/queens-pink-contrast/variables.scss
@@ -0,0 +1,81 @@
+// my theme customizations
+$queen-highlight-color: #b82493;
+$queen-bg: #3a002f;
+$queen-radius: 16px;
+$queen-radius-pfp: 25%;
+$queen-border-thickness: .1rem;
+
+
+
+// built-in defaults
+$black: #000000;
+$white: #ffffff;
+$success-green: #79bd9a;
+$error-red: #df405a;
+$warning-red: #ff5050;
+$gold-star: #ca8f04; // favorite colour
+
+
+// Values from the classic Mastodon UI
+$classic-base-color: $queen-bg;         // Midnight Express
+$classic-primary-color: desaturate($queen-highlight-color, 15%);      // Echo Blue
+$classic-secondary-color: #d9e1e8;    // Pattens Blue
+$classic-highlight-color: #2b90d9;    // Summer Sky
+
+
+// Variables for defaults in UI
+$base-shadow-color: $black !default;
+$base-overlay-background: $black !default;
+$base-border-color: $white !default;
+$simple-background-color: $white !default;
+$valid-value-color: $success-green !default;
+$error-value-color: $error-red !default;
+
+$ui-base-color: $queen-bg !default;
+$ui-base-lighter-color: lighten($queen-highlight-color, 25%) !default; // Lighter darkest
+$ui-primary-color: $classic-primary-color !default;
+$ui-secondary-color: $classic-secondary-color !default;
+$ui-highlight-color: $classic-highlight-color !default;
+
+$ui-highlight-color: $queen-highlight-color;
+
+
+// Variables for texts
+$primary-text-color: $white !default;
+$darker-text-color: $ui-primary-color !default;
+$dark-text-color: $ui-base-lighter-color !default;
+$secondary-text-color: $ui-secondary-color !default;
+$highlight-text-color: $ui-highlight-color !default;
+$action-button-color: $ui-base-lighter-color !default;
+$action-button-color-lighter: lighten($ui-base-lighter-color, 10%) !default;
+$action-button-color-darker: darken($ui-base-lighter-color, 5%) !default;
+
+$passive-text-color: $gold-star !default;
+$active-passive-text-color: $success-green !default;
+// For texts on inverted backgrounds
+$inverted-text-color: $ui-base-color !default;
+$lighter-text-color: $ui-base-lighter-color !default;
+$light-text-color: $ui-primary-color !default;
+
+$solar-disabled: darken($action-button-color, 10%) !default;
+
+// Language codes that uses CJK fonts
+$cjk-langs: ja, ko, zh-CN, zh-HK, zh-TW;
+
+// Variables for components
+$media-modal-media-max-width: 100%;
+// put margins on top and bottom of image to avoid the screen covered by image.
+$media-modal-media-max-height: 80%;
+
+$no-gap-breakpoint: 415px;
+
+$font-sans-serif: 'mastodon-font-sans-serif' !default;
+$font-display: 'mastodon-font-display' !default;
+$font-monospace: 'mastodon-font-monospace' !default;
+
+// Avatar border size (8% default, 100% for rounded avatars)
+$ui-avatar-border-size: 8%;
+
+// More variables
+$dismiss-overlay-width: 4rem;
+
diff --git a/app/javascript/skins/blobfox/solarpunk/common.scss b/app/javascript/skins/blobfox/solarpunk/common.scss
new file mode 100644
index 00000000000000..8dcb5f04f0386a
--- /dev/null
+++ b/app/javascript/skins/blobfox/solarpunk/common.scss
@@ -0,0 +1,3 @@
+@import 'variables';
+@import 'flavours/blobfox/styles/index';
+@import 'diff';
diff --git a/app/javascript/skins/blobfox/solarpunk/diff.scss b/app/javascript/skins/blobfox/solarpunk/diff.scss
new file mode 100644
index 00000000000000..bac39ce234f104
--- /dev/null
+++ b/app/javascript/skins/blobfox/solarpunk/diff.scss
@@ -0,0 +1,143 @@
+@import 'variables';
+
+body {
+  background-color: $solar-bg;
+}
+
+.media-modal__overlay
+.picture-in-picture__footer 
+button.icon-button {
+  i.fa-retweet {
+    background: url("data:image/svg+xml;utf8,<svg width='18.7' height='94' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 657.07 3289.9' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-.77735 -6.0669)'><g transform='translate(29.186 28.067)' fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($white)}'/></g></g></g><g fill='#{hex-color($solar-recycling)}'><g transform='translate(27.733 1347)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l97.831-167.4-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-21.415 39.593-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172 40.453-1.5167v148.02l-3.6203 1.5167zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l110.62 189.47 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078l-196.7-1.5167v51.57l-75.004-125.23 75.004-125.27v52.258l261.67 1.5167zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-9.2306-16.843 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(691.01 5483.9)'><g fill='#{hex-color($solar-recycling)}'/></g><g transform='translate(30.324 2009.8)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l11.377-17.243-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-107.87 189.75-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h216.39v148.02h-179.56zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l13.551 2.912 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-20.759v51.57l-75.004-125.23 75.004-125.27v52.258h85.731zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-116.92-210.98 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(27.915 2669.7)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g><g transform='translate(29.186 687.36)' fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l76.66-134.68-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-42.586 72.315-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h111.57v148.02h-74.736zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l73.027 129.71 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-125.58v51.57l-75.004-125.23 75.004-125.27v52.258h190.56zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-46.825-76.605 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g></svg>") no-repeat;
+  }
+
+  &.reblogPrivate i.fa-retweet {
+    background: url("data:image/svg+xml;utf8,<svg width='19' height='19' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 630.25 615.51' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><defs><mask id='mask-powermask-path-effect929' maskUnits='userSpaceOnUse'><path id='mask-powermask-path-effect929_box' d='m-1-1h630.25v615.51h-630.25z' fill='%23fff'/><g transform='matrix(1.8197 0 0 1.8197 467.68 255.09)' style=''><rect x='-219.52' y='91.198' width='114.31' height='71.892' fill='none' stroke='%23000' stroke-width='34.869'/><path d='m-211.31 76.273c0-2.2289-0.45078-28.654-0.20046-30.833 1.9095-16.621 12.096-32.827 25.749-41.276 15.443-9.5572 34.438-9.4492 49.787 0.28295 13.762 8.7258 22.704 23.991 24.271 40.915 0.18075 1.9519-0.0259 28.68-0.0453 30.668' fill='none' stroke='%23000' stroke-width='28.408'/></g></mask></defs><g transform='translate(-28.186 -5.0669)'><g transform='translate(29.186 6.0669)' fill='#{hex-color($white)}' mask='url(%23mask-powermask-path-effect929)'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($white)}' mask='none'/></g><path d='m28.186 5.0669h630.25v615.51h-630.25z' fill='none'/><g transform='translate(691.01 5483.9)' fill='%23009700'><g fill='%23009700'/></g><g transform='matrix(1.8197 0 0 1.8197 537.9 285.64)' stroke='#{hex-color($white)}'><rect x='-219.52' y='91.198' width='114.31' height='71.892' fill='#{hex-color($white)}' stroke-width='34.869'/><path d='m-211.31 76.273c0-2.2289-0.45078-28.654-0.20046-30.833 1.9095-16.621 12.096-32.827 25.749-41.276 15.443-9.5572 34.438-9.4492 49.787 0.28295 13.762 8.7258 22.704 23.991 24.271 40.915 0.18075 1.9519-0.0259 28.68-0.0453 30.668' fill='none' stroke-width='28.408'/></g></g></svg>");
+  }
+
+  i.fa-star {
+    background: url("data:image/svg+xml;utf8,<svg width='6.299mm' height='28.591mm' version='1.1' viewBox='0 0 4.9132 22.301' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-72.638 -93.656)'><g transform='translate(.47668 5.3239)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.63 92.871-0.44504 1.5146 0.31204 0.77128 0.57329-0.77889z' stroke-width='1.0644'/><path d='m74.205 95.719 0.41002-0.27757 0.41153 0.30391-0.37017 1.3919' stroke-width='1.1036'/><path d='m72.271 95.052 1.6527 0.40144 0.90943-0.37981-0.85337-0.45872z'/><path d='m76.865 95.04-1.5894-0.43145-0.8651 0.42185 0.90715 0.47274z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.025125 .41834)'><g transform='translate(.49227 .42869)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.0678-2.4605c-1.0077-0.58067-1.6372-0.66239-2.5754-0.46052l-3.7943 1.2335 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515c1.573 0.58073 1.5464 0.26082 2.5538 0.4567l3.7344-1.0206-2.2515 3.1493c-1.1607 1.0484-0.49695 1.609-0.4567 2.5538z' fill='#{hex-color($white)}'/><g fill='#{hex-color($white)}'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 1.0348 0.36811 0.59405-0.35157-0.59987-0.37789z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='#{hex-color($white)}' stroke-width='.20427'/></g></g><g transform='translate(.51611 14.249)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m2.4281 13.801-6.2849-5.3807-1.3051 3.847-0.17157-4.0062-6.9762 2.8655 5.1741-5.3193-3.5064-1.4065 3.8314-0.094818-3.2175-7.6125 5.8984 5.7429 1.3543-3.5156 0.19915 3.8497 7.3536-2.8701-5.754 5.5572 3.6813 1.3297-4.0124-0.09844z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.51271 18.705)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.50609 9.7746)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0428'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g></g></svg>") no-repeat top;
+  }
+}
+
+.media-modal__overlay
+.picture-in-picture__footer 
+button.icon-button.active {
+  &.reblogPrivate.active i.fa-retweet,
+  i.fa-retweet {
+    background: url("data:image/svg+xml;utf8,<svg width='18.7' height='94' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 657.07 3289.9' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-.77735 -6.0669)'><g transform='translate(29.186 28.067)' fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($white)}'/></g></g></g><g fill='#{hex-color($solar-recycling)}'><g transform='translate(27.733 1347)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l97.831-167.4-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-21.415 39.593-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172 40.453-1.5167v148.02l-3.6203 1.5167zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l110.62 189.47 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078l-196.7-1.5167v51.57l-75.004-125.23 75.004-125.27v52.258l261.67 1.5167zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-9.2306-16.843 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(691.01 5483.9)'><g fill='#{hex-color($solar-recycling)}'/></g><g transform='translate(30.324 2009.8)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l11.377-17.243-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-107.87 189.75-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h216.39v148.02h-179.56zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l13.551 2.912 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-20.759v51.57l-75.004-125.23 75.004-125.27v52.258h85.731zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-116.92-210.98 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(27.915 2669.7)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g><g transform='translate(29.186 687.36)' fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><g fill='#{hex-color($white)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l76.66-134.68-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-42.586 72.315-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h111.57v148.02h-74.736zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l73.027 129.71 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-125.58v51.57l-75.004-125.23 75.004-125.27v52.258h190.56zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-46.825-76.605 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g></svg>") no-repeat;
+    background-position: 50% 100%;
+  }
+
+  i.fa-star {
+    background-position: 50% 100%;
+  }
+}
+
+button.icon-button {
+  &:hover i.fa-retweet,
+  &.reblogPrivate.active i.fa-retweet,
+  &:hover.reblogPrivate.active i.fa-retweet,
+  i.fa-retweet {
+      background: url("data:image/svg+xml;utf8,<svg width='18.7' height='94' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 657.07 3289.9' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-.77735 -6.0669)'><g transform='translate(29.186 28.067)' fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($action-button-color)}'/></g></g></g><g fill='#{hex-color($solar-recycling)}'><g transform='translate(27.733 1347)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l97.831-167.4-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-21.415 39.593-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172 40.453-1.5167v148.02l-3.6203 1.5167zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l110.62 189.47 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078l-196.7-1.5167v51.57l-75.004-125.23 75.004-125.27v52.258l261.67 1.5167zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-9.2306-16.843 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(691.01 5483.9)'><g fill='#{hex-color($solar-recycling)}'/></g><g transform='translate(30.324 2009.8)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l11.377-17.243-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-107.87 189.75-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h216.39v148.02h-179.56zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l13.551 2.912 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-20.759v51.57l-75.004-125.23 75.004-125.27v52.258h85.731zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-116.92-210.98 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(27.915 2669.7)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g><g transform='translate(29.186 687.36)' fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l76.66-134.68-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-42.586 72.315-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h111.57v148.02h-74.736zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l73.027 129.71 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-125.58v51.57l-75.004-125.23 75.004-125.27v52.258h190.56zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-46.825-76.605 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g></svg>") no-repeat;
+    width: 19px;
+    height: 18.5px;
+    transition: background-position .6s steps(4);
+  }
+
+  &.reblogPrivate:hover i.fa-retweet,
+  &.reblogPrivate i.fa-retweet {
+    background: url("data:image/svg+xml;utf8,<svg width='19' height='19' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 630.25 615.51' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><defs><mask id='mask-powermask-path-effect929' maskUnits='userSpaceOnUse'><path id='mask-powermask-path-effect929_box' d='m-1-1h630.25v615.51h-630.25z' fill='%23fff'/><g transform='matrix(1.8197 0 0 1.8197 467.68 255.09)' style=''><rect x='-219.52' y='91.198' width='114.31' height='71.892' fill='none' stroke='%23000' stroke-width='34.869'/><path d='m-211.31 76.273c0-2.2289-0.45078-28.654-0.20046-30.833 1.9095-16.621 12.096-32.827 25.749-41.276 15.443-9.5572 34.438-9.4492 49.787 0.28295 13.762 8.7258 22.704 23.991 24.271 40.915 0.18075 1.9519-0.0259 28.68-0.0453 30.668' fill='none' stroke='%23000' stroke-width='28.408'/></g></mask></defs><g transform='translate(-28.186 -5.0669)'><g transform='translate(29.186 6.0669)' fill='#{hex-color($action-button-color)}' mask='url(%23mask-powermask-path-effect929)'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($action-button-color)}' mask='none'/></g><path d='m28.186 5.0669h630.25v615.51h-630.25z' fill='none'/><g transform='translate(691.01 5483.9)' fill='%23009700'><g fill='%23009700'/></g><g transform='matrix(1.8197 0 0 1.8197 537.9 285.64)' stroke='#{hex-color($action-button-color)}'><rect x='-219.52' y='91.198' width='114.31' height='71.892' fill='#{hex-color($action-button-color)}' stroke-width='34.869'/><path d='m-211.31 76.273c0-2.2289-0.45078-28.654-0.20046-30.833 1.9095-16.621 12.096-32.827 25.749-41.276 15.443-9.5572 34.438-9.4492 49.787 0.28295 13.762 8.7258 22.704 23.991 24.271 40.915 0.18075 1.9519-0.0259 28.68-0.0453 30.668' fill='none' stroke-width='28.408'/></g></g></svg>");
+  }
+
+  &.reblogPrivate.active i.fa-retweet,
+  &:hover.reblogPrivate.active i.fa-retweet {
+    background-position: bottom;
+  }
+
+  &:hover i.fa-star,
+  i.fa-star {
+    background: url("data:image/svg+xml;utf8,<svg width='6.299mm' height='28.591mm' version='1.1' viewBox='0 0 4.9132 22.301' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-72.638 -93.656)'><g transform='translate(.47668 5.3239)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.63 92.871-0.44504 1.5146 0.31204 0.77128 0.57329-0.77889z' stroke-width='1.0644'/><path d='m74.205 95.719 0.41002-0.27757 0.41153 0.30391-0.37017 1.3919' stroke-width='1.1036'/><path d='m72.271 95.052 1.6527 0.40144 0.90943-0.37981-0.85337-0.45872z'/><path d='m76.865 95.04-1.5894-0.43145-0.8651 0.42185 0.90715 0.47274z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.025125 .41834)'><g transform='translate(.49227 .42869)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.0678-2.4605c-1.0077-0.58067-1.6372-0.66239-2.5754-0.46052l-3.7943 1.2335 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515c1.573 0.58073 1.5464 0.26082 2.5538 0.4567l3.7344-1.0206-2.2515 3.1493c-1.1607 1.0484-0.49695 1.609-0.4567 2.5538z' fill='#{hex-color($action-button-color)}'/><g fill='#{hex-color($action-button-color-darker)}'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 1.0348 0.36811 0.59405-0.35157-0.59987-0.37789z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='#{hex-color($action-button-color-lighter)}' stroke-width='.20427'/></g></g><g transform='translate(.51611 14.249)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m2.4281 13.801-6.2849-5.3807-1.3051 3.847-0.17157-4.0062-6.9762 2.8655 5.1741-5.3193-3.5064-1.4065 3.8314-0.094818-3.2175-7.6125 5.8984 5.7429 1.3543-3.5156 0.19915 3.8497 7.3536-2.8701-5.754 5.5572 3.6813 1.3297-4.0124-0.09844z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.51271 18.705)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.50609 9.7746)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0428'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g></g></svg>") no-repeat top;
+    height: 21.4px;
+    margin: 1px 0;
+    transition: background-position .6s steps(4);
+  }
+
+  &.active i.fa-star {
+    background-position: 50% 100%;
+  }
+
+  /* Remove the font-awesome icon */
+  .fa-star::before {
+    content: "";
+  }
+
+  &.disabled {
+    i.fa-retweet,
+    &:hover i.fa-retweet {
+      background-image: url("data:image/svg+xml;utf8,<svg width='18.7' height='18.7' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 628.25 613.51' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><defs><mask id='mask5629' maskUnits='userSpaceOnUse'><rect x='-156.41' y='-66.33' width='941.77' height='797.17' fill='#{hex-color($white)}' stroke-width='35'/><g transform='matrix(.33928 .4906 -.82248 .5688 533.91 -40.843)' fill='#{hex-color($solar-disabled)}'><rect x='285' y='13.387' width='105.67' height='767.51' fill='#{hex-color($black)}' stroke-width='35'/></g></mask></defs><g transform='translate(-29.186 -6.0669)'><g transform='translate(29.186 6.0669)'><g fill='#{hex-color($solar-disabled)}' mask='url(%23mask5629)'><g fill='#{hex-color($solar-disabled)}' mask='none'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($solar-disabled)}'/></g></g><rect transform='matrix(.57488 .81824 -.81993 .57247 0 0)' x='409.24' y='-521.16' width='66.549' height='882.4' ry='6.6154' fill='#{hex-color($solar-disabled)}' stroke-width='46.779'/></g><g transform='translate(691.01 5483.9)'></g></g></svg>");
+    }
+  }
+
+  &.active i.fa-retweet {
+    background-position: 50% 100%;
+  }
+}
+
+.no-reduce-motion .icon-button.star-icon {
+  &.deactivate > .fa-star,
+  &.activate > .fa-star {
+    animation: none;
+    -webkit-animation: none;
+  }
+}
+
+.detailed-status__link {
+  .fa-star {
+    background: url("data:image/svg+xml;utf8,<svg width='5mm' height='20mm' version='1.1' viewBox='0 0 4.9132 22.301' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-72.638 -93.656)'><g transform='translate(.47668 5.3239)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.63 92.871-0.44504 1.5146 0.31204 0.77128 0.57329-0.77889z' stroke-width='1.0644'/><path d='m74.205 95.719 0.41002-0.27757 0.41153 0.30391-0.37017 1.3919' stroke-width='1.1036'/><path d='m72.271 95.052 1.6527 0.40144 0.90943-0.37981-0.85337-0.45872z'/><path d='m76.865 95.04-1.5894-0.43145-0.8651 0.42185 0.90715 0.47274z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.025125 .41834)'><g transform='translate(.49227 .42869)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.0678-2.4605c-1.0077-0.58067-1.6372-0.66239-2.5754-0.46052l-3.7943 1.2335 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515c1.573 0.58073 1.5464 0.26082 2.5538 0.4567l3.7344-1.0206-2.2515 3.1493c-1.1607 1.0484-0.49695 1.609-0.4567 2.5538z' fill='#{hex-color($action-button-color)}'/><g fill='#{hex-color($action-button-color-darker)}'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 1.0348 0.36811 0.59405-0.35157-0.59987-0.37789z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='#{hex-color($action-button-color-lighter)}' stroke-width='.20427'/></g></g><g transform='translate(.51611 14.249)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m2.4281 13.801-6.2849-5.3807-1.3051 3.847-0.17157-4.0062-6.9762 2.8655 5.1741-5.3193-3.5064-1.4065 3.8314-0.094818-3.2175-7.6125 5.8984 5.7429 1.3543-3.5156 0.19915 3.8497 7.3536-2.8701-5.754 5.5572 3.6813 1.3297-4.0124-0.09844z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.51271 18.705)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0253'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g><g transform='translate(.50609 9.7746)'><path transform='matrix(.04592 .26057 -.26057 .04592 76.244 95.826)' d='m-0.09375 11.156-3.1493-2.2515-1.9189 3.3623-0.63485-3.819-3.7344 1.0206 2.2515-3.1493-3.3623-1.9189 3.819-0.63485-1.0206-3.7344 3.1493 2.2515 1.9189-3.3623 0.63485 3.819 3.7344-1.0206-2.2515 3.1493 3.3623 1.9189-3.819 0.63485z' fill='%23ffb271'/><g fill='%23ff8e7c'><path d='m74.612 93.227-0.34726 1.0005 0.25632 0.51263 0.43822-0.52917z'/><path d='m74.249 95.856 0.27765-0.50436 0.42515 0.5209-0.34112 1.0428'/><path d='m72.802 95.054 0.95912 0.339 0.66973-0.32246-0.66973-0.37207z'/><path d='m76.403 95.054-0.95912-0.339-0.66973 0.32246 0.66973 0.37207z'/></g><circle cx='74.637' cy='95.054' r='.86816' fill='%23f9e59a' stroke-width='.20427'/></g></g></svg>") no-repeat top;
+    width: 12px;
+    height: 13px;
+    margin-bottom: -1px;
+
+    &::before {
+      content: "";
+    }
+  }
+
+  .fa-retweet {
+      background: url("data:image/svg+xml;utf8,<svg width='3.5mm' height='70' enable-background='new 0 0 628.254 613.516' version='1.1' viewBox='0 0 657.07 3289.9' xml:space='preserve' xmlns='http://www.w3.org/2000/svg'><g transform='translate(-.77735 -6.0669)'><g transform='translate(29.186 28.067)' fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($action-button-color)}'/></g></g></g><g fill='#{hex-color($solar-recycling)}'><g transform='translate(27.733 1347)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l97.831-167.4-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-21.415 39.593-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172 40.453-1.5167v148.02l-3.6203 1.5167zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l110.62 189.47 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078l-196.7-1.5167v51.57l-75.004-125.23 75.004-125.27v52.258l261.67 1.5167zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-9.2306-16.843 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(691.01 5483.9)'><g fill='#{hex-color($solar-recycling)}'/></g><g transform='translate(30.324 2009.8)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l11.377-17.243-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-107.87 189.75-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h216.39v148.02h-179.56zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l13.551 2.912 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-20.759v51.57l-75.004-125.23 75.004-125.27v52.258h85.731zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-116.92-210.98 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g><g transform='translate(27.915 2669.7)'><g fill='#{hex-color($solar-recycling)}'><g fill='#{hex-color($solar-recycling)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l40.195-70.328-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-79.051 136.66-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h158.76v148.02h-121.92zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l40.852 69.648 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-78.395v51.57l-75.004-125.23 75.004-125.27v52.258h143.37zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-79-136.66 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g><g transform='translate(29.186 687.36)' fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><g fill='#{hex-color($action-button-color)}'><path d='m99.777 446.09c-6.699 12.031-12.031 30.133-12.031 41.539 0 2.648 0 6.016 0.656 10.688l-83.726-143.99c-2.68-4.672-4.676-11.375-4.676-17.414 0-6.047 1.996-13.398 4.676-18.078l76.66-134.68-44.871-25.43 146.02-2.703 70.984 127.91-45.527-26.117zm64.313-405.86c12.715-22.125 33.496-34.18 58.926-34.18 27.48 0 48.918 12.742 64.312 38.828l22.777 38.172-42.586 72.315-127.91-74.352zm9.351 521.72c-38.172 0-69.645-31.477-69.645-69.648 0-10.719 4.703-28.82 11.402-40.195l21.41-38.172h111.57v148.02h-74.736zm127.29-525.76c-10.036-17.391-23.434-29.477-39.512-36.18h164.75c14.738 0 26.113 6.047 32.84 17.445l73.027 129.71 44.191-26.141-71.016 127.28-145.3-2.047 44.871-25.43zm253.86 379.09c20.07 0 36.832-5.359 50.887-16.055l-83.07 144.65c-6.699 11.375-18.73 18.078-32.789 18.078h-125.58v51.57l-75.004-125.23 75.004-125.27v52.258h190.56zm64.258-120.56c6.043 10.719 9.406 22.094 9.406 34.156 0 24.117-15.422 49.57-36.832 61.602-10.062 5.391-24.145 8.75-38.172 8.75h-44.242l-46.825-76.605 127.92-73.008z' fill='#{hex-color($solar-recycling)}'/></g></g></g></g></svg>") no-repeat;
+      width: 13px;
+      height: 14px;
+      margin-bottom: -1px;
+
+      &::before {
+        content: "";
+      }
+   }
+}
+
+.drawer .drawer__inner {
+  background: lighten($ui-base-color, 13%) url("data:image/svg+xml;utf8,<svg width='391.91mm' height='62.185mm' version='1.1' viewBox='0 0 391.91 62.185' xmlns='http://www.w3.org/2000/svg'><defs><clipPath id='clipPath1458'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath><clipPath id='clipath_lpe_path-effect2057'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath><clipPath id='clipath_lpe_path-effect2076'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath><clipPath id='clipath_lpe_path-effect2095'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath><clipPath id='clipath_lpe_path-effect2114'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath><clipPath id='clipath_lpe_path-effect2133'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath></defs><g transform='translate(-4.7406 -70.901)' fill='%2321234a' stroke='%2321234a' stroke-linejoin='bevel' stroke-width='10.222'><path x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipPath1458)' style='paint-order:normal'/><path transform='translate(63.308)' x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipath_lpe_path-effect2057)' style='paint-order:normal'/><path transform='translate(126.62)' x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipath_lpe_path-effect2076)' style='paint-order:normal'/><path transform='translate(189.92)' x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipath_lpe_path-effect2095)' style='paint-order:normal'/><path transform='translate(253.23)' x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipath_lpe_path-effect2114)' style='paint-order:normal'/><path transform='translate(316.54)' x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipath_lpe_path-effect2133)' style='paint-order:normal'/></g></svg>") 0 9px;
+  background-size: 100%;
+
+  &.darker {
+    background: $ui-base-color;
+  }
+}
+
+.drawer__inner__mastodon {
+  background-color: transparent !important;
+}
+
+.single-column,
+.auto-columns {
+  .column-header {
+    background: lighten($ui-base-color, 4%) url("data:image/svg+xml;utf8,<svg width='61.371mm' height='62.185mm' version='1.1' viewBox='0 0 61.371 62.185' xmlns='http://www.w3.org/2000/svg'><defs><clipPath id='clipPath1458'><g display='none'><rect x='20.515' y='65.617' width='3.914' height='73.915' d='m 20.514772,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='60.408' y='65.617' width='3.914' height='73.915' d='m 60.408058,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='33.813' y='65.617' width='3.914' height='73.915' d='m 33.812534,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/><rect x='47.11' y='65.617' width='3.914' height='73.915' d='m 47.110298,65.61702 h 3.91404 v 73.91547 h -3.91404 z' stroke-width='7.5232'/></g><path class='powerclip' d='m9.7406 68.901h65.371v66.185h-65.371zm10.774-3.2844v73.915h3.914v-73.915zm39.893 0v73.915h3.914v-73.915zm-26.596 0v73.915h3.914v-73.915zm13.298 0v73.915h3.914v-73.915z'/></clipPath></defs><g transform='translate(-11.741 -70.901)'><path x='19.851646' y='79.012375' width='45.14875' height='45.963497' d='m19.852 79.012h45.149v45.963h-45.149z' clip-path='url(%23clipPath1458)' fill='#{hex-color($solar-panel-header)}' stroke='#{hex-color($solar-panel-header)}' stroke-linejoin='bevel' stroke-width='10.222' style='paint-order:normal'/></g></svg>") 2px 2px;
+    background-size: 44.25px;
+  }
+
+  .column-header__button,
+  .column-header__back-button {
+    background-color: transparent;
+  }
+}
diff --git a/app/javascript/skins/blobfox/solarpunk/names.yml b/app/javascript/skins/blobfox/solarpunk/names.yml
new file mode 100644
index 00000000000000..dd6e2af12deb2b
--- /dev/null
+++ b/app/javascript/skins/blobfox/solarpunk/names.yml
@@ -0,0 +1,4 @@
+en:
+  skins:
+    blobfox:
+      solarpunk: Solarpunk
diff --git a/app/javascript/skins/blobfox/solarpunk/variables.scss b/app/javascript/skins/blobfox/solarpunk/variables.scss
new file mode 100644
index 00000000000000..038f6d15bfca87
--- /dev/null
+++ b/app/javascript/skins/blobfox/solarpunk/variables.scss
@@ -0,0 +1,75 @@
+$black: #000000;
+$white: #ffffff;
+$success-green: #79bd9a;
+$error-red: #df405a;
+$warning-red: #ff5050;
+$gold-star: #ca8f04;
+
+// Values from the classic Mastodon UI
+$classic-base-color: #282c37;         // Midnight Express
+$classic-primary-color: #9baec8;      // Echo Blue
+$classic-secondary-color: #d9e1e8;    // Pattens Blue
+$classic-highlight-color: #2b90d9;    // Summer Sky
+
+// our theme customizations
+$solar-highlight-color: #378d2a;
+$solar-bg: #060e0c; // dark green
+$solar-recycling: #009700;
+$solar-panel-header: #202c4d;
+
+// Variables for defaults in UI
+$base-shadow-color: $black !default;
+$base-overlay-background: $black !default;
+$base-border-color: $white !default;
+$simple-background-color: $white !default;
+$valid-value-color: $success-green !default;
+$error-value-color: $error-red !default;
+
+$ui-base-color: #161c2c;
+$ui-base-lighter-color: #5370a2 !default; // Lighter darkest
+$ui-primary-color: $classic-primary-color !default;
+$ui-secondary-color: $classic-secondary-color !default;
+$ui-highlight-color: $classic-highlight-color !default;
+
+$ui-highlight-color: $solar-highlight-color;
+
+
+// Variables for texts
+$primary-text-color: $white !default;
+$darker-text-color: $ui-primary-color !default;
+$dark-text-color: $ui-base-lighter-color !default;
+$secondary-text-color: $ui-secondary-color !default;
+$highlight-text-color: $ui-highlight-color !default;
+$action-button-color: $ui-base-lighter-color !default;
+$action-button-color-lighter: lighten($ui-base-lighter-color, 10%) !default;
+$action-button-color-darker: darken($ui-base-lighter-color, 5%) !default;
+
+$passive-text-color: $gold-star !default;
+$active-passive-text-color: $success-green !default;
+// For texts on inverted backgrounds
+$inverted-text-color: $ui-base-color !default;
+$lighter-text-color: $ui-base-lighter-color !default;
+$light-text-color: $ui-primary-color !default;
+
+$solar-disabled: darken($action-button-color, 10%) !default;
+
+// Language codes that uses CJK fonts
+$cjk-langs: ja, ko, zh-CN, zh-HK, zh-TW;
+
+// Variables for components
+$media-modal-media-max-width: 100%;
+// put margins on top and bottom of image to avoid the screen covered by image.
+$media-modal-media-max-height: 80%;
+
+$no-gap-breakpoint: 415px;
+
+$font-sans-serif: 'mastodon-font-sans-serif' !default;
+$font-display: 'mastodon-font-display' !default;
+$font-monospace: 'mastodon-font-monospace' !default;
+
+// Avatar border size (8% default, 100% for rounded avatars)
+$ui-avatar-border-size: 8%;
+
+// More variables
+$dismiss-overlay-width: 4rem;
+