diff --git a/app/controllers/api/v1/emoji_reactions_controller.rb b/app/controllers/api/v1/emoji_reactions_controller.rb
new file mode 100644
index 0000000000..a0f4e59ccf
--- /dev/null
+++ b/app/controllers/api/v1/emoji_reactions_controller.rb
@@ -0,0 +1,61 @@
+# frozen_string_literal: true
+
+class Api::V1::EmojiReactionsController < Api::BaseController
+ before_action -> { doorkeeper_authorize! :read, :'read:favourites' }
+ before_action :require_user!
+ after_action :insert_pagination_headers
+
+ def index
+ @statuses = load_statuses
+ render json: @statuses, each_serializer: REST::StatusSerializer, relationships: StatusRelationshipsPresenter.new(@statuses, current_user&.account_id)
+ end
+
+ private
+
+ def load_statuses
+ cached_emoji_reactions
+ end
+
+ def cached_emoji_reactions
+ cache_collection(results.map(&:status), EmojiReaction)
+ end
+
+ def results
+ @_results ||= account_emoji_reactions.joins(:status).eager_load(:status).to_a_paginated_by_id(
+ limit_param(DEFAULT_STATUSES_LIMIT),
+ params_slice(:max_id, :since_id, :min_id)
+ )
+ end
+
+ def account_emoji_reactions
+ current_account.emoji_reactions
+ end
+
+ def insert_pagination_headers
+ set_pagination_headers(next_path, prev_path)
+ end
+
+ def next_path
+ api_v1_emoji_reactions_url pagination_params(max_id: pagination_max_id) if records_continue?
+ end
+
+ def prev_path
+ api_v1_emoji_reactions_url pagination_params(min_id: pagination_since_id) unless results.empty?
+ end
+
+ def pagination_max_id
+ results.last.id
+ end
+
+ def pagination_since_id
+ results.first.id
+ end
+
+ def records_continue?
+ results.size == limit_param(DEFAULT_STATUSES_LIMIT)
+ end
+
+ def pagination_params(core_params)
+ params.slice(:limit).permit(:limit).merge(core_params)
+ end
+end
diff --git a/app/javascript/mastodon/actions/emoji_reactions.js b/app/javascript/mastodon/actions/emoji_reactions.js
new file mode 100644
index 0000000000..a58e53d2fe
--- /dev/null
+++ b/app/javascript/mastodon/actions/emoji_reactions.js
@@ -0,0 +1,94 @@
+import api, { getLinks } from '../api';
+import { importFetchedStatuses } from './importer';
+
+export const EMOJI_REACTED_STATUSES_FETCH_REQUEST = 'EMOJI_REACTED_STATUSES_FETCH_REQUEST';
+export const EMOJI_REACTED_STATUSES_FETCH_SUCCESS = 'EMOJI_REACTED_STATUSES_FETCH_SUCCESS';
+export const EMOJI_REACTED_STATUSES_FETCH_FAIL = 'EMOJI_REACTED_STATUSES_FETCH_FAIL';
+
+export const EMOJI_REACTED_STATUSES_EXPAND_REQUEST = 'EMOJI_REACTED_STATUSES_EXPAND_REQUEST';
+export const EMOJI_REACTED_STATUSES_EXPAND_SUCCESS = 'EMOJI_REACTED_STATUSES_EXPAND_SUCCESS';
+export const EMOJI_REACTED_STATUSES_EXPAND_FAIL = 'EMOJI_REACTED_STATUSES_EXPAND_FAIL';
+
+export function fetchEmojiReactedStatuses() {
+ return (dispatch, getState) => {
+ if (getState().getIn(['status_lists', 'emoji_reactions', 'isLoading'])) {
+ return;
+ }
+
+ dispatch(fetchEmojiReactedStatusesRequest());
+
+ api(getState).get('/api/v1/emoji_reactions').then(response => {
+ console.dir(response.data)
+ const next = getLinks(response).refs.find(link => link.rel === 'next');
+ dispatch(importFetchedStatuses(response.data));
+ dispatch(fetchEmojiReactedStatusesSuccess(response.data, next ? next.uri : null));
+ }).catch(error => {
+ dispatch(fetchEmojiReactedStatusesFail(error));
+ });
+ };
+}
+
+export function fetchEmojiReactedStatusesRequest() {
+ return {
+ type: EMOJI_REACTED_STATUSES_FETCH_REQUEST,
+ skipLoading: true,
+ };
+}
+
+export function fetchEmojiReactedStatusesSuccess(statuses, next) {
+ return {
+ type: EMOJI_REACTED_STATUSES_FETCH_SUCCESS,
+ statuses,
+ next,
+ skipLoading: true,
+ };
+}
+
+export function fetchEmojiReactedStatusesFail(error) {
+ return {
+ type: EMOJI_REACTED_STATUSES_FETCH_FAIL,
+ error,
+ skipLoading: true,
+ };
+}
+
+export function expandEmojiReactedStatuses() {
+ return (dispatch, getState) => {
+ const url = getState().getIn(['status_lists', 'emoji_reactions', 'next'], null);
+
+ if (url === null || getState().getIn(['status_lists', 'emoji_reactions', 'isLoading'])) {
+ return;
+ }
+
+ dispatch(expandEmojiReactedStatusesRequest());
+
+ api(getState).get(url).then(response => {
+ const next = getLinks(response).refs.find(link => link.rel === 'next');
+ dispatch(importFetchedStatuses(response.data));
+ dispatch(expandEmojiReactedStatusesSuccess(response.data, next ? next.uri : null));
+ }).catch(error => {
+ dispatch(expandEmojiReactedStatusesFail(error));
+ });
+ };
+}
+
+export function expandEmojiReactedStatusesRequest() {
+ return {
+ type: EMOJI_REACTED_STATUSES_EXPAND_REQUEST,
+ };
+}
+
+export function expandEmojiReactedStatusesSuccess(statuses, next) {
+ return {
+ type: EMOJI_REACTED_STATUSES_EXPAND_SUCCESS,
+ statuses,
+ next,
+ };
+}
+
+export function expandEmojiReactedStatusesFail(error) {
+ return {
+ type: EMOJI_REACTED_STATUSES_EXPAND_FAIL,
+ error,
+ };
+}
diff --git a/app/javascript/mastodon/features/compose/components/action_bar.jsx b/app/javascript/mastodon/features/compose/components/action_bar.jsx
index ee584cb1ba..714071c875 100644
--- a/app/javascript/mastodon/features/compose/components/action_bar.jsx
+++ b/app/javascript/mastodon/features/compose/components/action_bar.jsx
@@ -10,6 +10,7 @@ const messages = defineMessages({
preferences: { id: 'navigation_bar.preferences', defaultMessage: 'Preferences' },
follow_requests: { id: 'navigation_bar.follow_requests', defaultMessage: 'Follow requests' },
favourites: { id: 'navigation_bar.favourites', defaultMessage: 'Favourites' },
+ emoji_reactions: { id: 'navigation_bar.emoji_reactions', defaultMessage: 'Emoji Reactions' },
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' },
@@ -44,6 +45,7 @@ class ActionBar extends React.PureComponent {
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.emoji_reactions), to: '/emoji_reactions' });
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' });
diff --git a/app/javascript/mastodon/features/emoji_reacted_statuses/index.jsx b/app/javascript/mastodon/features/emoji_reacted_statuses/index.jsx
new file mode 100644
index 0000000000..6d878d3da3
--- /dev/null
+++ b/app/javascript/mastodon/features/emoji_reacted_statuses/index.jsx
@@ -0,0 +1,108 @@
+import { debounce } from 'lodash';
+import PropTypes from 'prop-types';
+import React from 'react';
+import { Helmet } from 'react-helmet';
+import ImmutablePropTypes from 'react-immutable-proptypes';
+import ImmutablePureComponent from 'react-immutable-pure-component';
+import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
+import { connect } from 'react-redux';
+import { addColumn, removeColumn, moveColumn } from 'mastodon/actions/columns';
+import { fetchEmojiReactedStatuses, expandEmojiReactedStatuses } from 'mastodon/actions/emoji_reactions';
+import ColumnHeader from 'mastodon/components/column_header';
+import StatusList from 'mastodon/components/status_list';
+import Column from 'mastodon/features/ui/components/column';
+
+const messages = defineMessages({
+ heading: { id: 'column.emoji_reactions', defaultMessage: 'Emoji Reactions' },
+});
+
+const mapStateToProps = state => ({
+ statusIds: state.getIn(['status_lists', 'emoji_reactions', 'items']),
+ isLoading: state.getIn(['status_lists', 'emoji_reactions', 'isLoading'], true),
+ hasMore: !!state.getIn(['status_lists', 'emoji_reactions', 'next']),
+});
+
+export default @connect(mapStateToProps)
+@injectIntl
+class EmojiReactions 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,
+ };
+
+ componentWillMount () {
+ this.props.dispatch(fetchEmojiReactedStatuses());
+ }
+
+ handlePin = () => {
+ const { columnId, dispatch } = this.props;
+
+ if (columnId) {
+ dispatch(removeColumn(columnId));
+ } else {
+ dispatch(addColumn('EMOJI_REACTIONS', {}));
+ }
+ };
+
+ 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(expandEmojiReactedStatuses());
+ }, 300, { leading: true });
+
+ render () {
+ const { intl, statusIds, columnId, multiColumn, hasMore, isLoading } = this.props;
+ const pinned = !!columnId;
+
+ const emptyMessage =