[Twitter]全部入りセット

説明。

目前为 2023-10-26 提交的版本。查看 最新版本

  1. // ==UserScript==
  2. // @name [Twitter]全部入りセット
  3. // @namespace https://gf.qytechs.cn/ja/users/1023652
  4. // @version 0.1.0.7_beta
  5. // @description 説明。
  6. // @author ゆにてぃー
  7. // @match https://twitter.com/*
  8. // @match https://mobile.twitter.com/*
  9. // @match https://x.com/*
  10. // @match https://X.com/*
  11. // @connect twitter.com
  12. // @connect api.twitter.com
  13. // @connect api.fanbox.cc
  14. // @connect pbs.twimg.com
  15. // @connect abs.twimg.com
  16. // @connect video.twimg.com
  17. // @connect discord.com
  18. // @connect booth.pm
  19. // @connect carrd.co
  20. // @connect creatorlink.net
  21. // @connect fantia.jp
  22. // @connect html.co.jp
  23. // @connect linktr.ee
  24. // @connect lit.link
  25. // @connect potofu.me
  26. // @connect profcard.info
  27. // @connect skeb.jp
  28. // @connect sketch.pixiv.net
  29. // @connect tumblr.com
  30. // @connect twpf.jp
  31. // @connect lab.syncer.jp
  32. // @connect geek-website.com
  33. // @icon 
  34. // @grant GM_xmlhttpRequest
  35. // @grant GM_registerMenuCommand
  36. // @license MIT
  37. // @run-at document-idle
  38. // ==/UserScript==
  39.  
  40. (function(){
  41. 'use strict';
  42. const commonselectors = {
  43. 'tweet_field': 'article[data-testid="tweet"]',
  44. 'retweeted': '[data-testid="socialContext"]',
  45. 'liked_color': 'r-vkub15',
  46. 'liked': 'M20.884 13.19c-1.351 2.48-4.001 5.12-8.379 7.67l-.503.3-.504-.3c-4.379-2.55-7.029-5.19-8.382-7.67-1.36-2.5-1.41-4.86-.514-6.67.887-1.79 2.647-2.91 4.601-3.01 1.651-.09 3.368.56 4.798 2.01 1.429-1.45 3.146-2.1 4.796-2.01 1.954.1 3.714 1.22 4.601 3.01.896 1.81.846 4.17-.514 6.67z',
  47. 'info_field': '.css-1dbjc4n.r-1d09ksm.r-1471scf.r-18u37iz.r-1wbh5a2',
  48. 'click_media_field': '.css-1dbjc4n.r-1p0dtai.r-1mlwlqe.r-1d2f490.r-dnmrzs.r-1udh08x.r-u8s1d.r-zchlnj.r-ipm5af.r-417010',
  49. 'profile_field_Header_Items': '[data-testid="UserProfileHeader_Items"]',
  50. 'engagementsTextColor': {
  51. "0": {"count": "rgb(15, 20, 25)","text": "rgb(83, 100, 113)"},
  52. "1": {"count": "rgb(247, 249, 249)","text": "rgb(139, 152, 165)"},
  53. "2": {"count": "rgb(231, 233, 234)","text": "rgb(113, 118, 123)"}
  54. },
  55. }
  56. const desktop_selectors = {
  57. 'time_line_media_field': '.css-1dbjc4n.r-1p0dtai.r-1mlwlqe.r-1d2f490.r-11wrixw.r-61z16t.r-1udh08x.r-u8s1d.r-zchlnj.r-ipm5af.r-417010',
  58. 'media_field': '.css-1dbjc4n.r-1ssbvtb.r-1s2bzr4',
  59. 'profileField': '.css-1dbjc4n.r-1ifxtd0.r-ymttw5.r-ttdzmv',
  60. 'followersLink': '.css-4rbku5.css-18t94o4.css-901oao.r-vlxjld.r-1loqt21.r-1tl8opc.r-a023e6.r-16dba41.r-rjixqe.r-bcqeeo.r-qvutc0',
  61. };
  62. const mobile_selectors = {
  63. 'time_line_media_field': '.css-1dbjc4n.r-1p0dtai.r-1mlwlqe.r-1d2f490.r-1udh08x.r-u8s1d.r-zchlnj.r-ipm5af.r-417010',
  64. 'media_field': '.css-1dbjc4n.r-1s367qj.r-a1ub67',
  65. 'profileField': '.css-1dbjc4n.r-ku1wi2.r-1j3t67a.r-1b3ntt7',
  66. 'followersLink': '.css-4rbku5.css-18t94o4.css-901oao.r-vlxjld.r-1loqt21.r-1tl8opc.r-1b43r93.r-16dba41.r-hjklzo.r-bcqeeo.r-qvutc0',
  67. };
  68. const deny_names = ["home", "explore", "notifications", "messages", "i", "settings", "tos", "privacy", "compose", "search"];
  69. const denyNamesRegex = new RegExp(`https://twitter\\.com/((?!${deny_names.join('|')})[^/]+)`, 'i');
  70. let currentUrl = document.location.href;
  71. let updating = false;
  72. const userAgent = navigator.userAgent || navigator.vendor || window.opera;
  73. const timeZoneObject = Intl.DateTimeFormat().resolvedOptions();
  74. let env_selector;
  75. let isMobile = /Android|webOS|iPhone|iPad|iPod|BlackBerry|IEMobile|Opera Mini/i.test(userAgent);
  76. if(isMobile){
  77. env_selector = {...commonselectors,...mobile_selectors};
  78. }else{
  79. env_selector = {...commonselectors,...desktop_selectors};
  80. }
  81. let fetchedTweets = {};
  82. let fetchedTweetsUserData = {};
  83. let fetchedTweetsUserDataByUserName = {};
  84. let isFunctionRunning = {
  85. Hello_tweet_where_are_you_from: false,
  86. Engagement_Restorer: false,
  87. };
  88. let storedSettings = {
  89. 'webhook_brings_tweets_to_discord': JSON.parse(localStorage.getItem('webhook_brings_tweets_to_discord') || '{}'),
  90. 'Hello_tweet_where_are_you_from': JSON.parse(localStorage.getItem('Hello_tweet_where_are_you_from') || '{}'),
  91. 'Show_me_your_Pixiv': JSON.parse(localStorage.getItem('Show_me_your_Pixiv') || '{}'),
  92. 'Note_Tweet_expander': JSON.parse(localStorage.getItem('Note_Tweet_expander') || '{}'),
  93. 'Show_me_big_pics': JSON.parse(localStorage.getItem('Show_me_big_pics') || '{}'),
  94. 'sneakilyFavorite': JSON.parse(localStorage.getItem('sneakilyFavorite') || '{}'),
  95. 'Engagement_Restorer': JSON.parse(localStorage.getItem('Engagement_Restorer') || '{}'),
  96. };
  97. let script_settings = {};
  98. script_settings['webhook_brings_tweets_to_discord'] = {
  99. "displayMethod": storedSettings['webhook_brings_tweets_to_discord'].displayMethod || 'method1',
  100. "lang": storedSettings['webhook_brings_tweets_to_discord'].lang || GetCookie("lang") || 'en',
  101. "defaultWebhook": storedSettings['webhook_brings_tweets_to_discord'].defaultWebhook,
  102. "downloadVideo": storedSettings['webhook_brings_tweets_to_discord'].downloadVideo || false,
  103. "webHooks": (function(data){
  104. let webhooks = {};
  105. if(data && Array.isArray(data)){
  106. data.forEach(item => {
  107. if(item.name && item.value){
  108. webhooks[item.name] = item.value;
  109. }
  110. });
  111. }
  112. return webhooks;
  113. })(storedSettings['webhook_brings_tweets_to_discord'].data),
  114. };
  115. let scriptDataStore = {
  116. "Show_me_your_Pixiv": JSON.parse(localStorage.getItem('user_pixvi_link_collection')) || {},
  117. };
  118. let Text = {};
  119. Text.ja = {
  120. "settings": "の設定",
  121. "close": "閉じる",
  122. "general": "全体",
  123. "webhook_brings_tweets_to_discord": {
  124. "functionName": "WebhookがTweetを連れてくるわ今日も",
  125. "link_to_tweet": "ツイートへ",
  126. "link_to_image": "画像へのリンク",
  127. "engagement": "エンゲージメント",
  128. "likes": "いいね",
  129. "retweets": "リツイート",
  130. "units": "万",
  131. "roundingScale": 10000,
  132. "decimalPlaces": 2,
  133. "postedDate": "投稿日時",
  134. "quotedTweet": "↓♻️引用元♻️↓",
  135. "submit": "送信",
  136. "display_everywhere": "どこでも表示する",
  137. "tweet_details_only": "詳細表示したときだけ",
  138. "when_webhook_name_duplicate": "Webhookの名前が重複しています。",
  139. "cancel": "キャンセル",
  140. "save_settings": "設定を保存",
  141. "display_method": "表示方法",
  142. "default": "デフォルトの",
  143. "language": "送信時言語",
  144. "webhook_not_set": "ウェブフックが設定されていません。",
  145. "when_webhook_url_invalid": "正しいDiscordのWebhookのURLではありません。",
  146. "when_post_failed": "以下のURLのポストに失敗しました。",
  147. "various_links": "各種リンク",
  148. },
  149. "Engagement_Restorer": {
  150. "functionName": "返ってこい!リツイート欄!",
  151. "retweet": "リツイート",
  152. "quoted": "件の引用",
  153. "like": "いいね",
  154. "units": "万",
  155. "roundingScale": 10000,
  156. "decimalPlaces": 2,
  157. },
  158. "sneakilyFavorite": {
  159. "functionName": "こっそりいいね",
  160. "favorite": "いいね!",
  161. },
  162. };
  163.  
  164. let env_Text = Text[script_settings.lang] || Text.ja;
  165.  
  166. locationChange();
  167.  
  168. async function main(refresh = false){
  169. const selector = refresh ? 'article[data-testid="tweet"]' : 'article[data-testid="tweet"]:not(.checked-tweet)';
  170. const tweetNodes = Array.from(document.querySelectorAll(selector)).map(tweet => {
  171. const link = tweet.querySelector(`[data-testid="User-Name"] a[aria-label], ${env_selector.info_field} a[aria-label]`);
  172. if(link){
  173. const match = link.href.match(/twitter\.com\/[^/]+\/status\/(\d+)/);
  174. if(match){
  175. tweet.classList.add('checked-tweet');
  176. return { id: match[1], link: link.href, node: tweet };
  177. }
  178. }
  179. }).filter(Boolean);
  180. console.log(tweetNodes)
  181. const isTweetDetail = !!currentUrl.match(/twitter\.com\/[\w]*\/status\/[0-9]*/);
  182. if(isTweetDetail){
  183. Hello_tweet_where_are_you_from();
  184. }
  185. if(tweetNodes){
  186. Note_Tweet_expander(tweetNodes);
  187. }
  188. if(tweetNodes){
  189. Show_me_your_Pixiv(tweetNodes);
  190. }
  191. if(isTweetDetail){
  192. Engagement_Restorer();
  193. }
  194. if(tweetNodes){
  195. sneakilyFavorite(tweetNodes);
  196. }
  197. if(tweetNodes){
  198. webhook_brings_tweets_to_discord(tweetNodes);
  199. }
  200. if(true){
  201. showFollowers();
  202. }
  203. if(tweetNodes){
  204. hideAnalytics(tweetNodes);
  205. }
  206. if(isMobile){
  207. shareTweet_Restorer_for_mobile(tweetNodes);
  208. }
  209. //if(currentUrl.match(/https?:\/\/twitter\.com\/[\w]*\/status\/[0-9]*/))console.log(await waitForTweetData(extractTweetId(currentUrl)))
  210. }
  211. function shareTweet_Restorer_for_mobile(tweetNodes){
  212. tweetNodes.forEach(async (tweet)=>{
  213. let color = ['rgb(83, 100, 113)','rgb(139, 152, 165)','rgb(113, 118, 123)'];
  214. let footer = tweet.node.querySelector('div[id][role="group"]');
  215. let lastChild = footer.lastElementChild;
  216. let clonedNode = lastChild.cloneNode(true);
  217. clonedNode.style.marginLeft = "1em";
  218. let clonedSvg = clonedNode.querySelector('svg');
  219. while(clonedSvg.firstChild){
  220. clonedSvg.removeChild(clonedSvg.firstChild);
  221. }
  222. let newPath = document.createElementNS('http://www.w3.org/2000/svg', 'path');
  223. newPath.setAttribute('d', 'M17 4c-1.1 0-2 .9-2 2 0 .33.08.65.22.92C15.56 7.56 16.23 8 17 8c1.1 0 2-.9 2-2s-.9-2-2-2zm-4 2c0-2.21 1.79-4 4-4s4 1.79 4 4-1.79 4-4 4c-1.17 0-2.22-.5-2.95-1.3l-4.16 2.37c.07.3.11.61.11.93s-.04.63-.11.93l4.16 2.37c.73-.8 1.78-1.3 2.95-1.3 2.21 0 4 1.79 4 4s-1.79 4-4 4-4-1.79-4-4c0-.32.04-.63.11-.93L8.95 14.7C8.22 15.5 7.17 16 6 16c-2.21 0-4-1.79-4-4s1.79-4 4-4c1.17 0 2.22.5 2.95 1.3l4.16-2.37c-.07-.3-.11-.61-.11-.93zm-7 4c-1.1 0-2 .9-2 2s.9 2 2 2c.77 0 1.44-.44 1.78-1.08.14-.27.22-.59.22-.92s-.08-.65-.22-.92C7.44 10.44 6.77 10 6 10zm11 6c-.77 0-1.44.44-1.78 1.08-.14.27-.22.59-.22.92 0 1.1.9 2 2 2s2-.9 2-2-.9-2-2-2z');
  224. clonedSvg.appendChild(newPath);
  225. clonedSvg.addEventListener('click', () => {
  226. copyToClipboard(tweet.link);
  227. });
  228. clonedNode.addEventListener('click', (event) => {
  229. event.stopPropagation();
  230. });
  231. footer.appendChild(clonedNode);
  232. });
  233. function copyToClipboard(text){
  234. navigator.clipboard.writeText(text).then(function() {
  235. //console.log('クリップボードにコピーしました!');
  236. }).catch(function(err) {
  237. console.error('コピーに失敗しました:', err);
  238. });
  239. }
  240. }
  241. async function showFollowers(){
  242. try{
  243. const screenName = extractUserName(currentUrl);
  244. if(!currentUrl.match(new RegExp(`${screenName}$`)))return;
  245. (await wait_load_Element(`${env_selector.profileField} ${env_selector.followersLink}`,500,5)).forEach(e=>{
  246. if(e.href.match(/verified_followers$/)){
  247. e.href = e.href.replace(/verified_followers$/,'followers');
  248. e.addEventListener('click',async (event) => {
  249. (await wait_load_Element('a[role="tab"]')).forEach(t=>{
  250. if(t.href.match(/followers$/))t.click();
  251. });
  252. });
  253. }
  254. });
  255. }catch(error){console.error(error)}
  256. }
  257. function hideAnalytics(tweetNodes){
  258. try{
  259. tweetNodes.forEach(t=>{
  260. t.node.querySelector('div[id][role="group"] a[role="link"]').parentNode.style.display = "none";
  261. });
  262. }catch(error){console.error(error)}
  263. }
  264. async function webhook_brings_tweets_to_discord(tweetNodes){
  265. let textData = env_Text.webhook_brings_tweets_to_discord;
  266. let thisScriptSettings = script_settings['webhook_brings_tweets_to_discord'];
  267. tweetNodes.forEach(function(tweetNode){
  268. let element = tweetNode.node;
  269. if(element.querySelector(".quickDimg")) return;
  270. let tweet_link = tweetNode.link;
  271. let fotter = element.querySelector('div[id][role="group"]');
  272. const flexContainer = document.createElement('div');
  273. flexContainer.classList.add('quickDimg');
  274. flexContainer.style.display = 'flex';
  275.  
  276. // 1つ目のドロップダウン(サーバー選択)
  277. const dropdown_select_server = document.createElement('select');
  278. dropdown_select_server.className = "quickDimgPullDown quickDimgPullDown1";
  279. for(const server in thisScriptSettings.webHooks){
  280. const option = document.createElement('option');
  281. option.value = thisScriptSettings.webHooks[server];
  282. option.textContent = server;
  283. if(server == thisScriptSettings.defaultWebhook){
  284. option.selected = true;
  285. }
  286. dropdown_select_server.appendChild(option);
  287. }
  288. flexContainer.appendChild(dropdown_select_server);
  289. dropdown_select_server.addEventListener('click', (event) => {
  290. event.stopPropagation();
  291. });
  292.  
  293. const dropdown_send_image = document.createElement('select');
  294. dropdown_send_image.className = "quickDimgPullDown quickDimgPullDown2";
  295. for(let i = 1; i <= 5; i++){
  296. const option = document.createElement('option');
  297. option.value = i;
  298. option.textContent = i;
  299. if(i === 5){
  300. option.selected = true;
  301. }
  302. dropdown_send_image.appendChild(option);
  303. }
  304. flexContainer.appendChild(dropdown_send_image);
  305.  
  306. dropdown_send_image.addEventListener('click', (event) => {
  307. event.stopPropagation();
  308. });
  309. const dropdown_post_quote = document.createElement('select');
  310. dropdown_post_quote.className = "quickDimgPullDown quickDimgPullDown3";
  311. ['false','true'].forEach(value => {
  312. const option = document.createElement('option');
  313. option.value = value;
  314. option.textContent = value;
  315. dropdown_post_quote.appendChild(option);
  316. });
  317. flexContainer.appendChild(dropdown_post_quote);
  318.  
  319. dropdown_post_quote.addEventListener('click', (event) => {
  320. event.stopPropagation();
  321. });
  322.  
  323. const dropdown_use_graphql = document.createElement('select');
  324. dropdown_use_graphql.className = "quickDimgPullDown quickDimgPullDown4";
  325. ['false','true'].forEach(value => {
  326. const option = document.createElement('option');
  327. option.value = value;
  328. option.textContent = value;
  329. dropdown_use_graphql.appendChild(option);
  330. });
  331. flexContainer.appendChild(dropdown_use_graphql);
  332.  
  333. dropdown_use_graphql.addEventListener('click', (event) => {
  334. event.stopPropagation();
  335. });
  336. // ボタンを作成
  337. const button = document.createElement('button');
  338. button.className = "quickDimgButton";
  339. button.textContent = textData.submit;
  340. flexContainer.appendChild(button);
  341.  
  342. dropdown_select_server.addEventListener('change', () => {
  343. button.disabled = false;
  344. });
  345. dropdown_send_image.addEventListener('change', () => {
  346. button.disabled = false;
  347. });
  348. dropdown_post_quote.addEventListener('change', () => {
  349. button.disabled = false;
  350. });
  351. dropdown_use_graphql.addEventListener('change', () => {
  352. button.disabled = false;
  353. });
  354.  
  355. // ボタンのクリックイベントを監視
  356. button.addEventListener('click',async function(){
  357. // ここでドロップダウンの選択値に基づいて処理を行う
  358. this.disabled = true;
  359. const selectedServer = dropdown_select_server.value;
  360. const selectedNumber = dropdown_send_image.value;
  361. const send_post_tweet = dropdown_post_quote.value === 'true';
  362. const useGraphql = dropdown_use_graphql.value === 'true';
  363. if(!selectedServer){
  364. customAlert(textData.webhook_not_set);
  365. return;
  366. }
  367. let send_page;
  368. if(selectedNumber != 5){
  369. send_page = [selectedNumber-1]
  370. }else{
  371. send_page = [0,1,2,3]
  372. }
  373. const body = await make_send_data(tweet_link,send_page,send_post_tweet,useGraphql);
  374. await sleep(300)
  375. for(let target in body){
  376. let formData = new FormData();
  377. let payload = {};
  378. let tmp = body[target];
  379. if(tmp.embeds){
  380. payload.embeds = tmp.embeds;
  381. }
  382. if(tmp.content){
  383. payload.content = tmp.content;
  384. }
  385. formData.append('payload_json', JSON.stringify(payload));
  386. //console.log(formData)
  387. if(tmp.files){
  388. tmp.files.forEach((file, index) => {
  389. formData.append(`file${index}`, file.attachment, file.name);
  390. });
  391. }
  392. try{
  393. let res = await request(new sendObject_to_discord_webhook(selectedServer,formData));
  394. if(res.statusText == "Bad Request"){
  395. customAlert(`${textData.when_post_failed}`,payload.embeds[0].url);
  396. }
  397. }catch(error){
  398. customAlert(`${textData.when_post_failed}`,payload.embeds[0].url);
  399. console.log(error);
  400. }
  401. //console.log(res)
  402. await sleep(1000)
  403. }
  404. });
  405. fotter.parentNode.appendChild(flexContainer);
  406. });
  407. async function make_send_data(tweet_link,select_pages = [1],send_quoted_tweet,use_graphQL){
  408. const tweet_id = tweet_link.match(/https?:\/\/twitter\.com\/\w+\/status\/(\d+)/)[1];
  409. let tweet_data,return_object,apiType;
  410. try{
  411. if(use_graphQL){
  412. apiType = "graphQL";
  413. }else{
  414. apiType = "1_1"
  415. }
  416. tweet_data = await getTweetData(tweet_id,apiType);
  417. return_object = await make_embeds();
  418. }catch(error){
  419. customAlert(`${textData.when_post_failed}`,tweet_link);
  420. console.error(error);
  421. }
  422. if(send_quoted_tweet && return_object[return_object.length - 1].quoted_tweet_data){
  423. tweet_data = return_object.pop().quoted_tweet_data;
  424. return_object = return_object.concat([{content: textData.quotedTweet}],await make_embeds(1));
  425. }
  426. return return_object;
  427. async function make_embeds(quoted_tweet_mode = 0){
  428. let embeds = [];
  429. let tmpEmbed = {};
  430. let tmp_return_object = [];
  431. let twitter_user_data = {};
  432. let twitter_tweet_data = {};
  433. let tweet_user_data_json = {};
  434. let tweet_tweet_data_json = {};
  435. tweet_user_data_json = tweet_data.core?.user_results?.result || tweet_data.user;
  436. tweet_tweet_data_json = tweet_data.legacy || tweet_data;
  437. twitter_user_data.ID = tweet_user_data_json.rest_id || tweet_user_data_json.id_str;
  438. twitter_user_data.screen_name = tweet_user_data_json.legacy?.screen_name || tweet_user_data_json.screen_name;
  439. twitter_user_data.name = tweet_user_data_json.legacy?.name || tweet_user_data_json.name;
  440. twitter_user_data.profile_image = tweet_user_data_json.legacy?.profile_image_url_https.replace('_normal.','.') || tweet_user_data_json.profile_image_url_https.replace('_normal.','.');
  441. twitter_user_data.urls = tweet_user_data_json.legacy?.entities || tweet_user_data_json.entities || [];
  442. twitter_tweet_data.hashtags = tweet_tweet_data_json.entities.hashtags || [];
  443. twitter_tweet_data.user_mentions = tweet_tweet_data_json.entities.user_mentions || [];
  444. twitter_tweet_data.symbols = tweet_tweet_data_json.entities.symbols || [];
  445. try{
  446. twitter_user_data.pixiv_url = await find_pixiv_link(extractUrls(twitter_user_data.urls));
  447. }catch(error){
  448. console.log("pixivのURLの取得に失敗しました。");
  449. throw(error);
  450. }
  451. twitter_tweet_data.full_text = tweet_tweet_data_json.full_text || "";
  452. twitter_tweet_data.extended_entities = tweet_tweet_data_json.extended_entities;
  453. twitter_tweet_data.retweet_count = tweet_tweet_data_json.retweet_count;
  454. twitter_tweet_data.favorite_count = tweet_tweet_data_json.favorite_count;
  455. twitter_tweet_data.id = tweet_tweet_data_json.id_str;
  456. twitter_tweet_data.created_at = new Date(tweet_tweet_data_json.created_at).toLocaleString(timeZoneObject.locale, { timeZone: timeZoneObject.timeZone });
  457. twitter_tweet_data.urls = tweet_tweet_data_json.entities.urls;
  458. twitter_tweet_data.media = make_media_list(twitter_tweet_data.extended_entities,select_pages);
  459. try{
  460. //文が長すぎるとエラーになるので一定の長さで切る。
  461. //普通のツイートではそんなことありえないが、Blueでは長いツイートが可能なのでそれに対応している。
  462. if(tweet_data.API_type == "graphql"){
  463. let note_tweet = tweet_data.result.note_tweet?.note_tweet_results.result||tweet_data.result.tweet.note_tweet.note_tweet_results.result;
  464. twitter_tweet_data.full_text = note_tweet.text;
  465. twitter_tweet_data.urls = note_tweet.entity_set.urls;
  466. twitter_tweet_data.hashtags = get_only_particular_key_value(note_tweet.entity_set,"hashtags",[]);
  467. twitter_tweet_data.user_mentions = get_only_particular_key_value(note_tweet.entity_set,"user_mentions",[]);
  468. twitter_tweet_data.symbols = get_only_particular_key_value(note_tweet.entity_set,"symbols",[]);
  469. }
  470. }catch{}
  471. //console.log(twitter_tweet_data.full_text)
  472. // hashtags, mentions, symbolsを一つの配列に結合
  473. let combined = [].concat(
  474. twitter_tweet_data.hashtags.map(tag => ({
  475. type: 'hashtag',
  476. indices: tag.indices,
  477. text: tag.text
  478. })),
  479. twitter_tweet_data.user_mentions.map(mention => ({
  480. type: 'mention',
  481. indices: mention.indices,
  482. text: mention.screen_name
  483. })),
  484. twitter_tweet_data.symbols.map(symbol => ({
  485. type: 'symbol',
  486. indices: symbol.indices,
  487. text: symbol.text
  488. }))
  489. );
  490. // combinedをindicesの順にソート
  491. combined.sort((a, b) => b.indices[0] - a.indices[0]);
  492. let transformedText = twitter_tweet_data.full_text;
  493. function countSurrogatePairs(str){
  494. return Array.from(str).filter(char => char.match(/[\uD800-\uDBFF][\uDC00-\uDFFF]/)).length;
  495. }
  496. const currentTimeMillis = new Date().getTime();
  497. const linkTextStart = `linkTextStart${currentTimeMillis}`;
  498. const linkTextEnd = `linkTextEnd${currentTimeMillis}`;
  499. const linkUrlStart = `linkUrlStart${currentTimeMillis}`;
  500. const linkUrlEnd = `linkUrlEnd${currentTimeMillis}`;
  501. const hashtag = `hashtag${currentTimeMillis}`
  502. combined.forEach(item => {
  503. let start = item.indices[0];
  504. let end = item.indices[1];
  505. // サロゲートペアの数をカウントして調整
  506. const adjustment = countSurrogatePairs(transformedText.slice(0, end));
  507. start += adjustment;
  508. end += adjustment;
  509. let replacement = '';
  510. switch(item.type){
  511. case 'hashtag':
  512. replacement = `${linkTextStart}${hashtag}${item.text}${linkTextEnd}${linkUrlStart}https://twitter.com/hashtag/${item.text}${linkUrlEnd}`;
  513. break;
  514. case 'mention':
  515. replacement = `${linkTextStart}@${item.text}${linkTextEnd}${linkUrlStart}https://twitter.com/${item.text}${linkUrlEnd}`;
  516. break;
  517. case 'symbol':
  518. replacement = `${linkTextStart}$${item.text}${linkTextEnd}${linkUrlStart}https://twitter.com/search?q=%24${item.text}&src=cashtag_click${linkUrlEnd}`;
  519. break;
  520. }
  521. transformedText = transformedText.slice(0, start) + replacement + transformedText.slice(end);
  522. });
  523. twitter_tweet_data.full_text = str_max_length(transformedText,7000);
  524. //マークダウンにならないでほしいやつのエスケープ
  525. let escapeCharacters = ['|', '*', '_', '`', '~', '[', ']', '(', ')', '>', '#', '-'];
  526. escapeCharacters.forEach(char => {
  527. let regExp = new RegExp('\\' + char, 'g');
  528. twitter_tweet_data.full_text = twitter_tweet_data.full_text.replace(regExp, '\\' + char);
  529. });
  530. //マークダウンになって欲しいやつは戻す
  531. twitter_tweet_data.full_text = twitter_tweet_data.full_text.replace(new RegExp(linkTextStart, 'g'), '[');
  532. twitter_tweet_data.full_text = twitter_tweet_data.full_text.replace(new RegExp(linkTextEnd, 'g'), ']');
  533. twitter_tweet_data.full_text = twitter_tweet_data.full_text.replace(new RegExp(linkUrlStart, 'g'), '(');
  534. twitter_tweet_data.full_text = twitter_tweet_data.full_text.replace(new RegExp(linkUrlEnd, 'g'), ')');
  535. twitter_tweet_data.full_text = twitter_tweet_data.full_text.replace(new RegExp(hashtag, 'g'), '#');
  536. /*
  537. try{
  538. if(select_pages.length > 1 && ! twitter_tweet_data.media.every(v => v.media_type == "photo")){
  539. twitter_tweet_data.images = [twitter_tweet_data.media[0]];
  540. }
  541. }catch{}
  542. try{
  543. if(twitter_tweet_data.media.every(v => v.media_type.match(/(video|animated_gif)/))){
  544. twitter_tweet_data.videos = [twitter_tweet_data.media[0]];
  545. }
  546. }catch{}
  547. */
  548. let tweet_url;
  549. if(!quoted_tweet_mode == 1){
  550. tweet_url = tweet_link;
  551. }else{
  552. tweet_url = `https://twitter.com/${twitter_user_data.screen_name}/status/${twitter_tweet_data.id}`;
  553. }
  554. tmpEmbed.color = 1940464;
  555. tmpEmbed.title = "Tweet";
  556. tmpEmbed.url = tweet_url;
  557. tmpEmbed.author = {
  558. "name": `${twitter_user_data.name} (@${twitter_user_data.screen_name})`,
  559. "url": `https://twitter.com/${twitter_user_data.screen_name}`,
  560. "icon_url": `attachment://profile_images.${twitter_user_data.profile_image.split('.').pop()}`
  561. };
  562. tmpEmbed.description = replace_t_co_to_original_url(twitter_tweet_data.full_text,twitter_tweet_data.urls,twitter_tweet_data.media);
  563. tmpEmbed.thumbnail = {"url": "https://pbs.twimg.com/profile_images/1488548719062654976/u6qfBBkF_400x400.jpg"};
  564. tmpEmbed.fields = [
  565. {
  566. "name": `${textData.various_links}:link:`,
  567. "value": `[${textData.link_to_tweet}](${tweet_url})` + `\n[TwitterID: ${twitter_user_data.ID}](https://twitter.com/intent/user?user_id=${twitter_user_data.ID})` + if_exsit_return_text(twitter_tweet_data.media[0]?.url,`\n[${textData.link_to_image}](${image_url_to_original(twitter_tweet_data.media[0]?.url)})`) + if_exsit_return_text(twitter_user_data.pixiv_url,`\n[Pixiv](${twitter_user_data.pixiv_url})`)
  568. },{
  569. "name": textData.engagement,
  570. "value": `${textData.retweets} ${round_half_up(twitter_tweet_data.retweet_count,textData.roundingScale,textData.decimalPlaces,textData.units)}:recycle: ${textData.likes} ${round_half_up(twitter_tweet_data.favorite_count,textData.roundingScale,textData.decimalPlaces,textData.units)}:heart:`
  571. },{
  572. "name": textData.postedDate,
  573. "value": twitter_tweet_data.created_at
  574. }
  575. ];
  576. if(twitter_tweet_data.media.images[0]?.url){
  577. tmpEmbed.image = {
  578. "url": `attachment://${twitter_tweet_data.media.images[0].url.split('/').pop()}`
  579. }
  580. }
  581. embeds.push(tmpEmbed);
  582. if(twitter_tweet_data.media.images[1]?.url){
  583. for(let i=1;i<twitter_tweet_data.images.length;i++){
  584. embeds[i] = {
  585. "url": tweet_url,
  586. "image": {"url": `attachment://${twitter_tweet_data.media.images[i].url.split('/').pop()}`}
  587. }
  588. }
  589. }
  590. tmp_return_object.push({"embeds": embeds, "files": await fetchImages(twitter_tweet_data.media.images.concat([{"url": twitter_user_data.profile_image}]))});
  591. if(twitter_tweet_data.media.videos?.length >= 1 && thisScriptSettings.downloadVideo){
  592. const promises = twitter_tweet_data.media.videos.map(video => downloadVideo(video.url));
  593. await Promise.all(promises)
  594. .then(results => {
  595. results.forEach(obj => {
  596. tmp_return_object.push(obj);
  597. });
  598. })
  599. .catch(error => {
  600. console.error("Error downloading videos:", error);
  601. });
  602. }else if(twitter_tweet_data.media.videos?.length >= 1){
  603. twitter_tweet_data.media.videos.forEach(video => {
  604. tmp_return_object.push({"content": video.url});
  605. });
  606. }
  607. if(send_quoted_tweet && quoted_tweet_mode == 0 && (tweet_tweet_data_json.result?.quoted_status_result||tweet_tweet_data_json.quoted_status)){
  608. let tmp_quoted_data = tweet_tweet_data_json.result?.quoted_status_result||tweet_tweet_data_json.quoted_status;
  609. tmp_quoted_data.APIsource = tweet_data.APIsource;
  610. tmp_return_object.push({"quoted_tweet_data": tmp_quoted_data});
  611. }
  612. return tmp_return_object;
  613. }
  614. async function downloadVideo(url){
  615. return new Promise(async (resolve) => {
  616. if((await request(new requestObject_binary_head(url))).responseHeaders.match(/content-length: ?(\d+)/)[1] < 24117249){
  617. return resolve({"files": [{attachment: (await request(new requestObject_binary_data(url))).response, name: url.split('/').pop()}]});
  618. }else{
  619. return resolve({"content": url});
  620. }
  621. });
  622. }
  623. function extractUrls(entities){
  624. let urls = [];
  625. if(entities.description && entities.description.urls){
  626. entities.description.urls.forEach(urlObj => {
  627. urls.push(urlObj.expanded_url || urlObj.url);
  628. });
  629. }
  630. if(entities.url && entities.url.urls){
  631. entities.url.urls.forEach(urlObj => {
  632. urls.push(urlObj.expanded_url || urlObj.url);
  633. });
  634. }
  635. return urls;
  636. }
  637. function replace_t_co_to_original_url(full_text,urls,media_urls){
  638. //ツイート内のt.coで短縮されたリンクをもとにのリンクにもどす。
  639. try{
  640. if(typeof full_text !== "undefined"){
  641. full_text = decodeHtml(full_text);
  642. if(typeof urls !== "undefined"){
  643. for(let i=0;i<=urls.length-1;i++){
  644. if(urls[i].expanded_url.length > 200){
  645. full_text = full_text.replace(urls[i].url,`[${decodeURI(urls[i].expanded_url).slice(0,150)}](${decodeURI(urls[i].expanded_url)}) ... `);
  646. }else{
  647. full_text = full_text.replace(urls[i].url,decodeURI(urls[i].expanded_url));
  648. }
  649. }
  650. }
  651. //メディアがくっついてるツイートは末尾にメディアのURLが付随しているためそれを消す。
  652. if(typeof media_urls !== "undefined"){
  653. for(let i=0;i<=media_urls.length-1;i++){
  654. full_text = full_text.replace(media_urls[i].tco_url,"");
  655. }
  656. }
  657. }
  658. }catch{}
  659. return full_text;
  660. }
  661. function if_exsit_return_text(variable,return_text){
  662. if(!(variable === null || variable === undefined || variable === "")){
  663. return return_text
  664. }
  665. return "";
  666. }
  667. function make_media_list(extended_entities, select_pages){
  668. var result = {images: [],videos: []};
  669. if(extended_entities){
  670. for(const target in select_pages){
  671. try{
  672. if(extended_entities.media.length > select_pages[target]){
  673. let mediaItem = extended_entities.media[select_pages[target]];
  674. let tmp_object = {media_type: mediaItem.type, tco_url: mediaItem.url};
  675. if(tmp_object.media_type == "animated_gif" || tmp_object.media_type == "video"){
  676. tmp_object.url = mediaItem.video_info.variants.filter(function(obj){return obj.content_type == "video/mp4";}).reduce((a, b) => a.bitrate > b.bitrate ? a : b).url.split('?')[0];
  677. result.videos.push(tmp_object);
  678. }else if(tmp_object.media_type == "photo"){
  679. tmp_object.url = mediaItem.media_url_https;
  680. result.images.push(tmp_object);
  681. }
  682. }
  683. }catch(error){
  684. console.error("メディアリストの作成に失敗しました。:\n" + error);
  685. }
  686. }
  687. }
  688. return result;
  689. }
  690. function str_max_length(text, max_length,defaultValue = "……以下Discordの字数オーバー。"){
  691. var r = 0;
  692. for(var i = 0; i < text.length; i++){
  693. var c = text.charCodeAt(i);
  694. if((c >= 0x0 && c < 0x81) || (c == 0xf8f0) || (c >= 0xff61 && c < 0xffa0) || (c >= 0xf8f1 && c < 0xf8f4)){
  695. r += 1;
  696. }else{
  697. r += 2;
  698. }
  699. if(r >= max_length){
  700. text = `${text.slice(0, i - 1)} ${defaultValue}`;
  701. break;
  702. }
  703. }
  704. return text;
  705. }
  706. function image_url_to_original(image_url){
  707. //apiから帰ってくるURLをそのまま開くと小さい画像になってしまうので最大サイズの画像をダウンロードできるようにする。
  708. if(typeof image_url !== "undefined"){
  709. var extension = image_url.split(".").pop();
  710. return `${image_url.replace(`.${extension}`,"")}?format=${extension}&name=orig`
  711. }
  712. }
  713. async function fetchImages(mediaUrlArray){
  714. if(mediaUrlArray?.length == 0) return;
  715. let downloadPromises = mediaUrlArray.map(fetchImage);
  716. return remove_null_from_array(await Promise.all(downloadPromises));
  717. async function fetchImage(target) {
  718. let retryCount = 5; // リトライ回数を設定
  719. while(retryCount > 0){
  720. if(!target.url) return;
  721. let image;
  722. let name;
  723. if(target.url.match(/https?:\/\/pbs\.twimg\.com\/media\//)){
  724. image = await request(new requestObject_binary_data(image_url_to_original(target.url)),3);
  725. name = target.url.split('/').pop();
  726. }else{
  727. image = await request(new requestObject_binary_data(target.url),3);
  728. name = "profile_images." + target.url.split('.').pop();
  729. }
  730. // ダウンロードした画像データのサイズを確認
  731. if(image.response.size > 1024){
  732. return {
  733. "attachment": image.response,
  734. "name": name,
  735. };
  736. }else{
  737. retryCount--;
  738. }
  739. }
  740. console.warn(`Failed to download image after multiple retries: ${target.url}`);
  741. return null;
  742. }
  743. }
  744. }
  745. }
  746. async function sneakilyFavorite(tweetNodes){
  747. tweetNodes.forEach(function(element){
  748. const node = element.node;
  749. if(node.querySelector(".sneakilyFavorite") || ! node.querySelector(env_selector.retweeted) || !node.querySelector('[data-testid="like"]')) return;
  750. let tweet_link = element.link;
  751. let button = document.createElement('button');
  752. let fotter = node.querySelector('div[id][role="group"]');
  753. let like_element = fotter.querySelector('[data-testid="like"]');
  754. button.textContent = env_Text.sneakilyFavorite.favorite;
  755. button.style.fontSize = '0.7em';
  756. button.style.whiteSpace = 'nowrap';
  757. button.classList.add('sneakilyFavorite');
  758. button.addEventListener('click',async function(event){
  759. this.disabled = true;
  760. const status = await request(new requestObject_twitter_FavoriteTweet(tweet_link));
  761. if(status.response.data.favorite_tweet == "Done"){
  762. like_element.querySelector('div[dir="ltr"]').classList.add(env_selector.liked_color);
  763. like_element.querySelector('div[dir="ltr"]').style.color = "rgb(249, 24, 128)";
  764. like_element.querySelector("path").setAttribute('d',env_selector.liked);
  765. like_element.setAttribute('data-testid', 'unlike');
  766. }
  767. this.remove();
  768. });
  769. fotter.insertBefore(button, like_element.parentElement.nextSibling);
  770. });
  771. }
  772. async function Engagement_Restorer(){
  773. if(!currentUrl.match(/https?\:\/\/twitter\.com\/\w*\/status\/[0-9]*($|\?.*)/) || document.getElementById('restoreEngagements') || isFunctionRunning.Engagement_Restorer)return;
  774. isFunctionRunning.Engagement_Restorer = true;
  775. try{
  776. const tweetLink = currentUrl.match(/https?\:\/\/twitter\.com\/\w*\/status\/[0-9].*/)[0];
  777. const tweetId = tweetLink.match(/\/status\/(\d+)/)[1];
  778. const response = (await getTweetData(tweetId,"graphQL")).legacy;
  779. const engagemants = {"favorite_count": response.favorite_count,"quote_count": response.quote_count,"retweet_count": response.retweet_count};
  780. const target_node = Array.from((await wait_load_Element('article[data-testid="tweet"]'))).find(node => {
  781. const timeParents = Array.from(node.querySelectorAll('time')).map(time => time.parentNode);
  782. return timeParents.some(parent => parent.href && parent.href.match(tweetId));
  783. });
  784. const engagemants_aria = target_node.querySelector('[role="group"]');
  785. if(!engagemants_aria)return;
  786. let envEngagementsTextColor = env_selector.engagementsTextColor[GetCookie("night_mode") || "0"];
  787. const flexContainer = document.createElement('div');
  788. flexContainer.style.display = 'flex';
  789. flexContainer.style.justifyContent = 'space-between';
  790. flexContainer.style.width = '70%';
  791. flexContainer.id = 'restoreEngagements';
  792. const links = [
  793. {
  794. "name": "retweets",
  795. "href": tweetLink + "/retweets",
  796. "count": round_half_up(engagemants.retweet_count,env_Text.Engagement_Restorer.roundingScale,env_Text.Engagement_Restorer.decimalPlaces,env_Text.Engagement_Restorer.units),
  797. "text": env_Text.Engagement_Restorer.retweet
  798. },
  799. {
  800. "name": "quotes",
  801. "href": tweetLink + "/quotes",
  802. "count": round_half_up(engagemants.quote_count,env_Text.Engagement_Restorer.roundingScale,env_Text.Engagement_Restorer.decimalPlaces,env_Text.Engagement_Restorer.units),
  803. "text": env_Text.Engagement_Restorer.quoted,
  804. },
  805. {
  806. "name": "likes",
  807. "href": tweetLink + "/likes",
  808. "count": round_half_up(engagemants.favorite_count,env_Text.Engagement_Restorer.roundingScale,env_Text.Engagement_Restorer.decimalPlaces,env_Text.Engagement_Restorer.units),
  809. "text": env_Text.Engagement_Restorer.like,
  810. },
  811. ];
  812. links.forEach((a, index) => {
  813. const newLink = document.createElement('a');
  814. newLink.style.textDecoration = 'none';
  815. newLink.href = a.href;
  816. const countText = document.createElement('span');
  817. countText.textContent = a.count;
  818. countText.style.color = envEngagementsTextColor.count;
  819. newLink.appendChild(countText);
  820.  
  821. const textPart = document.createElement('span');
  822. textPart.textContent = " " + a.text;
  823. textPart.style.color = envEngagementsTextColor.text;
  824. newLink.appendChild(textPart);
  825.  
  826. newLink.addEventListener('click', (e) => {
  827. e.preventDefault();
  828. clickTab(a.name,target_node);
  829. });
  830. flexContainer.appendChild(newLink);
  831. });
  832. if(currentUrl.match(/https?\:\/\/twitter\.com\/\w*\/status\/[0-9]*($|\?.*)/))engagemants_aria.parentNode.prepend(flexContainer);
  833. }catch(error){
  834. console.error(error)
  835. }finally{
  836. isFunctionRunning.Engagement_Restorer = false;
  837. }
  838. async function clickTab(name,target_node){
  839. target_node.querySelector('[data-testid="caret"]').click();
  840. document.querySelector('[data-testid="tweetEngagements"]').click();
  841. const engagemants_aria = (await wait_load_Element('nav[aria-live="polite"]'))[0];
  842. const regex = new RegExp(name + '$');
  843. engagemants_aria.querySelectorAll('[role="presentation"] a').forEach((e)=>{
  844. if(e.href.match(regex)) e.click();
  845. });
  846. }
  847. }
  848. async function Hello_tweet_where_are_you_from(){
  849. if(document.querySelector('.display_twitter_client') || isFunctionRunning.Hello_tweet_where_are_you_from)return;
  850. isFunctionRunning.Hello_tweet_where_are_you_from = true;
  851. try{
  852. const tweet_data = await getTweetData(extractTweetId(currentUrl),"graphQL");
  853. const target_node = (await wait_load_Element(env_selector.info_field))[0];
  854. const info_selector = target_node.childNodes[0].className;
  855. let client_text = document.createElement("div");
  856. client_text.dir = "ltr";
  857. client_text.innerHTML = `<span class="display_twitter_client ${info_selector}" style="">${tweet_data.source.replace(/<("[^"]*"|'[^']*'|[^'">])*>/g,'')}</span>`;
  858. target_node.appendChild(client_text);
  859. const media_data = tweet_data.legacy?.extended_entities?.media || tweet_data.extended_entities?.media;
  860. if(!media_data)return;
  861. const video_data = media_data.map((mediaItem, index) => {
  862. if (['video', 'animated_gif'].includes(mediaItem.type)){
  863. const highestBitrateVariant = mediaItem.video_info.variants.filter(obj => obj.content_type !== 'application/x-mpegURL').reduce((a, b) => a.bitrate > b.bitrate ? a : b);
  864. return {
  865. index: index,
  866. url: highestBitrateVariant.url.split('?')[0]
  867. };
  868. }
  869. }).filter(Boolean)
  870. if(video_data.length === 0)return;
  871. let show_video_url = document.createElement("span");
  872. show_video_url.className = "display_video_url";
  873. show_video_url.style.whiteSpace = "pre";
  874. for(let video of video_data){
  875. let video_url = document.createElement("a");
  876. video_url.href = video.url;
  877. video_url.textContent = `${video.index + 1}:video_url`;
  878. video_url.className = "css-4rbku5 css-18t94o4 css-901oao css-16my406 r-1cvl2hr r-1loqt21 r-poiln3 r-bcqeeo r-qvutc0";
  879. video_url.style.color = "rgb(29, 155, 240)";
  880. video_url.target = "_blank";
  881. video_url.rel = "noopener";
  882. show_video_url.appendChild(video_url);
  883. if(video !== video_data[video_data.length - 1]){
  884. show_video_url.appendChild(document.createTextNode(' '));
  885. }
  886. }
  887. client_text.appendChild(document.createElement("br"));
  888. client_text.appendChild(show_video_url);
  889. }catch(error){
  890. console.error(error);
  891. }finally{
  892. isFunctionRunning.Hello_tweet_where_are_you_from = false;
  893. }
  894. }
  895.  
  896. async function Note_Tweet_expander(tweetNodes){
  897. const link_class = "r-18u37iz css-4rbku5 css-18t94o4 css-901oao css-16my406 r-1cvl2hr r-1loqt21 r-poiln3 r-bcqeeo r-qvutc0";
  898. tweetNodes.forEach(async function(target){
  899. const tweet_node = target.node;
  900. const elements = tweet_node.querySelectorAll('[data-testid="tweetText"]');
  901. const show_more_link = tweet_node.querySelectorAll('[data-testid="tweet-text-show-more-link"]');
  902. if(show_more_link[0])show_more_link[0].style.display = "none";
  903. if(show_more_link[1])show_more_link[1].style.display = "none";
  904. elements.forEach(async (element,index) =>{
  905. if(!element.innerText.match(/…$/)){
  906. element.classList.add('tweetExpanderChecked');
  907. //if(element.innerText.split('\n').length >= 10)
  908. element.style.webkitLineClamp = null;
  909. return;
  910. }
  911. let tweet_id,tweet_data,note_tweet;
  912. if(index == 0){
  913. tweet_id = target.id;
  914. tweet_data = await getTweetData(tweet_id,"graphQL");
  915. note_tweet = tweet_data.note_tweet?.note_tweet_results.result || tweet_data.note_tweet?.note_tweet_results.result || null;
  916. }else{
  917. tweet_data = await getTweetData(target.id,"graphQL");
  918. tweet_data = await getTweetData(tweet_data.legacy.quoted_status_id_str,"graphQL");
  919. note_tweet = tweet_data.note_tweet?.note_tweet_results.result || tweet_data.note_tweet?.note_tweet_results.result || null;
  920. }
  921. if(!note_tweet){
  922. element.style.webkitLineClamp = null;
  923. element.classList.add('tweetExpanderChecked');
  924. return;
  925. }
  926. const hashtags = note_tweet.entity_set?.hashtags || [];
  927. const user_mentions = note_tweet.entity_set?.user_mentions || [];
  928. const symbols = note_tweet.entity_set?.symbols || [];
  929. const urls = note_tweet.entity_set?.urls;
  930. var new_tweet_text = note_tweet.text;
  931. function countSurrogatePairs(str){
  932. return Array.from(str).filter(char => char.match(/[\uD800-\uDBFF][\uDC00-\uDFFF]/)).length;
  933. }
  934. let combined = [].concat(
  935. hashtags.map(tag => ({
  936. type: 'hashtag',
  937. indices: tag.indices,
  938. text: tag.text
  939. })),
  940. user_mentions.map(mention => ({
  941. type: 'mention',
  942. indices: mention.indices,
  943. text: mention.screen_name
  944. })),
  945. symbols.map(symbol => ({
  946. type: 'symbol',
  947. indices: symbol.indices,
  948. text: symbol.text
  949. }))
  950. );
  951. // combinedをindicesの順にソート
  952. combined.sort((a, b) => b.indices[0] - a.indices[0]);
  953. let transformedText = new_tweet_text;
  954.  
  955. combined.forEach(item => {
  956. let start = item.indices[0];
  957. let end = item.indices[1];
  958.  
  959. // サロゲートペアの数をカウントして調整
  960. const adjustment = countSurrogatePairs(transformedText.slice(0, end));
  961. start += adjustment;
  962. end += adjustment;
  963.  
  964. let replacement = '';
  965. switch(item.type){
  966. case 'hashtag':
  967. replacement = `<a class="${link_class}" dir="ltr" role="link" href="https://twitter.com/hashtag/${item.text}" target="_blank" rel="noopener">#${item.text}</a>`;
  968. break;
  969. case 'mention':
  970. replacement = `<a class="${link_class}" dir="ltr" role="link" href="https://twitter.com/${item.text}" target="_blank" rel="noopener">@${item.text}</a>`;
  971. break;
  972. case 'symbol':
  973. replacement = `<a class="${link_class}" dir="ltr" role="link" href="https://twitter.com/search?q=%24${item.text}&src=cashtag_click" target="_blank" rel="noopener">$${item.text}</a>`;
  974. break;
  975. }
  976. transformedText = transformedText.slice(0, start) + replacement + transformedText.slice(end);
  977. });
  978. new_tweet_text = transformedText;
  979. urls.forEach(target =>{
  980. new_tweet_text = new_tweet_text.replace(new RegExp(`${target.url}(?=(\\s|$|\\u3000|\\W)(?!\\.|,))`, 'gu'), `<a class="${link_class}" dir="ltr" role="link" href="${target.url}" target="_blank" rel="noopener">${target.display_url}</a>`);
  981. });
  982. var new_tweet_node = document.createElement("span");
  983. new_tweet_node.className = 'css-901oao css-16my406 r-1tl8opc r-bcqeeo r-qvutc0';
  984. new_tweet_node.innerHTML = new_tweet_text;
  985. while(element.firstChild){
  986. element.removeChild(element.firstChild);
  987. }
  988. element.appendChild(new_tweet_node);
  989. element.style.webkitLineClamp = null;
  990. });
  991. });
  992. }
  993.  
  994. async function Show_me_your_Pixiv(tweetNodes){
  995. let todo = [];
  996. tweetNodes.forEach(item => {
  997. const node = item.node.querySelector(`${env_selector.media_field}:not(.display_pixiv_link)`);
  998. if(node){
  999. todo.push({
  1000. "node": node,
  1001. "name": item.node.querySelector('[data-testid="User-Name"]>div>div>a').href.split("/").pop()
  1002. });
  1003. }
  1004. });
  1005. let currentScreenName = extractUserName(currentUrl);
  1006. const uniqueScreenNames = [...new Set(todo.map(item => item.name))];
  1007. if(currentScreenName){
  1008. uniqueScreenNames.push(currentScreenName);
  1009. }
  1010. let promises = uniqueScreenNames.map(async screen_name => {
  1011. if(!((scriptDataStore.Show_me_your_Pixiv[screen_name]?.pixiv_url) === undefined))return "Already exists";
  1012. if(((scriptDataStore.Show_me_your_Pixiv[screen_name]?.Create_date || 0) + 604800000) <= new Date().getTime()){
  1013. let userEntitiesData = await getTweetData(screen_name,"user_1_1");
  1014. userEntitiesData = userEntitiesData.legacy?.entities || userEntitiesData.entities;
  1015. const end_stat = await find_pixiv_link(extractUrls(userEntitiesData));
  1016. if(end_stat == "Too Many Requests"){
  1017. console.log("API limit.");
  1018. }else if(!end_stat){
  1019. scriptDataStore.Show_me_your_Pixiv[screen_name] = {"pixiv_url": null,"Create_date": new Date().getTime()};
  1020. return `${screen_name}: Pixivリンクなし`;
  1021. }else{
  1022. scriptDataStore.Show_me_your_Pixiv[screen_name] = {"pixiv_url": end_stat.replace(/^https?/,'https'),"Create_date": new Date().getTime()};
  1023. return `${screen_name}: ${end_stat}`;
  1024. }
  1025. }else{
  1026. return "Not yet time";
  1027. }
  1028. });
  1029. const results = await Promise.allSettled(promises);
  1030. /*
  1031. results.forEach(result => {
  1032. if(result.status === 'fulfilled'){debug_log(`${result.value}`);
  1033. }else{debug_log(`Failure: ${result.reason}`);}
  1034. });
  1035. */
  1036. todo.forEach(item => {
  1037. const node = item.node;
  1038. const screen_name = item.name;
  1039. const pixiv_url = scriptDataStore.Show_me_your_Pixiv[screen_name]?.pixiv_url;
  1040. if(pixiv_url && node && !(node?.querySelector(".display_pixiv_link")))node.appendChild(makeLinkElement(pixiv_url,"Pixiv🔗","display_pixiv_link"));
  1041. });
  1042. if(scriptDataStore.Show_me_your_Pixiv[currentScreenName]?.pixiv_url){
  1043. const profile_field = (await wait_load_Element(env_selector.profile_field_Header_Items))[0];
  1044. currentScreenName = extractUserName(currentUrl);
  1045. if(!profile_field.querySelector(".display_pixiv_link") && scriptDataStore.Show_me_your_Pixiv[currentScreenName]?.pixiv_url){
  1046. let brElement = document.createElement("br");
  1047. profile_field.appendChild(brElement);
  1048. profile_field.appendChild(makeLinkElement(scriptDataStore.Show_me_your_Pixiv[currentScreenName]?.pixiv_url, "Pixiv🔗", "display_pixiv_link"));
  1049. }
  1050. }
  1051. localStorage.setItem('user_pixvi_link_collection', JSON.stringify(scriptDataStore.Show_me_your_Pixiv));
  1052. function makeLinkElement(href,text,additionalClass){
  1053. let new_content = document.createElement("a");
  1054. new_content.style.color = "rgb(29, 155, 240)";
  1055. new_content.href = href;
  1056. new_content.textContent = text;
  1057. new_content.className = `${additionalClass} css-4rbku5 css-18t94o4 css-901oao css-16my406 r-1cvl2hr r-1loqt21 r-poiln3 r-bcqeeo r-qvutc0`;
  1058. new_content.target = "_blank";
  1059. new_content.rel = "noopener";
  1060. return new_content;
  1061. }
  1062. function extractUrls(entities){
  1063. let urls = [];
  1064. if (entities.description && entities.description.urls){
  1065. entities.description.urls.forEach(urlObj => {
  1066. urls.push(urlObj.expanded_url || urlObj.url);
  1067. });
  1068. }
  1069. if (entities.url && entities.url.urls){
  1070. entities.url.urls.forEach(urlObj => {
  1071. urls.push(urlObj.expanded_url || urlObj.url);
  1072. });
  1073. }
  1074. return urls;
  1075. }
  1076. }
  1077.  
  1078.  
  1079. /////////////////////////////////////////////////////////////////////
  1080. ///////////////////////////ここから汎用関数///////////////////////////
  1081. /////////////////////////////////////////////////////////////////////
  1082.  
  1083. function update(){
  1084. if(updating) return;
  1085. updating = true;
  1086. console.log({user: fetchedTweetsUserDataByUserName,tweets: fetchedTweets})
  1087. main();
  1088. setTimeout(() => {updating = false;}, 600);
  1089. }
  1090. function extractTweetId(url){
  1091. const match = url.match(/twitter\.com\/[^/]+\/status\/(\d+)/);
  1092. return match ? match[1] : null;
  1093. }
  1094. function extractUserName(url){
  1095. const match = url.match(denyNamesRegex);
  1096. return match ? match[1] : null;
  1097. }
  1098. function findParent(element, selector){
  1099. let current = element;
  1100. while(current !== null){
  1101. if(current.matches(selector)){
  1102. return current;
  1103. }
  1104. current = current.parentNode;
  1105. }
  1106. return null;
  1107. }
  1108. function locationChange(){
  1109. const observer = new MutationObserver(mutations => {
  1110. if(currentUrl !== document.location.href){
  1111. currentUrl = document.location.href;
  1112. try{
  1113. if(extractTweetId(currentUrl))getTweetData(extractTweetId(currentUrl),"graphQL");
  1114. if(extractUserName(currentUrl))getTweetData(extractUserName(currentUrl),"user");
  1115. }catch(error){console.error(error)}
  1116. main();
  1117. }
  1118. });
  1119. const target = document.getElementById("react-root");
  1120. const config = {childList: true,subtree: true};
  1121. observer.observe(target, config);
  1122. }
  1123. function wait_load_Element(Element_Name,interval = 100,retry = 25){
  1124. return new Promise((resolve, reject) => {
  1125. const MAX_RETRY_COUNT = retry;
  1126. let retry_counter = 0;
  1127. let set_interval_id = setInterval(find_target_element, interval);
  1128. function find_target_element(){
  1129. retry_counter++;
  1130. if(retry_counter > MAX_RETRY_COUNT){
  1131. clearInterval(set_interval_id);
  1132. return reject("Max retry count reached");
  1133. }
  1134. let target_elements = document.querySelectorAll(`${Element_Name}`);
  1135. if(target_elements.length > 0){
  1136. clearInterval(set_interval_id);
  1137. return resolve(target_elements);
  1138. }
  1139. }
  1140. });
  1141. }
  1142. function GetCookie(name){
  1143. let arr, reg = new RegExp("(^| )" + name + "=([^;]*)(;|$)");
  1144. if(arr = document.cookie.match(reg)){
  1145. return decodeURIComponent(arr[2]);
  1146. }else{
  1147. return null;
  1148. }
  1149. }
  1150. function remove_null_from_array(arr){
  1151. return arr.filter(function(x){return !(x === null || x === undefined || x === "")});
  1152. }
  1153. function get_only_particular_key_value(object, path, defaultValue = undefined){
  1154. const isArray = Array.isArray;
  1155. if(object == null || typeof object != 'object') return defaultValue;
  1156. return (isArray(object)) ? object.map(createProcessFunction(path)) : createProcessFunction(path)(object);
  1157. function createProcessFunction(path){
  1158. if(typeof path == 'string') path = path.split('.');
  1159. if(!isArray(path)) path = [path];
  1160. return function(object){
  1161. let index = 0,
  1162. length = path.length;
  1163. while(index < length){
  1164. const key = toString_(path[index++]);
  1165. if(object === undefined){
  1166. return defaultValue;
  1167. }
  1168. // 配列に対する処理
  1169. if(isArray(object)){
  1170. object = object.map(item => item[key]);
  1171. }else{
  1172. object = object[key];
  1173. }
  1174. }
  1175. return (index && index == length) ? object : void 0;
  1176. };
  1177. }
  1178. function toString_(value){
  1179. if(value == null) return '';
  1180. if(typeof value == 'string') return value;
  1181. if(isArray(value)) return value.map(toString) + '';
  1182. let result = value + '';
  1183. return '0' == result && 1 / value == -(1 / 0) ? '-0' : result;
  1184. }
  1185. }
  1186. async function sleep(time){
  1187. return new Promise((resolve)=>{
  1188. setTimeout(()=>{return resolve(time)}, time);
  1189. });
  1190. }
  1191. async function expand_shortening_link(urls){
  1192. let isInputArray = true;
  1193. if(typeof urls === 'string'){
  1194. urls = [urls];
  1195. isInputArray = false;
  1196. }
  1197. function isValidURL(url){
  1198. return url.match(/https?:\/\/[\w!\?/\+\-_~=;\.,\*&@#\$%\(\)'\[\]]+/);
  1199. }
  1200. async function expandURL(url){
  1201. if(!isValidURL(url)){
  1202. throw new Error(`Invalid URL: ${url}`);
  1203. }
  1204. const response = await request(new requestObject_url_expand(url));
  1205. return{
  1206. original: url,
  1207. expanded: response.response
  1208. };
  1209. }
  1210. const results = await Promise.all(urls.map(url => expandURL(url)));
  1211. if(!isInputArray){
  1212. return results[0];
  1213. }
  1214. return results;
  1215. }
  1216. async function find_pixiv_link(urls){
  1217. const Pixiv_url_regex = /^https?:\/\/(((www|touch)\.)?pixiv\.(net\/([a-z]{2}\/)?((member(_illust)?\.php\?id\=|(users|u)\/)[0-9]*)|me\/.*))/;
  1218. const Fanbox_url_regex = /^https?:(\/\/www\.pixiv\.net\/fanbox\/creator\/[0-9]*|\/\/.*\.fanbox\.cc\/?)/;
  1219. return new Promise(async function(resolve){
  1220. let tmp_urls = urls;
  1221. let Pixiv_url;
  1222. if(tmp_urls.length > 0){
  1223. Pixiv_url = await find_pixiv_link_from_profile_urls(tmp_urls);
  1224. if(Pixiv_url === undefined || Pixiv_url === null || Pixiv_url === false){
  1225. tmp_urls = get_only_particular_key_value((await expand_shortening_link(tmp_urls)),"expanded",[]);
  1226. Pixiv_url = await find_pixiv_link_from_profile_urls(tmp_urls);
  1227. return resolve(Pixiv_url);
  1228. }else{
  1229. return resolve(Pixiv_url);
  1230. }
  1231. }
  1232. return resolve(null);
  1233. });
  1234. async function find_pixiv_link_from_profile_urls(urls_in_profile){
  1235. return new Promise(async function(resolve,reject){
  1236. let tmp_pixiv_url;
  1237. tmp_pixiv_url = findMatch_from_array(urls_in_profile,Pixiv_url_regex,true);
  1238. if (tmp_pixiv_url !== undefined) return resolve(tmp_pixiv_url);
  1239. if(findMatch_from_array(urls_in_profile,Fanbox_url_regex) !== undefined){
  1240. tmp_pixiv_url = await when_fanbox(findMatch_from_array(urls_in_profile,Fanbox_url_regex,true));
  1241. if(Pixiv_url_regex.test(tmp_pixiv_url)){
  1242. return resolve(tmp_pixiv_url);
  1243. }
  1244. }else{
  1245. let get_url_promise_list = [];
  1246. urls_in_profile.forEach(target=>{
  1247. switch(true){
  1248. case /^https?:\/\/((skeb\.jp\/\@.*)|(fantia\.jp\/(fanclubs\/[0-9])?.*)|(.*\.booth\.pm)|(.*linktr\.ee)|(.*profcard\.info)|(.*lit\.link)|(potofu\.me)|(.*\.carrd\.co)|(.*\.tumblr\.com$)|(twpf\.jp))\/?/.test(target):
  1249. get_url_promise_list.push(new Promise(
  1250. async function(resolve,reject){
  1251. try{
  1252. return resolve(await when_general(target));
  1253. }catch(error){
  1254. return reject(error);
  1255. }
  1256. }
  1257. ));
  1258. break;
  1259. case /^https?:\/\/.*\.creatorlink\.net(\/.*)?/.test(target):
  1260. get_url_promise_list.push(new Promise(
  1261. async function(resolve,reject){
  1262. try{
  1263. return resolve(await when_general(`${target.match(/^https?:\/\/.*\.creatorlink\.net/)[0]}\/Contact`));
  1264. }catch(error){
  1265. return reject(error);
  1266. }
  1267. }
  1268. ));
  1269. break;
  1270. case /^https?:\/\/sketch\.pixiv\.net\//.test(target):
  1271. get_url_promise_list.push(new Promise(
  1272. async function(resolve,reject){
  1273. try{
  1274. return resolve(await when_pixiv_sketch(target));
  1275. }catch(error){
  1276. return reject(error);
  1277. }
  1278. }
  1279. ));
  1280. break;
  1281. default:
  1282. break;
  1283. }
  1284. });
  1285. if(get_url_promise_list.length > 0){
  1286. await Promise.any(get_url_promise_list).then((value) => {tmp_pixiv_url = value}).catch(() => {tmp_pixiv_url = undefined});
  1287. if(!Pixiv_url_regex.test(tmp_pixiv_url)) return resolve(undefined);
  1288. return resolve(tmp_pixiv_url.replace(/^https?/,'https').replace(/(\/|\\)$/,''));
  1289. }
  1290. }
  1291. return resolve(null);
  1292. async function when_general(target_url){
  1293. return new Promise(async function(resolve,reject){
  1294. const response_data = await request(new requestObject(target_url.replace(/^https?/,"https")));
  1295. let response_data_urls = response_data.response.split(/\"|\<|\>/).filter(function(data_str){return data_str.match(/^https?:(\/\/(((www|touch)\.)?pixiv\.(net\/([a-z]{2}\/)?((member(_illust)?\.php\?id\=|(users|u|fanbox\/creator)\/)[0-9].*)|me\/.*))|.*\.fanbox\.cc\/?)$/)});
  1296. if(response_data_urls.find(function(element){return element.match(Pixiv_url_regex)}) !== undefined){
  1297. return resolve(response_data_urls.find(function(element){return element.match(Pixiv_url_regex)}));
  1298. }else if(response_data_urls.find(function(element){return element.match(Fanbox_url_regex)}) !== undefined){
  1299. return resolve(await when_fanbox(response_data_urls.find(function(element){return element.match(Fanbox_url_regex)})));
  1300. }else{
  1301. return reject(undefined);
  1302. }
  1303. });
  1304. }
  1305. async function when_pixiv_sketch(target_url){
  1306. return new Promise(async function(resolve,reject){
  1307. const response_data = await request(new requestObject(target_url));
  1308. let User_id = response_data.response.split(',').filter(function(data_str){return data_str.match(/\\"pixiv_user_id\\":\\"[0-9]*\\"/)});
  1309. if(User_id){
  1310. return resolve("https://www.pixiv.net/users/" + User_id[0].split(/\"|\\/)[6]);
  1311. }else{
  1312. return reject(undefined);
  1313. }
  1314. });
  1315. }
  1316. async function when_fanbox(fanbox_url){
  1317. return new Promise(async function(resolve,reject){
  1318. if(fanbox_url.match(/^https?:\/\/www\.pixiv\.net\/fanbox\/creator\/[0-9]*/)){
  1319. return resolve(fanbox_url.replace('fanbox/creator', 'users'));
  1320. }else{
  1321. const fanbox_response = await request(new requestObject_fanbox(`https://api.fanbox.cc/creator.get?creatorId=${fanbox_url.replace(/(https?:\/\/|\.fanbox.*)/g,'')}`,fanbox_url.replace(/^http:/, 'https:').replace(/\/$/, '')));
  1322. if(fanbox_response.status == "404") return reject(undefined);
  1323. tmp_pixiv_url = findMatch_from_array(fanbox_response.response.body.profileLinks,Pixiv_url_regex,true);
  1324. if(tmp_pixiv_url !== undefined){
  1325. return resolve(tmp_pixiv_url);
  1326. }else{
  1327. return resolve(`https://www.pixiv.net/users/${fanbox_response.response.body.user.userId}`);
  1328. }
  1329. }
  1330. })
  1331. }
  1332. }
  1333. )}
  1334. }
  1335. function round_half_up(original_value,where_round_off,decimal_place = 0,unit_str = ""){
  1336. //四捨五入関数。
  1337. /*
  1338. original_valu: 元の値
  1339. where_round_off: どこで四捨五入するか(0.1, 1, 10, 100, 1000など)
  1340. decimal_place: 小数点以下を何桁にするか(1, 2, 3, 4, 5など)
  1341. unit_str: 単位を末尾につける(千,万など)
  1342. */
  1343. if(Number(original_value)>=Number(where_round_off)){
  1344. let tmp_value;
  1345. tmp_value = Math.round(Number(original_value) / Number(where_round_off) * Math.pow(10,Number(decimal_place))) / Math.pow(10,Number(decimal_place));
  1346. if(unit_str == ""){
  1347. return tmp_value;
  1348. }else{
  1349. return `${tmp_value}${unit_str}`
  1350. }
  1351. }else{
  1352. return original_value;
  1353. }
  1354. }
  1355. function findMatch_from_array(arr, regex, is_strict = false){
  1356. for(let i = 0; i < arr.length; i++){
  1357. if(regex.test(arr[i])){
  1358. if(is_strict === true){
  1359. return arr[i].match(regex)[0];
  1360. }else{
  1361. return arr[i];
  1362. }
  1363. }
  1364. }
  1365. return undefined;
  1366. }
  1367. async function fetchAndProcessTwitterApi(method,id = undefined,forceFetch = false){
  1368. return new Promise(async (resolve, reject) => {
  1369. try{
  1370. switch(method){
  1371. case 'TL':
  1372. case 'forYou':
  1373. await getTL();
  1374. break;
  1375. case 'graphQL':
  1376. if(fetchedTweets[id] && !(forceFetch === true))return resolve(fetchedTweets[id]);
  1377. await graphQL();
  1378. break;
  1379. case 'user':
  1380. await getUser();
  1381. break;
  1382. case 'user_1_1':
  1383. await getUser1_1();
  1384. break;
  1385. case '1_1':
  1386. await get1_1();
  1387. break;
  1388. default:
  1389. console.warn("なにか間違ってないか?")
  1390. return reject("something wrong.");
  1391. }
  1392. return resolve("OK!");
  1393. }catch(error){
  1394. console.error(error);
  1395. throw new Error(`Failed to fetch API data.\nmethod: ${method}\nid: ${id}`);
  1396. }
  1397. });
  1398. async function getTL(){
  1399. let requestObject;
  1400. if(method == 'TL'){
  1401. requestObject = new requestObject_twitter_time_line();
  1402. }else{
  1403. requestObject = new requestObject_twitter_time_line_forYou();
  1404. }
  1405. const response = await request(requestObject);
  1406. if(!response.status === "200")throw new Error(`Failed to fetch`);
  1407. processgraphQL(response.response.data.home.home_timeline_urt.instructions[0].entries);
  1408. }
  1409. async function graphQL(){
  1410. const response = await request(new requestObject_twitter_api_graphql(id));
  1411. if(!response.status === "200")throw new Error(`Failed to fetch`);
  1412. processgraphQL(response.response.data.threaded_conversation_with_injections_v2.instructions[0].entries);
  1413. }
  1414. async function getUser(){
  1415. if(fetchedTweetsUserDataByUserName[id]?.API_type === "graphQL")return;
  1416. const response = await request(new requestObject_twitter_get_user_by_screenname(id));
  1417. if(!response.status === "200")throw new Error(`Failed to fetch`);
  1418. const userData = response.response.data.user.result;
  1419. fetchedTweetsUserData[userData.rest_id] = {...userData,"API_type": "graphQL"};
  1420. fetchedTweetsUserDataByUserName[userData.legacy.screen_name] = fetchedTweetsUserData[userData.rest_id];
  1421. }
  1422. async function getUser1_1(){
  1423. if(fetchedTweetsUserDataByUserName[id])return;
  1424. const response = await request(new requestObject_twitter_get_user_by_screenname_1_1(id));
  1425. if(!response.status === "200")throw new Error(`Failed to fetch`);
  1426. if(response.response.status && !(fetchedTweets[response.response.status?.id_str]?.API_type === "graphQL")){
  1427. fetchedTweets[response.response.status.id_str] = {...response.response.status,"API_type": "1_1"};
  1428. response.response.status = fetchedTweets[response.response.status.id_str];
  1429. }
  1430. fetchedTweetsUserData[response.response.id_str] = {...response.response,"API_type": "1_1"};
  1431. fetchedTweetsUserDataByUserName[response.response.screen_name] = fetchedTweetsUserData[response.response.id_str];
  1432. }
  1433. async function get1_1(){
  1434. const idsToFetch = forceFetch ? id : id.filter(singleId => !(fetchedTweets[singleId]?.API_type === "graphQL"));
  1435. if(idsToFetch.length === 0) return;
  1436. const response = await request(new requestObject_twitter_api_v1_1(idsToFetch.join(",")));
  1437. if(!response.status === "200")throw new Error(`Failed to fetch`);
  1438. response.response.forEach((tweetData)=>{
  1439. if(tweetData.quoted_status){
  1440. let quoted = tweetData.quoted_status;
  1441. if(!(fetchedTweetsUserData[quoted.user.id_str]?.API_type === "graphQL")){
  1442. fetchedTweetsUserData[quoted.user.id_str] = {...quoted.user,"API_type": "1_1"};
  1443. fetchedTweetsUserDataByUserName[quoted.user.screen_name] = fetchedTweetsUserData[quoted.user.id_str];
  1444. }
  1445. quoted.user = fetchedTweetsUserData[quoted.user.id_str];
  1446. if(!(fetchedTweets[quoted.id_str]?.API_type === "graphQL")){
  1447. fetchedTweets[quoted.id_str] = {...quoted,"API_type": "1_1"};
  1448. }
  1449. tweetData.quoted_status = fetchedTweets[quoted.id_str];
  1450. }
  1451. if(!(fetchedTweetsUserData[tweetData.user.id_str]?.API_type === "graphQL")){
  1452. fetchedTweetsUserData[tweetData.user.id_str] = {...tweetData.user,"API_type": "1_1"};
  1453. fetchedTweetsUserDataByUserName[tweetData.user.screen_name] = fetchedTweetsUserData[tweetData.user.id_str];
  1454. }
  1455. tweetData.user = fetchedTweetsUserData[tweetData.user.id_str];
  1456. fetchedTweets[tweetData.id_str] = {...tweetData,"API_type": "1_1"};
  1457. });
  1458. }
  1459. function processgraphQL(entries){
  1460. if(!entries)return null;
  1461. entries.forEach(entry=>{
  1462. let tweetData = entry.content?.itemContent?.tweet_results?.result?.tweet || entry.content?.itemContent?.tweet_results?.result;
  1463. if(!tweetData)return;
  1464. if(tweetData.quoted_status_result){
  1465. let quoted = tweetData.quoted_status_result.result;
  1466. fetchedTweetsUserData[quoted.core.user_results.result.rest_id] = {...quoted.core.user_results.result,"API_type": "graphQL"};
  1467. fetchedTweetsUserDataByUserName[quoted.core.user_results.result.legacy.screen_name] = fetchedTweetsUserData[quoted.core.user_results.result.rest_id];
  1468. quoted.core.user_results.result = fetchedTweetsUserData[quoted.core.user_results.result.rest_id];
  1469. fetchedTweets[quoted.rest_id] = {...quoted,"API_type": "graphQL"};
  1470. tweetData.quoted_status_result.result = fetchedTweets[quoted.rest_id];
  1471. }
  1472. fetchedTweetsUserData[tweetData.core.user_results.result.rest_id] = {...tweetData.core.user_results.result,"API_type": "graphQL"};
  1473. fetchedTweetsUserDataByUserName[tweetData.core.user_results.result.legacy.screen_name] = fetchedTweetsUserData[tweetData.core.user_results.result.rest_id];
  1474. tweetData.core.user_results.result = fetchedTweetsUserData[tweetData.core.user_results.result.rest_id];
  1475. fetchedTweets[tweetData.rest_id] = {...tweetData,"API_type": "graphQL"};
  1476. });
  1477. }
  1478. }
  1479. async function getTweetData(id, method = '1_1', forceFetch = false){
  1480. const dataStore = method === 'user' || method === 'user_1_1' ? fetchedTweetsUserDataByUserName : fetchedTweets;
  1481. let ids;
  1482. if(typeof id === 'string'){
  1483. if(dataStore[id] && !forceFetch) return dataStore[id];
  1484. ids = [id];
  1485. }else if(Array.isArray(id)){
  1486. if(!forceFetch && id.every(singleId => dataStore[singleId])) return id.map(singleId => dataStore[singleId]);
  1487. ids = id;
  1488. }else{
  1489. throw new Error("Invalid ID type.");
  1490. }
  1491. if(method == "1_1") ids = [ids];
  1492. let promises = ids.map(singleId => fetchAndProcessTwitterApi(method, singleId, forceFetch));
  1493. await Promise.all(promises);
  1494. if(typeof id === 'string'){
  1495. if(dataStore[id]){
  1496. return dataStore[id];
  1497. }else{
  1498. throw new Error("Failed to fetch tweet data for ID: " + id);
  1499. }
  1500. }else if(Array.isArray(id)){
  1501. if(id.every(singleId => dataStore[singleId])){
  1502. return id.map(singleId => dataStore[singleId]);
  1503. }else{
  1504. throw new Error("Failed to fetch tweet data for some IDs.");
  1505. }
  1506. }
  1507. }
  1508. async function waitForTweetData(id, method = "1_1", retry = 30, span = 100){
  1509. let dataStore = method === 'user' || method === 'user_1_1' ? fetchedTweetsUserDataByUserName : fetchedTweets;
  1510. if(typeof id === 'string'){
  1511. if(dataStore[id]?.API_type === method)return dataStore[id];
  1512. id = [id];
  1513. }else if(Array.isArray(id)){
  1514. if(id.every(singleId => dataStore[singleId]?.API_type === method))return id.map(singleId => dataStore[singleId]);
  1515. }else{
  1516. throw new Error(`Invalid ID type: ${typeof id}. Expected string or array.`);
  1517. }
  1518. return new Promise((resolve, reject) => {
  1519. const MAX_RETRY_COUNT = retry;
  1520. let retry_counter = 0;
  1521. let set_interval_id = setInterval(isThereTweetData, span);
  1522. async function isThereTweetData(){
  1523. retry_counter++;
  1524. if(dataStore[id]){
  1525. clearInterval(set_interval_id);
  1526. return resolve(dataStore[id]);
  1527. }
  1528. if(retry_counter > MAX_RETRY_COUNT){
  1529. await getTweetData(id, method);
  1530. if(typeof id === 'string'){
  1531. if(dataStore[id]){
  1532. return resolve(dataStore[id]);
  1533. }else{
  1534. return reject("Max retry count reached");
  1535. }
  1536. }else if(Array.isArray(id)){
  1537. if(id.every(singleId => dataStore[singleId])){
  1538. return resolve(id.map(singleId => dataStore[singleId]));
  1539. }else{
  1540. return reject("Max retry count reached");
  1541. }
  1542. }
  1543. }
  1544. }
  1545. });
  1546. }
  1547. async function getFileSize(url){
  1548. return (await request(new requestObject_binary_head(url))).responseHeaders.match(/content-length: ?(\d+)/)[1];
  1549. }
  1550. function decodeHtml(html) {
  1551. var txt = document.createElement("textarea");
  1552. txt.innerHTML = html;
  1553. return txt.value;
  1554. }
  1555. function encodeBase64(data){
  1556. return btoa(data);
  1557. }
  1558. function decodeBase64(encodedData){
  1559. return atob(encodedData);
  1560. }
  1561. function openSettings(){
  1562. let container = document.createElement('div');
  1563. container.style.position = 'fixed';
  1564. container.style.width = '70vw';
  1565. container.style.height = '80vh';
  1566. container.style.backgroundColor = 'rgba(220, 200, 200, 0.9)';
  1567. container.style.zIndex = '9999';
  1568. container.style.display = 'flex';
  1569. container.style.maxHeight = '80vh';
  1570. container.style.top = '50%'; // 画面の上から50%の位置
  1571. container.style.left = '50%'; // 画面の左から50%の位置
  1572. container.style.transform = 'translate(-50%, -50%)'; // 要素の中心を基準に位置を調整
  1573. container.style.boxShadow = '0 0 10px rgba(0, 0, 0, 0.5)';
  1574. container.style.flexDirection = 'column';
  1575. container.className = `settingWindowContainer`
  1576. document.body.appendChild(container);
  1577. let pages = {};
  1578. let nav = document.createElement('div');
  1579. nav.style.position = 'static';
  1580. nav.style.top = '0';
  1581. nav.style.left = '0';
  1582. nav.style.width = '100%';
  1583. nav.style.zIndex = '2';
  1584. nav.style.display = 'flex';
  1585. nav.style.flexWrap = 'wrap';
  1586. container.appendChild(nav);
  1587. let contentWrapper = document.createElement('div');
  1588. contentWrapper.style.flexGrow = '1';
  1589. contentWrapper.style.position = 'relative';
  1590. contentWrapper.style.flexDirection = 'column';
  1591. contentWrapper.style.display = 'flex';
  1592. contentWrapper.style.overflowY = 'auto';
  1593. container.appendChild(contentWrapper);
  1594. let fotter = document.createElement('div');
  1595. fotter.style.position = 'static';
  1596. fotter.style.width = '100%';
  1597. fotter.style.zIndex = '2';
  1598. fotter.style.display = 'flex';
  1599. fotter.style.flexWrap = 'wrap';
  1600. fotter.style.left = '1em';
  1601. fotter.style.bottom = '1em';
  1602. container.appendChild(fotter);
  1603. let saveAllButton = document.createElement('button');
  1604. saveAllButton.textContent = 'Save All';
  1605. saveAllButton.addEventListener('click', saveAll);
  1606. fotter.appendChild(saveAllButton);
  1607. let cancelButton = document.createElement('button');
  1608. cancelButton.textContent = 'キャンセル';
  1609. cancelButton.addEventListener('click', () => {
  1610. container.remove();
  1611. });
  1612. fotter.appendChild(cancelButton);
  1613. /////////////////////////////////////////////
  1614. /*
  1615. function createHogePage(){
  1616. const pageName = "hogehoge";
  1617. const textData = env_Text[pageName];
  1618. const thisStoredSettings = storedSettings[pageName];
  1619. let thisScriptSettings = script_settings[pageName];
  1620. let contents = createPage(pageName,pageName,textData.functionName);
  1621. let mainContent = contents.main;
  1622. mainContent.innerHTML = `
  1623. <div class="${pageName}MainContents">
  1624. <span class="item_name">hogehoge</span>
  1625. </div>
  1626. `;
  1627. document.head.querySelectorAll(`style.settingCss.${pageName}`).forEach(e=>e.remove());
  1628. let style = document.createElement('style');
  1629. style.className = `settingCss ${pageName}`
  1630. style.textContent = `
  1631.  
  1632. `;
  1633. document.head.appendChild(style);
  1634. pages[pageName].saveFunction = save;
  1635. function save(){
  1636. console.log(pageName)
  1637. }
  1638. }
  1639. */
  1640. function creategGeneralPage(){
  1641. const pageName = "general"
  1642. let contents = createPage(pageName,env_Text.general,env_Text.general);
  1643. let mainContent = contents.main;
  1644. let header = contents.header;
  1645. mainContent.innerHTML = `
  1646. <div class="${pageName}MainContents">
  1647. <span class="item_name">home</span>
  1648. </div>
  1649. `;
  1650. document.head.querySelectorAll(`style.settingCss.${pageName}`).forEach(e=>e.remove());
  1651. let style = document.createElement('style');
  1652. style.className = `settingCss ${pageName}`
  1653. style.textContent = `
  1654. `;
  1655. document.head.appendChild(style);
  1656. pages[pageName].saveFunction = save;
  1657. function save(){
  1658. console.log(pageName)
  1659. }
  1660. }
  1661. function createStylePage(){
  1662. const pageName = "style";
  1663. const textData = env_Text[pageName];
  1664. const thisStoredSettings = storedSettings[pageName];
  1665. let thisScriptSettings = script_settings[pageName];
  1666. let contents = createPage(pageName,pageName,textData.functionName);
  1667. let mainContent = contents.main;
  1668. mainContent.innerHTML = `
  1669. <div class="${pageName}MainContents">
  1670. <span class="item_name">hogehoge</span>
  1671. </div>
  1672. `;
  1673. document.head.querySelectorAll(`style.settingCss.${pageName}`).forEach(e=>e.remove());
  1674. let style = document.createElement('style');
  1675. style.className = `settingCss ${pageName}`
  1676. style.textContent = `
  1677.  
  1678. `;
  1679. document.head.appendChild(style);
  1680. pages[pageName].saveFunction = save;
  1681. function save(){
  1682. console.log(pageName)
  1683. }
  1684. }
  1685. function createWebhook_brings_tweets_to_discordPage(){
  1686. const pageName = "webhook_brings_tweets_to_discord"
  1687. const textData = env_Text.webhook_brings_tweets_to_discord;
  1688. let contents = createPage(pageName,"webhook",textData.functionName);
  1689. let mainContent = contents.main;
  1690. const thisStoredSettings = storedSettings[pageName];
  1691. let thisScriptSettings = script_settings[pageName];
  1692. mainContent.innerHTML = `
  1693. <div class="${pageName}MainContents webhook-container" style="">
  1694. <span class="item_name">Webhooks</span>
  1695. <div id="webhooks" style="">
  1696. <!-- ここにWebhookの設定行が動的に追加されます -->
  1697. </div>
  1698. <div class="addWebhookButton">
  1699. <button id="addWebhook">+</button>
  1700. </div>
  1701. <div class="defaultWebhook">
  1702. <span class="item_name">${textData.default}Webhook</span><br>
  1703. <select id="defaultWebhook"></select>
  1704. </div>
  1705. <div class="displayMethod">
  1706. <span class="item_name">${textData.display_method}</span><br>
  1707. <select id="displayMethod">
  1708. <option value="method1">${textData.display_everywhere}</option>
  1709. <option value="method2">${textData.tweet_details_only}</option>
  1710. </select>
  1711. </div>
  1712. <div class="sendLangage">
  1713. <span class="item_name">${textData.language}</span><br>
  1714. <select id="languageSelect">
  1715. ${Object.keys(Text).map(lang_ => `<option value="${lang_}" ${lang_ === script_settings.webhook_brings_tweets_to_discord.lang ? 'selected' : ''}>${lang_}</option>`).join('')}
  1716. </select>
  1717. </div>
  1718. <div class="downloadVideo">
  1719. <span class="item_name">Video Download Option</span><br>
  1720. <input type="radio" id="downloadYes" name="downloadOption" value="true" ${thisStoredSettings.downloadVideo === true ? 'checked' : ''}>
  1721. <label for="downloadYes">Yes</label>
  1722. <input type="radio" id="downloadNo" name="downloadOption" value="false" ${thisStoredSettings.downloadVideo === false ? 'checked' : ''}>
  1723. <label for="downloadNo">No</label>
  1724. </div>
  1725. <br>
  1726. </div>
  1727. `;
  1728. let style = document.createElement('style');
  1729. style.className = `settingCss ${pageName}`
  1730. style.textContent = `
  1731. .webhook-container {
  1732. display: flex;
  1733. flex-direction: column;
  1734. }
  1735. .webhook-row {
  1736. display: flex;
  1737. align-items: center; /* 子要素を中央揃えにする */
  1738. flex-wrap: wrap; /* 必要に応じて折り返す */
  1739. }
  1740. .webhook-name,.webhook-url{
  1741. margin-right: 5px; /* 余白を追加 */
  1742. }
  1743. `;
  1744. document.head.appendChild(style);
  1745. document.getElementById('addWebhook').addEventListener('click', () => {
  1746. addWebhookRow();
  1747. });
  1748. function addWebhookRow(name = '', url = ''){
  1749. let row = document.createElement('div');
  1750. row.className = 'webhook-row'; // flexboxのスタイルを適用するためのクラスを追加
  1751. row.innerHTML = `
  1752. <div style="display:flex;border-bottom: 1px solid;flex-wrap: wrap;">
  1753. <div class="WebhookName">
  1754. <label>Name:</label>
  1755. <input type="text" class="webhook-name" value="${name}">
  1756. </div>
  1757. <div class=">WebhookUrl">
  1758. <label>URL:</label>
  1759. <input type="text" class="webhook-url" value="${url}">
  1760. </div>
  1761. <div class="removeWebhookButton">
  1762. <button class="removeWebhook">X</button>
  1763. </div>
  1764. </div>
  1765. `;
  1766. row.querySelector('.removeWebhook').addEventListener('click', () => {
  1767. row.remove();
  1768. updateDefaultWebhookOptions();
  1769. });
  1770. document.getElementById('webhooks').appendChild(row);
  1771. }
  1772. //console.log(thisStoredSettings.data)
  1773. function updateDefaultWebhookOptions(){
  1774. let select = document.getElementById('defaultWebhook');
  1775. select.innerHTML = '';
  1776. let webhookElements = document.getElementById('webhooks').children;
  1777. for(let elem of webhookElements){
  1778. let name = elem.querySelector('.webhook-name').value;
  1779. if(name){
  1780. let option = document.createElement('option');
  1781. option.value = name;
  1782. option.textContent = name;
  1783. select.appendChild(option);
  1784. }
  1785. }
  1786. }
  1787. for(let item of thisStoredSettings.data){
  1788. let decodedUrl = 'https://discord.com/api/webhooks/' + decodeBase64(item.value);
  1789. addWebhookRow(item.name, decodedUrl);
  1790. let option = document.createElement('option');
  1791. option.value = item.name;
  1792. option.textContent = item.name;
  1793. document.getElementById('defaultWebhook').appendChild(option);
  1794. }
  1795. pages[pageName].saveFunction = save;
  1796. function save(){
  1797. let data = [];
  1798. let names = [];
  1799. let hasDuplicate = false;
  1800. let hasInvalidWebhook = false;
  1801. let webhookElements = document.getElementById('webhooks').children;
  1802. for(let elem of webhookElements){
  1803. let name = elem.querySelector('.webhook-name').value;
  1804. let url = elem.querySelector('.webhook-url').value;
  1805. if(name && url){
  1806. if(!/^https:\/\/discord\.com\/api\/webhooks\/[\w-]+\/[\w-]+$/.test(url)){
  1807. elem.querySelector('.webhook-url').style.backgroundColor = 'red';
  1808. hasInvalidWebhook = true; // 無効なWebhookを検出
  1809. continue;
  1810. }else{
  1811. elem.querySelector('.webhook-url').style.backgroundColor = '';
  1812. }
  1813. // URLをBase64エンコード
  1814. let encodedUrl = encodeBase64(url.replace('https://discord.com/api/webhooks/', ''));
  1815. if(names.includes(name)){
  1816. hasDuplicate = true;
  1817. elem.querySelector('.webhook-name').style.backgroundColor = 'red';
  1818. }else{
  1819. names.push(name);
  1820. elem.querySelector('.webhook-name').style.backgroundColor = '';
  1821. data.push({ name, value: encodedUrl });
  1822. }
  1823. }
  1824. }
  1825. if(hasDuplicate){
  1826. customAlert(textData.when_webhook_name_duplicate);
  1827. return;
  1828. }
  1829. if(hasInvalidWebhook){
  1830. customAlert(textData.when_webhook_url_invalid);
  1831. return;
  1832. }
  1833. // 設定をJSON形式で保存
  1834. let selectedLanguage = document.getElementById('languageSelect').value;
  1835. let settings = {
  1836. data: data,
  1837. downloadVideo: getDownloadOption(),
  1838. defaultWebhook: document.getElementById('defaultWebhook').value,
  1839. displayMethod: document.getElementById('displayMethod').value,
  1840. lang: selectedLanguage
  1841. };
  1842. localStorage.setItem('webhook_brings_tweets_to_discord', JSON.stringify(settings));
  1843. // 保存した設定を再度読み込む
  1844. storedSettings[pageName] = JSON.parse(localStorage.getItem('webhook_brings_tweets_to_discord') || '{}');
  1845. script_settings[pageName] = {
  1846. "displayMethod": storedSettings[pageName].displayMethod || 'method1',
  1847. "lang": storedSettings[pageName].lang || GetCookie("lang") || 'en',
  1848. "defaultWebhook": storedSettings[pageName].defaultWebhook,
  1849. "webHooks": (function(data){
  1850. let webhooks = {};
  1851. if(data && Array.isArray(data)){
  1852. data.forEach(item => {
  1853. if(item.name && item.value){
  1854. webhooks[item.name] = item.value;
  1855. }
  1856. });
  1857. }
  1858. return webhooks;
  1859. })(storedSettings[pageName].data)
  1860. };
  1861. updateDefaultWebhookOptions();
  1862. document.querySelectorAll('.quickDimg').forEach((target)=>{target.remove()});
  1863. function getDownloadOption(){
  1864. const radioButtons = document.querySelectorAll('.downloadVideo input[name="downloadOption"]');
  1865. for(let i = 0; i < radioButtons.length; i++){
  1866. if(radioButtons[i].checked){
  1867. return radioButtons[i].value === "true";
  1868. }
  1869. }
  1870. return null;
  1871. }
  1872. };
  1873. }
  1874.  
  1875. function createPage(name,buttonText = undefined,headerText = undefined){
  1876. let page = document.createElement('div');
  1877. page.style.position = 'relative';
  1878. page.style.display = 'flex';
  1879. page.style.flexDirection = 'column';
  1880. page.style.flexGrow = '0';
  1881. page.style.flexShrink = '1';
  1882. page.style.top = '0';
  1883. page.style.left = '0';
  1884. page.style.width = 'auto';
  1885. page.style.height = 'auto';
  1886. page.style.zIndex = '-1';
  1887. page.style.overflowY = 'auto';
  1888. page.style.border = '1px solid black';
  1889. page.style.padding = "0.5em";
  1890. page.className = `${name} settingChildWindow`;
  1891. page.setAttribute('settingName', name);
  1892. let header = document.createElement('div');
  1893. header.className = 'pageHeader';
  1894. header.innerHTML = `<span style="font-size: 2em; font-weight: 700">${headerText || name} ${env_Text.settings}</span>`;
  1895. page.appendChild(header);
  1896. let contentContainer = document.createElement('div');
  1897. contentContainer.className = 'pageContent';
  1898. contentContainer.style.display = 'flex';
  1899. contentContainer.style.flexDirection = 'column';
  1900. contentContainer.style.overflowY = 'auto';
  1901. page.appendChild(contentContainer);
  1902. let footer = document.createElement('div');
  1903. footer.className = 'pageFooter';
  1904. footer.style.marginTop = "1em";
  1905. let saveButton = document.createElement('button');
  1906. saveButton.textContent = 'Save';
  1907. saveButton.addEventListener('click', function(){
  1908. if(pages[name] && typeof pages[name].saveFunction === 'function'){
  1909. pages[name].saveFunction();
  1910. }
  1911. });
  1912. footer.appendChild(saveButton);
  1913. page.appendChild(footer);
  1914. pages[name] = {
  1915. "page": page,
  1916. "saveFunction": function(){}
  1917. };
  1918. contentWrapper.appendChild(page);
  1919. makeChangePageButton(name,buttonText)
  1920. return {"all": page,"header": header,"main": contentContainer,"footer": footer};
  1921. }
  1922. function showPage(name){
  1923. for(let key in pages){
  1924. if(key === name){
  1925. pages[key].page.style.zIndex = '1';
  1926. pages[key].page.style.display = 'flex';
  1927. }else{
  1928. pages[key].page.style.zIndex = '-1';
  1929. pages[key].page.style.display = 'none';
  1930. }
  1931. }
  1932. }
  1933. function makeChangePageButton(name,buttonText = undefined){
  1934. if(buttonText === undefined)buttonText = name;
  1935. let homeButton = document.createElement('button');
  1936. homeButton.textContent = buttonText;
  1937. homeButton.addEventListener('click', () => showPage(name));
  1938. nav.appendChild(homeButton);
  1939. }
  1940. function saveAll(){
  1941. for(let key in pages){
  1942. pages[key].saveFunction();
  1943. }
  1944. container.remove();
  1945. }
  1946. let style = document.createElement('style');
  1947. style.textContent = `
  1948. .pageHeader, .pageFooter {
  1949. flex-shrink: 0;
  1950. }
  1951. .pageContent {
  1952. flex-grow: 1;
  1953. overflow-y: auto;
  1954. }
  1955. .item_name {
  1956. font-size: 1.5em;
  1957. }
  1958. `;
  1959. document.head.appendChild(style);
  1960. creategGeneralPage();
  1961. createWebhook_brings_tweets_to_discordPage();
  1962. showPage('general');
  1963. }
  1964. GM_registerMenuCommand('script settings', openSettings);
  1965. function customAlert(message, url){
  1966. let overlay = document.createElement('div');
  1967. overlay.style.position = 'fixed';
  1968. overlay.style.top = '0';
  1969. overlay.style.left = '0';
  1970. overlay.style.width = '100%';
  1971. overlay.style.height = '100%';
  1972. overlay.style.backgroundColor = 'rgba(0, 0, 0, 0.5)';
  1973. overlay.style.zIndex = '10000';
  1974. let alertBox = document.createElement('div');
  1975. alertBox.style.position = 'absolute';
  1976. alertBox.style.top = '50%';
  1977. alertBox.style.left = '50%';
  1978. alertBox.style.transform = 'translate(-50%, -50%)';
  1979. alertBox.style.padding = '20px';
  1980. alertBox.style.backgroundColor = 'white';
  1981. alertBox.style.border = '1px solid black';
  1982. alertBox.style.zIndex = '10001';
  1983. alertBox.style.maxWidth = '80%'; // ボックスの最大幅を設定してテキストを折り返します
  1984. alertBox.style.whiteSpace = 'pre-wrap'; // 改行と空白を保持します
  1985.  
  1986. let alertMessage = document.createElement('p');
  1987. alertMessage.innerHTML = message.replace(/\n/g, '<br>'); // \nを<br>に置き換えて改行を表示します
  1988.  
  1989. let closeButton = document.createElement('button');
  1990. closeButton.textContent = env_Text.close;
  1991. alertBox.addEventListener('click', (e) => {
  1992. e.stopPropagation(); // これにより、イベントがoverlayまで伝播しなくなります
  1993. });
  1994. overlay.addEventListener('click', () => {
  1995. document.body.removeChild(overlay);
  1996. });
  1997. closeButton.addEventListener('click', () => {
  1998. document.body.removeChild(overlay);
  1999. });
  2000.  
  2001. alertBox.appendChild(alertMessage);
  2002.  
  2003. // URLが提供された場合、それを表示するa要素を作成します
  2004. if(url){
  2005. let urlElement = document.createElement('a');
  2006. urlElement.href = url;
  2007. urlElement.textContent = url;
  2008. urlElement.style.display = 'block';
  2009. urlElement.style.marginTop = '10px';
  2010. urlElement.target = '_blank';
  2011. urlElement.rel = 'noopener';
  2012. alertBox.appendChild(urlElement);
  2013. }
  2014. alertBox.appendChild(closeButton);
  2015. overlay.appendChild(alertBox);
  2016. document.body.appendChild(overlay);
  2017. }
  2018. async function request(object, maxRetries = 0, timeout = 60000){
  2019. let retryCount = 0;
  2020. while(retryCount <= maxRetries){
  2021. try{
  2022. return await new Promise((resolve, reject) => {
  2023. GM_xmlhttpRequest({
  2024. method: object.method,
  2025. url: object.url,
  2026. headers: object.headers,
  2027. responseType: object.respType,
  2028. data: object.body,
  2029. anonymous: object.anonymous,
  2030. timeout: timeout,
  2031. onload: function(responseDetails){
  2032. return resolve(responseDetails);
  2033. },
  2034. ontimeout: function(responseDetails){
  2035. reject(`[request]time out:\nresponse ${responseDetails}`);
  2036. },
  2037. onerror: function(responseDetails){
  2038. reject(`[request]error:\nresponse ${responseDetails}`);
  2039. }
  2040. });
  2041. });
  2042. }catch(error){
  2043. retryCount++;
  2044. console.warn(`Retry ${retryCount}: Failed to fetch ${object.url}. Reason: ${error}`);
  2045. if(retryCount === maxRetries){
  2046. throw new Error(`Failed to fetch ${object.url} after ${maxRetries} retries.`);
  2047. }
  2048. }
  2049. }
  2050. }
  2051. class requestObject_twitter_api_v1_1{
  2052. constructor(ID,endpoints = "lookup"){
  2053. this.method = 'GET';
  2054. this.respType = 'json';
  2055. this.url = `https://api.twitter.com/1.1/statuses/${endpoints}.json?id=${ID}&tweet_mode=extended`;
  2056. this.body = null;
  2057. this.headers = {
  2058. "Content-Type": "application/json",
  2059. 'User-agent': userAgent,
  2060. 'accept': '*/*',
  2061. 'Accept-Encoding': 'br, gzip, deflate',
  2062. 'Referer': "https://twitter.com/",
  2063. 'Host': 'api.twitter.com',
  2064. 'authorization': 'Bearer AAAAAAAAAAAAAAAAAAAAAIK1zgAAAAAA2tUWuhGZ2JceoId5GwYWU5GspY4%3DUq7gzFoCZs1QfwGoVdvSac3IniczZEYXIcDyumCauIXpcAPorE',
  2065. 'x-csrf-token': GetCookie("ct0"),
  2066. };
  2067. this.package = null;
  2068. this.anonymous = false;
  2069. }
  2070. }
  2071. class requestObject_twitter_api_graphql{
  2072. constructor(ID){
  2073. this.method = 'GET';
  2074. this.respType = 'json';
  2075. this.url = `https://twitter.com/i/api/graphql/TuC3CinYecrqAyqccUyFhw/TweetDetail?variables=%7B%22focalTweetId%22%3A%22${ID}%22%2C%22referrer%22%3A%22home%22%2C%22with_rux_injections%22%3Afalse%2C%22includePromotedContent%22%3Atrue%2C%22withCommunity%22%3Atrue%2C%22withQuickPromoteEligibilityTweetFields%22%3Atrue%2C%22withArticleRichContent%22%3Atrue%2C%22withBirdwatchNotes%22%3Atrue%2C%22withVoice%22%3Atrue%2C%22withV2Timeline%22%3Atrue%7D&features=%7B%22rweb_lists_timeline_redesign_enabled%22%3Atrue%2C%22responsive_web_graphql_exclude_directive_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22tweetypie_unmention_optimization_enabled%22%3Atrue%2C%22responsive_web_edit_tweet_api_enabled%22%3Atrue%2C%22graphql_is_translatable_rweb_tweet_is_translatable_enabled%22%3Atrue%2C%22view_counts_everywhere_api_enabled%22%3Atrue%2C%22longform_notetweets_consumption_enabled%22%3Atrue%2C%22responsive_web_twitter_article_tweet_consumption_enabled%22%3Atrue%2C%22tweet_awards_web_tipping_enabled%22%3Afalse%2C%22freedom_of_speech_not_reach_fetch_enabled%22%3Atrue%2C%22standardized_nudges_misinfo%22%3Atrue%2C%22tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled%22%3Atrue%2C%22longform_notetweets_rich_text_read_enabled%22%3Atrue%2C%22longform_notetweets_inline_media_enabled%22%3Atrue%2C%22responsive_web_media_download_video_enabled%22%3Atrue%2C%22responsive_web_enhance_cards_enabled%22%3Afalse%7D&fieldToggles=%7B%22withArticleRichContentState%22%3Atrue%7D`;
  2076. this.body = null;
  2077. this.headers = {
  2078. "Content-Type": "application/json",
  2079. 'User-agent': userAgent,
  2080. 'accept': '*/*',
  2081. 'Accept-Encoding': 'br, gzip, deflate',
  2082. 'authorization': `Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA`,
  2083. 'x-csrf-token': GetCookie("ct0"),
  2084. };
  2085. this.package = null;
  2086. this.anonymous = false;
  2087. }
  2088. }
  2089. class requestObject_twitter_get_user_by_screenname{
  2090. constructor(ID){
  2091. this.method = 'GET';
  2092. this.respType = 'json';
  2093. this.url = `https://api.twitter.com/graphql/rePnxwe9LZ51nQ7Sn_xN_A/UserByScreenName?variables=%7B%22screen_name%22%3A%22${ID}%22%2C%22withSafetyModeUserFields%22%3Afalse%2C%22withSuperFollowsUserFields%22%3Afalse%7D&features=%7B%22responsive_web_twitter_blue_verified_badge_is_enabled%22%3Afalse%2C%22responsive_web_graphql_exclude_directive_enabled%22%3Afalse%2C%22verified_phone_label_enabled%22%3Afalse%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Afalse%7D`;
  2094. this.body = null;
  2095. this.headers = {
  2096. "Content-Type": "application/json",
  2097. 'User-agent': userAgent,
  2098. 'accept': '*/*',
  2099. 'Referer': "https://twitter.com/",
  2100. 'Host': 'api.twitter.com',
  2101. 'Accept-Encoding': 'br, gzip, deflate',
  2102. 'authorization': `Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA`,
  2103. 'x-csrf-token': GetCookie("ct0"),
  2104. };
  2105. this.package = null;
  2106. this.anonymous = false;
  2107. this.screen_name = ID;
  2108. }
  2109. }
  2110. class requestObject_twitter_get_user_by_screenname_1_1{
  2111. constructor(ID) {
  2112. this.method = 'GET';
  2113. this.respType = 'json';
  2114. this.url = `https://api.twitter.com/1.1/users/show.json?screen_name=${ID}`;
  2115. this.body = null;
  2116. this.headers = {
  2117. "Content-Type": "application/json",
  2118. 'User-agent': userAgent,
  2119. 'accept': '*/*',
  2120. 'Referer': "https://twitter.com/",
  2121. 'Host': 'api.twitter.com',
  2122. 'Accept-Encoding': 'br, gzip, deflate',
  2123. 'authorization': `Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA`,
  2124. 'x-csrf-token': GetCookie("ct0"),
  2125. };
  2126. this.package = null;
  2127. this.anonymous = false;
  2128. this.screen_name = ID;
  2129. }
  2130. }
  2131. class requestObject_twitter_time_line{
  2132. constructor(){
  2133. this.method = 'POST';
  2134. this.respType = 'json';
  2135. this.url = `https://twitter.com/i/api/graphql/HnVOsy-Rh_0Cuq06_z9lGA/HomeLatestTimeline`;
  2136. this.body = JSON.stringify({
  2137. "variables": {
  2138. "count": 100,
  2139. "includePromotedContent": false,
  2140. "latestControlAvailable": true,
  2141. "requestContext": "launch"
  2142. },
  2143. "features": {
  2144. "responsive_web_graphql_exclude_directive_enabled": true,
  2145. "verified_phone_label_enabled": false,
  2146. "responsive_web_home_pinned_timelines_enabled": true,
  2147. "creator_subscriptions_tweet_preview_api_enabled": true,
  2148. "responsive_web_graphql_timeline_navigation_enabled": true,
  2149. "responsive_web_graphql_skip_user_profile_image_extensions_enabled": false,
  2150. "tweetypie_unmention_optimization_enabled": true,
  2151. "responsive_web_edit_tweet_api_enabled": true,
  2152. "graphql_is_translatable_rweb_tweet_is_translatable_enabled": true,
  2153. "view_counts_everywhere_api_enabled": true,
  2154. "longform_notetweets_consumption_enabled": true,
  2155. "responsive_web_twitter_article_tweet_consumption_enabled": false,
  2156. "tweet_awards_web_tipping_enabled": false,
  2157. "freedom_of_speech_not_reach_fetch_enabled": true,
  2158. "standardized_nudges_misinfo": true,
  2159. "tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled": true,
  2160. "longform_notetweets_rich_text_read_enabled": true,
  2161. "longform_notetweets_inline_media_enabled": true,
  2162. "responsive_web_media_download_video_enabled": true,
  2163. "responsive_web_enhance_cards_enabled": false
  2164. },
  2165. "queryId": "HnVOsy-Rh_0Cuq06_z9lGA"
  2166. });
  2167. this.headers = {
  2168. "Content-Type": "application/json",
  2169. 'User-agent': userAgent,
  2170. 'accept': '*/*',
  2171. 'Referer': "https://twitter.com/",
  2172. 'Host': 'twitter.com',
  2173. 'Accept-Encoding': 'br, gzip, deflate',
  2174. 'authorization': `Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA`,
  2175. 'x-csrf-token': GetCookie("ct0"),
  2176. };
  2177. this.package = null;
  2178. this.anonymous = false;
  2179. }
  2180. }
  2181. class requestObject_twitter_time_line_forYou{
  2182. constructor(){
  2183. this.method = 'POST';
  2184. this.respType = 'json';
  2185. this.url = `https://twitter.com/i/api/graphql/-8TWbLqVU1ROq-eeErVc2w/HomeTimeline`;
  2186. this.body = JSON.stringify({
  2187. "variables": {
  2188. "count": 100,
  2189. "includePromotedContent": false,
  2190. "latestControlAvailable": true,
  2191. "requestContext": "launch",
  2192. "withCommunity": true,
  2193. },
  2194. "features": {
  2195. "responsive_web_graphql_exclude_directive_enabled": true,
  2196. "verified_phone_label_enabled": false,
  2197. "responsive_web_home_pinned_timelines_enabled": true,
  2198. "creator_subscriptions_tweet_preview_api_enabled": true,
  2199. "responsive_web_graphql_timeline_navigation_enabled": true,
  2200. "responsive_web_graphql_skip_user_profile_image_extensions_enabled": false,
  2201. "tweetypie_unmention_optimization_enabled": true,
  2202. "responsive_web_edit_tweet_api_enabled": true,
  2203. "graphql_is_translatable_rweb_tweet_is_translatable_enabled": true,
  2204. "view_counts_everywhere_api_enabled": true,
  2205. "longform_notetweets_consumption_enabled": true,
  2206. "responsive_web_twitter_article_tweet_consumption_enabled": false,
  2207. "tweet_awards_web_tipping_enabled": false,
  2208. "freedom_of_speech_not_reach_fetch_enabled": true,
  2209. "standardized_nudges_misinfo": true,
  2210. "tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled": true,
  2211. "longform_notetweets_rich_text_read_enabled": true,
  2212. "longform_notetweets_inline_media_enabled": true,
  2213. "responsive_web_media_download_video_enabled": false,
  2214. "responsive_web_enhance_cards_enabled": false
  2215. },
  2216. "queryId": "-8TWbLqVU1ROq-eeErVc2w"
  2217. });
  2218. this.headers = {
  2219. "Content-Type": "application/json",
  2220. 'User-agent': userAgent,
  2221. 'accept': '*/*',
  2222. 'Referer': "https://twitter.com/",
  2223. 'Host': 'twitter.com',
  2224. 'Accept-Encoding': 'br, gzip, deflate',
  2225. 'authorization': `Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA`,
  2226. 'x-csrf-token': GetCookie("ct0"),
  2227. };
  2228. this.package = null;
  2229. this.anonymous = false;
  2230. }
  2231. }
  2232. class requestObject_twitter_FavoriteTweet{
  2233. constructor(URL){
  2234. this.method = 'POST';
  2235. this.respType = 'json';
  2236. this.url = 'https://twitter.com/i/api/graphql/lI07N6Otwv1PhnEgXILM7A/FavoriteTweet';
  2237. this.body = `{"variables":{"tweet_id":"${URL.split('/').pop()}"},"queryId":"lI07N6Otwv1PhnEgXILM7A"}`;
  2238. this.headers = {
  2239. 'Content-Type': 'application/json',
  2240. 'User-agent': userAgent,
  2241. 'accept': '*/*',
  2242. 'Referer': URL,
  2243. 'Accept-Encoding': 'br, gzip, deflate',
  2244. 'Origin': 'https://twitter.com',
  2245. 'authorization': 'Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA',
  2246. "x-csrf-token": GetCookie("ct0"),
  2247. 'Sec-Fetch-Site': 'same-origin',
  2248. 'Sec-Fetch-Mode': 'navigate',
  2249. };
  2250. this.package = null;
  2251. this.anonymous = false;
  2252. }
  2253. }
  2254. class requestObject_url_expand{
  2255. constructor(URL){
  2256. this.method = 'POST';
  2257. this.respType = 'text';
  2258. this.url = `https://geek-website.com/tool/shortlink_open/request.php`;
  2259. this.body = `shortlink=${encodeURIComponent(URL)}`;
  2260. this.headers = {
  2261. "Content-Type": "application/x-www-form-urlencoded; charset=UTF-8",
  2262. 'User-agent': userAgent,
  2263. "Accept": "text/plain, */*; q=0.01",
  2264. 'Accept-Encoding': 'br, gzip, deflate',
  2265. 'Referer': "https://geek-website.com/tool/shortlink_open/",
  2266. 'Host': 'geek-website.com',
  2267. "Sec-Fetch-Dest": "empty",
  2268. "Sec-Fetch-Mode": "cors",
  2269. "Sec-Fetch-Site": "same-origin"
  2270. };
  2271. this.package = null;
  2272. this.anonymous = true;
  2273. }
  2274. }
  2275. class requestObject_fanbox{
  2276. constructor(URL,fanbox_URL){
  2277. this.method = 'GET';
  2278. this.respType = 'json';
  2279. this.url = `${URL}`;
  2280. this.body = null;
  2281. this.headers = {
  2282. 'User-agent': userAgent,
  2283. 'origin': fanbox_URL,
  2284. 'Accept-Encoding': 'br, gzip, deflate',
  2285. 'Host': 'api.fanbox.cc',
  2286. };
  2287. }
  2288. }
  2289. class sendObject_to_discord_webhook{
  2290. constructor(webhook,sendEmbeds){
  2291. this.method = 'POST';
  2292. this.respType = 'json';
  2293. this.url = "https://discord.com/api/webhooks/" + atob(webhook);
  2294. this.headers = {
  2295. };
  2296. this.package = null;
  2297. this.anonymous = true;
  2298. this.body = sendEmbeds;
  2299. }
  2300. }
  2301. class requestObject_binary_data{
  2302. constructor(URL,addtional_cookie = undefined){
  2303. this.method = 'GET';
  2304. this.respType = 'blob';
  2305. this.url = `${URL}`;
  2306. this.body = null;
  2307. this.encoding = null;
  2308. this.headers = {
  2309. "Content-Type": "*/*",
  2310. 'User-agent': navigator.userAgent || navigator.vendor || window.opera,
  2311. 'accept': '*/*',
  2312. 'Accept-Encoding': 'br, gzip, deflate',
  2313. 'Referer': URL,
  2314. "Sec-Fetch-Mode": "navigate",
  2315. 'cookie': `${addtional_cookie}`
  2316. };
  2317. this.package = null;
  2318. }
  2319. }
  2320. class requestObject_binary_head{
  2321. constructor(URL, addtional_cookie = undefined) {
  2322. this.method = 'HEAD'; // ここを 'HEAD' に変更
  2323. this.url = `${URL}`;
  2324. this.body = null;
  2325. this.encoding = null;
  2326. this.headers = {
  2327. "Content-Type": "*/*",
  2328. 'User-agent': navigator.userAgent || navigator.vendor || window.opera,
  2329. 'accept': '*/*',
  2330. 'Accept-Encoding': 'br, gzip, deflate',
  2331. 'Referer': URL,
  2332. "Sec-Fetch-Mode": "navigate",
  2333. 'cookie': `${addtional_cookie}`
  2334. };
  2335. this.package = null;
  2336. }
  2337. }
  2338. class requestObject{
  2339. constructor(URL){
  2340. this.method = 'GET';
  2341. this.respType = '';
  2342. this.url = `${URL}`;
  2343. this.headers = {
  2344. "Content-Type": "*/*",
  2345. 'Accept-Encoding': 'br, gzip, deflate',
  2346. 'User-agent': userAgent,
  2347. 'accept': '*/*',
  2348. 'Referer': URL,
  2349. "Sec-Fetch-Mode": "navigate",
  2350. "Sec-Fetch-Dest": "empty",
  2351. "Sec-Fetch-Mode": "cors",
  2352. "Sec-Fetch-Site": "same-origin"
  2353. };
  2354. this.package = null;
  2355. }
  2356. }
  2357. async function test(){
  2358. }
  2359. //test();
  2360. main();
  2361. if(currentUrl.match(/https?:\/\/twitter\.com\/[\w]*\/status\/[0-9]*/))fetchAndProcessTwitterApi("graphQL",extractTweetId(currentUrl));
  2362. window.addEventListener("scroll", update);
  2363. })();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址