tweet-body.ts 5.0 KB

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