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