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.

97 lines
3.6 KiB

  1. import escapeTextContentForBrowser from 'escape-html';
  2. import emojify from '../../features/emoji/emoji';
  3. import { unescapeHTML } from '../../utils/html';
  4. import { expandSpoilers } from '../../initial_state';
  5. const domParser = new DOMParser();
  6. const makeEmojiMap = record => record.emojis.reduce((obj, emoji) => {
  7. obj[`:${emoji.shortcode}:`] = emoji;
  8. return obj;
  9. }, {});
  10. export function searchTextFromRawStatus (status) {
  11. const spoilerText = status.spoiler_text || '';
  12. const searchContent = ([spoilerText, status.content].concat((status.poll && status.poll.options) ? status.poll.options.map(option => option.title) : [])).join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n');
  13. return domParser.parseFromString(searchContent, 'text/html').documentElement.textContent;
  14. }
  15. export function normalizeAccount(account) {
  16. account = { ...account };
  17. const emojiMap = makeEmojiMap(account);
  18. const displayName = account.display_name.trim().length === 0 ? account.username : account.display_name;
  19. account.display_name_html = emojify(escapeTextContentForBrowser(displayName), emojiMap);
  20. account.note_emojified = emojify(account.note, emojiMap);
  21. if (account.fields) {
  22. account.fields = account.fields.map(pair => ({
  23. ...pair,
  24. name_emojified: emojify(escapeTextContentForBrowser(pair.name), emojiMap),
  25. value_emojified: emojify(pair.value, emojiMap),
  26. value_plain: unescapeHTML(pair.value),
  27. }));
  28. }
  29. if (account.moved) {
  30. account.moved = account.moved.id;
  31. }
  32. return account;
  33. }
  34. export function normalizeStatus(status, normalOldStatus) {
  35. const normalStatus = { ...status };
  36. normalStatus.account = status.account.id;
  37. if (status.reblog && status.reblog.id) {
  38. normalStatus.reblog = status.reblog.id;
  39. }
  40. if (status.poll && status.poll.id) {
  41. normalStatus.poll = status.poll.id;
  42. }
  43. // Only calculate these values when status first encountered
  44. // Otherwise keep the ones already in the reducer
  45. if (normalOldStatus) {
  46. normalStatus.search_index = normalOldStatus.get('search_index');
  47. normalStatus.contentHtml = normalOldStatus.get('contentHtml');
  48. normalStatus.spoilerHtml = normalOldStatus.get('spoilerHtml');
  49. normalStatus.hidden = normalOldStatus.get('hidden');
  50. } else {
  51. const spoilerText = normalStatus.spoiler_text || '';
  52. const searchContent = ([spoilerText, status.content].concat((status.poll && status.poll.options) ? status.poll.options.map(option => option.title) : [])).join('\n\n').replace(/<br\s*\/?>/g, '\n').replace(/<\/p><p>/g, '\n\n');
  53. const emojiMap = makeEmojiMap(normalStatus);
  54. normalStatus.search_index = domParser.parseFromString(searchContent, 'text/html').documentElement.textContent;
  55. normalStatus.contentHtml = emojify(normalStatus.content, emojiMap);
  56. normalStatus.spoilerHtml = emojify(escapeTextContentForBrowser(spoilerText), emojiMap);
  57. normalStatus.hidden = expandSpoilers ? false : spoilerText.length > 0 || normalStatus.sensitive;
  58. }
  59. return normalStatus;
  60. }
  61. export function normalizePoll(poll) {
  62. const normalPoll = { ...poll };
  63. const emojiMap = makeEmojiMap(normalPoll);
  64. normalPoll.options = poll.options.map((option, index) => ({
  65. ...option,
  66. voted: poll.own_votes && poll.own_votes.includes(index),
  67. title_emojified: emojify(escapeTextContentForBrowser(option.title), emojiMap),
  68. }));
  69. return normalPoll;
  70. }
  71. export function normalizeAnnouncement(announcement) {
  72. const normalAnnouncement = { ...announcement };
  73. const emojiMap = makeEmojiMap(normalAnnouncement);
  74. normalAnnouncement.contentHtml = emojify(normalAnnouncement.content, emojiMap);
  75. return normalAnnouncement;
  76. }