You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

140 lines
4.1 KiB

7 years ago
7 years ago
  1. import {
  2. REBLOG_REQUEST,
  3. REBLOG_SUCCESS,
  4. REBLOG_FAIL,
  5. UNREBLOG_SUCCESS,
  6. FAVOURITE_REQUEST,
  7. FAVOURITE_SUCCESS,
  8. FAVOURITE_FAIL,
  9. UNFAVOURITE_SUCCESS,
  10. PIN_SUCCESS,
  11. UNPIN_SUCCESS,
  12. } from '../actions/interactions';
  13. import {
  14. STATUS_FETCH_SUCCESS,
  15. CONTEXT_FETCH_SUCCESS,
  16. STATUS_MUTE_SUCCESS,
  17. STATUS_UNMUTE_SUCCESS,
  18. STATUS_REVEAL,
  19. STATUS_HIDE,
  20. } from '../actions/statuses';
  21. import {
  22. TIMELINE_REFRESH_SUCCESS,
  23. TIMELINE_UPDATE,
  24. TIMELINE_DELETE,
  25. TIMELINE_EXPAND_SUCCESS,
  26. } from '../actions/timelines';
  27. import {
  28. NOTIFICATIONS_UPDATE,
  29. NOTIFICATIONS_REFRESH_SUCCESS,
  30. NOTIFICATIONS_EXPAND_SUCCESS,
  31. } from '../actions/notifications';
  32. import {
  33. FAVOURITED_STATUSES_FETCH_SUCCESS,
  34. FAVOURITED_STATUSES_EXPAND_SUCCESS,
  35. } from '../actions/favourites';
  36. import {
  37. PINNED_STATUSES_FETCH_SUCCESS,
  38. } from '../actions/pin_statuses';
  39. import { SEARCH_FETCH_SUCCESS } from '../actions/search';
  40. import emojify from '../features/emoji/emoji';
  41. import { Map as ImmutableMap, fromJS } from 'immutable';
  42. import escapeTextContentForBrowser from 'escape-html';
  43. const domParser = new DOMParser();
  44. const normalizeStatus = (state, status) => {
  45. if (!status) {
  46. return state;
  47. }
  48. const normalStatus = { ...status };
  49. normalStatus.account = status.account.id;
  50. if (status.reblog && status.reblog.id) {
  51. state = normalizeStatus(state, status.reblog);
  52. normalStatus.reblog = status.reblog.id;
  53. }
  54. const searchContent = [status.spoiler_text, status.content].join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n');
  55. const emojiMap = normalStatus.emojis.reduce((obj, emoji) => {
  56. obj[`:${emoji.shortcode}:`] = emoji;
  57. return obj;
  58. }, {});
  59. normalStatus.search_index = domParser.parseFromString(searchContent, 'text/html').documentElement.textContent;
  60. normalStatus.contentHtml = emojify(normalStatus.content, emojiMap);
  61. normalStatus.spoilerHtml = emojify(escapeTextContentForBrowser(normalStatus.spoiler_text || ''), emojiMap);
  62. normalStatus.hidden = normalStatus.sensitive;
  63. return state.update(status.id, ImmutableMap(), map => map.mergeDeep(fromJS(normalStatus)));
  64. };
  65. const normalizeStatuses = (state, statuses) => {
  66. statuses.forEach(status => {
  67. state = normalizeStatus(state, status);
  68. });
  69. return state;
  70. };
  71. const deleteStatus = (state, id, references) => {
  72. references.forEach(ref => {
  73. state = deleteStatus(state, ref[0], []);
  74. });
  75. return state.delete(id);
  76. };
  77. const initialState = ImmutableMap();
  78. export default function statuses(state = initialState, action) {
  79. switch(action.type) {
  80. case TIMELINE_UPDATE:
  81. case STATUS_FETCH_SUCCESS:
  82. case NOTIFICATIONS_UPDATE:
  83. return normalizeStatus(state, action.status);
  84. case REBLOG_SUCCESS:
  85. case UNREBLOG_SUCCESS:
  86. case FAVOURITE_SUCCESS:
  87. case UNFAVOURITE_SUCCESS:
  88. case PIN_SUCCESS:
  89. case UNPIN_SUCCESS:
  90. return normalizeStatus(state, action.response);
  91. case FAVOURITE_REQUEST:
  92. return state.setIn([action.status.get('id'), 'favourited'], true);
  93. case FAVOURITE_FAIL:
  94. return state.setIn([action.status.get('id'), 'favourited'], false);
  95. case REBLOG_REQUEST:
  96. return state.setIn([action.status.get('id'), 'reblogged'], true);
  97. case REBLOG_FAIL:
  98. return state.setIn([action.status.get('id'), 'reblogged'], false);
  99. case STATUS_MUTE_SUCCESS:
  100. return state.setIn([action.id, 'muted'], true);
  101. case STATUS_UNMUTE_SUCCESS:
  102. return state.setIn([action.id, 'muted'], false);
  103. case STATUS_REVEAL:
  104. return state.withMutations(map => {
  105. action.ids.forEach(id => map.setIn([id, 'hidden'], false));
  106. });
  107. case STATUS_HIDE:
  108. return state.withMutations(map => {
  109. action.ids.forEach(id => map.setIn([id, 'hidden'], true));
  110. });
  111. case TIMELINE_REFRESH_SUCCESS:
  112. case TIMELINE_EXPAND_SUCCESS:
  113. case CONTEXT_FETCH_SUCCESS:
  114. case NOTIFICATIONS_REFRESH_SUCCESS:
  115. case NOTIFICATIONS_EXPAND_SUCCESS:
  116. case FAVOURITED_STATUSES_FETCH_SUCCESS:
  117. case FAVOURITED_STATUSES_EXPAND_SUCCESS:
  118. case PINNED_STATUSES_FETCH_SUCCESS:
  119. case SEARCH_FETCH_SUCCESS:
  120. return normalizeStatuses(state, action.statuses);
  121. case TIMELINE_DELETE:
  122. return deleteStatus(state, action.id, action.references);
  123. default:
  124. return state;
  125. }
  126. };