tweet-body.ts 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. import { Component, Input } from "@angular/core";
  2. import twittertext from "twitter-text";
  3. @Component({
  4. selector: "tweet-body",
  5. templateUrl: "tweet-body.html"
  6. })
  7. export class TweetBodyComponent {
  8. @Input()
  9. data: any[];
  10. constructor() {}
  11. get full_text(): string {
  12. if (this.data["retweeted_status"]) {
  13. return this.data["retweeted_status"]["full_text"];
  14. } else {
  15. return this.data["full_text"];
  16. }
  17. }
  18. get entities() {
  19. if (this.data["retweeted_status"]) {
  20. return this.data["retweeted_status"]["entities"];
  21. } else {
  22. return this.data["entities"];
  23. }
  24. }
  25. get extended_entities() {
  26. if (this.data["retweeted_status"]) {
  27. return this.data["retweeted_status"]["extended_entities"];
  28. } else {
  29. return this.data["extended_entities"];
  30. }
  31. }
  32. get range() {
  33. if (this.data["retweeted_status"]) {
  34. return this.data["retweeted_status"]["display_text_range"];
  35. } else {
  36. return this.data["display_text_range"];
  37. }
  38. }
  39. get hasPhoto() {
  40. return (
  41. !this.data["private_tweet"] &&
  42. !this.isGif &&
  43. (this.entities["media"] && this.entities["media"][0]["type"] == "photo")
  44. );
  45. }
  46. get isGif() {
  47. return (
  48. !this.data["private_tweet"] &&
  49. this.extended_entities &&
  50. this.extended_entities["media"] &&
  51. this.extended_entities["media"][0]["type"] === "animated_gif"
  52. );
  53. }
  54. get status() {
  55. // Cut off beginning
  56. let status = this.full_text.substring(this.range[0]);
  57. // Cut off end (URLs)
  58. this.urlsToRemoveFromStatus().forEach(
  59. url => (status = status.replace(url, ""))
  60. );
  61. return status.trim();
  62. }
  63. get tweetArray() {
  64. const extractedEntites = twittertext.extractEntitiesWithIndices(
  65. this.status
  66. );
  67. let tweetArray = [];
  68. tweetArray = tweetArray.concat(
  69. this.getHashtagsForTweetArray(
  70. extractedEntites.filter(element => element["hashtag"])
  71. )
  72. );
  73. tweetArray = tweetArray.concat(
  74. this.getMentionsForTweetArray(
  75. extractedEntites.filter(element => element["screenName"])
  76. )
  77. );
  78. tweetArray = tweetArray.concat(
  79. this.getUrlsForTweetArray(
  80. extractedEntites.filter(element => element["url"])
  81. )
  82. );
  83. tweetArray = tweetArray.concat(this.getTextParts(tweetArray));
  84. return tweetArray.sort((a, b) => a["start"] - b["start"]);
  85. }
  86. private urlsToRemoveFromStatus(): string[] {
  87. const res = [];
  88. if (this.data["quoted_status_permalink"]) {
  89. res.push(this.data["quoted_status_permalink"]["url"]);
  90. }
  91. if (this.extended_entities) {
  92. this.extended_entities["media"].forEach(element => {
  93. res.push(element["url"]);
  94. });
  95. }
  96. return res.filter(entry => entry.length);
  97. }
  98. private getHashtagsForTweetArray(hashtags) {
  99. const res = [];
  100. hashtags.forEach(element => {
  101. res.push({
  102. start: element.indices[0],
  103. stop: element.indices[1],
  104. type: "hashtag",
  105. text: "#" + element["hashtag"]
  106. });
  107. });
  108. return res;
  109. }
  110. private getMentionsForTweetArray(mentions) {
  111. const res = [];
  112. mentions.forEach(element => {
  113. const apiEntity = this.entities.user_mentions.filter(
  114. el =>
  115. el.screen_name.toLowerCase() === element["screenName"].toLowerCase()
  116. )[0];
  117. if (apiEntity) {
  118. res.push({
  119. start: element.indices[0],
  120. stop: element.indices[1],
  121. type: "user_mention",
  122. text: "@" + element["screenName"],
  123. userId: apiEntity["id_str"]
  124. });
  125. }
  126. });
  127. return res;
  128. }
  129. private getUrlsForTweetArray(urls) {
  130. const res = [];
  131. urls.forEach(element => {
  132. const apiEntity = this.entities.urls.filter(
  133. el => el.url.toLowerCase() === element["url"].toLowerCase()
  134. )[0];
  135. res.push({
  136. start: element.indices[0],
  137. stop: element.indices[1],
  138. type: "url",
  139. url: element["url"],
  140. text: apiEntity["display_url"] || element["url"]
  141. });
  142. });
  143. return res;
  144. }
  145. private getTextParts(tweetArray) {
  146. const sortedTweetArray = tweetArray.sort((a, b) => a["start"] - b["start"]);
  147. const textParts = [];
  148. let prevEnd = 0;
  149. for (let i = 0; i < sortedTweetArray.length; i++) {
  150. if (sortedTweetArray[i]["start"] !== prevEnd) {
  151. const text = this.status.substring(
  152. prevEnd,
  153. sortedTweetArray[i]["start"]
  154. );
  155. textParts.push({
  156. start: prevEnd,
  157. stop: sortedTweetArray[i]["start"],
  158. type: "text",
  159. text: text
  160. });
  161. }
  162. prevEnd = sortedTweetArray[i]["stop"];
  163. }
  164. if (prevEnd != this.status.length) {
  165. textParts.push({
  166. start: prevEnd,
  167. stop: this.status.length,
  168. type: "text",
  169. text: this.status.substring(prevEnd, this.status.length)
  170. });
  171. }
  172. return textParts;
  173. }
  174. }