Greasy Fork is available in English.
長いツイートを「更に表示」を押さなくてもTLで展開します。
// ==UserScript==// @name [Twitter]長いツイートをTLで展開// @name:ja [Twitter]長いツイートをTLで展開// @name:en [Twitter]Note_Tweet expander// @version ######1919810.0.21// @description 長いツイートを「更に表示」を押さなくてもTLで展開します。// @description:ja 長いツイートを「更に表示」を押さなくてもTLで展開します。// @description:en Long tweets will expand in the TimeLine without having to press "Show More".// @author ゆにてぃー// @match https://twitter.com/*// @match https://mobile.twitter.com/*// @match https://x.com/*// @match https://X.com/*// @connect api.twitter.com// @icon  @grant GM_xmlhttpRequest// @license MIT// @namespace https://greasyfork.org/ja/users/1023652// ==/UserScript==(function() {'use strict';let updating = false;const userAgent = navigator.userAgent || navigator.vendor || window.opera;const isFirefox = !!userAgent.match("Firefox")let fetchedTweets = {};let fetchedTweetsUserData = {};let fetchedTweetsUserDataByUserName = {};window.addEventListener("scroll", update);init();async function main(){const link_class = "css-1jxf684 r-bcqeeo r-1ttztb7 r-qvutc0 r-poiln3 r-1loqt21";document.querySelectorAll('article[data-testid="tweet"]:not(.tweetExpanderChecked)').forEach(async function(tweet){tweet.classList.add('tweetExpanderChecked');const elements = tweet.querySelectorAll('[data-testid="tweetText"]');const show_more_link = tweet.querySelectorAll('[data-testid="tweet-text-show-more-link"]');const tweetId = tweet.querySelector(`[data-testid="User-Name"] a[aria-label], .css-1dbjc4n.r-1d09ksm.r-1471scf.r-18u37iz.r-1wbh5a2 a[aria-label]`)?.href.match(/[\w]{1,}\.com\/[^/]+\/status\/(\d+)/)[1];if(!tweetId)return;if(show_more_link[0])show_more_link[0].style.display = "none";if(show_more_link[1])show_more_link[1].style.display = "none";elements.forEach(async (element,index) =>{if(!(element.innerText.match(/…$/) || (show_more_link[index]?.tagName.toLowerCase().match(/div|button/)))){element.classList.add('tweetExpanderChecked');//if(element.innerText.split('\n').length >= 10)element.style.webkitLineClamp = null;return;}let tweet_data,note_tweet;if(index == 0){tweet_data = await getTweetData(tweetId,"graphQL");note_tweet = tweet_data.note_tweet?.note_tweet_r###lts.r###lt || tweet_data.note_tweet?.note_tweet_r###lts.r###lt || null;}else{tweet_data = await getTweetData(tweetId,"graphQL");tweet_data = await getTweetData(tweet_data.legacy.quoted_status_id_str,"graphQL");note_tweet = tweet_data.note_tweet?.note_tweet_r###lts.r###lt || tweet_data.note_tweet?.note_tweet_r###lts.r###lt || null;}if(!note_tweet){element.style.webkitLineClamp = null;element.classList.add('tweetExpanderChecked');return;}const hashtags = note_tweet.entity_set?.hashtags || [];const user_mentions = note_tweet.entity_set?.user_mentions || [];const symbols = note_tweet.entity_set?.symbols || [];const urls = note_tweet.entity_set?.urls;var new_tweet_text = note_tweet.text;function countSurrogatePairs(str){return Array.from(str).filter(char => char.match(/[\uD800-\uDBFF][\uDC00-\uDFFF]/)).length;}let combined = [].concat(hashtags.map(tag => ({type: 'hashtag',indices: tag.indices,text: tag.text})),user_mentions.map(mention => ({type: 'mention',indices: mention.indices,text: mention.screen_name})),symbols.map(symbol => ({type: 'symbol',indices: symbol.indices,text: symbol.text})));// combinedをindicesの順にソートcombined.sort((a, b) => b.indices[0] - a.indices[0]);let transformedText = new_tweet_text;combined.forEach(item => {let start = item.indices[0];let end = item.indices[1];// サロゲートペアの数をカウントして調整const adjustment = countSurrogatePairs(transformedText.slice(0, end));start += adjustment;end += adjustment;let replacement = '';switch(item.type){case 'hashtag':replacement = `<a class="${link_class}" style="text-decoration: none;color:rgb(29, 155, 240)" dir="ltr" role="link" href="https://twitter.com/hashtag/${item.text}" target="_blank" rel="noopener">#${item.text}</a>`;break;case 'mention':replacement = `<a class="${link_class}" style="text-decoration: none;color:rgb(29, 155, 240)" dir="ltr" role="link" href="https://twitter.com/${item.text}" target="_blank" rel="noopener">@${item.text}</a>`;break;case 'symbol':replacement = `<a class="${link_class}" style="text-decoration: none;color:rgb(29, 155, 240)" dir="ltr" role="link" href="https://twitter.com/search?q=%24${item.text}&src=cashtag_click" target="_blank" rel="noopener">$${item.text}</a>`;break;}transformedText = transformedText.slice(0, start) + replacement + transformedText.slice(end);});new_tweet_text = transformedText;const seen = new Set();urls.filter(target => !seen.has(target.url) && seen.add(target.url)).forEach(target =>{new_tweet_text = new_tweet_text.replace(new RegExp(`${target.url}(?=(\\s|$|\\u3000|\\W)(?!\\.|,))`, 'gu'), `<a class="${link_class}" style="text-decoration: none;color:rgb(29, 155, 240)" dir="ltr" role="link" href="${target.url}" target="_blank" rel="noopener">${target.display_url}</a>`);});var new_tweet_node = document.createElement("span");new_tweet_node.className = 'css-901oao css-16my406 r-1tl8opc r-bcqeeo r-qvutc0';new_tweet_node.innerHTML = new_tweet_text;while(element.firstChild){element.removeChild(element.firstChild);}element.appendChild(new_tweet_node);element.style.webkitLineClamp = null;});});}function init() {main();}function update() {if(updating) return;updating = true;init();setTimeout(() => {updating = false;}, 1500);}function GetCookie(name){let arr, reg = new RegExp("(^| )" + name + "=([^;]*)(;|$)");if(arr = document.cookie.match(reg)){return decodeURIComponent(arr[2]);}else{return null;}}function findParent(element, selector){let current = element;while(current !== null){if(current.matches(selector)){return current;}current = current.parentNode;}return null;}async function fetchAndProcessTwitterApi(method,id = undefined,forceFetch = false){return new Promise(async (resolve, reject) => {try{switch(method){case 'graphQL':if(fetchedTweets[id] && !(forceFetch === true))return resolve(fetchedTweets[id]);await graphQL();break;default:console.warn("なにか間違ってないか?")return reject("something wrong.");}return resolve("OK!");}catch(error){console.error(error);throw new Error(`Failed to fetch API data.\nmethod: ${method}\nid: ${id}`);}});async function graphQL(){const response = await request(new requestObject_twitter_api_graphql(id));if(!response.status === "200")throw new Error(`Failed to fetch`);processgraphQL(response.response.data.threaded_conversation_with_injections_v2.instructions[0].entries);}function processgraphQL(entries){if(!entries)return null;entries.forEach(entry=>{const tmpData = entry.content?.itemContent?.tweet_r###lts || entry.item?.itemContent?.tweet_r###lts;let tweetData = tmpData?.r###lt?.tweet || tmpData?.r###lt;if(tweetData?.tombstone)return;if(!tweetData)return;try{if(tweetData.quoted_status_r###lt){let quoted = tweetData.quoted_status_r###lt.r###lt?.tweet || tweetData.quoted_status_r###lt.tweet || tweetData.quoted_status_r###lt?.r###lt;fetchedTweetsUserData[quoted.core.user_r###lts.r###lt.rest_id] = {...quoted.core.user_r###lts.r###lt,"API_type": "graphQL"};fetchedTweetsUserDataByUserName[quoted.core.user_r###lts.r###lt.legacy.screen_name] = fetchedTweetsUserData[quoted.core.user_r###lts.r###lt.rest_id];quoted.core.user_r###lts.r###lt = fetchedTweetsUserData[quoted.core.user_r###lts.r###lt.rest_id];fetchedTweets[quoted.rest_id] = {...quoted,"API_type": "graphQL"};tweetData.quoted_status_r###lt.r###lt = fetchedTweets[quoted.rest_id];}fetchedTweetsUserData[tweetData.core.user_r###lts.r###lt.rest_id] = {...tweetData.core.user_r###lts.r###lt,"API_type": "graphQL"};fetchedTweetsUserDataByUserName[tweetData.core.user_r###lts.r###lt.legacy.screen_name] = fetchedTweetsUserData[tweetData.core.user_r###lts.r###lt.rest_id];tweetData.core.user_r###lts.r###lt = fetchedTweetsUserData[tweetData.core.user_r###lts.r###lt.rest_id];fetchedTweets[tweetData.rest_id] = {...tweetData,"API_type": "graphQL"};}catch(error){console.errorconsole.error({error: `processgraphQL error.\ndetails: ${error}`,apiResponse:tweetData});}});}}async function getTweetData(id, method = 'graphQL', forceFetch = false){const dataStore = fetchedTweets;if(typeof id === 'string'){if(dataStore[id] && !forceFetch)return dataStore[id];}else{throw new Error("Invalid ID type.");}await fetchAndProcessTwitterApi(method, id, forceFetch);if(dataStore[id]){return dataStore[id];}else{throw new Error("Failed to fetch tweet data for ID: " + id);}}async function request(object, maxRetries = 0, timeout = 60000){let retryCount = 0;while(retryCount <= maxRetries){try{return await new Promise((resolve, reject) => {GM_xmlhttpRequest({method: object.method,url: object.url,headers: object.headers,responseType: object.respType,data: object.body,anonymous: object.anonymous,timeout: timeout,onload: function(responseDetails){return resolve(responseDetails);},ontimeout: function(responseDetails){reject(`[request]time out:\nresponse ${responseDetails}`);},onerror: function(responseDetails){reject(`[request]error:\nresponse ${responseDetails}`);}});});}catch(error){retryCount++;console.warn(`Retry ${retryCount}: Failed to fetch ${object.url}. Reason: ${error}`);if(retryCount === maxRetries){throw new Error(`Failed to fetch ${object.url} after ${maxRetries} retries.`);}}}}class requestObject_twitter_api_graphql{constructor(ID) {this.method = 'GET';this.respType = 'json';this.url = `https://${window.location.hostname}/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_r###lts_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`;this.body = null;this.headers = {"Content-Type": "application/json",'User-agent': navigator.userAgent || navigator.vendor || window.opera,'accept': '*/*','authorization': `Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA`,'Accept-Encoding': 'br, gzip, deflate','x-csrf-token': GetCookie("ct0"),};this.package = null;this.anonymous = false;}}})();