Extension to return old Twitter layout from 2015.
at master 190 kB view raw
1const linkRegex = /(\s|^)(http(s)?:\/\/.)?(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,60}\.(삼성|닷컴|닷넷|香格里拉|餐厅|食品|飞利浦|電訊盈科|集团|通販|购物|谷歌|诺基亚|联通|网络|网站|网店|网址|组织机构|移动|珠宝|点看|游戏|淡马锡|机构|書籍|时尚|新闻|政府|政务|招聘|手表|手机|我爱你|慈善|微博|广东|工行|家電|娱乐|天主教|大拿|大众汽车|在线|嘉里大酒店|嘉里|商标|商店|商城|公益|公司|八卦|健康|信息|佛山|企业|亚马逊|中文网|中信|世界|ポイント|ファッション|セール|ストア|コム|グーグル|クラウド|アマゾン|みんな|คอม|संगठन|नेट|कॉम|همراه|موقع|موبايلي|كوم|كاثوليك|عرب|شبكة|بيتك|بازار|العليان|ارامكو|اتصالات|ابوظبي|קום|сайт|рус|орг|онлайн|москва|ком|католик|дети|zuerich|zone|zippo|zip|zero|zara|zappos|yun|youtube|you|yokohama|yoga|yodobashi|yandex|yamaxun|yahoo|yachts|xyz|xxx|xperia|xin|xihuan|xfinity|xerox|xbox|wtf|wtc|wow|world|works|work|woodside|wolterskluwer|wme|winners|wine|windows|win|williamhill|wiki|wien|whoswho|weir|weibo|wedding|wed|website|weber|webcam|weatherchannel|weather|watches|watch|warman|wanggou|wang|walter|walmart|wales|vuelos|voyage|voto|voting|vote|volvo|volkswagen|vodka|vlaanderen|vivo|viva|vistaprint|vista|vision|visa|virgin|vip|vin|villas|viking|vig|video|viajes|vet|versicherung|vermögensberatung|vermögensberater|verisign|ventures|vegas|vanguard|vana|vacations|ups|uol|uno|university|unicom|uconnect|ubs|ubank|tvs|tushu|tunes|tui|tube|trv|trust|travelersinsurance|travelers|travelchannel|travel|training|trading|trade|toys|toyota|town|tours|total|toshiba|toray|top|tools|tokyo|today|tmall|tkmaxx|tjx|tjmaxx|tirol|tires|tips|tiffany|tienda|tickets|tiaa|theatre|theater|thd|teva|tennis|temasek|telefonica|telecity|tel|technology|tech|team|tdk|tci|taxi|tax|tattoo|tatar|tatamotors|target|taobao|talk|taipei|tab|systems|symantec|sydney|swiss|swiftcover|swatch|suzuki|surgery|surf|support|supply|supplies|sucks|style|study|studio|stream|store|storage|stockholm|stcgroup|stc|statoil|statefarm|statebank|starhub|star|staples|stada|srt|srl|spreadbetting|spot|sport|spiegel|space|soy|sony|song|solutions|solar|sohu|software|softbank|social|soccer|sncf|smile|smart|sling|skype|sky|skin|ski|site|singles|sina|silk|shriram|showtime|show|shouji|shopping|shop|shoes|shiksha|shia|shell|shaw|sharp|shangrila|sfr|sexy|sex|sew|seven|ses|services|sener|select|seek|security|secure|seat|search|scot|scor|scjohnson|science|schwarz|schule|school|scholarships|schmidt|schaeffler|scb|sca|sbs|sbi|saxo|save|sas|sarl|sapo|sap|sanofi|sandvikcoromant|sandvik|samsung|samsclub|salon|sale|sakura|safety|safe|saarland|ryukyu|rwe|run|ruhr|rugby|rsvp|room|rogers|rodeo|rocks|rocher|rmit|rip|rio|ril|rightathome|ricoh|richardli|rich|rexroth|reviews|review|restaurant|rest|republican|report|repair|rentals|rent|ren|reliance|reit|reisen|reise|rehab|redumbrella|redstone|red|recipes|realty|realtor|realestate|read|raid|radio|racing|qvc|quest|quebec|qpon|pwc|pub|prudential|pru|protection|property|properties|promo|progressive|prof|productions|prod|pro|prime|press|praxi|pramerica|post|porn|politie|poker|pohl|pnc|plus|plumbing|playstation|play|place|pizza|pioneer|pink|ping|pin|pid|pictures|pictet|pics|piaget|physio|photos|photography|photo|phone|philips|phd|pharmacy|pfizer|pet|pccw|pay|passagens|party|parts|partners|pars|paris|panerai|panasonic|pamperedchef|page|ovh|ott|otsuka|osaka|origins|orientexpress|organic|org|orange|oracle|open|ooo|onyourside|online|onl|ong|one|omega|ollo|oldnavy|olayangroup|olayan|okinawa|office|off|observer|obi|nyc|ntt|nrw|nra|nowtv|nowruz|now|norton|northwesternmutual|nokia|nissay|nissan|ninja|nikon|nike|nico|nhk|ngo|nfl|nexus|nextdirect|next|news|newholland|new|neustar|network|netflix|netbank|net|nec|nba|navy|natura|nationwide|name|nagoya|nadex|nab|mutuelle|mutual|museum|mtr|mtpc|mtn|msd|movistar|movie|mov|motorcycles|moto|moscow|mortgage|mormon|mopar|montblanc|monster|money|monash|mom|moi|moe|moda|mobily|mobile|mobi|mma|mls|mlb|mitsubishi|mit|mint|mini|mil|microsoft|miami|metlife|merckmsd|meo|menu|men|memorial|meme|melbourne|meet|media|med|mckinsey|mcdonalds|mcd|mba|mattel|maserati|marshalls|marriott|markets|marketing|market|map|mango|management|man|makeup|maison|maif|madrid|macys|luxury|luxe|lupin|lundbeck|ltda|ltd|lplfinancial|lpl|love|lotto|lotte|london|lol|loft|locus|locker|loans|loan|llp|llc|lixil|living|live|lipsy|link|linde|lincoln|limo|limited|lilly|like|lighting|lifestyle|lifeinsurance|life|lidl|liaison|lgbt|lexus|lego|legal|lefrak|leclerc|lease|lds|lawyer|law|latrobe|latino|lat|lasalle|lanxess|landrover|land|lancome|lancia|lancaster|lamer|lamborghini|ladbrokes|lacaixa|kyoto|kuokgroup|kred|krd|kpn|kpmg|kosher|komatsu|koeln|kiwi|kitchen|kindle|kinder|kim|kia|kfh|kerryproperties|kerrylogistics|kerryhotels|kddi|kaufen|juniper|juegos|jprs|jpmorgan|joy|jot|joburg|jobs|jnj|jmp|jll|jlc|jio|jewelry|jetzt|jeep|jcp|jcb|java|jaguar|iwc|iveco|itv|itau|istanbul|ist|ismaili|iselect|irish|ipiranga|investments|intuit|international|intel|int|insure|insurance|institute|ink|ing|info|infiniti|industries|inc|immobilien|immo|imdb|imamat|ikano|iinet|ifm|ieee|icu|ice|icbc|ibm|hyundai|hyatt|hughes|htc|hsbc|how|house|hotmail|hotels|hoteles|hot|hosting|host|hospital|horse|honeywell|honda|homesense|homes|homegoods|homedepot|holiday|holdings|hockey|hkt|hiv|hitachi|hisamitsu|hiphop|hgtv|hermes|here|helsinki|help|healthcare|health|hdfcbank|hdfc|hbo|haus|hangout|hamburg|hair|guru|guitars|guide|guge|gucci|guardian|group|grocery|gripe|green|gratis|graphics|grainger|gov|got|gop|google|goog|goodyear|goodhands|goo|golf|goldpoint|gold|godaddy|gmx|gmo|gmbh|gmail|globo|global|gle|glass|glade|giving|gives|gifts|gift|ggee|george|genting|gent|gea|gdn|gbiz|gay|garden|gap|games|game|gallup|gallo|gallery|gal|fyi|futbol|furniture|fund|fun|fujixerox|fujitsu|ftr|frontier|frontdoor|frogans|frl|fresenius|free|fox|foundation|forum|forsale|forex|ford|football|foodnetwork|food|foo|fly|flsmidth|flowers|florist|flir|flights|flickr|fitness|fit|fishing|fish|firmdale|firestone|fire|financial|finance|final|film|fido|fidelity|fiat|ferrero|ferrari|feedback|fedex|fast|fashion|farmers|farm|fans|fan|family|faith|fairwinds|fail|fage|extraspace|express|exposed|expert|exchange|everbank|events|eus|eurovision|etisalat|esurance|estate|esq|erni|ericsson|equipment|epson|epost|enterprises|engineering|engineer|energy|emerck|email|education|edu|edeka|eco|eat|earth|dvr|dvag|durban|dupont|duns|dunlop|duck|dubai|dtv|drive|download|dot|doosan|domains|doha|dog|dodge|doctor|docs|dnp|diy|dish|discover|discount|directory|direct|digital|diet|diamonds|dhl|dev|design|desi|dentist|dental|democrat|delta|deloitte|dell|delivery|degree|deals|dealer|deal|dds|dclk|day|datsun|dating|date|data|dance|dad|dabur|cyou|cymru|cuisinella|csc|cruises|cruise|crs|crown|cricket|creditunion|creditcard|credit|cpa|courses|coupons|coupon|country|corsica|coop|cool|cookingchannel|cooking|contractors|contact|consulting|construction|condos|comsec|computer|compare|company|community|commbank|comcast|com|cologne|college|coffee|codes|coach|clubmed|club|cloud|clothing|clinique|clinic|click|cleaning|claims|cityeats|city|citic|citi|citadel|cisco|circle|cipriani|church|chrysler|chrome|christmas|chloe|chintai|cheap|chat|chase|charity|channel|chanel|cfd|cfa|cern|ceo|center|ceb|cbs|cbre|cbn|cba|catholic|catering|cat|casino|cash|caseih|case|casa|cartier|cars|careers|career|care|cards|caravan|car|capitalone|capital|capetown|canon|cancerresearch|camp|camera|cam|calvinklein|call|cal|cafe|cab|bzh|buzz|buy|business|builders|build|bugatti|budapest|brussels|brother|broker|broadway|bridgestone|bradesco|box|boutique|bot|boston|bostik|bosch|boots|booking|book|boo|bond|bom|bofa|boehringer|boats|bnpparibas|bnl|bmw|bms|blue|bloomberg|blog|blockbuster|blanco|blackfriday|black|biz|bio|bingo|bing|bike|bid|bible|bharti|bet|bestbuy|best|berlin|bentley|beer|beauty|beats|bcn|bcg|bbva|bbt|bbc|bayern|bauhaus|basketball|baseball|bargains|barefoot|barclays|barclaycard|barcelona|bar|bank|band|bananarepublic|banamex|baidu|baby|azure|axa|aws|avianca|autos|auto|author|auspost|audio|audible|audi|auction|attorney|athleta|associates|asia|asda|arte|art|arpa|army|archi|aramco|arab|aquarelle|apple|app|apartments|aol|anz|anquan|android|analytics|amsterdam|amica|amfam|amex|americanfamily|americanexpress|amazon|alstom|alsace|ally|allstate|allfinanz|alipay|alibaba|alfaromeo|akdn|airtel|airforce|airbus|aigo|aig|agency|agakhan|africa|afl|afamilycompany|aetna|aero|aeg|adult|ads|adac|actor|active|aco|accountants|accountant|accenture|academy|abudhabi|abogado|able|abc|abbvie|abbott|abb|abarth|aarp|aaa|onion)\b([-a-zA-Z0-9@:%_\+.~#?&/=]*)/gi; 2const hashtagRegex = /(#|#)([a-z0-9_\u00c0-\u00d6\u00d8-\u00f6\u00f8-\u00ff\u0100-\u024f\u0253-\u0254\u0256-\u0257\u0300-\u036f\u1e00-\u1eff\u0400-\u04ff\u0500-\u0527\u2de0-\u2dff\ua640-\ua69f\u0591-\u05bf\u05c1-\u05c2\u05c4-\u05c5\u05d0-\u05ea\u05f0-\u05f4\ufb12-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb40-\ufb41\ufb43-\ufb44\ufb46-\ufb4f\u0610-\u061a\u0620-\u065f\u066e-\u06d3\u06d5-\u06dc\u06de-\u06e8\u06ea-\u06ef\u06fa-\u06fc\u0750-\u077f\u08a2-\u08ac\u08e4-\u08fe\ufb50-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe70-\ufe74\ufe76-\ufefc\u200c-\u200c\u0e01-\u0e3a\u0e40-\u0e4e\u1100-\u11ff\u3130-\u3185\ua960-\ua97f\uac00-\ud7af\ud7b0-\ud7ff\uffa1-\uffdc\u30a1-\u30fa\u30fc-\u30fe\uff66-\uff9f\uff10-\uff19\uff21-\uff3a\uff41-\uff5a\u3041-\u3096\u3099-\u309e\u3400-\u4dbf\u4e00-\u9fff\u20000-\u2a6df\u2a700-\u2b73f\u2b740-\u2b81f\u2f800-\u2fa1f]*[a-z_\u00c0-\u00d6\u00d8-\u00f6\u00f8-\u00ff\u0100-\u024f\u0253-\u0254\u0256-\u0257\u0300-\u036f\u1e00-\u1eff\u0400-\u04ff\u0500-\u0527\u2de0-\u2dff\ua640-\ua69f\u0591-\u05bf\u05c1-\u05c2\u05c4-\u05c5\u05d0-\u05ea\u05f0-\u05f4\ufb12-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb40-\ufb41\ufb43-\ufb44\ufb46-\ufb4f\u0610-\u061a\u0620-\u065f\u066e-\u06d3\u06d5-\u06dc\u06de-\u06e8\u06ea-\u06ef\u06fa-\u06fc\u0750-\u077f\u08a2-\u08ac\u08e4-\u08fe\ufb50-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe70-\ufe74\ufe76-\ufefc\u200c-\u200c\u0e01-\u0e3a\u0e40-\u0e4e\u1100-\u11ff\u3130-\u3185\ua960-\ua97f\uac00-\ud7af\ud7b0-\ud7ff\uffa1-\uffdc\u30a1-\u30fa\u30fc-\u30fe\uff66-\uff9f\uff10-\uff19\uff21-\uff3a\uff41-\uff5a\u3041-\u3096\u3099-\u309e\u3400-\u4dbf\u4e00-\u9fff\u20000-\u2a6df\u2a700-\u2b73f\u2b740-\u2b81f\u2f800-\u2fa1f][a-z0-9_\u00c0-\u00d6\u00d8-\u00f6\u00f8-\u00ff\u0100-\u024f\u0253-\u0254\u0256-\u0257\u0300-\u036f\u1e00-\u1eff\u0400-\u04ff\u0500-\u0527\u2de0-\u2dff\ua640-\ua69f\u0591-\u05bf\u05c1-\u05c2\u05c4-\u05c5\u05d0-\u05ea\u05f0-\u05f4\ufb12-\ufb28\ufb2a-\ufb36\ufb38-\ufb3c\ufb40-\ufb41\ufb43-\ufb44\ufb46-\ufb4f\u0610-\u061a\u0620-\u065f\u066e-\u06d3\u06d5-\u06dc\u06de-\u06e8\u06ea-\u06ef\u06fa-\u06fc\u0750-\u077f\u08a2-\u08ac\u08e4-\u08fe\ufb50-\ufbb1\ufbd3-\ufd3d\ufd50-\ufd8f\ufd92-\ufdc7\ufdf0-\ufdfb\ufe70-\ufe74\ufe76-\ufefc\u200c-\u200c\u0e01-\u0e3a\u0e40-\u0e4e\u1100-\u11ff\u3130-\u3185\ua960-\ua97f\uac00-\ud7af\ud7b0-\ud7ff\uffa1-\uffdc\u30a1-\u30fa\u30fc-\u30fe\uff66-\uff9f\uff10-\uff19\uff21-\uff3a\uff41-\uff5a\u3041-\u3096\u3099-\u309e\u3400-\u4dbf\u4e00-\u9fff\u20000-\u2a6df\u2a700-\u2b73f\u2b740-\u2b81f\u2f800-\u2fa1f]*)/gi; 3const rtlLanguages = ['ar', 'arc', 'dv', 'fa', 'ha', 'he', 'khw', 'ks', 'ku', 'ps', 'ur', 'yi']; 4 5function arrayBufferToBase64(buffer) { 6 let binary = ''; 7 let bytes = new Uint8Array(buffer); 8 let len = bytes.byteLength; 9 for (let i = 0; i < len; i++) { 10 binary += String.fromCharCode(bytes[i]); 11 } 12 return window.btoa(binary); 13} 14function sleep(ms) { 15 return new Promise(resolve => setTimeout(resolve, ms)); 16} 17function createModal(html, className, onclose, canclose) { 18 let modal = document.createElement('div'); 19 modal.classList.add('modal'); 20 let modal_content = document.createElement('div'); 21 modal_content.classList.add('modal-content'); 22 if(className) modal_content.classList.add(className); 23 modal_content.innerHTML = html; 24 modal.appendChild(modal_content); 25 let close = document.createElement('span'); 26 close.classList.add('modal-close'); 27 close.title = "ESC"; 28 close.innerHTML = '&times;'; 29 document.body.style.overflowY = 'hidden'; 30 function removeModal() { 31 modal.remove(); 32 let event = new Event('findActiveTweet'); 33 document.dispatchEvent(event); 34 document.removeEventListener('keydown', escapeEvent); 35 if(onclose) onclose(); 36 let modals = document.getElementsByClassName('modal'); 37 if(modals.length === 0) { 38 document.body.style.overflowY = 'auto'; 39 } 40 } 41 modal.removeModal = removeModal; 42 function escapeEvent(e) { 43 if(document.querySelector('.viewer-in')) return; 44 if(e.key === 'Escape' || (e.altKey && e.keyCode === 78)) { 45 removeModal(); 46 } 47 } 48 close.addEventListener('click', removeModal); 49 modal.addEventListener('click', e => { 50 if(e.target === modal) { 51 if(!canclose || canclose()) removeModal(); 52 } 53 }); 54 document.addEventListener('keydown', escapeEvent); 55 modal_content.appendChild(close); 56 document.body.appendChild(modal); 57 return modal; 58} 59async function handleFiles(files, mediaArray, mediaContainer) { 60 let images = []; 61 let videos = []; 62 let gifs = []; 63 for (let i = 0; i < files.length; i++) { 64 let file = files[i]; 65 if (file.type.includes('gif')) { 66 // max 15 mb 67 if (file.size > 15000000) { 68 return alert(LOC.gifs_max.message); 69 } 70 gifs.push(file); 71 } else if (file.type.includes('video')) { 72 // max 500 mb 73 if (file.size > 500000000) { 74 return alert(LOC.videos_max.message); 75 } 76 videos.push(file); 77 } else if (file.type.includes('image')) { 78 // max 5 mb 79 if (file.size > 5000000) { 80 // convert png to jpeg 81 await new Promise(resolve => { 82 let canvas = document.createElement('canvas'); 83 let ctx = canvas.getContext('2d'); 84 let img = new Image(); 85 img.onload = function () { 86 canvas.width = img.width; 87 canvas.height = img.height; 88 ctx.drawImage(img, 0, 0); 89 let dataURL = canvas.toDataURL('image/jpeg', 0.9); 90 let blobBin = atob(dataURL.split(',')[1]); 91 let array = []; 92 for (let i = 0; i < blobBin.length; i++) { 93 array.push(blobBin.charCodeAt(i)); 94 } 95 file = new Blob([new Uint8Array(array)], { type: 'image/jpeg' }); 96 resolve(); 97 }; 98 img.src = URL.createObjectURL(file); 99 }); 100 if(file.size > 5000000) { 101 return alert(LOC.images_max.message); 102 } 103 } 104 images.push(file); 105 } 106 } 107 // either up to 4 images or 1 video or 1 gif 108 if (images.length > 0) { 109 if (images.length > 4) { 110 images = images.slice(0, 4); 111 } 112 if (videos.length > 0 || gifs.length > 0) { 113 return alert(LOC.max_count.message); 114 } 115 } 116 if (videos.length > 0) { 117 if (images.length > 0 || gifs.length > 0 || videos.length > 1) { 118 return alert(LOC.max_count.message); 119 } 120 } 121 if (gifs.length > 0) { 122 if (images.length > 0 || videos.length > 0 || gifs.length > 1) { 123 return alert(LOC.max_count.message); 124 } 125 } 126 // get base64 data 127 let media = [...images, ...videos, ...gifs]; 128 let base64Data = []; 129 for (let i = 0; i < media.length; i++) { 130 let file = media[i]; 131 let reader = new FileReader(); 132 reader.readAsArrayBuffer(file); 133 reader.onload = () => { 134 base64Data.push(reader.result); 135 if (base64Data.length === media.length) { 136 while (mediaArray.length >= 4) { 137 mediaArray.pop(); 138 mediaContainer.lastChild.remove(); 139 } 140 base64Data.forEach(data => { 141 let div = document.createElement('div'); 142 let img = document.createElement('img'); 143 div.title = file.name; 144 div.id = `new-tweet-media-img-${Date.now()}${Math.random()}`.replace('.', '-'); 145 div.className = "new-tweet-media-img-div"; 146 img.className = "new-tweet-media-img"; 147 let progress = document.createElement('span'); 148 progress.hidden = true; 149 progress.className = "new-tweet-media-img-progress"; 150 let remove = document.createElement('span'); 151 remove.className = "new-tweet-media-img-remove"; 152 let alt; 153 if (!file.type.includes('video')) { 154 alt = document.createElement('span'); 155 alt.className = "new-tweet-media-img-alt"; 156 alt.innerText = "ALT"; 157 alt.addEventListener('click', () => { 158 mediaObject.alt = prompt(LOC.alt_text.message, mediaObject.alt || ''); 159 }); 160 } 161 let cw = document.createElement('span'); 162 cw.className = "new-tweet-media-img-cw"; 163 cw.innerText = "CW"; 164 cw.addEventListener('click', () => { 165 createModal(` 166 <div class="cw-modal" style="color:var(--almost-black)"> 167 <h2 class="nice-header">${LOC.content_warnings.message}</h2> 168 <br> 169 <input type="checkbox" id="cw-modal-graphic_violence"${mediaObject.cw.includes('graphic_violence') ? ' checked' : ''}> <label for="cw-modal-graphic_violence">${LOC.graphic_violence.message}</label><br> 170 <input type="checkbox" id="cw-modal-adult_content"${mediaObject.cw.includes('adult_content') ? ' checked' : ''}> <label for="cw-modal-adult_content">${LOC.adult_content.message}</label><br> 171 <input type="checkbox" id="cw-modal-other"${mediaObject.cw.includes('other') ? ' checked' : ''}> <label for="cw-modal-other">${LOC.sensitive_content.message}</label><br> 172 </div> 173 `); 174 let graphic_violence = document.getElementById('cw-modal-graphic_violence'); 175 let adult_content = document.getElementById('cw-modal-adult_content'); 176 let sensitive_content = document.getElementById('cw-modal-other'); 177 [graphic_violence, adult_content, sensitive_content].forEach(checkbox => { 178 checkbox.addEventListener('change', () => { 179 if (checkbox.checked) { 180 mediaObject.cw.push(checkbox.id.slice(9)); 181 } else { 182 let index = mediaObject.cw.indexOf(checkbox.id.slice(9)); 183 if (index > -1) { 184 mediaObject.cw.splice(index, 1); 185 } 186 } 187 }); 188 }); 189 }); 190 191 let dataBase64 = arrayBufferToBase64(data); 192 let mediaObject = { 193 div, img, 194 id: img.id, 195 data: data, 196 dataBase64: dataBase64, 197 type: file.type, 198 cw: [], 199 category: file.type.includes('gif') ? 'tweet_gif' : file.type.includes('video') ? 'tweet_video' : 'tweet_image' 200 }; 201 mediaArray.push(mediaObject); 202 img.src = file.type.includes('video') ? '' : `data:${file.type};base64,${dataBase64}`; 203 remove.addEventListener('click', () => { 204 div.remove(); 205 for (let i = mediaArray.length - 1; i >= 0; i--) { 206 let m = mediaArray[i]; 207 if (m.id === img.id) mediaArray.splice(i, 1); 208 } 209 }); 210 div.append(img, progress, remove); 211 if (!file.type.includes('video')) { 212 img.addEventListener('click', () => { 213 if(!img.src.endsWith('?name=orig') && !img.src.startsWith('data:')) { 214 img.src += '?name=orig'; 215 } 216 new Viewer(mediaContainer, { 217 transition: false 218 }); 219 }); 220 div.append(alt); 221 } else { 222 cw.style.marginLeft = '-53px'; 223 } 224 div.append(cw); 225 mediaContainer.append(div); 226 }); 227 } 228 } 229 } 230} 231let isURL = (str) => { 232 try { 233 new URL(str); 234 return true; 235 } catch (_) { 236 return false; 237 } 238} 239function handleDrop(event, mediaArray, mediaContainer) { 240 let text = event.dataTransfer.getData("Text").trim(); 241 if(text.length <= 1) { 242 event.stopPropagation(); 243 event.preventDefault(); 244 let files = event.dataTransfer.files; 245 handleFiles(files, mediaArray, mediaContainer); 246 } 247} 248function getMedia(mediaArray, mediaContainer) { 249 let input = document.createElement('input'); 250 input.type = 'file'; 251 input.multiple = true; 252 input.accept = 'image/png,image/jpeg,image/gif,video/mp4,video/mov'; 253 input.addEventListener('change', () => { 254 handleFiles(input.files, mediaArray, mediaContainer); 255 }); 256 input.click(); 257}; 258function getDMMedia(mediaArray, mediaContainer, modalElement) { 259 let input = document.createElement('input'); 260 input.type = 'file'; 261 input.accept = 'image/*'; 262 input.addEventListener('change', async () => { 263 let files = input.files; 264 let images = []; 265 let gifs = []; 266 for (let i = 0; i < files.length; i++) { 267 let file = files[i]; 268 if (file.type.includes('gif')) { 269 // max 15 mb 270 if (file.size > 15000000) { 271 return alert(LOC.gifs_max.message); 272 } 273 gifs.push(file); 274 } else if (file.type.includes('image')) { 275 // max 5 mb 276 if (file.size > 5000000) { 277 return alert(LOC.images_max.message); 278 } 279 images.push(file); 280 } 281 } 282 // get base64 data 283 let media = [...images, ...gifs]; 284 let base64Data = []; 285 for (let i = 0; i < media.length; i++) { 286 let file = media[i]; 287 let reader = new FileReader(); 288 reader.readAsArrayBuffer(file); 289 reader.onload = () => { 290 base64Data.push(reader.result); 291 if (base64Data.length === media.length) { 292 mediaContainer.innerHTML = ''; 293 while (mediaArray.length > 0) { 294 mediaArray.pop(); 295 } 296 base64Data.forEach(data => { 297 let div = document.createElement('div'); 298 let img = document.createElement('img'); 299 div.title = file.name; 300 div.id = `new-tweet-media-img-${Date.now()}${Math.random()}`.replace('.', '-'); 301 div.className = "new-tweet-media-img-div"; 302 img.className = "new-tweet-media-img"; 303 let progress = document.createElement('span'); 304 progress.hidden = true; 305 progress.className = "new-tweet-media-img-progress"; 306 let remove = document.createElement('span'); 307 remove.className = "new-tweet-media-img-remove"; 308 let dataBase64 = arrayBufferToBase64(data); 309 let mediaObject = { 310 div, img, 311 id: img.id, 312 data: data, 313 dataBase64: dataBase64, 314 type: file.type, 315 category: file.type.includes('gif') ? 'tweet_gif' : file.type.includes('video') ? 'tweet_video' : 'tweet_image' 316 }; 317 mediaArray.push(mediaObject); 318 img.src = file.type.includes('video') ? '' : `data:${file.type};base64,${dataBase64}`; 319 remove.addEventListener('click', () => { 320 div.remove(); 321 for (let i = mediaArray.length - 1; i >= 0; i--) { 322 let m = mediaArray[i]; 323 if (m.id === img.id) mediaArray.splice(i, 1); 324 } 325 }); 326 div.append(img, progress, remove); 327 if (!file.type.includes('video')) { 328 img.addEventListener('click', () => { 329 if(!img.src.endsWith('?name=orig') && !img.src.startsWith('data:')) { 330 img.src += '?name=orig'; 331 } 332 new Viewer(mediaContainer, { 333 transition: false 334 }); 335 }); 336 } 337 mediaContainer.append(div); 338 setTimeout(() => modalElement.scrollTop = modalElement.scrollHeight, 50); 339 }); 340 } 341 } 342 } 343 }); 344 input.click(); 345}; 346function timeElapsed(targetTimestamp) { 347 let currentDate = new Date(); 348 let currentTimeInms = currentDate.getTime(); 349 let targetDate = new Date(targetTimestamp); 350 let targetTimeInms = targetDate.getTime(); 351 let elapsed = Math.floor((currentTimeInms - targetTimeInms) / 1000); 352 const MonthNames = [ 353 LOC.january.message, 354 LOC.february.message, 355 LOC.march.message, 356 LOC.april.message, 357 LOC.may.message, 358 LOC.june.message, 359 LOC.july.message, 360 LOC.august.message, 361 LOC.september.message, 362 LOC.october.message, 363 LOC.november.message, 364 LOC.december.message 365 ]; 366 if (elapsed < 1) { 367 return LOC.s.message.replace('$NUMBER$', 0); 368 } 369 if (elapsed < 60) { //< 60 sec 370 return LOC.s.message.replace('$NUMBER$', elapsed); 371 } 372 if (elapsed < 3600) { //< 60 minutes 373 return LOC.m.message.replace('$NUMBER$', Math.floor(elapsed / (60))); 374 } 375 if (elapsed < 86400) { //< 24 hours 376 return LOC.h.message.replace('$NUMBER$', Math.floor(elapsed / (3600))); 377 } 378 if (elapsed < 604800) { //<7 days 379 return LOC.d.message.replace('$NUMBER$', Math.floor(elapsed / (86400))); 380 } 381 if (elapsed < 2628000) { //<1 month 382 return MonthNames[targetDate.getMonth()].replace('$NUMBER$', targetDate.getDate()); 383 } 384 return `${MonthNames[targetDate.getMonth()].replace('$NUMBER$', targetDate.getDate())}, ${targetDate.getFullYear()}`; //more than a monh 385} 386function openInNewTab(href) { 387 Object.assign(document.createElement('a'), { 388 target: '_blank', 389 rel: 'noopener noreferrer', 390 href: href, 391 }).click(); 392} 393function onVisibilityChange(callback) { 394 var visible = true; 395 396 if (!callback) { 397 throw new Error('no callback given'); 398 } 399 400 function focused() { 401 if (!visible) { 402 callback(visible = true); 403 } 404 } 405 406 function unfocused() { 407 if (visible) { 408 callback(visible = false); 409 } 410 } 411 412 // Standards: 413 if ('hidden' in document) { 414 visible = !document.hidden; 415 document.addEventListener('visibilitychange', 416 function () { (document.hidden ? unfocused : focused)() }); 417 } 418 if ('mozHidden' in document) { 419 visible = !document.mozHidden; 420 document.addEventListener('mozvisibilitychange', 421 function () { (document.mozHidden ? unfocused : focused)() }); 422 } 423 if ('webkitHidden' in document) { 424 visible = !document.webkitHidden; 425 document.addEventListener('webkitvisibilitychange', 426 function () { (document.webkitHidden ? unfocused : focused)() }); 427 } 428 if ('msHidden' in document) { 429 visible = !document.msHidden; 430 document.addEventListener('msvisibilitychange', 431 function () { (document.msHidden ? unfocused : focused)() }); 432 } 433 // IE 9 and lower: 434 if ('onfocusin' in document) { 435 document.onfocusin = focused; 436 document.onfocusout = unfocused; 437 } 438 // All others: 439 window.onpageshow = window.onfocus = focused; 440 window.onpagehide = window.onblur = unfocused; 441}; 442function escapeHTML(unsafe) { 443 return unsafe 444 .replace(/</g, "&lt;") 445 .replace(/>/g, "&gt;") 446 .replace(/"/g, "&quot;") 447 .replace(/'/g, "’"); 448} 449async function renderTweetBodyHTML(full_text, entities, display_text_range, is_quote_tweet=false) { 450 let result = "", 451 last_pos = 0, 452 index_map = {}; // {start_position: [end_position, replacer_func]} 453 hashflags = []; 454 455 if (vars.enableHashflags) { 456 hashflags = await API.discover.getHashflagsV2(); 457 } 458 459 full_text_array = Array.from(full_text); 460 461 if (is_quote_tweet) { // for quoted tweet we need only hashflags and readable urls 462 if (entities.hashtags) { 463 entities.hashtags.forEach(hashtag => { 464 let hashflag = hashflags.find(h => h.hashtag.toLowerCase() === hashtag.text.toLowerCase()); 465 index_map[hashtag.indices[0]] = [hashtag.indices[1], text => 466 `#${escapeHTML(hashtag.text)}`+ 467 `${hashflag ? `<img src="${hashflag.asset_url}" class="hashflag">` : ''}` 468 ]; 469 }); 470 } 471 472 if (entities.urls) { 473 entities.urls.forEach(url => { 474 index_map[url.indices[0]] = [url.indices[1], text => `${escapeHTML(url.display_url)}`]; 475 }); 476 } 477 } else { 478 if (entities.hashtags) { 479 entities.hashtags.forEach(hashtag => { 480 let hashflag = hashflags.find(h => h.hashtag.toLowerCase() === hashtag.text.toLowerCase()); 481 index_map[hashtag.indices[0]] = [hashtag.indices[1], text => `<a href="https://twitter.com/hashtag/${escapeHTML(hashtag.text)}">`+ 482 `#${escapeHTML(hashtag.text)}`+ 483 `${hashflag ? `<img src="${hashflag.asset_url}" class="hashflag">` : ''}`+ 484 `</a>`]; 485 }); 486 } 487 488 if (entities.symbols) { 489 entities.symbols.forEach(symbol => { 490 index_map[symbol.indices[0]] = [symbol.indices[1], text => `<a href="https://twitter.com/search?q=%24${escapeHTML(symbol.text)}">`+ 491 `$${escapeHTML(symbol.text)}`+ 492 `</a>`]; 493 }); 494 } 495 496 if (entities.urls) { 497 entities.urls.forEach(url => { 498 index_map[url.indices[0]] = [url.indices[1], text => 499 `<a href="${escapeHTML(url.expanded_url)}" title="${escapeHTML(url.expanded_url)}" target="_blank" rel="noopener noreferrer">`+ 500 `${escapeHTML(url.display_url)}</a>`]; 501 }); 502 } 503 504 if (entities.user_mentions) { 505 entities.user_mentions.forEach(user => { 506 index_map[user.indices[0]] = [user.indices[1], text => `<a href="https://twitter.com/${escapeHTML(user.screen_name)}">${escapeHTML(text)}</a>`]; 507 }); 508 } 509 } 510 511 let display_start = display_text_range !== undefined ? display_text_range[0] : 0; 512 let display_end = display_text_range !== undefined ? display_text_range[1] : full_text_array.length; 513 for (let [current_pos, _] of full_text_array.entries()) { 514 if (current_pos < display_start) { // do not render first part of message 515 last_pos = current_pos + 1; // to start copy from next symbol 516 continue; 517 } 518 if (current_pos == display_end || // reached the end of visible part 519 current_pos == full_text_array.length - 1) { // reached the end of tweet itself 520 if (display_end == full_text_array.length) current_pos++; // dirty hack to include last element of slice 521 result += escapeHTML(full_text_array.slice(last_pos, current_pos).join('')); 522 break; 523 } 524 if (current_pos > display_end) { 525 break; // do not render last part of message 526 } 527 528 if (current_pos in index_map) { 529 let [end, func] = index_map[current_pos]; 530 531 if (current_pos > last_pos) { 532 result += escapeHTML(full_text_array.slice(last_pos, current_pos).join('')); // store chunk of untouched text 533 } 534 result += func(full_text_array.slice(current_pos, end).join('')); // run replacer func on corresponding range 535 last_pos = end; 536 } 537 } 538 return result 539} 540function arrayInsert(arr, index, value) { 541 return [...arr.slice(0, index), value, ...arr.slice(index)]; 542} 543function generatePoll(tweet, tweetElement, user) { 544 let pollElement = tweetElement.getElementsByClassName('tweet-card')[0]; 545 pollElement.innerHTML = ''; 546 let poll = tweet.card.binding_values; 547 let choices = Object.keys(poll).filter(key => key.endsWith('label')).map((key, i) => ({ 548 label: poll[key].string_value, 549 count: poll[key.replace('label', 'count')] ? +poll[key.replace('label', 'count')].string_value : 0, 550 id: i+1 551 })); 552 let voteCount = choices.reduce((acc, cur) => acc + cur.count, 0); 553 if(poll.selected_choice || user.id_str === tweet.user.id_str || (poll.counts_are_final && poll.counts_are_final.boolean_value)) { 554 for(let i in choices) { 555 let choice = choices[i]; 556 if(user.id_str !== tweet.user.id_str && poll.selected_choice && choice.id === +poll.selected_choice.string_value) { 557 choice.selected = true; 558 } 559 choice.percentage = Math.round(choice.count / voteCount * 100); 560 let choiceElement = document.createElement('div'); 561 choiceElement.classList.add('choice'); 562 choiceElement.innerHTML = ` 563 <div class="choice-bg" style="width:${choice.percentage}%" data-percentage="${choice.percentage}"></div> 564 <div class="choice-label"> 565 <span>${escapeHTML(choice.label)}</span> 566 ${choice.selected ? `<span class="choice-selected"></span>` : ''} 567 </div> 568 ${isFinite(choice.percentage) ? `<div class="choice-count">${choice.count} (${choice.percentage}%)</div>` : '<div class="choice-count">0</div>'} 569 `; 570 pollElement.append(choiceElement); 571 } 572 } else { 573 for(let i in choices) { 574 let choice = choices[i]; 575 let choiceElement = document.createElement('div'); 576 choiceElement.classList.add('choice', 'choice-unselected'); 577 choiceElement.innerHTML = ` 578 <div class="choice-bg" style="width:100%"></div> 579 <div class="choice-label">${escapeHTML(choice.label)}</div> 580 `; 581 choiceElement.addEventListener('click', async () => { 582 let newCard = await API.tweet.vote(poll.api.string_value, tweet.id_str, tweet.card.url, tweet.card.name, choice.id); 583 tweet.card = newCard.card; 584 generateCard(tweet, tweetElement, user); 585 }); 586 pollElement.append(choiceElement); 587 } 588 } 589 if(tweet.card.url.startsWith('card://')) { 590 let footer = document.createElement('span'); 591 footer.classList.add('poll-footer'); 592 let endsAtMessage; 593 if(LOC.ends_at.message.includes("$DATE$")) { 594 endsAtMessage = LOC.ends_at.message.replace('$DATE$', new Date(poll.end_datetime_utc.string_value).toLocaleString()); 595 } else { 596 endsAtMessage = `${LOC.ends_at.message} ${new Date(poll.end_datetime_utc.string_value).toLocaleString()}`; 597 } 598 footer.innerHTML = `${voteCount} ${voteCount === 1 ? LOC.vote.message : LOC.votes.message}${(!poll.counts_are_final || !poll.counts_are_final.boolean_value) && poll.end_datetime_utc ? `${endsAtMessage}` : ''}`; 599 pollElement.append(footer); 600 } 601} 602function generateCard(tweet, tweetElement, user) { 603 if(!tweet.card) return; 604 if(tweet.card.name === 'promo_image_convo' || tweet.card.name === 'promo_video_convo') { 605 let vals = tweet.card.binding_values; 606 let a = document.createElement('a'); 607 a.href = vals.thank_you_url ? vals.thank_you_url.string_value : "#"; 608 a.target = '_blank'; 609 a.title = vals.thank_you_text.string_value; 610 let img = document.createElement('img'); 611 let imgValue = vals.promo_image; 612 if(!imgValue) { 613 imgValue = vals.cover_promo_image_original; 614 } 615 if(!imgValue) { 616 imgValue = vals.cover_promo_image_large; 617 } 618 if(!imgValue) { 619 return; 620 } 621 img.src = imgValue.image_value.url; 622 img.width = sizeFunctions[1](imgValue.image_value.width, imgValue.image_value.height)[0]; 623 img.height = sizeFunctions[1](imgValue.image_value.width, imgValue.image_value.height)[1]; 624 img.className = 'tweet-media-element'; 625 let ctas = []; 626 if(vals.cta_one) { 627 ctas.push([vals.cta_one, vals.cta_one_tweet]); 628 } 629 if(vals.cta_two) { 630 ctas.push([vals.cta_two, vals.cta_two_tweet]); 631 } 632 if(vals.cta_three) { 633 ctas.push([vals.cta_three, vals.cta_three_tweet]); 634 } 635 if(vals.cta_four) { 636 ctas.push([vals.cta_four, vals.cta_four_tweet]); 637 } 638 let buttonGroup = document.createElement('div'); 639 buttonGroup.classList.add('tweet-button-group'); 640 for(let b of ctas) { 641 let button = document.createElement('button'); 642 button.className = `nice-button tweet-app-button`; 643 button.innerText = `${LOC.tweet_verb.message} ${b[0].string_value}`; 644 button.addEventListener('click', async () => { 645 let modal = createModal(` 646 <p style="color:var(--almost-black);margin-top:0">${LOC.do_you_want_to_tweet.message.replace("$TWEET_TEXT$", b[1].string_value)}</p> 647 <button class="nice-button">${LOC.tweet_verb.message}</button> 648 `); 649 modal.getElementsByClassName('nice-button')[0].addEventListener('click', async () => { 650 modal.removeModal(); 651 try { 652 await API.tweet.postV2({ 653 "text": b[1].string_value, 654 "card_uri": tweet.card.url, 655 }); 656 } catch(e) { 657 console.error(e); 658 alert(String(e)); 659 } 660 }); 661 }); 662 buttonGroup.append(button); 663 } 664 a.append(img); 665 tweetElement.getElementsByClassName('tweet-card')[0].append(a); 666 tweetElement.getElementsByClassName('tweet-card')[0].append(buttonGroup); 667 } else if(tweet.card.name === "player") { 668 let iframe = document.createElement('iframe'); 669 iframe.src = tweet.card.binding_values.player_url.string_value.replace("autoplay=true", "autoplay=false"); 670 iframe.classList.add('tweet-player'); 671 iframe.width = 450; 672 iframe.height = 250; 673 tweetElement.getElementsByClassName('tweet-card')[0].innerHTML = ''; 674 tweetElement.getElementsByClassName('tweet-card')[0].append(iframe); 675 } else if(tweet.card.name === "unified_card") { 676 let uc = JSON.parse(tweet.card.binding_values.unified_card.string_value); 677 for(let cn of uc.components) { 678 let co = uc.component_objects[cn]; 679 if(co.type === "media") { 680 let media = uc.media_entities[co.data.id]; 681 let video = document.createElement('video'); 682 video.className = 'tweet-media-element tweet-media-element-one'; 683 let [w, h] = sizeFunctions[1](media.original_info.width, media.original_info.height); 684 video.width = w; 685 video.height = h; 686 video.crossOrigin = 'anonymous'; 687 video.loading = 'lazy'; 688 video.controls = true; 689 if(!media.video_info) { 690 console.log(`bug found in ${tweet.id_str}, please report this message to https://github.com/dimdenGD/OldTwitter/issues`, tweet); 691 continue; 692 }; 693 let variants = media.video_info.variants.sort((a, b) => { 694 if(!b.bitrate) return -1; 695 return b.bitrate-a.bitrate; 696 }); 697 if(typeof(vars.savePreferredQuality) !== 'boolean') { 698 chrome.storage.sync.set({ 699 savePreferredQuality: true 700 }, () => {}); 701 vars.savePreferredQuality = true; 702 } 703 if(localStorage.preferredQuality && vars.savePreferredQuality) { 704 let closestQuality = variants.filter(v => v.bitrate).reduce((prev, curr) => { 705 return (Math.abs(parseInt(curr.url.match(/\/(\d+)x/)[1]) - parseInt(localStorage.preferredQuality)) < Math.abs(parseInt(prev.url.match(/\/(\d+)x/)[1]) - parseInt(localStorage.preferredQuality)) ? curr : prev); 706 }); 707 let preferredQualityVariantIndex = variants.findIndex(v => v.url === closestQuality.url); 708 if(preferredQualityVariantIndex !== -1) { 709 let preferredQualityVariant = variants[preferredQualityVariantIndex]; 710 variants.splice(preferredQualityVariantIndex, 1); 711 variants.unshift(preferredQualityVariant); 712 } 713 } 714 for(let v in variants) { 715 let source = document.createElement('source'); 716 source.src = variants[v].url; 717 source.type = variants[v].content_type; 718 video.append(source); 719 } 720 tweetElement.getElementsByClassName('tweet-card')[0].append(video, document.createElement('br')); 721 } else if(co.type === "app_store_details") { 722 let app = uc.app_store_data[uc.destination_objects[co.data.destination].data.app_id][0]; 723 let appElement = document.createElement('div'); 724 appElement.classList.add('tweet-app-info'); 725 appElement.innerHTML = ` 726 <h3>${escapeHTML(app.title.content)}</h3> 727 <span>${escapeHTML(app.category.content)}</span> 728 <br><br> 729 `; 730 tweetElement.getElementsByClassName('tweet-card')[0].append(appElement); 731 } else if(co.type === "button_group") { 732 let buttonGroup = document.createElement('div'); 733 buttonGroup.classList.add('tweet-button-group'); 734 for(let b of co.data.buttons) { 735 let app = uc.app_store_data[uc.destination_objects[b.destination].data.app_id][0]; 736 let button = document.createElement('a'); 737 button.href = `http://play.google.com/store/apps/details?id=${app.id}`; 738 button.target = '_blank'; 739 button.className = `nice-button tweet-app-button tweet-app-button-${b.style}` 740 button.innerText = b.action[0].toUpperCase() + b.action.slice(1); 741 buttonGroup.append(button); 742 } 743 tweetElement.getElementsByClassName('tweet-card')[0].append(buttonGroup); 744 } 745 } 746 } else if(tweet.card.name === "summary" || tweet.card.name === "summary_large_image") { 747 let vals = tweet.card.binding_values; 748 let a = document.createElement('a'); 749 let url = vals.card_url.string_value; 750 if(tweet.entities && tweet.entities.urls) { 751 let urlEntity = tweet.entities.urls.find(u => u.url === url); 752 if(urlEntity) { 753 url = urlEntity.expanded_url; 754 } 755 } 756 a.target = '_blank'; 757 a.href = url; 758 a.className = 'tweet-card-link box'; 759 a.innerHTML = ` 760 ${vals.thumbnail_image ? `<img src="${vals.thumbnail_image.image_value.url}" class="tweet-card-link-thumbnail">` : ''} 761 <div class="tweet-card-link-text"> 762 ${vals.vanity_url ? `<span class="tweet-card-link-vanity">${escapeHTML(vals.vanity_url.string_value)}</span><br>` : ''} 763 ${vals.title ? `<h3 class="tweet-card-link-title">${escapeHTML(vals.title.string_value)}</h3>` : ''} 764 ${vals.description ? `<span class="tweet-card-link-description">${escapeHTML(vals.description.string_value)}</span>` : ''} 765 </div> 766 `; 767 tweetElement.getElementsByClassName('tweet-card')[0].append(a); 768 } else if(tweet.card.url.startsWith('card://')) { 769 generatePoll(tweet, tweetElement, user); 770 } 771} 772function createEmojiPicker(container, input, style = {}) { 773 let picker = new EmojiPicker({ 774 i18n: { 775 "categories": { 776 "custom": LOC.custom.message, 777 "smileys-emotion": LOC.smileys_emotion.message, 778 "people-body": LOC.people_body.message, 779 "animals-nature": LOC.animals_nature.message, 780 "food-drink": LOC.food_drink.message, 781 "travel-places": LOC.travel_places.message, 782 "activities": LOC.activities.message, 783 "objects": LOC.objects.message, 784 "symbols": LOC.symbols.message, 785 "flags": LOC.flags.message 786 }, 787 "categoriesLabel": LOC.categories.message, 788 "emojiUnsupportedMessage": LOC.unsupported_emoji.message, 789 "favoritesLabel": LOC.favorites.message, 790 "loadingMessage": LOC.loading.message, 791 "networkErrorMessage": LOC.cant_load_emoji.message, 792 "regionLabel": LOC.emoji_picker.message, 793 "searchDescription": LOC.emoji_search_description.message, 794 "searchLabel": LOC.search.message, 795 "searchResultsLabel": LOC.search_results.message, 796 "skinToneDescription": "When expanded, press up or down to select and enter to choose.", 797 "skinToneLabel": LOC.skin_tone_label.message.replace("$SKIN_TONE$", "{skinTone}"), 798 "skinTones": [ 799 "Default", 800 "Light", 801 "Medium-Light", 802 "Medium", 803 "Medium-Dark", 804 "Dark" 805 ], 806 "skinTonesLabel": LOC.skin_tones_label.message 807 } 808 }); 809 for(let i in style) { 810 picker.style[i] = style[i]; 811 } 812 picker.className = isDarkModeEnabled ? 'dark' : 'light'; 813 picker.addEventListener('emoji-click', e => { 814 let pos = input.selectionStart; 815 let text = input.value; 816 input.value = text.slice(0, pos) + e.detail.unicode + text.slice(pos); 817 input.selectionStart = pos + e.detail.unicode.length; 818 }); 819 container.append(picker); 820 821 let observer; 822 if(vars.enableTwemoji) { 823 const style = document.createElement('style'); 824 style.textContent = `.twemoji { 825 width: var(--emoji-size); 826 height: var(--emoji-size); 827 pointer-events: none; 828 }`; 829 picker.shadowRoot.appendChild(style); 830 831 observer = new MutationObserver(() => { 832 for (const emoji of picker.shadowRoot.querySelectorAll('.emoji')) { 833 // Avoid infinite loops of MutationObserver 834 if (!emoji.querySelector('.twemoji')) { 835 // Do not use default 'emoji' class name because it conflicts with emoji-picker-element's 836 twemoji.parse(emoji, { className: 'twemoji' }) 837 } 838 } 839 }) 840 observer.observe(picker.shadowRoot, { 841 subtree: true, 842 childList: true 843 }); 844 } 845 846 setTimeout(() => { 847 function oc (e) { 848 if (picker.contains(e.target)) return; 849 if(observer) { 850 observer.disconnect(); 851 } 852 picker.remove(); 853 document.removeEventListener('click', oc); 854 picker.database.close(); 855 } 856 document.addEventListener('click', oc); 857 }, 100); 858 859 return picker; 860} 861 862const RED = 0.2126; 863const GREEN = 0.7152; 864const BLUE = 0.0722; 865 866const GAMMA = 2.4; 867 868function luminance(r, g, b) { 869 const a = [r, g, b].map(v => { 870 v /= 255; 871 return v <= 0.03928 872 ? v / 12.92 873 : Math.pow((v + 0.055) / 1.055, GAMMA); 874 }); 875 return a[0] * RED + a[1] * GREEN + a[2] * BLUE; 876} 877 878function contrast(rgb1, rgb2) { 879 const lum1 = luminance(...rgb1); 880 const lum2 = luminance(...rgb2); 881 const brightest = Math.max(lum1, lum2); 882 const darkest = Math.min(lum1, lum2); 883 return (brightest + 0.05) / (darkest + 0.05); 884} 885function hex2rgb(hex) { 886 if(!hex.startsWith('#')) hex = `#${hex}`; 887 const r = parseInt(hex.slice(1, 3), 16) 888 const g = parseInt(hex.slice(3, 5), 16) 889 const b = parseInt(hex.slice(5, 7), 16) 890 return [r, g, b]; 891} 892function rgb2hex(r, g, b) { 893 return `#${r.toString(16).padStart(2, '0')}${g.toString(16).padStart(2, '0')}${b.toString(16).padStart(2, '0')}`; 894} 895function rgbToHsl(r, g, b) { 896 r /= 255; g /= 255; b /= 255; 897 let max = Math.max(r, g, b); 898 let min = Math.min(r, g, b); 899 let d = max - min; 900 let h; 901 if (d === 0) h = 0; 902 else if (max === r) h = ((((g - b) / d) % 6)+6)%6; 903 else if (max === g) h = (b - r) / d + 2; 904 else if (max === b) h = (r - g) / d + 4; 905 let l = (min + max) / 2; 906 let s = d === 0 ? 0 : d / (1 - Math.abs(2 * l - 1)); 907 return [h * 60, s, l]; 908} 909 910function hslToRgb(h, s, l) { 911 let c = (1 - Math.abs(2 * l - 1)) * s; 912 let hp = h / 60.0; 913 let x = c * (1 - Math.abs((hp % 2) - 1)); 914 let rgb1; 915 if (isNaN(h)) rgb1 = [0, 0, 0]; 916 else if (hp <= 1) rgb1 = [c, x, 0]; 917 else if (hp <= 2) rgb1 = [x, c, 0]; 918 else if (hp <= 3) rgb1 = [0, c, x]; 919 else if (hp <= 4) rgb1 = [0, x, c]; 920 else if (hp <= 5) rgb1 = [x, 0, c]; 921 else if (hp <= 6) rgb1 = [c, 0, x]; 922 let m = l - c * 0.5; 923 return [ 924 Math.round(255 * (rgb1[0] + m)), 925 Math.round(255 * (rgb1[1] + m)), 926 Math.round(255 * (rgb1[2] + m)) 927 ]; 928} 929function getBackgroundColor() { 930 let root = document.documentElement; 931 let bg_color = getComputedStyle(root).getPropertyValue('--background-color'); 932 if(bg_color === 'white') { 933 bg_color = '#ffffff'; 934 } else if(bg_color === 'black') { 935 bg_color = '#000000'; 936 } else if(bg_color.startsWith('rgb(')) { 937 let rgb = bg_color.slice(4, -1).split(',').map(v => parseInt(v)); 938 bg_color = rgb2hex(...rgb); 939 } 940 if(!bg_color) bg_color = '#ffffff'; 941 return bg_color; 942} 943function makeSeeableColor(color, bg_color = getBackgroundColor()) { 944 let bg_rgb = hex2rgb(bg_color); 945 let rgb = hex2rgb(color); 946 let c = contrast(bg_rgb, rgb); 947 let hsl = rgbToHsl(...rgb); 948 let bg_hsl = rgbToHsl(...bg_rgb); 949 if(c < 4.5) { 950 if(bg_hsl[2] > 0.7) { 951 if(hsl[2] > 0.7) { 952 hsl[2] = 0.4; 953 if(hsl[1] >= 0.1) hsl[1] -= 0.1; 954 } 955 } 956 if(bg_hsl[2] < 0.4) { 957 if(c < 2.9) { 958 if(hsl[2] <= 0.6) { 959 hsl[2] = 0.6; 960 if(hsl[1] >= 0.1) hsl[1] -= 0.1; 961 } 962 } 963 } 964 } 965 return rgb2hex(...hslToRgb(...hsl)); 966} 967 968const getLinkColors = ids => { 969 if(typeof ids === "string") ids = ids.split(","); 970 ids = [...new Set(ids)]; 971 return new Promise(async (resolve, reject) => { 972 chrome.storage.local.get(["linkColors"], async data => { 973 let linkColors = data.linkColors || {}; 974 let toFetch = []; 975 let fetched = []; 976 for(let id of ids) { 977 if(typeof linkColors[id] === "undefined") { 978 toFetch.push(id); 979 } else { 980 if(linkColors[id]) fetched.push({id, color: linkColors[id]}); 981 } 982 } 983 if(toFetch.length === 0) { 984 return resolve(fetched); 985 } 986 987 try { 988 let res = await fetch("https://dimden.dev/services/twitter_link_colors/v2/get_multiple/"+toFetch.join(",")); 989 let json = await res.json(); 990 for(let id in json) { 991 if(json[id] === 'none' || json[id] === '4595b5') { 992 continue; 993 } 994 fetched.push({id, color: json[id]}); 995 linkColors[id] = json[id]; 996 } 997 for(let id of ids) { 998 if(typeof linkColors[id] === "undefined") { 999 linkColors[id] = 0; 1000 } 1001 } 1002 let keys = Object.keys(linkColors); 1003 if(keys.length > 20000) { 1004 chrome.storage.local.set({linkColors: {}}, () => {}); 1005 } else { 1006 chrome.storage.local.set({linkColors}, () => {}); 1007 } 1008 return resolve(fetched); 1009 } catch(e) { 1010 return resolve(fetched); 1011 } 1012 }); 1013 }); 1014} 1015 1016function getOtAuthToken(cache = true) { 1017 return new Promise((resolve, reject) => { 1018 chrome.storage.local.get(['otPrivateTokens'], async data => { 1019 if(!data.otPrivateTokens) { 1020 data.otPrivateTokens = {}; 1021 } 1022 if(data.otPrivateTokens[user.id_str] && cache) { 1023 resolve(data.otPrivateTokens[user.id_str]); 1024 } else { 1025 let tokens = await fetch(`https://dimden.dev/services/twitter_link_colors/v2/request_token`, {method: 'post'}).then(r => r.json()); 1026 let tweet; 1027 try { 1028 tweet = await API.tweet.postV2({ 1029 status: `otauth=${tokens.public_token}` 1030 }); 1031 let res = await fetch(`https://dimden.dev/services/twitter_link_colors/v2/verify_token`, { 1032 method: 'POST', 1033 headers: { 1034 'Content-Type': 'application/json' 1035 }, 1036 body: JSON.stringify({ 1037 tweet, 1038 public_token: tokens.public_token, 1039 private_token: tokens.private_token 1040 }) 1041 }).then(i => i.text()); 1042 if(res === 'success') { 1043 data.otPrivateTokens[user.id_str] = tokens.private_token; 1044 chrome.storage.local.set({otPrivateTokens: data.otPrivateTokens}, () => { 1045 resolve(tokens.private_token); 1046 }); 1047 } else { 1048 console.error(res); 1049 alert(res); 1050 reject(res); 1051 } 1052 } catch(e) { 1053 console.error(e); 1054 alert(e); 1055 reject(e); 1056 } finally { 1057 API.tweet.delete(tweet.id_str).catch(e => { 1058 console.error(e); 1059 setTimeout(() => { 1060 API.tweet.delete(tweet.id_str); 1061 }, 1000); 1062 }); 1063 } 1064 } 1065 }); 1066 }); 1067} 1068 1069function isProfilePath(path) { 1070 path = path.split('?')[0].split('#')[0]; 1071 if(path.endsWith('/')) path = path.slice(0, -1); 1072 if(path.split('/').length > 2) return false; 1073 if(path.length <= 1) return false; 1074 if(['/home', '/notifications', '/messages', '/settings', '/explore', '/login', '/register', '/signin', '/signup', '/logout', '/i', '/old', '/search', '/donate'].includes(path)) return false; 1075 return /^\/[A-z-0-9-_]{1,15}$/.test(path); 1076} 1077function isSticky(el) { 1078 while(el !== document.body.parentElement) { 1079 let pos = getComputedStyle(el).position; 1080 if(pos === 'sticky' || pos === 'fixed') return true; 1081 el = el.parentElement; 1082 } 1083 return false; 1084} 1085function onVisible(element, callback) { 1086 new IntersectionObserver((entries, observer) => { 1087 entries.forEach(entry => { 1088 if(entry.intersectionRatio > 0) { 1089 callback(element); 1090 observer.disconnect(); 1091 } 1092 }); 1093 }).observe(element); 1094} 1095function updateUnfollows(res) { 1096 return new Promise(async (resolve, reject) => { 1097 let data = res[user.id_str]; 1098 let cursor = "-1"; 1099 let followers = [], following = []; 1100 1101 data.lastUpdate = Date.now(); 1102 chrome.storage.local.set({unfollows: res}); 1103 1104 while(cursor !== "0") { 1105 let data = await API.user.getFollowersIds(cursor); 1106 cursor = data.next_cursor_str; 1107 followers = followers.concat(data.ids); 1108 } 1109 cursor = "-1"; 1110 while(cursor !== "0") { 1111 let data = await API.user.getFollowingIds(cursor); 1112 cursor = data.next_cursor_str; 1113 following = following.concat(data.ids); 1114 } 1115 1116 let unfollowers = data.followers.filter(f => !followers.includes(f)); 1117 data.followers = followers; 1118 if(unfollowers.length > 0 && unfollowers.length < 100) { 1119 unfollowers = unfollowers.map(u => [u, Date.now()]); 1120 data.unfollowers = data.unfollowers.concat(unfollowers); 1121 if(data.unfollowers.length > 100) { 1122 data.unfollowers = data.unfollowers.slice(data.unfollowers.length - 100); 1123 } 1124 } 1125 let unfollowings = data.following.filter(f => !following.includes(f)); 1126 data.following = following; 1127 if(unfollowings.length > 0 && unfollowings.length < 100) { 1128 unfollowings = unfollowings.map(u => [u, Date.now()]); 1129 data.unfollowings = data.unfollowings.concat(unfollowings); 1130 if(data.unfollowings.length > 100) { 1131 data.unfollowings = data.unfollowings.slice(data.unfollowings.length - 100); 1132 } 1133 } 1134 chrome.storage.local.set({unfollows: res}, () => resolve(res)); 1135 }); 1136} 1137function getTimeZone() { 1138 let offset = new Date().getTimezoneOffset(), o = Math.abs(offset); 1139 return (offset < 0 ? "+" : "-") + ("00" + Math.floor(o / 60)).slice(-2) + ":" + ("00" + (o % 60)).slice(-2); 1140} 1141 1142const mediaClasses = [ 1143 undefined, 1144 'tweet-media-element-one', 1145 'tweet-media-element-two', 1146 'tweet-media-element-three', 1147 'tweet-media-element-two', 1148]; 1149const sizeFunctions = [ 1150 undefined, 1151 (w, h) => [w > 450 ? 450 : w < 150 ? 150 : w, h > 500 ? 500 : h < 150 ? 150 : h], 1152 (w, h) => [w > 200 ? 200 : w < 150 ? 150 : w, h > 400 ? 400 : h < 150 ? 150 : h], 1153 (w, h) => [150, h > 250 ? 250 : h < 150 ? 150 : h], 1154 // (w, h) => [w > 100 ? 100 : w, h > 150 ? 150 : h], 1155 (w, h) => [w > 200 ? 200 : w < 150 ? 150 : w, h > 400 ? 400 : h < 150 ? 150 : h], 1156]; 1157const quoteSizeFunctions = [ 1158 undefined, 1159 (w, h) => [w > 400 ? 400 : w, h > 400 ? 400 : h], 1160 (w, h) => [w > 200 ? 200 : w, h > 400 ? 400 : h], 1161 (w, h) => [w > 125 ? 125 : w, h > 200 ? 200 : h], 1162 (w, h) => [w > 100 ? 100 : w, h > 150 ? 150 : h], 1163]; 1164 1165async function renderTrends(compact = false, cache = true) { 1166 if(vars.hideTrends) return; 1167 let [trendsData, hashflags] = await Promise.allSettled([API.discover[vars.disablePersonalizedTrends ? 'getTrends' : 'getTrendsV2'](cache), API.discover.getHashflags()]); 1168 let trends = trendsData.value.modules; 1169 hashflags = hashflags.value ? hashflags.value : []; 1170 let trendsContainer = document.getElementById('trends-list'); 1171 trendsContainer.innerHTML = ''; 1172 let max = 7; 1173 if(innerHeight < 650) max = 3; 1174 trends.slice(0, max).forEach(({ trend }) => { 1175 let hashflag = hashflags.find(h => h.hashtag.toLowerCase() === trend.name.slice(1).toLowerCase()); 1176 let trendDiv = document.createElement('div'); 1177 trendDiv.className = 'trend' + (compact ? ' compact-trend' : ''); 1178 trendDiv.innerHTML = compact ? /*html*/`<a href="https://twitter.com/search?q=${escapeHTML(trend.name)}" class="trend-name">${escapeHTML(trend.name)}</a>` : /*html*/` 1179 <b> 1180 <a href="https://twitter.com/search?q=${escapeHTML(trend.name)}" class="trend-name"> 1181 ${escapeHTML(trend.name)} 1182 ${hashflag ? `<img src="${hashflag.asset_url}" class="hashflag" width="16" height="16">` : ''} 1183 </a> 1184 </b><br> 1185 <span class="trend-description">${trend.meta_description ? escapeHTML(trend.meta_description) : ''}</span> 1186 `; 1187 trendsContainer.append(trendDiv); 1188 if(vars.enableTwemoji) twemoji.parse(trendDiv); 1189 }); 1190} 1191async function renderDiscovery(cache = true) { 1192 if(vars.hideWtf) return; 1193 let discover = await API.discover.getPeople(cache); 1194 let discoverContainer = document.getElementById('wtf-list'); 1195 discoverContainer.innerHTML = ''; 1196 try { 1197 let usersData = discover.globalObjects.users; 1198 let max = 6; 1199 if(innerHeight < 700) max = 5; 1200 if(innerHeight < 650) max = 3; 1201 let usersSuggestions = discover.timeline.instructions[0].addEntries.entries[0].content.timelineModule.items.map(s => s.entryId.slice('user-'.length)).slice(0, max); // why is it so deep 1202 usersSuggestions.forEach(userId => { 1203 let userData = usersData[userId]; 1204 if (!userData) return; 1205 if(vars.twitterBlueCheckmarks && userData.ext && userData.ext.isBlueVerified && userData.ext.isBlueVerified.r && userData.ext.isBlueVerified.r.ok) { 1206 userData.verified_type = "Blue"; 1207 } 1208 if(userData.ext && userData.ext.verifiedType && userData.ext.verifiedType.r && userData.ext.verifiedType.r.ok) { 1209 userData.verified_type = userData.ext.verifiedType.r.ok; 1210 } 1211 let udiv = document.createElement('div'); 1212 udiv.className = 'wtf-user'; 1213 udiv.dataset.userId = userId; 1214 udiv.innerHTML = ` 1215 <a class="tweet-avatar-link" href="https://twitter.com/${userData.screen_name}"><img src="${`${(userData.default_profile_image && vars.useOldDefaultProfileImage) ? chrome.runtime.getURL(`images/default_profile_images/default_profile_${Number(userData.id_str) % 7}_normal.png`): userData.profile_image_url_https}`.replace("_normal", "_bigger")}" alt="${escapeHTML(userData.name)}" class="tweet-avatar" width="48" height="48"></a> 1216 <div class="tweet-header wtf-header"> 1217 <a class="tweet-header-info wtf-user-link" href="https://twitter.com/${userData.screen_name}"> 1218 <b class="tweet-header-name wtf-user-name${userData.verified || userData.verified_type ? ' user-verified' : userData.id_str === '1123203847776763904' ? ' user-verified user-verified-dimden' : ''} ${userData.verified_type === 'Government' ? 'user-verified-gray' : userData.verified_type === 'Business' ? 'user-verified-yellow' : userData.verified_type === 'Blue' ? 'user-verified-blue' : ''}">${escapeHTML(userData.name)}</b> 1219 <span class="tweet-header-handle wtf-user-handle">@${userData.screen_name}</span> 1220 </a> 1221 <br> 1222 <button class="nice-button discover-follow-btn ${userData.following ? 'following' : 'follow'}" style="position:relative;bottom: 1px;">${userData.following ? LOC.following_btn.message : LOC.follow.message}</button> 1223 </div> 1224 `; 1225 const followBtn = udiv.querySelector('.discover-follow-btn'); 1226 followBtn.addEventListener('click', async () => { 1227 if (followBtn.className.includes('following')) { 1228 try { 1229 await API.user.unfollow(userData.screen_name); 1230 } catch(e) { 1231 console.error(e); 1232 alert(e); 1233 return; 1234 } 1235 followBtn.classList.remove('following'); 1236 followBtn.classList.add('follow'); 1237 followBtn.innerText = LOC.follow.message; 1238 userData.following = false; 1239 } else { 1240 try { 1241 await API.user.follow(userData.screen_name); 1242 } catch(e) { 1243 console.error(e); 1244 alert(e); 1245 return; 1246 } 1247 followBtn.classList.add('following'); 1248 followBtn.classList.remove('follow'); 1249 followBtn.innerText = LOC.following_btn.message; 1250 userData.following = true; 1251 } 1252 chrome.storage.local.set({ 1253 discoverData: { 1254 date: Date.now(), 1255 data: discover 1256 } 1257 }, () => { }) 1258 }); 1259 discoverContainer.append(udiv); 1260 if(vars.enableTwemoji) twemoji.parse(udiv); 1261 }); 1262 } catch (e) { 1263 console.warn(e); 1264 } 1265} 1266 1267function renderMedia(t) { 1268 let html = ''; 1269 if(!t.extended_entities || !t.extended_entities.media) return ''; 1270 1271 let cws = []; 1272 1273 for(let i = 0; i < t.extended_entities.media.length; i++) { 1274 let m = t.extended_entities.media[i]; 1275 let toCensor = !vars.displaySensitiveContent && t.possibly_sensitive; 1276 if(m.sensitive_media_warning) { 1277 if(m.sensitive_media_warning.graphic_violence) { 1278 cws.push(LOC.graphic_violence.message); 1279 toCensor = !vars.uncensorGraphicViolenceAutomatically; 1280 } 1281 if(m.sensitive_media_warning.adult_content) { 1282 cws.push(LOC.adult_content.message); 1283 toCensor = !vars.uncensorAdultContentAutomatically; 1284 } 1285 if(m.sensitive_media_warning.other) { 1286 cws.push(LOC.sensitive_content.message); 1287 toCensor = !vars.uncensorSensitiveContentAutomatically; 1288 } 1289 } 1290 if(m.type === 'photo') { 1291 html += /*html*/` 1292 <img 1293 ${m.ext_alt_text ? `alt="${escapeHTML(m.ext_alt_text)}" title="${escapeHTML(m.ext_alt_text)}"` : ''} 1294 crossorigin="anonymous" 1295 width="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[0]}" 1296 height="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[1]}" 1297 loading="lazy" 1298 src="${m.media_url_https + (vars.showOriginalImages && (m.media_url_https.endsWith('.jpg') || m.media_url_https.endsWith('.png')) ? '?name=orig' : '')}" 1299 class="tweet-media-element ${mediaClasses[t.extended_entities.media.length]} ${toCensor ? 'tweet-media-element-censor' : ''}" 1300 >`; 1301 } else if(m.type === 'animated_gif') { 1302 html += /*html*/` 1303 <video 1304 ${m.ext_alt_text ? `alt="${escapeHTML(m.ext_alt_text)}" title="${escapeHTML(m.ext_alt_text)}"` : ''} 1305 crossorigin="anonymous" 1306 width="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[0]}" 1307 height="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[1]}" 1308 loop 1309 onclick="if(this.paused) this.play(); else this.pause()" 1310 ${vars.disableGifAutoplay ? '' : 'autoplay'} 1311 muted 1312 class="tweet-media-element tweet-media-gif ${mediaClasses[t.extended_entities.media.length]} ${toCensor ? 'tweet-media-element-censor' : ''}" 1313 > 1314 ${m.video_info.variants.map(v => `<source src="${v.url}" type="${v.content_type}">`).join('\n')} 1315 ${LOC.unsupported_video.message} 1316 </video> 1317 `; 1318 } else if(m.type === 'video') { 1319 html += /*html*/` 1320 <video 1321 ${m.ext_alt_text ? `alt="${escapeHTML(m.ext_alt_text)}" title="${escapeHTML(m.ext_alt_text)}"` : ''} 1322 crossorigin="anonymous" 1323 width="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[0]}" 1324 height="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[1]}" 1325 preload="none" 1326 ${t.extended_entities.media.length > 1 ? 'controls' : ''} 1327 poster="${m.media_url_https}" 1328 class="tweet-media-element ${mediaClasses[t.extended_entities.media.length]} ${toCensor ? 'tweet-media-element-censor' : ''}" 1329 > 1330 ${m.video_info.variants.map(v => `<source src="${v.url}" type="${v.content_type}">`).join('\n')} 1331 ${LOC.unsupported_video.message} 1332 </video> 1333 `; 1334 } 1335 if(i === 1 && t.extended_entities.media.length > 3) { 1336 html += '<br>'; 1337 } 1338 } 1339 1340 if(cws.length > 0) { 1341 cws = [...new Set(cws)]; 1342 cws = LOC.content_warning.message.replace('$WARNINGS$', cws.join(', ')); 1343 html += `<br><div class="tweet-media-cws">${cws}</div>`; 1344 } 1345 return html; 1346} 1347 1348async function appendUser(u, container, label) { 1349 let userElement = document.createElement('div'); 1350 userElement.classList.add('user-item'); 1351 if(vars.twitterBlueCheckmarks && u.ext && u.ext.isBlueVerified && u.ext.isBlueVerified.r && u.ext.isBlueVerified.r.ok) { 1352 u.verified_type = "Blue"; 1353 } 1354 if(u.ext && u.ext.verifiedType && u.ext.verifiedType.r && u.ext.verifiedType.r.ok) { 1355 u.verified_type = u.ext.verifiedType.r.ok; 1356 } 1357 userElement.innerHTML = ` 1358 <div> 1359 <a href="https://twitter.com/${u.screen_name}" class="user-item-link"> 1360 <img src="${(u.default_profile_image && vars.useOldDefaultProfileImage) ? chrome.runtime.getURL(`images/default_profile_images/default_profile_${Number(u.id_str) % 7}_normal.png`): u.profile_image_url_https}" alt="${u.screen_name}" class="user-item-avatar tweet-avatar" width="48" height="48"> 1361 <div class="user-item-text"> 1362 <span${u.id_str === '1123203847776763904' ? ' title="Old Twitter Layout extension developer"' : ''} class="tweet-header-name user-item-name${u.protected ? ' user-protected' : ''}${u.verified || u.verified_type ? ' user-verified' : u.id_str === '1123203847776763904' ? ' user-verified user-verified-dimden' : ''} ${u.verified_type === 'Government' ? 'user-verified-gray' : u.verified_type === 'Business' ? 'user-verified-yellow' : u.verified_type === 'Blue' ? 'user-verified-blue' : ''}">${escapeHTML(u.name)}</span><br> 1363 <span class="tweet-header-handle">@${u.screen_name}</span> 1364 ${u.followed_by ? `<span class="follows-you-label">${LOC.follows_you.message}</span>` : ''} 1365 ${label ? `<br><span class="user-item-additional">${escapeHTML(label)}</span>` : ''} 1366 </div> 1367 </a> 1368 </div> 1369 <div> 1370 <button class="user-item-btn nice-button ${u.following ? 'following' : 'follow'}">${u.following ? LOC.following_btn.message : LOC.follow.message}</button> 1371 </div> 1372 `; 1373 1374 let followButton = userElement.querySelector('.user-item-btn'); 1375 followButton.addEventListener('click', async () => { 1376 if (followButton.classList.contains('following')) { 1377 try { 1378 await API.user.unfollow(u.screen_name); 1379 } catch(e) { 1380 console.error(e); 1381 alert(e); 1382 return; 1383 } 1384 followButton.classList.remove('following'); 1385 followButton.classList.add('follow'); 1386 followButton.innerText = LOC.follow.message; 1387 } else { 1388 try { 1389 await API.user.follow(u.screen_name); 1390 } catch(e) { 1391 console.error(e); 1392 alert(e); 1393 return; 1394 } 1395 followButton.classList.remove('follow'); 1396 followButton.classList.add('following'); 1397 followButton.innerText = LOC.following_btn.message; 1398 } 1399 }); 1400 1401 container.appendChild(userElement); 1402 if(vars.enableTwemoji) twemoji.parse(userElement); 1403} 1404 1405let lastTweetErrorDate = 0; 1406async function appendTweet(t, timelineContainer, options = {}) { 1407 if(typeof t !== 'object') { 1408 console.error('Tweet is undefined', t, timelineContainer, options); 1409 return; 1410 } 1411 if(typeof t.user !== 'object') { 1412 console.error('Tweet user is undefined', t, timelineContainer, options); 1413 return; 1414 } 1415 try { 1416 if(typeof seenReplies !== 'undefined') { 1417 if(seenReplies.includes(t.id_str)) return; 1418 seenReplies.push(t.id_str); 1419 } 1420 if(typeof seenThreads !== 'undefined') { 1421 if(seenThreads.includes(t.id_str)) return; 1422 } 1423 // if(t.entities && t.entities.urls) { 1424 // let webUrl = t.entities.urls.find(u => u.expanded_url.startsWith('https://twitter.com/i/web/status/')); 1425 // if(webUrl) { 1426 // try { 1427 // let source = t.source; 1428 // t = await API.tweet.getV2(t.id_str); 1429 // t.source = source; 1430 // } catch(e) {} 1431 // } 1432 // } 1433 if(t.socialContext) { 1434 options.top = {}; 1435 if(t.socialContext.description) { 1436 options.top.text = `<a target="_blank" href="https://twitter.com/i/topics/${t.socialContext.topic_id}">${t.socialContext.name}</a>`; 1437 options.top.icon = "\uf008"; 1438 options.top.color = isDarkModeEnabled ? "#7e5eff" : "#3300FF"; 1439 } else if(t.socialContext.contextType === "Like") { 1440 options.top.text = `<${t.socialContext.landingUrl.url.split('=')[1] ? `a href="https://twitter.com/i/user/${t.socialContext.landingUrl.url.split('=')[1]}"` : 'span'}>${!vars.heartsNotStars ? t.socialContext.text.replace(' liked', ' favorited') : t.socialContext.text}</a>`; 1441 if(vars.heartsNotStars) { 1442 options.top.icon = "\uf015"; 1443 options.top.color = "rgb(249, 24, 128)"; 1444 } else { 1445 options.top.icon = "\uf001"; 1446 options.top.color = "#ffac33"; 1447 } 1448 } else if(t.socialContext.contextType === "Follow") { 1449 options.top.text = t.socialContext.text; 1450 options.top.icon = "\uf002"; 1451 options.top.color = isDarkModeEnabled ? "#7e5eff" : "#3300FF"; 1452 } else if(t.socialContext.contextType === "Conversation") { 1453 options.top.text = t.socialContext.text; 1454 options.top.icon = "\uf005"; 1455 options.top.color = isDarkModeEnabled ? "#7e5eff" : "#3300FF"; 1456 } else { 1457 console.log(t.socialContext); 1458 } 1459 } 1460 if(vars.twitterBlueCheckmarks && t.user.ext && t.user.ext.isBlueVerified && t.user.ext.isBlueVerified.r && t.user.ext.isBlueVerified.r.ok) { 1461 t.user.verified_type = "Blue"; 1462 } 1463 if(t.user && t.user.ext && t.user.ext.verifiedType && t.user.ext.verifiedType.r && t.user.ext.verifiedType.r.ok) { 1464 t.user.verified_type = t.user.ext.verifiedType.r.ok; 1465 } 1466 if(typeof tweets !== 'undefined') tweets.push(['tweet', t, options]); 1467 const tweet = document.createElement('div'); 1468 t.element = tweet; 1469 t.options = options; 1470 1471 if(!options.mainTweet && typeof mainTweetLikers !== 'undefined' && !location.pathname.includes("retweets/with_comments")) { 1472 tweet.addEventListener('click', async e => { 1473 if(e.target.className && (e.target.className.startsWith('tweet tweet-id-') || e.target.classList.contains('tweet-body') || e.target.classList.contains('tweet-reply-to') || e.target.className === 'tweet-interact')) { 1474 document.getElementById('loading-box').hidden = false; 1475 savePageData(); 1476 history.pushState({}, null, `https://twitter.com/${t.user.screen_name}/status/${t.id_str}`); 1477 updateSubpage(); 1478 mediaToUpload = []; 1479 linkColors = {}; 1480 cursor = undefined; 1481 seenReplies = []; 1482 mainTweetLikers = []; 1483 let restored = await restorePageData(); 1484 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 1485 if(subpage === 'tweet' && !restored) { 1486 updateReplies(id); 1487 } else if(subpage === 'likes') { 1488 updateLikes(id); 1489 } else if(subpage === 'retweets') { 1490 updateRetweets(id); 1491 } else if(subpage === 'retweets_with_comments') { 1492 updateRetweetsWithComments(id); 1493 } 1494 renderDiscovery(); 1495 renderTrends(); 1496 currentLocation = location.pathname; 1497 } 1498 }); 1499 tweet.addEventListener('mousedown', e => { 1500 if(e.button === 1) { 1501 e.preventDefault(); 1502 if(e.target.className && (e.target.className.startsWith('tweet tweet-id-') || e.target.classList.contains('tweet-body') || e.target.classList.contains('tweet-reply-to') || e.target.className === 'tweet-interact')) { 1503 openInNewTab(`https://twitter.com/${t.user.screen_name}/status/${t.id_str}`); 1504 } 1505 } 1506 }); 1507 } else { 1508 if(!options.mainTweet) { 1509 tweet.addEventListener('click', e => { 1510 if(e.target.className && (e.target.className.startsWith('tweet tweet-id-') || e.target.classList.contains('tweet-body') || e.target.classList.contains('tweet-reply-to') || e.target.className === 'tweet-interact')) { 1511 let tweetData = t; 1512 if(tweetData.retweeted_status) tweetData = tweetData.retweeted_status; 1513 tweet.classList.add('tweet-preload'); 1514 new TweetViewer(user, tweetData); 1515 } 1516 }); 1517 tweet.addEventListener('mousedown', e => { 1518 if(e.button === 1) { 1519 e.preventDefault(); 1520 if(e.target.className && (e.target.className.startsWith('tweet tweet-id-') || e.target.classList.contains('tweet-body') || e.target.classList.contains('tweet-reply-to') || e.target.className === 'tweet-interact')) { 1521 openInNewTab(`https://twitter.com/${t.user.screen_name}/status/${t.id_str}`); 1522 } 1523 } 1524 }); 1525 } 1526 } 1527 tweet.tabIndex = -1; 1528 tweet.className = `tweet tweet-id-${t.id_str} ${options.mainTweet ? 'tweet-main' : location.pathname.includes('/status/') ? 'tweet-replying' : ''}`; 1529 tweet.dataset.tweetId = t.id_str; 1530 tweet.dataset.userId = t.user.id_str; 1531 try { 1532 if(!activeTweet) { 1533 tweet.classList.add('tweet-active'); 1534 activeTweet = tweet; 1535 } 1536 } catch(e) {}; 1537 1538 if(t.nonReply) { 1539 tweet.classList.add('tweet-non-reply'); 1540 } 1541 1542 if(t.threadContinuation) { 1543 options.threadContinuation = true; 1544 } 1545 if(t.noTop) { 1546 options.noTop = true; 1547 } 1548 if (options.threadContinuation) tweet.classList.add('tweet-self-thread-continuation'); 1549 if (options.selfThreadContinuation) tweet.classList.add('tweet-self-thread-continuation'); 1550 1551 if (options.noTop) tweet.classList.add('tweet-no-top'); 1552 if(vars.linkColorsInTL && typeof linkColors !== 'undefined') { 1553 if(linkColors[t.user.id_str]) { 1554 let sc = makeSeeableColor(linkColors[t.user.id_str]); 1555 tweet.style.setProperty('--link-color', sc); 1556 } else { 1557 if(t.user.profile_link_color && t.user.profile_link_color !== '1DA1F2') { 1558 let sc = makeSeeableColor(t.user.profile_link_color); 1559 tweet.style.setProperty('--link-color', sc); 1560 } 1561 } 1562 } 1563 let full_text = t.full_text ? t.full_text : ''; 1564 let strippedDownText = full_text 1565 .replace(/(?:https?|ftp):\/\/[\n\S]+/g, '') //links 1566 .replace(/(?<!\w)@([\w+]{1,15}\b)/g, '') //mentions 1567 .replace(/[\p{Extended_Pictographic}]/gu, '') //emojis (including ones that arent colored) 1568 .replace(/[\u200B-\u200D\uFE0E\uFE0F]/g, '') //sometimes emojis leave these behind 1569 .replace(/\d+/g, '') //numbers 1570 .trim(); 1571 let detectedLanguage = strippedDownText.length < 1 ? {languages:[{language:LANGUAGE, percentage:100}]} : await chrome.i18n.detectLanguage(strippedDownText); 1572 if(!detectedLanguage.languages[0]) detectedLanguage = {languages:[{language:t.lang, percentage:100}]}; //fallback to what twitter says 1573 let isEnglish = detectedLanguage.languages[0] && detectedLanguage.languages[0].percentage > 60 && detectedLanguage.languages[0].language.startsWith(LANGUAGE); 1574 let videos = t.extended_entities && t.extended_entities.media && t.extended_entities.media.filter(m => m.type === 'video'); 1575 if(!videos || videos.length === 0) { 1576 videos = undefined; 1577 } 1578 if(videos) { 1579 for(let v of videos) { 1580 if(!v.video_info) continue; 1581 v.video_info.variants = v.video_info.variants.sort((a, b) => { 1582 if(!b.bitrate) return -1; 1583 return b.bitrate-a.bitrate; 1584 }); 1585 if(typeof(vars.savePreferredQuality) !== 'boolean') { 1586 chrome.storage.sync.set({ 1587 savePreferredQuality: true 1588 }, () => {}); 1589 vars.savePreferredQuality = true; 1590 } 1591 if(localStorage.preferredQuality && vars.savePreferredQuality) { 1592 let closestQuality = v.video_info.variants.filter(v => v.bitrate).reduce((prev, curr) => { 1593 return (Math.abs(parseInt(curr.url.match(/\/(\d+)x/)[1]) - parseInt(localStorage.preferredQuality)) < Math.abs(parseInt(prev.url.match(/\/(\d+)x/)[1]) - parseInt(localStorage.preferredQuality)) ? curr : prev); 1594 }); 1595 let preferredQualityVariantIndex = v.video_info.variants.findIndex(v => v.url === closestQuality.url); 1596 if(preferredQualityVariantIndex !== -1) { 1597 let preferredQualityVariant = v.video_info.variants[preferredQualityVariantIndex]; 1598 v.video_info.variants.splice(preferredQualityVariantIndex, 1); 1599 v.video_info.variants.unshift(preferredQualityVariant); 1600 } 1601 } 1602 } 1603 } 1604 if(full_text.includes("Learn more")) { 1605 console.log(t); 1606 } 1607 if(t.withheld_in_countries && (t.withheld_in_countries.includes("XX") || t.withheld_in_countries.includes("XY"))) { 1608 full_text = ""; 1609 } 1610 if(t.quoted_status_id_str && !t.quoted_status && options.mainTweet) { //t.quoted_status is undefined if the user blocked the quoter (this also applies to deleted/private tweets too, but it just results in original behavior then) 1611 try { 1612 if(t.quoted_status_result && t.quoted_status_result.result.tweet) { 1613 t.quoted_status = t.quoted_status_result.result.tweet.legacy; 1614 t.quoted_status.user = t.quoted_status_result.result.tweet.core.user_results.result.legacy; 1615 } else { 1616 t.quoted_status = await API.tweet.getV2(t.quoted_status_id_str); 1617 } 1618 } catch { 1619 t.quoted_status = undefined; 1620 } 1621 } 1622 let followUserText, unfollowUserText, blockUserText, unblockUserText; 1623 let mentionedUserText = ``; 1624 let quoteMentionedUserText = ``; 1625 if( 1626 LOC.follow_user.message.includes('$SCREEN_NAME$') && LOC.unfollow_user.message.includes('$SCREEN_NAME$') && 1627 LOC.block_user.message.includes('$SCREEN_NAME$') && LOC.unblock_user.message.includes('$SCREEN_NAME$') 1628 ) { 1629 followUserText = `${LOC.follow_user.message.replace('$SCREEN_NAME$', t.user.screen_name)}`; 1630 unfollowUserText = `${LOC.unfollow_user.message.replace('$SCREEN_NAME$', t.user.screen_name)}`; 1631 blockUserText = `${LOC.block_user.message.replace('$SCREEN_NAME$', t.user.screen_name)}`; 1632 unblockUserText = `${LOC.unblock_user.message.replace('$SCREEN_NAME$', t.user.screen_name)}`; 1633 } else { 1634 followUserText = `${LOC.follow_user.message} @${t.user.screen_name}`; 1635 unfollowUserText = `${LOC.unfollow_user.message} @${t.user.screen_name}`; 1636 blockUserText = `${LOC.block_user.message} @${t.user.screen_name}`; 1637 unblockUserText = `${LOC.unblock_user.message} @${t.user.screen_name}`; 1638 } 1639 if(t.in_reply_to_screen_name && t.display_text_range) { 1640 t.entities.user_mentions.forEach(user_mention => { 1641 if(user_mention.indices[0] < t.display_text_range[0]){ 1642 mentionedUserText += `<a href="https://twitter.com/${user_mention.screen_name}">@${user_mention.screen_name}</a> ` 1643 } 1644 //else this is not reply but mention 1645 }); 1646 } 1647 if(t.quoted_status && t.quoted_status.in_reply_to_screen_name && t.display_text_range) { 1648 t.quoted_status.entities.user_mentions.forEach(user_mention => { 1649 if(user_mention.indices[0] < t.display_text_range[0]){ 1650 quoteMentionedUserText += `@${user_mention.screen_name} ` 1651 } 1652 //else this is not reply but mention 1653 }); 1654 } 1655 // i fucking hate this thing 1656 tweet.innerHTML = /*html*/` 1657 <div class="tweet-top" hidden></div> 1658 <a class="tweet-avatar-link" href="https://twitter.com/${t.user.screen_name}"> 1659 <img 1660 onerror="this.src = '${vars.useOldDefaultProfileImage ? chrome.runtime.getURL(`images/default_profile_images/default_profile_bigger.png`) : 'https://abs.twimg.com/sticky/default_profile_images/default_profile_bigger.png'}'" 1661 src="${`${(t.user.default_profile_image && vars.useOldDefaultProfileImage) ? chrome.runtime.getURL(`images/default_profile_images/default_profile_${Number(t.user.id_str) % 7}_normal.png`) : t.user.profile_image_url_https}`.replace("_normal.", "_bigger.")}" 1662 alt="${t.user.name}" 1663 class="tweet-avatar" 1664 width="48" 1665 height="48" 1666 > 1667 </a> 1668 <div class="tweet-header ${options.mainTweet ? 'tweet-header-main' : ''}"> 1669 <a class="tweet-header-info ${options.mainTweet ? 'tweet-header-info-main' : ''}" href="https://twitter.com/${t.user.screen_name}"> 1670 <b 1671 ${t.user.id_str === '1123203847776763904' ? 'title="Old Twitter Layout extension developer" ' : ''} 1672 class="tweet-header-name ${options.mainTweet ? 'tweet-header-name-main' : ''} ${t.user.verified || t.user.verified_type ? 'user-verified' : t.user.id_str === '1123203847776763904' ? 'user-verified user-verified-dimden' : ''} ${t.user.protected ? 'user-protected' : ''} ${t.user.verified_type === 'Government' ? 'user-verified-gray' : t.user.verified_type === 'Business' ? 'user-verified-yellow' : t.user.verified_type === 'Blue' ? 'user-verified-blue' : ''}" 1673 >${escapeHTML(t.user.name)}</b> 1674 <span class="tweet-header-handle">@${t.user.screen_name}</span> 1675 </a> 1676 <a ${options.mainTweet ? 'hidden' : ''} class="tweet-time" data-timestamp="${new Date(t.created_at).getTime()}" title="${new Date(t.created_at).toLocaleString()}" href="https://twitter.com/${t.user.screen_name}/status/${t.id_str}">${timeElapsed(new Date(t.created_at).getTime())}</a> 1677 ${location.pathname.split("?")[0].split("#")[0] === '/i/bookmarks' ? `<span class="tweet-delete-bookmark${!isEnglish ? ' tweet-delete-bookmark-lower' : ''}">&times;</span>` : ''} 1678 ${options.mainTweet && t.user.id_str !== user.id_str ? `<button class='nice-button tweet-header-follow ${t.user.following ? 'following' : 'follow'}'>${t.user.following ? LOC.following_btn.message : LOC.follow.message}</button>` : ''} 1679 ${!options.mainTweet && !isEnglish ? `<span class="tweet-translate-after">${`${t.user.name} ${t.user.screen_name} 1 Sept`.length < 40 ? LOC.view_translation.message : ''}</span>` : ''} 1680 </div> 1681 ${mentionedUserText !== `` && 1682 !options.threadContinuation && 1683 !options.noTop && 1684 !location.pathname.includes('/status/') && 1685 !vars.useOldStyleReply ? /*html*/` 1686 <div class="tweet-reply-to"><span>${LOC.replying_to_user.message.replace('$SCREEN_NAME$', mentionedUserText.trim().replaceAll(`> <`, `>${LOC.replying_to_comma.message}<`).replace(`>${LOC.replying_to_comma.message}<`, `>${LOC.replying_to_and.message}<`))}</span></div> 1687 `: ''} 1688 <div class="tweet-body ${options.mainTweet ? 'tweet-body-main' : ''}"> 1689 <span class="tweet-body-text ${vars.noBigFont || t.full_text.length > 280 || !options.bigFont || (!options.mainTweet && location.pathname.includes('/status/')) ? 'tweet-body-text-long' : 'tweet-body-text-short'}">${vars.useOldStyleReply ? /*html*/mentionedUserText: ''}${full_text ? await renderTweetBodyHTML(full_text, t.entities, t.display_text_range) : ''}</span> 1690 ${!isEnglish && options.mainTweet ? /*html*/` 1691 <br> 1692 <span class="tweet-translate">${LOC.view_translation.message}</span> 1693 ` : ``} 1694 ${t.extended_entities && t.extended_entities.media ? /*html*/` 1695 <div class="tweet-media"> 1696 ${t.extended_entities.media.length === 1 && t.extended_entities.media[0].type === 'video' ? /*html*/` 1697 <div class="tweet-media-video-overlay"> 1698 <svg viewBox="0 0 24 24" class="tweet-media-video-overlay-play"> 1699 <g> 1700 <path class="svg-play-path" d="M8 5v14l11-7z"></path> 1701 <path d="M0 0h24v24H0z" fill="none"></path> 1702 </g> 1703 </svg> 1704 </div> 1705 ` : ''} 1706 ${renderMedia(t)} 1707 </div> 1708 ${t.extended_entities && t.extended_entities.media && t.extended_entities.media.some(m => m.type === 'animated_gif') ? /*html*/`<div class="tweet-media-controls">GIF</div>` : ''} 1709 ${videos ? /*html*/` 1710 <div class="tweet-media-controls"> 1711 ${videos[0].ext && videos[0].ext.mediaStats && videos[0].ext.mediaStats.r && videos[0].ext.mediaStats.r.ok ? `<span class="tweet-video-views">${Number(videos[0].ext.mediaStats.r.ok.viewCount).toLocaleString().replace(/\s/g, ',')} ${LOC.views.message}</span> • ` : ''}<span class="tweet-video-reload">${LOC.reload.message}</span> • 1712 ${videos[0].video_info.variants.filter(v => v.bitrate).map(v => `<span class="tweet-video-quality" data-url="${v.url}">${v.url.match(/\/(\d+)x/)[1] + 'p'}</span> `).join(" / ")} 1713 </div> 1714 ` : ``} 1715 <span class="tweet-media-data"></span> 1716 ` : ``} 1717 ${t.card ? `<div class="tweet-card"></div>` : ''} 1718 ${t.quoted_status ? /*html*/` 1719 <a class="tweet-body-quote" target="_blank" href="https://twitter.com/${t.quoted_status.user.screen_name}/status/${t.quoted_status.id_str}"> 1720 <img src="${(t.quoted_status.user.default_profile_image && vars.useOldDefaultProfileImage) ? chrome.runtime.getURL(`images/default_profile_images/default_profile_${Number(t.quoted_status.user.id_str) % 7}_normal.png`): t.quoted_status.user.profile_image_url_https}" alt="${escapeHTML(t.quoted_status.user.name)}" class="tweet-avatar-quote" width="24" height="24"> 1721 <div class="tweet-header-quote"> 1722 <span class="tweet-header-info-quote"> 1723 <b class="tweet-header-name-quote ${t.quoted_status.user.verified ? 'user-verified' : t.quoted_status.user.id_str === '1123203847776763904' ? 'user-verified user-verified-dimden' : ''} ${t.quoted_status.user.protected ? 'user-protected' : ''}">${escapeHTML(t.quoted_status.user.name)}</b> 1724 <span class="tweet-header-handle-quote">@${t.quoted_status.user.screen_name}</span> 1725 </span> 1726 </div> 1727 <span class="tweet-time-quote" data-timestamp="${new Date(t.quoted_status.created_at).getTime()}" title="${new Date(t.quoted_status.created_at).toLocaleString()}">${timeElapsed(new Date(t.quoted_status.created_at).getTime())}</span> 1728 ${quoteMentionedUserText !== `` && !vars.useOldStyleReply ? /*html*/` 1729 <span class="tweet-reply-to tweet-quote-reply-to">${LOC.replying_to_user.message.replace('$SCREEN_NAME$', quoteMentionedUserText.trim().replaceAll(` `,LOC.replying_to_comma.message).replace(LOC.replying_to_comma.message,LOC.replying_to_and.message))}</span> 1730 ` : ''} 1731 <span class="tweet-body-text tweet-body-text-quote tweet-body-text-long" style="color:var(--default-text-color)!important">${vars.useOldStyleReply? quoteMentionedUserText: ''}${t.quoted_status.full_text ? await renderTweetBodyHTML(t.quoted_status.full_text, t.quoted_status.entities, t.quoted_status.display_text_range, true) : ''}</span> 1732 ${t.quoted_status.extended_entities && t.quoted_status.extended_entities.media ? ` 1733 <div class="tweet-media-quote"> 1734 ${t.quoted_status.extended_entities.media.map(m => `<${m.type === 'photo' ? 'img' : 'video'} ${m.ext_alt_text ? `alt="${escapeHTML(m.ext_alt_text)}" title="${escapeHTML(m.ext_alt_text)}"` : ''} crossorigin="anonymous" width="${quoteSizeFunctions[t.quoted_status.extended_entities.media.length](m.original_info.width, m.original_info.height)[0]}" height="${quoteSizeFunctions[t.quoted_status.extended_entities.media.length](m.original_info.width, m.original_info.height)[1]}" loading="lazy" ${m.type === 'video' ? 'controls' : ''} ${m.type === 'animated_gif' ? 'loop muted onclick="if(this.paused) this.play(); else this.pause()"' : ''}${m.type === 'animated_gif' && !vars.disableGifAutoplay ? ' autoplay' : ''} src="${m.type === 'photo' ? m.media_url_https : m.video_info.variants.find(v => v.content_type === 'video/mp4').url}" class="tweet-media-element tweet-media-element-quote ${mediaClasses[t.quoted_status.extended_entities.media.length]} ${!vars.displaySensitiveContent && t.quoted_status.possibly_sensitive ? 'tweet-media-element-censor' : ''}">${m.type === 'video' ? '</video>' : ''}`).join('\n')} 1735 </div> 1736 ` : ''} 1737 </a> 1738 ` : ``} 1739 ${t.limited_actions === 'limit_trusted_friends_tweet' && (options.mainTweet || !location.pathname.includes('/status/')) ? /*html*/` 1740 <div class="tweet-limited"> 1741 ${LOC.circle_limited_tweet.message} 1742 <a href="https://help.twitter.com/en/using-twitter/twitter-circle" target="_blank">${LOC.learn_more.message}</a> 1743 </div> 1744 `.replace('$SCREEN_NAME$', tweetStorage[t.conversation_id_str] ? tweetStorage[t.conversation_id_str].user.screen_name : t.in_reply_to_screen_name ? t.in_reply_to_screen_name : t.user.screen_name) : ''} 1745 ${t.tombstone ? `<div class="tweet-warning">${t.tombstone}</div>` : ''} 1746 ${((t.withheld_in_countries && (t.withheld_in_countries.includes("XX") || t.withheld_in_countries.includes("XY"))) || t.withheld_scope) ? `<div class="tweet-warning">This Tweet has been withheld in response to a report from the copyright holder. <a href="https://help.twitter.com/en/rules-and-policies/copyright-policy" target="_blank">Learn more.</a></div>` : ''} 1747 ${t.conversation_control ? `<div class="tweet-warning">${t.limited_actions_text ? t.limited_actions_text : LOC.limited_tweet.message}${t.conversation_control.policy && (t.user.id_str === user.id_str || (t.conversation_control.policy.toLowerCase() === 'community' && (t.user.followed_by || (full_text && full_text.includes(`@${user.screen_name}`)))) || (t.conversation_control.policy.toLowerCase() === 'by_invitation' && full_text && full_text.includes(`@${user.screen_name}`))) ? ' ' + LOC.you_can_reply.message : ''}.</div>` : ''} 1748 ${options.mainTweet ? /*html*/` 1749 <div class="tweet-footer"> 1750 <div class="tweet-footer-stats"> 1751 <a href="https://twitter.com/${t.user.screen_name}/status/${t.id_str}" class="tweet-footer-stat tweet-footer-stat-o"> 1752 <span class="tweet-footer-stat-text">${LOC.replies.message}</span> 1753 <b class="tweet-footer-stat-count tweet-footer-stat-replies">${Number(t.reply_count).toLocaleString().replace(/\s/g, ',')}</b> 1754 </a> 1755 <a href="https://twitter.com/${t.user.screen_name}/status/${t.id_str}/retweets" class="tweet-footer-stat tweet-footer-stat-r"> 1756 <span class="tweet-footer-stat-text">${LOC.retweets.message}</span> 1757 <b class="tweet-footer-stat-count tweet-footer-stat-retweets">${Number(t.retweet_count).toLocaleString().replace(/\s/g, ',')}</b> 1758 </a> 1759 <a href="https://twitter.com/${t.user.screen_name}/status/${t.id_str}/likes" class="tweet-footer-stat tweet-footer-stat-f"> 1760 <span class="tweet-footer-stat-text">${vars.heartsNotStars ? LOC.likes.message : LOC.favorites.message}</span> 1761 <b class="tweet-footer-stat-count tweet-footer-stat-favorites">${Number(t.favorite_count).toLocaleString().replace(/\s/g, ',')}</b> 1762 </a> 1763 </div> 1764 <div class="tweet-footer-favorites"></div> 1765 </div> 1766 ` : ''} 1767 <a ${!options.mainTweet ? 'hidden' : ''} class="tweet-date" title="${new Date(t.created_at).toLocaleString()}" href="https://twitter.com/${t.user.screen_name}/status/${t.id_str}"><br>${new Date(t.created_at).toLocaleTimeString(undefined, { hour: 'numeric', minute: 'numeric' }).toLowerCase()} - ${new Date(t.created_at).toLocaleDateString(undefined, { day: 'numeric', month: 'short', year: 'numeric' })}  ・ ${t.source ? t.source.split('>')[1].split('<')[0] : 'Unknown'}</a> 1768 <div class="tweet-interact"> 1769 <span class="tweet-interact-reply" title="${LOC.reply_btn.message}${!vars.disableHotkeys ? ' (R)' : ''}" data-val="${t.reply_count}">${options.mainTweet ? '' : Number(t.reply_count).toLocaleString().replace(/\s/g, ',')}</span> 1770 <span title="${LOC.retweet_btn.message}" class="tweet-interact-retweet${t.retweeted ? ' tweet-interact-retweeted' : ''}${(t.user.protected || t.limited_actions === 'limit_trusted_friends_tweet') && t.user.id_str !== user.id_str ? ' tweet-interact-retweet-disabled' : ''}" data-val="${t.retweet_count}">${options.mainTweet ? '' : Number(t.retweet_count).toLocaleString().replace(/\s/g, ',')}</span> 1771 <div class="tweet-interact-retweet-menu dropdown-menu" hidden> 1772 <span class="tweet-interact-retweet-menu-retweet">${t.retweeted ? LOC.unretweet.message : LOC.retweet.message}${!vars.disableHotkeys ? ' (T)' : ''}</span> 1773 <span class="tweet-interact-retweet-menu-quote">${LOC.quote_tweet.message}${!vars.disableHotkeys ? ' (Q)' : ''}</span> 1774 ${options.mainTweet ? /*html*/` 1775 <span class="tweet-interact-retweet-menu-quotes">${LOC.see_quotes_big.message}</span> 1776 <span class="tweet-interact-retweet-menu-retweeters">${LOC.see_retweeters.message}</span> 1777 ` : ''} 1778 </div> 1779 <span title="${vars.heartsNotStars ? LOC.like_btn.message : LOC.favorite_btn.message}${!vars.disableHotkeys ? ' (L)' : ''}" class="tweet-interact-favorite ${t.favorited ? 'tweet-interact-favorited' : ''}" data-val="${t.favorite_count}">${options.mainTweet ? '' : Number(t.favorite_count).toLocaleString().replace(/\s/g, ',')}</span> 1780 ${vars.seeTweetViews && t.ext && t.ext.views && t.ext.views.r && t.ext.views.r.ok && t.ext.views.r.ok.count ? /*html*/`<span title="${LOC.views_count.message}" class="tweet-interact-views" data-val="${t.ext.views.r.ok.count}">${Number(t.ext.views.r.ok.count).toLocaleString().replace(/\s/g, ',')}</span>` : ''} 1781 ${t.bookmark_count && vars.showBookmarkCount && options.mainTweet ? 1782 /*html*/`<span title="${LOC.bookmarks_count.message}" class="tweet-interact-bookmark${t.bookmarked ? ' tweet-interact-bookmarked' : ''}" data-val="${t.bookmark_count}">${Number(t.bookmark_count).toLocaleString().replace(/\s/g, ',')}</span>` : 1783 ''} 1784 <span class="tweet-interact-more"></span> 1785 <div class="tweet-interact-more-menu dropdown-menu" hidden> 1786 <span class="tweet-interact-more-menu-copy">${LOC.copy_link.message}</span> 1787 <span class="tweet-interact-more-menu-embed">${LOC.embed_tweet.message}</span> 1788 ${navigator.canShare ? `<span class="tweet-interact-more-menu-share">${LOC.share_tweet.message}</span>` : ''} 1789 <span class="tweet-interact-more-menu-newtwitter">${LOC.open_tweet_newtwitter.message}</span> 1790 ${t.user.id_str === user.id_str ? /*html*/` 1791 <hr> 1792 <span class="tweet-interact-more-menu-analytics">${LOC.tweet_analytics.message}</span> 1793 <span class="tweet-interact-more-menu-delete">${LOC.delete_tweet.message}</span> 1794 ${typeof pageUser !== 'undefined' && pageUser.id_str === user.id_str ? /*html*/`<span class="tweet-interact-more-menu-pin">${pinnedTweet && pinnedTweet.id_str === t.id_str ? LOC.unpin_tweet.message : LOC.pin_tweet.message}</span>` : ''} 1795 ` : ''} 1796 ${t.conversation_id_str && tweetStorage[t.conversation_id_str] && tweetStorage[t.conversation_id_str].user.id_str === user.id_str && t.user.id_str !== user.id_str ? /*html*/` 1797 <span class="tweet-interact-more-menu-hide">${t.moderated ? LOC.unhide_tweet.message : LOC.hide_tweet.message}</span> 1798 `: ''} 1799 ${t.hasModeratedReplies ? /*html*/` 1800 <span class="tweet-interact-more-menu-hidden"><a target="_blank" href="/${t.user.screen_name}/status/${t.id_str}/hidden?newtwitter=true">${LOC.see_hidden_replies.message}</a></span> 1801 ` : ''} 1802 <hr> 1803 ${t.user.id_str !== user.id_str && !options.mainTweet ? /*html*/` 1804 <span class="tweet-interact-more-menu-follow"${t.user.blocking ? ' hidden' : ''}>${t.user.following ? unfollowUserText : followUserText}</span> 1805 ` : ''} 1806 ${t.user.id_str !== user.id_str ? /*html*/` 1807 <span class="tweet-interact-more-menu-block">${t.user.blocking ? unblockUserText : blockUserText}</span> 1808 ` : ''} 1809 ${!location.pathname.startsWith('/i/bookmarks') ? /*html*/`<span class="tweet-interact-more-menu-bookmark">${t.bookmarked ? LOC.remove_bookmark.message : LOC.bookmark_tweet.message}</span>` : ''} 1810 <span class="tweet-interact-more-menu-mute">${t.conversation_muted ? LOC.unmute_convo.message : LOC.mute_convo.message}</span> 1811 <hr> 1812 ${t.feedback ? t.feedback.map((f, i) => /*html*/`<span class="tweet-interact-more-menu-feedback" data-index="${i}">${f.prompt ? f.prompt : LOC.topic_not_interested.message}</span>`).join("\n") : ''} 1813 <span class="tweet-interact-more-menu-refresh">${LOC.refresh_tweet.message}</span> 1814 ${t.extended_entities && t.extended_entities.media.length === 1 && t.extended_entities.media[0].type === 'animated_gif' ? /*html*/`<span class="tweet-interact-more-menu-download-gif" data-gifno="1">${LOC.download_gif.message}</span>` : ``} 1815 ${t.extended_entities && t.extended_entities.media.length > 1 ? t.extended_entities.media.filter(m => m.type === 'animated_gif').map((m, i) => /*html*/`<span class="tweet-interact-more-menu-download-gif" data-gifno="${i+1}">${LOC.download_gif.message} (#${i+1})</span>`).join('\n') : ''} 1816 ${t.extended_entities && t.extended_entities.media.length > 0 ? /*html*/`<span class="tweet-interact-more-menu-download">${LOC.download_media.message}</span>` : ``} 1817 ${vars.developerMode ? /*html*/` 1818 <hr> 1819 <span class="tweet-interact-more-menu-copy-user-id">${LOC.copy_user_id.message}</span> 1820 <span class="tweet-interact-more-menu-copy-tweet-id">${LOC.copy_tweet_id.message}</span> 1821 <span class="tweet-interact-more-menu-log">Log tweet object</span> 1822 ` : ''} 1823 </div> 1824 ${options.selfThreadButton && t.self_thread && t.self_thread.id_str && !options.threadContinuation && !location.pathname.includes('/status/') ? /*html*/`<a class="tweet-self-thread-button tweet-thread-right" target="_blank" href="https://twitter.com/${t.user.screen_name}/status/${t.self_thread.id_str}">${LOC.show_this_thread.message}</a>` : ``} 1825 ${!options.noTop && !options.selfThreadButton && t.in_reply_to_status_id_str && !(options.threadContinuation || (options.selfThreadContinuation && t.self_thread && t.self_thread.id_str)) && !location.pathname.includes('/status/') ? `<a class="tweet-self-thread-button tweet-thread-right" target="_blank" href="https://twitter.com/${t.in_reply_to_screen_name}/status/${t.in_reply_to_status_id_str}">${LOC.show_this_thread.message}</a>` : ``} 1826 </div> 1827 <div class="tweet-reply" hidden> 1828 <br> 1829 <b style="font-size: 12px;display: block;margin-bottom: 5px;">${LOC.replying_to_tweet.message} <span ${!vars.disableHotkeys ? 'title="ALT+M"' : ''} class="tweet-reply-upload">${LOC.upload_media_btn.message}</span> <span class="tweet-reply-add-emoji">${LOC.emoji_btn.message}</span> <span ${!vars.disableHotkeys ? 'title="ALT+R"' : ''} class="tweet-reply-cancel">${LOC.cancel_btn.message}</span></b> 1830 <span class="tweet-reply-error" style="color:red"></span> 1831 <textarea maxlength="1000" class="tweet-reply-text" placeholder="${LOC.reply_example.message}"></textarea> 1832 <button title="CTRL+ENTER" class="tweet-reply-button nice-button">${LOC.reply.message}</button><br> 1833 <span class="tweet-reply-char">0/280</span><br> 1834 <div class="tweet-reply-media" style="padding-bottom: 10px;"></div> 1835 </div> 1836 <div class="tweet-quote" hidden> 1837 <br> 1838 <b style="font-size: 12px;display: block;margin-bottom: 5px;">${LOC.quote_tweet.message} <span ${!vars.disableHotkeys ? 'title="ALT+M"' : ''} class="tweet-quote-upload">${LOC.upload_media_btn.message}</span> <span class="tweet-quote-add-emoji">${LOC.emoji_btn.message}</span> <span ${!vars.disableHotkeys ? 'title="ALT+Q"' : ''} class="tweet-quote-cancel">${LOC.cancel_btn.message}</span></b> 1839 <span class="tweet-quote-error" style="color:red"></span> 1840 <textarea maxlength="1000" class="tweet-quote-text" placeholder="${LOC.quote_example.message}"></textarea> 1841 <button title="CTRL+ENTER" class="tweet-quote-button nice-button">${LOC.quote.message}</button><br> 1842 <span class="tweet-quote-char">0/280</span><br> 1843 <div class="tweet-quote-media" style="padding-bottom: 10px;"></div> 1844 </div> 1845 <div class="tweet-self-thread-div" ${(options.threadContinuation || (options.selfThreadContinuation && t.self_thread && t.self_thread.id_str)) ? '' : 'hidden'}> 1846 ${options.selfThreadContinuation && t.self_thread && t.self_thread.id_str && !location.pathname.includes('/status/') ? /*html*/`<br> 1847 <a class="tweet-self-thread-button" target="_blank" href="https://twitter.com/${t.user.screen_name}/status/${t.self_thread.id_str}"> 1848 ${LOC.show_this_thread.message} 1849 </a> 1850 <span class="tweet-self-thread-line" style="margin-left: -108px;margin-top: -5px;"></span> 1851 <div class="tweet-self-thread-line-dots" style="margin-left: -120px;margin-top: -3px;"></div> 1852 ` : /*html*/` 1853 ${location.pathname.includes('/status/') ? `<br><br>` : ''} 1854 <span ${location.pathname.includes('/status/') ? `style="margin-top:-10px;" ` : ''}class="tweet-self-thread-line"></span> 1855 <div ${location.pathname.includes('/status/') ? `style="margin-top:-8px;" ` : ''}class="tweet-self-thread-line-dots"></div> 1856 `} 1857 </div> 1858 </div> 1859 `; 1860 // video 1861 let vidOverlay = tweet.getElementsByClassName('tweet-media-video-overlay')[0]; 1862 if(vidOverlay) { 1863 vidOverlay.addEventListener('click', () => { 1864 let vid = Array.from(tweet.getElementsByClassName('tweet-media')[0].children).filter(e => e.tagName === 'VIDEO')[0]; 1865 vid.play(); 1866 vid.controls = true; 1867 vid.classList.remove('tweet-media-element-censor'); 1868 vidOverlay.style.display = 'none'; 1869 }); 1870 } 1871 if(videos) { 1872 let videoErrors = 0; 1873 let vids = Array.from(tweet.getElementsByClassName('tweet-media')[0].children).filter(e => e.tagName === 'VIDEO'); 1874 vids[0].addEventListener('error', () => { 1875 if(videoErrors >= 3) return; 1876 videoErrors++; 1877 setTimeout(() => { 1878 vids[0].load(); 1879 }, 25); 1880 }) 1881 vids[0].onloadstart = () => { 1882 let src = vids[0].currentSrc; 1883 Array.from(tweet.getElementsByClassName('tweet-video-quality')).forEach(el => { 1884 if(el.dataset.url === src) el.classList.add('tweet-video-quality-current'); 1885 }); 1886 tweet.getElementsByClassName('tweet-video-reload')[0].addEventListener('click', () => { 1887 let vid = Array.from(tweet.getElementsByClassName('tweet-media')[0].children).filter(e => e.tagName === 'VIDEO')[0]; 1888 let time = vid.currentTime; 1889 let paused = vid.paused; 1890 vid.load(); 1891 vid.onloadstart = () => { 1892 let src = vid.currentSrc; 1893 vid.currentTime = time; 1894 if(!paused) vid.play(); 1895 Array.from(tweet.getElementsByClassName('tweet-video-quality')).forEach(el => { 1896 if(el.dataset.url === src.split('&ttd=')[0]) el.classList.add('tweet-video-quality-current'); 1897 else el.classList.remove('tweet-video-quality-current'); 1898 }); 1899 } 1900 }); 1901 Array.from(tweet.getElementsByClassName('tweet-video-quality')).forEach(el => el.addEventListener('click', () => { 1902 if(el.className.includes('tweet-video-quality-current')) return; 1903 localStorage.preferredQuality = parseInt(el.innerText); 1904 let vid = Array.from(tweet.getElementsByClassName('tweet-media')[0].children).filter(e => e.tagName === 'VIDEO')[0]; 1905 let time = vid.currentTime; 1906 let paused = vid.paused; 1907 for(let v of videos) { 1908 let closestQuality = v.video_info.variants.filter(v => v.bitrate).reduce((prev, curr) => { 1909 return (Math.abs(parseInt(curr.url.match(/\/(\d+)x/)[1]) - parseInt(localStorage.preferredQuality)) < Math.abs(parseInt(prev.url.match(/\/(\d+)x/)[1]) - parseInt(localStorage.preferredQuality)) ? curr : prev); 1910 }); 1911 let preferredQualityVariantIndex = v.video_info.variants.findIndex(v => v.url === closestQuality.url); 1912 if(preferredQualityVariantIndex !== -1) { 1913 let preferredQualityVariant = v.video_info.variants[preferredQualityVariantIndex]; 1914 v.video_info.variants.splice(preferredQualityVariantIndex, 1); 1915 v.video_info.variants.unshift(preferredQualityVariant); 1916 } 1917 } 1918 tweet.getElementsByClassName('tweet-media')[0].innerHTML = /*html*/` 1919 ${t.extended_entities.media.map(m => `<${m.type === 'photo' ? 'img' : 'video'} ${m.ext_alt_text ? `alt="${escapeHTML(m.ext_alt_text)}" title="${escapeHTML(m.ext_alt_text)}"` : ''} crossorigin="anonymous" width="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[0]}" height="${sizeFunctions[t.extended_entities.media.length](m.original_info.width, m.original_info.height)[1]}" loading="lazy" ${m.type === 'video' ? 'controls' : ''} ${m.type === 'animated_gif' ? 'loop muted onclick="if(this.paused) this.play(); else this.pause()"' : ''}${m.type === 'animated_gif' && !vars.disableGifAutoplay ? ' autoplay' : ''} ${m.type === 'photo' ? `src="${m.media_url_https}"` : ''} class="tweet-media-element ${mediaClasses[t.extended_entities.media.length]} ${!vars.displaySensitiveContent && t.possibly_sensitive ? 'tweet-media-element-censor' : ''}">${m.type === 'video' || m.type === 'animated_gif' ? ` 1920 ${m.video_info.variants.map(v => `<source src="${v.url}&ttd=${Date.now()}" type="${v.content_type}">`).join('\n')} 1921 ${LOC.unsupported_video.message} 1922 </video>` : ''}`).join('\n')} 1923 `; 1924 vid = Array.from(tweet.getElementsByClassName('tweet-media')[0].children).filter(e => e.tagName === 'VIDEO')[0]; 1925 vid.onloadstart = () => { 1926 let src = vid.currentSrc; 1927 vid.currentTime = time; 1928 if(!paused) vid.play(); 1929 Array.from(tweet.getElementsByClassName('tweet-video-quality')).forEach(el => { 1930 if(el.dataset.url === src.split('&ttd=')[0]) el.classList.add('tweet-video-quality-current'); 1931 else el.classList.remove('tweet-video-quality-current'); 1932 }); 1933 } 1934 vid.addEventListener('mousedown', e => { 1935 if(e.button === 1) { 1936 e.preventDefault(); 1937 window.open(vid.currentSrc, '_blank'); 1938 } 1939 }); 1940 })); 1941 }; 1942 for(let vid of vids) { 1943 if(typeof vars.volume === 'number') { 1944 vid.volume = vars.volume; 1945 } 1946 vid.onvolumechange = () => { 1947 chrome.storage.sync.set({ 1948 volume: vid.volume 1949 }, () => { }); 1950 let allVids = document.getElementsByTagName('video'); 1951 for(let i = 0; i < allVids.length; i++) { 1952 allVids[i].volume = vid.volume; 1953 } 1954 }; 1955 vid.addEventListener('mousedown', e => { 1956 if(e.button === 1) { 1957 e.preventDefault(); 1958 window.open(vid.currentSrc, '_blank'); 1959 } 1960 }); 1961 } 1962 } 1963 1964 let footerFavorites = tweet.getElementsByClassName('tweet-footer-favorites')[0]; 1965 if(t.card) { 1966 generateCard(t, tweet, user); 1967 } 1968 if (options.top) { 1969 tweet.querySelector('.tweet-top').hidden = false; 1970 const icon = document.createElement('span'); 1971 icon.innerText = options.top.icon; 1972 icon.classList.add('tweet-top-icon'); 1973 icon.style.color = options.top.color; 1974 1975 const span = document.createElement("span"); 1976 span.classList.add("tweet-top-text"); 1977 span.innerHTML = options.top.text; 1978 if(options.top.class) { 1979 span.classList.add(options.top.class); 1980 tweet.classList.add(`tweet-top-${options.top.class}`); 1981 } 1982 tweet.querySelector('.tweet-top').append(icon, span); 1983 } 1984 if(options.mainTweet) { 1985 let likers = mainTweetLikers.slice(0, 8); 1986 for(let i in likers) { 1987 let liker = likers[i]; 1988 let a = document.createElement('a'); 1989 a.href = `https://twitter.com/${liker.screen_name}`; 1990 let likerImg = document.createElement('img'); 1991 likerImg.src = `${(liker.default_profile_image && vars.useOldDefaultProfileImage) ? chrome.runtime.getURL(`images/default_profile_images/default_profile_${Number(liker.id_str) % 7}_normal.png`): liker.profile_image_url_https}`; 1992 likerImg.classList.add('tweet-footer-favorites-img'); 1993 likerImg.title = liker.name + ' (@' + liker.screen_name + ')'; 1994 likerImg.width = 24; 1995 likerImg.height = 24; 1996 a.dataset.id = liker.id_str; 1997 a.appendChild(likerImg); 1998 footerFavorites.appendChild(a); 1999 } 2000 let likesLink = tweet.getElementsByClassName('tweet-footer-stat-f')[0]; 2001 likesLink.addEventListener('click', e => { 2002 e.preventDefault(); 2003 document.getElementById('loading-box').hidden = false; 2004 history.pushState({}, null, `https://twitter.com/${t.user.screen_name}/status/${t.id_str}/likes`); 2005 updateSubpage(); 2006 mediaToUpload = []; 2007 linkColors = {}; 2008 cursor = undefined; 2009 seenReplies = []; 2010 mainTweetLikers = []; 2011 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 2012 updateLikes(id); 2013 renderDiscovery(); 2014 renderTrends(); 2015 currentLocation = location.pathname; 2016 }); 2017 let retweetsLink = tweet.getElementsByClassName('tweet-footer-stat-r')[0]; 2018 retweetsLink.addEventListener('click', e => { 2019 e.preventDefault(); 2020 document.getElementById('loading-box').hidden = false; 2021 history.pushState({}, null, `https://twitter.com/${t.user.screen_name}/status/${t.id_str}/retweets`); 2022 updateSubpage(); 2023 mediaToUpload = []; 2024 linkColors = {}; 2025 cursor = undefined; 2026 seenReplies = []; 2027 mainTweetLikers = []; 2028 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 2029 updateRetweets(id); 2030 renderDiscovery(); 2031 renderTrends(); 2032 currentLocation = location.pathname; 2033 }); 2034 let repliesLink = tweet.getElementsByClassName('tweet-footer-stat-o')[0]; 2035 repliesLink.addEventListener('click', e => { 2036 e.preventDefault(); 2037 if(location.href === `https://twitter.com/${t.user.screen_name}/status/${t.id_str}`) return; 2038 document.getElementById('loading-box').hidden = false; 2039 history.pushState({}, null, `https://twitter.com/${t.user.screen_name}/status/${t.id_str}`); 2040 updateSubpage(); 2041 mediaToUpload = []; 2042 linkColors = {}; 2043 cursor = undefined; 2044 seenReplies = []; 2045 mainTweetLikers = []; 2046 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 2047 updateReplies(id); 2048 renderDiscovery(); 2049 renderTrends(); 2050 currentLocation = location.pathname; 2051 }); 2052 } 2053 if(options.mainTweet && t.user.id_str !== user.id_str) { 2054 const tweetFollow = tweet.getElementsByClassName('tweet-header-follow')[0]; 2055 tweetFollow.addEventListener('click', async () => { 2056 if(t.user.following) { 2057 try { 2058 await API.user.unfollow(t.user.screen_name); 2059 } catch(e) { 2060 console.error(e); 2061 alert(e); 2062 return; 2063 } 2064 tweetFollow.innerText = LOC.follow.message; 2065 tweetFollow.classList.remove('following'); 2066 tweetFollow.classList.add('follow'); 2067 t.user.following = false; 2068 } else { 2069 try { 2070 await API.user.follow(t.user.screen_name); 2071 } catch(e) { 2072 console.error(e); 2073 alert(e); 2074 return; 2075 } 2076 tweetFollow.innerText = LOC.unfollow.message; 2077 tweetFollow.classList.remove('follow'); 2078 tweetFollow.classList.add('following'); 2079 t.user.following = true; 2080 } 2081 }); 2082 } 2083 const tweetBody = tweet.getElementsByClassName('tweet-body')[0]; 2084 const tweetBodyText = tweet.getElementsByClassName('tweet-body-text')[0]; 2085 const tweetTranslate = tweet.getElementsByClassName('tweet-translate')[0]; 2086 const tweetTranslateAfter = tweet.getElementsByClassName('tweet-translate-after')[0]; 2087 const tweetBodyQuote = tweet.getElementsByClassName('tweet-body-quote')[0]; 2088 const tweetMediaQuote = tweet.getElementsByClassName('tweet-media-quote')[0]; 2089 const tweetBodyQuoteText = tweet.getElementsByClassName('tweet-body-text-quote')[0]; 2090 const tweetDeleteBookmark = tweet.getElementsByClassName('tweet-delete-bookmark')[0]; 2091 2092 const tweetReplyCancel = tweet.getElementsByClassName('tweet-reply-cancel')[0]; 2093 const tweetReplyUpload = tweet.getElementsByClassName('tweet-reply-upload')[0]; 2094 const tweetReplyAddEmoji = tweet.getElementsByClassName('tweet-reply-add-emoji')[0]; 2095 const tweetReply = tweet.getElementsByClassName('tweet-reply')[0]; 2096 const tweetReplyButton = tweet.getElementsByClassName('tweet-reply-button')[0]; 2097 const tweetReplyError = tweet.getElementsByClassName('tweet-reply-error')[0]; 2098 const tweetReplyText = tweet.getElementsByClassName('tweet-reply-text')[0]; 2099 const tweetReplyChar = tweet.getElementsByClassName('tweet-reply-char')[0]; 2100 const tweetReplyMedia = tweet.getElementsByClassName('tweet-reply-media')[0]; 2101 2102 const tweetInteract = tweet.getElementsByClassName('tweet-interact')[0]; 2103 const tweetInteractReply = tweet.getElementsByClassName('tweet-interact-reply')[0]; 2104 const tweetInteractRetweet = tweet.getElementsByClassName('tweet-interact-retweet')[0]; 2105 const tweetInteractFavorite = tweet.getElementsByClassName('tweet-interact-favorite')[0]; 2106 const tweetInteractBookmark = tweet.getElementsByClassName('tweet-interact-bookmark')[0]; 2107 const tweetInteractMore = tweet.getElementsByClassName('tweet-interact-more')[0]; 2108 2109 const tweetFooter = tweet.getElementsByClassName('tweet-footer')[0]; 2110 const tweetFooterReplies = tweet.getElementsByClassName('tweet-footer-stat-replies')[0]; 2111 const tweetFooterRetweets = tweet.getElementsByClassName('tweet-footer-stat-retweets')[0]; 2112 const tweetFooterFavorites = tweet.getElementsByClassName('tweet-footer-stat-favorites')[0]; 2113 2114 const tweetQuote = tweet.getElementsByClassName('tweet-quote')[0]; 2115 const tweetQuoteCancel = tweet.getElementsByClassName('tweet-quote-cancel')[0]; 2116 const tweetQuoteUpload = tweet.getElementsByClassName('tweet-quote-upload')[0]; 2117 const tweetQuoteAddEmoji = tweet.getElementsByClassName('tweet-quote-add-emoji')[0]; 2118 const tweetQuoteButton = tweet.getElementsByClassName('tweet-quote-button')[0]; 2119 const tweetQuoteError = tweet.getElementsByClassName('tweet-quote-error')[0]; 2120 const tweetQuoteText = tweet.getElementsByClassName('tweet-quote-text')[0]; 2121 const tweetQuoteChar = tweet.getElementsByClassName('tweet-quote-char')[0]; 2122 const tweetQuoteMedia = tweet.getElementsByClassName('tweet-quote-media')[0]; 2123 2124 const tweetInteractRetweetMenu = tweet.getElementsByClassName('tweet-interact-retweet-menu')[0]; 2125 const tweetInteractRetweetMenuRetweet = tweet.getElementsByClassName('tweet-interact-retweet-menu-retweet')[0]; 2126 const tweetInteractRetweetMenuQuote = tweet.getElementsByClassName('tweet-interact-retweet-menu-quote')[0]; 2127 const tweetInteractRetweetMenuQuotes = tweet.getElementsByClassName('tweet-interact-retweet-menu-quotes')[0]; 2128 const tweetInteractRetweetMenuRetweeters = tweet.getElementsByClassName('tweet-interact-retweet-menu-retweeters')[0]; 2129 2130 const tweetInteractMoreMenu = tweet.getElementsByClassName('tweet-interact-more-menu')[0]; 2131 const tweetInteractMoreMenuCopy = tweet.getElementsByClassName('tweet-interact-more-menu-copy')[0]; 2132 const tweetInteractMoreMenuCopyTweetId = tweet.getElementsByClassName('tweet-interact-more-menu-copy-tweet-id')[0]; 2133 const tweetInteractMoreMenuLog = tweet.getElementsByClassName('tweet-interact-more-menu-log')[0]; 2134 const tweetInteractMoreMenuCopyUserId = tweet.getElementsByClassName('tweet-interact-more-menu-copy-user-id')[0]; 2135 const tweetInteractMoreMenuEmbed = tweet.getElementsByClassName('tweet-interact-more-menu-embed')[0]; 2136 const tweetInteractMoreMenuShare = tweet.getElementsByClassName('tweet-interact-more-menu-share')[0]; 2137 const tweetInteractMoreMenuNewtwitter = tweet.getElementsByClassName('tweet-interact-more-menu-newtwitter')[0]; 2138 const tweetInteractMoreMenuAnalytics = tweet.getElementsByClassName('tweet-interact-more-menu-analytics')[0]; 2139 const tweetInteractMoreMenuRefresh = tweet.getElementsByClassName('tweet-interact-more-menu-refresh')[0]; 2140 const tweetInteractMoreMenuMute = tweet.getElementsByClassName('tweet-interact-more-menu-mute')[0]; 2141 const tweetInteractMoreMenuDownload = tweet.getElementsByClassName('tweet-interact-more-menu-download')[0]; 2142 const tweetInteractMoreMenuDownloadGifs = Array.from(tweet.getElementsByClassName('tweet-interact-more-menu-download-gif')); 2143 const tweetInteractMoreMenuDelete = tweet.getElementsByClassName('tweet-interact-more-menu-delete')[0]; 2144 const tweetInteractMoreMenuPin = tweet.getElementsByClassName('tweet-interact-more-menu-pin')[0]; 2145 const tweetInteractMoreMenuFollow = tweet.getElementsByClassName('tweet-interact-more-menu-follow')[0]; 2146 const tweetInteractMoreMenuBlock = tweet.getElementsByClassName('tweet-interact-more-menu-block')[0]; 2147 const tweetInteractMoreMenuBookmark = tweet.getElementsByClassName('tweet-interact-more-menu-bookmark')[0]; 2148 const tweetInteractMoreMenuFeedbacks = Array.from(tweet.getElementsByClassName('tweet-interact-more-menu-feedback')); 2149 const tweetInteractMoreMenuHide = tweet.getElementsByClassName('tweet-interact-more-menu-hide')[0]; 2150 2151 if(tweetInteractMoreMenuLog) tweetInteractMoreMenuLog.addEventListener('click', () => { 2152 console.log(t); 2153 }); 2154 2155 // moderating tweets 2156 if(tweetInteractMoreMenuHide) tweetInteractMoreMenuHide.addEventListener('click', async () => { 2157 if(t.moderated) { 2158 try { 2159 await API.tweet.unmoderate(t.id_str); 2160 } catch(e) { 2161 console.error(e); 2162 alert(e); 2163 return; 2164 } 2165 tweetInteractMoreMenuHide.innerText = LOC.hide_tweet.message; 2166 t.moderated = false; 2167 } else { 2168 let sure = confirm(LOC.hide_tweet_sure.message); 2169 if(!sure) return; 2170 try { 2171 await API.tweet.moderate(t.id_str); 2172 } catch(e) { 2173 console.error(e); 2174 alert(e); 2175 return; 2176 } 2177 tweetInteractMoreMenuHide.innerText = LOC.unhide_tweet.message; 2178 t.moderated = true; 2179 } 2180 }); 2181 2182 // community notes 2183 if(t.birdwatch && !vars.hideCommunityNotes) { 2184 if(t.birdwatch.subtitle) { 2185 let div = document.createElement('div'); 2186 div.classList.add('tweet-birdwatch', 'box'); 2187 let text = Array.from(escapeHTML(t.birdwatch.subtitle.text)); 2188 for(let e = t.birdwatch.subtitle.entities.length - 1; e >= 0; e--) { 2189 let entity = t.birdwatch.subtitle.entities[e]; 2190 if(!entity.ref) continue; 2191 text = arrayInsert(text, entity.toIndex, '</a>'); 2192 text = arrayInsert(text, entity.fromIndex, `<a href="${entity.ref.url}" target="_blank">`); 2193 } 2194 text = text.join(''); 2195 2196 div.innerHTML = /*html*/` 2197 <div class="tweet-birdwatch-header"> 2198 <span class="tweet-birdwatch-title">${escapeHTML(t.birdwatch.title)}</span> 2199 </div> 2200 <div class="tweet-birdwatch-body"> 2201 <span class="tweet-birdwatch-subtitle">${text}</span> 2202 </div> 2203 `; 2204 2205 if(tweetFooter) tweetFooter.before(div); 2206 else tweetInteract.before(div); 2207 } 2208 } 2209 2210 // rtl languages 2211 if(rtlLanguages.includes(t.lang)) { 2212 tweetBody.classList.add('rtl'); 2213 } 2214 2215 // Quote body 2216 if(tweetMediaQuote) tweetMediaQuote.addEventListener('click', e => { 2217 if(e && e.target && e.target.tagName === "VIDEO") { 2218 e.preventDefault(); 2219 e.stopPropagation(); 2220 e.stopImmediatePropagation(); 2221 if(e.target.paused) { 2222 e.target.play(); 2223 } else { 2224 e.target.pause(); 2225 } 2226 } 2227 }); 2228 if(tweetBodyQuote) { 2229 if(typeof mainTweetLikers !== 'undefined') { 2230 tweetBodyQuote.addEventListener('click', e => { 2231 e.preventDefault(); 2232 document.getElementById('loading-box').hidden = false; 2233 history.pushState({}, null, `https://twitter.com/${t.quoted_status.user.screen_name}/status/${t.quoted_status.id_str}`); 2234 updateSubpage(); 2235 mediaToUpload = []; 2236 linkColors = {}; 2237 cursor = undefined; 2238 seenReplies = []; 2239 mainTweetLikers = []; 2240 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 2241 if(subpage === 'tweet') { 2242 updateReplies(id); 2243 } else if(subpage === 'likes') { 2244 updateLikes(id); 2245 } else if(subpage === 'retweets') { 2246 updateRetweets(id); 2247 } else if(subpage === 'retweets_with_comments') { 2248 updateRetweetsWithComments(id); 2249 } 2250 renderDiscovery(); 2251 renderTrends(); 2252 currentLocation = location.pathname; 2253 }); 2254 } else { 2255 tweetBodyQuote.addEventListener('click', e => { 2256 e.preventDefault(); 2257 if(e.target.className && e.target.className.includes('tweet-media-element')) { 2258 if(!e.target.src.endsWith('?name=orig') && !e.target.src.startsWith('data:')) { 2259 e.target.src += '?name=orig'; 2260 } 2261 new Viewer(e.target, { 2262 transition: false 2263 }); 2264 e.target.click(); 2265 return; 2266 } 2267 new TweetViewer(user, t.quoted_status); 2268 }); 2269 } 2270 if(rtlLanguages.includes(t.quoted_status.lang)) { 2271 tweetBodyQuoteText.classList.add('rtl'); 2272 } else { 2273 tweetBodyQuoteText.classList.add('ltr'); 2274 } 2275 } 2276 if(tweetTranslate || tweetTranslateAfter) if(options.translate || vars.autotranslateProfiles.includes(t.user.id_str) || (typeof toAutotranslate !== 'undefined' && toAutotranslate)) { 2277 onVisible(tweet, () => { 2278 if(!t.translated) { 2279 if(tweetTranslate) tweetTranslate.click(); 2280 else if(tweetTranslateAfter) tweetTranslateAfter.click(); 2281 } 2282 }) 2283 } 2284 2285 // Translate 2286 t.translated = false; 2287 if(tweetTranslate || tweetTranslateAfter) (tweetTranslate ? tweetTranslate : tweetTranslateAfter).addEventListener('click', async () => { 2288 if(t.translated) return; 2289 let translated = await API.tweet.translate(t.id_str); 2290 t.translated = true; 2291 (tweetTranslate ? tweetTranslate : tweetTranslateAfter).hidden = true; 2292 let translatedMessage; 2293 if(LOC.translated_from.message.includes("$LANGUAGE$")) { 2294 translatedMessage = LOC.translated_from.message.replace("$LANGUAGE$", `[${translated.translated_lang}]`); 2295 } else { 2296 translatedMessage = `${LOC.translated_from.message} [${translated.translated_lang}]`; 2297 } 2298 tweetBodyText.innerHTML += `<br>`+ 2299 `<span style="font-size: 12px;color: var(--light-gray);">${translatedMessage}:</span>`+ 2300 `<br>`+ 2301 `<span class="tweet-translated-text">${await renderTweetBodyHTML(translated.text, translated.entities)}</span>`; 2302 if(vars.enableTwemoji) twemoji.parse(tweetBodyText); 2303 }); 2304 2305 // Bookmarks 2306 let switchingBookmark = false; 2307 let switchBookmark = () => { 2308 if(switchingBookmark) return; 2309 switchingBookmark = true; 2310 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2311 if(t.bookmarked) { 2312 API.bookmarks.delete(t.id_str).then(() => { 2313 toast.info(LOC.unbookmarked_tweet.message); 2314 switchingBookmark = false; 2315 if(tweetDeleteBookmark) { 2316 tweet.remove(); 2317 if(timelineContainer.children.length === 0) { 2318 timelineContainer.innerHTML = `<div style="color:var(--light-gray)">${LOC.empty.message}</div>`; 2319 document.getElementById('delete-all').hidden = true; 2320 } 2321 return; 2322 } 2323 t.bookmarked = false; 2324 t.bookmark_count--; 2325 tweetInteractMoreMenuBookmark.innerText = LOC.bookmark_tweet.message; 2326 if(tweetInteractBookmark) { 2327 tweetInteractBookmark.classList.remove('tweet-interact-bookmarked'); 2328 if(vars.bookmarkButton !== 'show_all_no_count') { 2329 tweetInteractBookmark.innerText = Number(t.bookmark_count).toLocaleString().replace(/\s/g, ','); 2330 } 2331 } 2332 }).catch(e => { 2333 switchingBookmark = false; 2334 console.error(e); 2335 alert(e); 2336 }); 2337 } else { 2338 API.bookmarks.create(t.id_str).then(() => { 2339 toast.info(LOC.bookmarked_tweet.message); 2340 switchingBookmark = false; 2341 t.bookmarked = true; 2342 t.bookmark_count++; 2343 tweetInteractMoreMenuBookmark.innerText = LOC.remove_bookmark.message; 2344 if(tweetInteractBookmark) { 2345 tweetInteractBookmark.classList.add('tweet-interact-bookmarked'); 2346 if(vars.bookmarkButton !== 'show_all_no_count') { 2347 tweetInteractBookmark.innerText = Number(t.bookmark_count).toLocaleString().replace(/\s/g, ','); 2348 } 2349 } 2350 }).catch(e => { 2351 switchingBookmark = false; 2352 console.error(e); 2353 alert(e); 2354 }); 2355 } 2356 }; 2357 if(tweetInteractBookmark) tweetInteractBookmark.addEventListener('click', switchBookmark); 2358 if(tweetInteractMoreMenuBookmark) tweetInteractMoreMenuBookmark.addEventListener('click', switchBookmark); 2359 if(tweetDeleteBookmark) tweetDeleteBookmark.addEventListener('click', async () => { 2360 await API.bookmarks.delete(t.id_str); 2361 tweet.remove(); 2362 if(timelineContainer.children.length === 0) { 2363 timelineContainer.innerHTML = `<div style="color:var(--light-gray)">${LOC.empty.message}</div>`; 2364 document.getElementById('delete-all').hidden = true; 2365 } 2366 }); 2367 2368 // Media 2369 if (t.extended_entities && t.extended_entities.media) { 2370 const tweetMedia = tweet.getElementsByClassName('tweet-media')[0]; 2371 tweetMedia.addEventListener('click', e => { 2372 if (e.target.className && e.target.className.includes('tweet-media-element-censor')) { 2373 return e.target.classList.remove('tweet-media-element-censor'); 2374 } 2375 if (e.target.tagName === 'IMG') { 2376 if(!e.target.src.endsWith('?name=orig') && !e.target.src.startsWith('data:')) { 2377 e.target.src += '?name=orig'; 2378 } 2379 new Viewer(tweetMedia, { 2380 transition: false 2381 }); 2382 e.target.click(); 2383 } 2384 }); 2385 if(typeof pageUser !== 'undefined' && !location.pathname.includes("/likes")) { 2386 let profileMediaDiv = document.getElementById('profile-media-div'); 2387 if(!options || !options.top || !options.top.text || !options.top.text.includes('retweeted')) t.extended_entities.media.forEach(m => { 2388 if(profileMediaDiv.children.length >= 6) return; 2389 let ch = Array.from(profileMediaDiv.children); 2390 if(ch.find(c => c.src === m.media_url_https)) return; 2391 const media = document.createElement('img'); 2392 media.classList.add('tweet-media-element', 'tweet-media-element-four', 'profile-media-preview'); 2393 if(!vars.displaySensitiveContent && t.possibly_sensitive) media.classList.add('tweet-media-element-censor'); 2394 media.src = m.media_url_https; 2395 if(m.ext_alt_text) media.alt = m.ext_alt_text; 2396 media.addEventListener('click', async () => { 2397 if(subpage !== 'profile' && subpage !== 'media') { 2398 document.getElementById('profile-stat-tweets-link').click(); 2399 while(!document.getElementsByClassName('tweet-id-' + t.id_str)[0]) await sleep(100); 2400 } 2401 document.getElementsByClassName('tweet-id-' + t.id_str)[0].scrollIntoView({behavior: 'smooth', block: 'center'}); 2402 }); 2403 profileMediaDiv.appendChild(media); 2404 }); 2405 } 2406 } 2407 2408 // Emojis 2409 [tweetReplyAddEmoji, tweetQuoteAddEmoji].forEach(e => { 2410 e.addEventListener('click', e => { 2411 let isReply = e.target.className === 'tweet-reply-add-emoji'; 2412 createEmojiPicker(isReply ? tweetReply : tweetQuote, isReply ? tweetReplyText : tweetQuoteText, {}); 2413 }); 2414 }); 2415 2416 // Reply 2417 tweetReplyCancel.addEventListener('click', () => { 2418 tweetReply.hidden = true; 2419 tweetInteractReply.classList.remove('tweet-interact-reply-clicked'); 2420 }); 2421 let replyMedia = []; 2422 tweetReply.addEventListener('drop', e => { 2423 handleDrop(e, replyMedia, tweetReplyMedia); 2424 }); 2425 tweetReply.addEventListener('paste', event => { 2426 let items = (event.clipboardData || event.originalEvent.clipboardData).items; 2427 for (let index in items) { 2428 let item = items[index]; 2429 if (item.kind === 'file') { 2430 let file = item.getAsFile(); 2431 handleFiles([file], replyMedia, tweetReplyMedia); 2432 } 2433 } 2434 }); 2435 tweetReplyUpload.addEventListener('click', () => { 2436 getMedia(replyMedia, tweetReplyMedia); 2437 tweetReplyText.focus(); 2438 }); 2439 tweetInteractReply.addEventListener('click', () => { 2440 if(options.mainTweet) { 2441 document.getElementById('new-tweet').click(); 2442 document.getElementById('new-tweet-text').focus(); 2443 return; 2444 } 2445 if (!tweetQuote.hidden) tweetQuote.hidden = true; 2446 if (tweetReply.hidden) { 2447 tweetInteractReply.classList.add('tweet-interact-reply-clicked'); 2448 } else { 2449 tweetInteractReply.classList.remove('tweet-interact-reply-clicked'); 2450 } 2451 tweetReply.hidden = !tweetReply.hidden; 2452 setTimeout(() => { 2453 tweetReplyText.focus(); 2454 }) 2455 }); 2456 tweetReplyText.addEventListener('keydown', e => { 2457 if (e.key === 'Enter' && e.ctrlKey) { 2458 tweetReplyButton.click(); 2459 } 2460 }); 2461 tweetReplyText.addEventListener('input', e => { 2462 let text = tweetReplyText.value.replace(linkRegex, ' https://t.co/xxxxxxxxxx').trim(); 2463 tweetReplyChar.innerText = `${text.length}/280`; 2464 if(text.length > 265) { 2465 tweetReplyChar.style.color = "#c26363"; 2466 } else { 2467 tweetReplyChar.style.color = ""; 2468 } 2469 if (text.length > 280) { 2470 tweetReplyChar.style.color = "red"; 2471 tweetReplyButton.disabled = true; 2472 } else { 2473 tweetReplyButton.disabled = false; 2474 } 2475 }); 2476 tweetReplyButton.addEventListener('click', async () => { 2477 tweetReplyError.innerHTML = ''; 2478 let text = tweetReplyText.value; 2479 if (text.length === 0 && replyMedia.length === 0) return; 2480 tweetReplyButton.disabled = true; 2481 let uploadedMedia = []; 2482 for (let i in replyMedia) { 2483 let media = replyMedia[i]; 2484 try { 2485 media.div.getElementsByClassName('new-tweet-media-img-progress')[0].hidden = false; 2486 let mediaId = await API.uploadMedia({ 2487 media_type: media.type, 2488 media_category: media.category, 2489 media: media.data, 2490 alt: media.alt, 2491 cw: media.cw, 2492 loadCallback: data => { 2493 media.div.getElementsByClassName('new-tweet-media-img-progress')[0].innerText = `${data.text} (${data.progress}%)`; 2494 } 2495 }); 2496 uploadedMedia.push(mediaId); 2497 } catch (e) { 2498 media.div.getElementsByClassName('new-tweet-media-img-progress')[0].hidden = true; 2499 console.error(e); 2500 alert(e); 2501 } 2502 } 2503 let tweetObject = { 2504 status: text, 2505 in_reply_to_status_id: t.id_str 2506 }; 2507 if (uploadedMedia.length > 0) { 2508 tweetObject.media_ids = uploadedMedia.join(','); 2509 } 2510 let tweetData; 2511 try { 2512 tweetData = await API.tweet.postV2(tweetObject) 2513 } catch (e) { 2514 tweetReplyError.innerHTML = (e && e.message ? e.message : e) + "<br>"; 2515 tweetReplyButton.disabled = false; 2516 return; 2517 } 2518 if (!tweetData) { 2519 tweetReplyButton.disabled = false; 2520 tweetReplyError.innerHTML = `${LOC.error_sending_tweet.message}<br>`; 2521 return; 2522 } 2523 tweetReplyChar.innerText = '0/280'; 2524 tweetReplyText.value = ''; 2525 tweetReply.hidden = true; 2526 tweetInteractReply.classList.remove('tweet-interact-reply-clicked'); 2527 if(!options.mainTweet) { 2528 tweetInteractReply.dataset.val = parseInt(tweetInteractReply.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1; 2529 tweetInteractReply.innerText = Number(parseInt(tweetInteractReply.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1).toLocaleString().replace(/\s/g, ','); 2530 } else { 2531 tweetFooterReplies.dataset.val = parseInt(tweetFooterReplies.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1; 2532 tweetFooterReplies.innerText = Number(parseInt(tweetFooterReplies.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1).toLocaleString().replace(/\s/g, ','); 2533 } 2534 tweetData._ARTIFICIAL = true; 2535 if(typeof timeline !== 'undefined') { 2536 timeline.data.unshift(tweetData); 2537 } 2538 if(tweet.getElementsByClassName('tweet-self-thread-div')[0]) tweet.getElementsByClassName('tweet-self-thread-div')[0].hidden = false; 2539 tweetReplyButton.disabled = false; 2540 tweetReplyMedia.innerHTML = []; 2541 replyMedia = []; 2542 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2543 appendTweet(tweetData, document.getElementById('timeline'), { 2544 noTop: true, 2545 after: tweet 2546 }); 2547 }); 2548 2549 // Retweet / Quote Tweet 2550 let retweetClicked = false; 2551 tweetQuoteCancel.addEventListener('click', () => { 2552 tweetQuote.hidden = true; 2553 }); 2554 tweetInteractRetweet.addEventListener('click', async () => { 2555 if(tweetInteractRetweet.classList.contains('tweet-interact-retweet-disabled')) { 2556 return; 2557 } 2558 if (!tweetQuote.hidden) { 2559 tweetQuote.hidden = true; 2560 return; 2561 } 2562 if (tweetInteractRetweetMenu.hidden) { 2563 tweetInteractRetweetMenu.hidden = false; 2564 } 2565 if(retweetClicked) return; 2566 retweetClicked = true; 2567 setTimeout(() => { 2568 document.body.addEventListener('click', () => { 2569 retweetClicked = false; 2570 setTimeout(() => tweetInteractRetweetMenu.hidden = true, 50); 2571 }, { once: true }); 2572 }, 50); 2573 }); 2574 t.renderRetweetsUp = (tweetData) => { 2575 tweetInteractRetweetMenuRetweet.innerText = LOC.unretweet.message; 2576 tweetInteractRetweet.classList.add('tweet-interact-retweeted'); 2577 t.retweeted = true; 2578 t.newTweetId = tweetData.id_str; 2579 if(!options.mainTweet) { 2580 tweetInteractRetweet.dataset.val = parseInt(tweetInteractRetweet.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1; 2581 tweetInteractRetweet.innerText = Number(parseInt(tweetInteractRetweet.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1).toLocaleString().replace(/\s/g, ','); 2582 } else { 2583 tweetFooterRetweets.innerText = Number(parseInt(tweetFooterRetweets.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1).toLocaleString().replace(/\s/g, ','); 2584 } 2585 } 2586 t.renderRetweetsDown = () => { 2587 tweetInteractRetweetMenuRetweet.innerText = LOC.retweet.message; 2588 tweetInteractRetweet.classList.remove('tweet-interact-retweeted'); 2589 t.retweeted = false; 2590 if(!options.mainTweet) { 2591 tweetInteractRetweet.dataset.val = parseInt(tweetInteractRetweet.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) - 1; 2592 tweetInteractRetweet.innerText = Number(parseInt(tweetInteractRetweet.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) - 1).toLocaleString().replace(/\s/g, ','); 2593 } else { 2594 tweetFooterRetweets.innerText = Number(parseInt(tweetFooterRetweets.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) - 1).toLocaleString().replace(/\s/g, ','); 2595 } 2596 delete t.newTweetId; 2597 } 2598 tweetInteractRetweetMenuRetweet.addEventListener('click', async () => { 2599 if (!t.retweeted) { 2600 let tweetData; 2601 try { 2602 tweetData = await API.tweet.retweet(t.id_str); 2603 } catch (e) { 2604 console.error(e); 2605 return; 2606 } 2607 if (!tweetData) { 2608 return; 2609 } 2610 t.renderRetweetsUp(tweetData); 2611 } else { 2612 let tweetData; 2613 try { 2614 tweetData = await API.tweet.unretweet(t.retweeted_status ? t.retweeted_status.id_str : t.id_str); 2615 } catch (e) { 2616 console.error(e); 2617 return; 2618 } 2619 if (!tweetData) { 2620 return; 2621 } 2622 if(t.current_user_retweet) { 2623 if(options.top && options.top.icon && options.top.icon === "\uf006") { 2624 tweet.remove(); 2625 if(typeof timeline !== 'undefined') { 2626 let index = timeline.data.findIndex((tweet) => tweet.retweeted_status && tweet.retweeted_status.id_str === t.id_str && !tweet.current_user_retweet); 2627 if(index > -1) { 2628 timeline.data.splice(index, 1); 2629 let originalTweet = timeline.data.find((tweet) => tweet.id_str === t.id_str); 2630 if(originalTweet) { 2631 delete originalTweet.current_user_retweet; 2632 originalTweet.renderRetweetsDown(); 2633 } 2634 } 2635 } 2636 } else { 2637 let retweetedElement = Array.from(document.getElementsByClassName('tweet')).find(te => te.dataset.tweetId === t.id_str && te.getElementsByClassName('retweet-label')[0]); 2638 if(retweetedElement) { 2639 retweetedElement.remove(); 2640 } 2641 if(typeof timeline !== 'undefined') { 2642 let index = timeline.data.findIndex((tweet) => tweet.retweeted_status && tweet.retweeted_status.id_str === t.id_str && !tweet.current_user_retweet); 2643 if(index > -1) { 2644 timeline.data.splice(index, 1); 2645 } 2646 } 2647 } 2648 } 2649 t.renderRetweetsDown(); 2650 } 2651 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2652 }); 2653 if(options.mainTweet) { 2654 tweetInteractRetweetMenuQuotes.addEventListener('click', async () => { 2655 document.getElementById('loading-box').hidden = false; 2656 history.pushState({}, null, `https://twitter.com/${t.user.screen_name}/status/${t.id_str}/retweets/with_comments`); 2657 updateSubpage(); 2658 mediaToUpload = []; 2659 linkColors = {}; 2660 cursor = undefined; 2661 seenReplies = []; 2662 mainTweetLikers = []; 2663 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 2664 if(subpage === 'tweet') { 2665 updateReplies(id); 2666 } else if(subpage === 'likes') { 2667 updateLikes(id); 2668 } else if(subpage === 'retweets') { 2669 updateRetweets(id); 2670 } else if(subpage === 'retweets_with_comments') { 2671 updateRetweetsWithComments(id); 2672 } 2673 renderDiscovery(); 2674 renderTrends(); 2675 currentLocation = location.pathname; 2676 }); 2677 tweetInteractRetweetMenuRetweeters.addEventListener('click', async () => { 2678 document.getElementById('loading-box').hidden = false; 2679 history.pushState({}, null, `https://twitter.com/${t.user.screen_name}/status/${t.id_str}/retweets`); 2680 updateSubpage(); 2681 mediaToUpload = []; 2682 linkColors = {}; 2683 cursor = undefined; 2684 seenReplies = []; 2685 mainTweetLikers = []; 2686 let id = location.pathname.match(/status\/(\d{1,32})/)[1]; 2687 if(subpage === 'tweet') { 2688 updateReplies(id); 2689 } else if(subpage === 'likes') { 2690 updateLikes(id); 2691 } else if(subpage === 'retweets') { 2692 updateRetweets(id); 2693 } else if(subpage === 'retweets_with_comments') { 2694 updateRetweetsWithComments(id); 2695 } 2696 renderDiscovery(); 2697 renderTrends(); 2698 currentLocation = location.pathname; 2699 }); 2700 } 2701 tweetInteractRetweetMenuQuote.addEventListener('click', async () => { 2702 if (!tweetReply.hidden) { 2703 tweetInteractReply.classList.remove('tweet-interact-reply-clicked'); 2704 tweetReply.hidden = true; 2705 } 2706 tweetQuote.hidden = false; 2707 setTimeout(() => { 2708 tweetQuoteText.focus(); 2709 }) 2710 }); 2711 let quoteMedia = []; 2712 tweetQuote.addEventListener('drop', e => { 2713 handleDrop(e, quoteMedia, tweetQuoteMedia); 2714 }); 2715 tweetQuote.addEventListener('paste', event => { 2716 let items = (event.clipboardData || event.originalEvent.clipboardData).items; 2717 for (let index in items) { 2718 let item = items[index]; 2719 if (item.kind === 'file') { 2720 let file = item.getAsFile(); 2721 handleFiles([file], quoteMedia, tweetQuoteMedia); 2722 } 2723 } 2724 }); 2725 tweetQuoteUpload.addEventListener('click', () => { 2726 getMedia(quoteMedia, tweetQuoteMedia); 2727 }); 2728 tweetQuoteText.addEventListener('keydown', e => { 2729 if (e.key === 'Enter' && e.ctrlKey) { 2730 tweetQuoteButton.click(); 2731 } 2732 }); 2733 tweetQuoteText.addEventListener('input', e => { 2734 let text = tweetQuoteText.value.replace(linkRegex, ' https://t.co/xxxxxxxxxx').trim(); 2735 tweetQuoteChar.innerText = `${text.length}/280`; 2736 if(text.length > 265) { 2737 tweetQuoteChar.style.color = "#c26363"; 2738 } else { 2739 tweetQuoteChar.style.color = ""; 2740 } 2741 if (text.length > 280) { 2742 tweetQuoteChar.style.color = "red"; 2743 tweetQuoteButton.disabled = true; 2744 } else { 2745 tweetQuoteButton.disabled = false; 2746 } 2747 }); 2748 tweetQuoteButton.addEventListener('click', async () => { 2749 let text = tweetQuoteText.value; 2750 tweetQuoteError.innerHTML = ''; 2751 if (text.length === 0 && quoteMedia.length === 0) return; 2752 tweetQuoteButton.disabled = true; 2753 let uploadedMedia = []; 2754 for (let i in quoteMedia) { 2755 let media = quoteMedia[i]; 2756 try { 2757 media.div.getElementsByClassName('new-tweet-media-img-progress')[0].hidden = false; 2758 let mediaId = await API.uploadMedia({ 2759 media_type: media.type, 2760 media_category: media.category, 2761 media: media.data, 2762 alt: media.alt, 2763 cw: media.cw, 2764 loadCallback: data => { 2765 media.div.getElementsByClassName('new-tweet-media-img-progress')[0].innerText = `${data.text} (${data.progress}%)`; 2766 } 2767 }); 2768 uploadedMedia.push(mediaId); 2769 } catch (e) { 2770 media.div.getElementsByClassName('new-tweet-media-img-progress')[0].hidden = true; 2771 console.error(e); 2772 alert(e); 2773 } 2774 } 2775 let tweetObject = { 2776 status: text, 2777 attachment_url: `https://twitter.com/${t.user.screen_name}/status/${t.id_str}` 2778 }; 2779 if (uploadedMedia.length > 0) { 2780 tweetObject.media_ids = uploadedMedia.join(','); 2781 } 2782 let tweetData; 2783 try { 2784 tweetData = await API.tweet.postV2(tweetObject) 2785 } catch (e) { 2786 tweetQuoteError.innerHTML = (e && e.message ? e.message : e) + "<br>"; 2787 tweetQuoteButton.disabled = false; 2788 return; 2789 } 2790 if (!tweetData) { 2791 tweetQuoteError.innerHTML = `${LOC.error_sending_tweet}<br>`; 2792 tweetQuoteButton.disabled = false; 2793 return; 2794 } 2795 tweetQuoteText.value = ''; 2796 tweetQuoteChar.innerText = '0/280'; 2797 tweetQuote.hidden = true; 2798 tweetData._ARTIFICIAL = true; 2799 quoteMedia = []; 2800 tweetQuoteButton.disabled = false; 2801 tweetQuoteMedia.innerHTML = ''; 2802 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2803 if(typeof timeline !== 'undefined') timeline.data.unshift(tweetData); 2804 else appendTweet(tweetData, timelineContainer, { prepend: true }); 2805 }); 2806 2807 // Favorite 2808 t.renderFavoritesDown = () => { 2809 t.favorited = false; 2810 t.favorite_count--; 2811 if(!options.mainTweet) { 2812 tweetInteractFavorite.dataset.val = parseInt(tweetInteractFavorite.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) - 1; 2813 tweetInteractFavorite.innerText = Number(parseInt(tweetInteractFavorite.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) - 1).toLocaleString().replace(/\s/g, ',');; 2814 } else { 2815 if(mainTweetLikers.find(liker => liker.id_str === user.id_str)) { 2816 mainTweetLikers.splice(mainTweetLikers.findIndex(liker => liker.id_str === user.id_str), 1); 2817 let likerImg = footerFavorites.querySelector(`a[data-id="${user.id_str}"]`); 2818 if(likerImg) likerImg.remove() 2819 } 2820 tweetFooterFavorites.innerText = Number(parseInt(tweetFooterFavorites.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) - 1).toLocaleString().replace(/\s/g, ','); 2821 } 2822 tweetInteractFavorite.classList.remove('tweet-interact-favorited'); 2823 } 2824 t.renderFavoritesUp = () => { 2825 t.favorited = true; 2826 t.favorite_count++; 2827 if(!options.mainTweet) { 2828 tweetInteractFavorite.dataset.val = parseInt(tweetInteractFavorite.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1; 2829 tweetInteractFavorite.innerText = Number(parseInt(tweetInteractFavorite.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1).toLocaleString().replace(/\s/g, ',');; 2830 } else { 2831 if(footerFavorites.children.length < 8 && !mainTweetLikers.find(liker => liker.id_str === user.id_str)) { 2832 let a = document.createElement('a'); 2833 a.href = `https://twitter.com/${user.screen_name}`; 2834 let likerImg = document.createElement('img'); 2835 likerImg.src = `${(user.default_profile_image && vars.useOldDefaultProfileImage) ? chrome.runtime.getURL(`images/default_profile_images/default_profile_${Number(user.id_str) % 7}_normal.png`): user.profile_image_url_https}`; 2836 likerImg.classList.add('tweet-footer-favorites-img'); 2837 likerImg.title = user.name + ' (@' + user.screen_name + ')'; 2838 likerImg.width = 24; 2839 likerImg.height = 24; 2840 a.dataset.id = user.id_str; 2841 a.appendChild(likerImg); 2842 footerFavorites.appendChild(a); 2843 mainTweetLikers.push(user); 2844 } 2845 tweetFooterFavorites.innerText = Number(parseInt(tweetFooterFavorites.innerText.replace(/\s/g, '').replace(/,/g, '').replace(/\./g, '')) + 1).toLocaleString().replace(/\s/g, ','); 2846 } 2847 tweetInteractFavorite.classList.add('tweet-interact-favorited'); 2848 } 2849 tweetInteractFavorite.addEventListener('click', () => { 2850 if (t.favorited) { 2851 API.tweet.unfavorite(t.id_str).catch(e => { 2852 console.error(e); 2853 alert(e); 2854 t.renderFavoritesUp(); 2855 }); 2856 t.renderFavoritesDown(); 2857 } else { 2858 API.tweet.favorite(t.id_str).catch(e => { 2859 console.error(e); 2860 alert(e); 2861 t.renderFavoritesDown(); 2862 }); 2863 t.renderFavoritesUp(); 2864 } 2865 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2866 }); 2867 2868 // More 2869 let moreClicked = false; 2870 tweetInteractMore.addEventListener('click', () => { 2871 if (tweetInteractMoreMenu.hidden) { 2872 tweetInteractMoreMenu.hidden = false; 2873 } 2874 if(moreClicked) return; 2875 moreClicked = true; 2876 setTimeout(() => { 2877 document.body.addEventListener('click', () => { 2878 moreClicked = false; 2879 setTimeout(() => tweetInteractMoreMenu.hidden = true, 50); 2880 }, { once: true }); 2881 }, 50); 2882 }); 2883 if(tweetInteractMoreMenuFollow) tweetInteractMoreMenuFollow.addEventListener('click', async () => { 2884 if (t.user.following) { 2885 try { 2886 await API.user.unfollow(t.user.screen_name); 2887 } catch(e) { 2888 console.error(e); 2889 alert(e); 2890 return; 2891 } 2892 t.user.following = false; 2893 tweetInteractMoreMenuFollow.innerText = followUserText; 2894 let event = new CustomEvent('tweetAction', { detail: { 2895 action: 'unfollow', 2896 tweet: t 2897 } }); 2898 document.dispatchEvent(event); 2899 } else { 2900 try { 2901 await API.user.follow(t.user.screen_name); 2902 } catch(e) { 2903 console.error(e); 2904 alert(e); 2905 return; 2906 } 2907 t.user.following = true; 2908 tweetInteractMoreMenuFollow.innerText = unfollowUserText; 2909 let event = new CustomEvent('tweetAction', { detail: { 2910 action: 'follow', 2911 tweet: t 2912 } }); 2913 document.dispatchEvent(event); 2914 } 2915 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2916 }); 2917 if(tweetInteractMoreMenuBlock) tweetInteractMoreMenuBlock.addEventListener('click', async () => { 2918 if (t.user.blocking) { 2919 await API.user.unblock(t.user.id_str); 2920 t.user.blocking = false; 2921 if(LOC.block_user.message.includes("$SCREEN_NAME$")) { 2922 tweetInteractMoreMenuBlock.innerText = LOC.block_user.message.replace("$SCREEN_NAME$", t.user.screen_name); 2923 } else { 2924 tweetInteractMoreMenuBlock.innerText = `${LOC.block_user.message} @${t.user.screen_name}`; 2925 } 2926 tweetInteractMoreMenuFollow.hidden = false; 2927 let event = new CustomEvent('tweetAction', { detail: { 2928 action: 'unblock', 2929 tweet: t 2930 } }); 2931 document.dispatchEvent(event); 2932 } else { 2933 let blockMessage; 2934 if(LOC.block_sure.message.includes("$SCREEN_NAME$")) { 2935 blockMessage = LOC.block_sure.message.replace("$SCREEN_NAME$", t.user.screen_name); 2936 } else { 2937 blockMessage = `${LOC.block_sure.message} @${t.user.screen_name}?`; 2938 } 2939 let c = confirm(blockMessage); 2940 if (!c) return; 2941 await API.user.block(t.user.id_str); 2942 t.user.blocking = true; 2943 if(LOC.unblock_user.message.includes("$SCREEN_NAME$")) { 2944 tweetInteractMoreMenuBlock.innerText = LOC.unblock_user.message.replace("$SCREEN_NAME$", t.user.screen_name); 2945 } else { 2946 tweetInteractMoreMenuBlock.innerText = `${LOC.unblock_user.message} @${t.user.screen_name}`; 2947 } 2948 tweetInteractMoreMenuFollow.hidden = true; 2949 t.user.following = false; 2950 tweetInteractMoreMenuFollow.innerText = followUserText; 2951 let event = new CustomEvent('tweetAction', { detail: { 2952 action: 'block', 2953 tweet: t 2954 } }); 2955 document.dispatchEvent(event); 2956 } 2957 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2958 }); 2959 tweetInteractMoreMenuCopy.addEventListener('click', () => { 2960 navigator.clipboard.writeText(`https://${vars.copyLinksAs}/${t.user.screen_name}/status/${t.id_str}`); 2961 }); 2962 if(tweetInteractMoreMenuCopyTweetId) tweetInteractMoreMenuCopyTweetId.addEventListener('click', () => { 2963 navigator.clipboard.writeText(t.id_str); 2964 }); 2965 if(tweetInteractMoreMenuCopyUserId) tweetInteractMoreMenuCopyUserId.addEventListener('click', () => { 2966 navigator.clipboard.writeText(t.user.id_str); 2967 }); 2968 if(tweetInteractMoreMenuShare) tweetInteractMoreMenuShare.addEventListener('click', () => { 2969 navigator.share({ url: `https://twitter.com/${t.user.screen_name}/status/${t.id_str}` }); 2970 }); 2971 tweetInteractMoreMenuNewtwitter.addEventListener('click', () => { 2972 openInNewTab(`https://twitter.com/${t.user.screen_name}/status/${t.id_str}?newtwitter=true`); 2973 }); 2974 tweetInteractMoreMenuEmbed.addEventListener('click', () => { 2975 openInNewTab(`https://publish.twitter.com/?query=https://twitter.com/${t.user.screen_name}/status/${t.id_str}&widget=Tweet`); 2976 }); 2977 if (t.user.id_str === user.id_str) { 2978 tweetInteractMoreMenuAnalytics.addEventListener('click', () => { 2979 openInNewTab(`https://twitter.com/${t.user.screen_name}/status/${t.id_str}/analytics?newtwitter=true`); 2980 }); 2981 tweetInteractMoreMenuDelete.addEventListener('click', async () => { 2982 let sure = confirm(LOC.delete_sure.message); 2983 if (!sure) return; 2984 try { 2985 await API.tweet.delete(t.id_str); 2986 } catch (e) { 2987 alert(e); 2988 console.error(e); 2989 return; 2990 } 2991 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 2992 Array.from(document.getElementsByClassName(`tweet-id-${t.id_str}`)).forEach(tweet => { 2993 tweet.remove(); 2994 }); 2995 if(options.mainTweet) { 2996 let tweets = Array.from(document.getElementsByClassName('tweet')); 2997 if(tweets.length === 0) { 2998 location.href = 'https://twitter.com/home'; 2999 } else { 3000 location.href = tweets[0].getElementsByClassName('tweet-time')[0].href; 3001 } 3002 } 3003 if(typeof timeline !== 'undefined') { 3004 timeline.data = timeline.data.filter(tweet => tweet.id_str !== t.id_str); 3005 } 3006 if(options.after && !options.disableAfterReplyCounter) { 3007 if(options.after.getElementsByClassName('tweet-self-thread-div')[0]) options.after.getElementsByClassName('tweet-self-thread-div')[0].hidden = true; 3008 if(!options.after.classList.contains('tweet-main')) options.after.getElementsByClassName('tweet-interact-reply')[0].innerText = (+options.after.getElementsByClassName('tweet-interact-reply')[0].innerText - 1).toString(); 3009 else options.after.getElementsByClassName('tweet-footer-stat-replies')[0].innerText = (+options.after.getElementsByClassName('tweet-footer-stat-replies')[0].innerText - 1).toString(); 3010 } 3011 }); 3012 if(tweetInteractMoreMenuPin) tweetInteractMoreMenuPin.addEventListener('click', async () => { 3013 if(pinnedTweet && pinnedTweet.id_str === t.id_str) { 3014 await API.tweet.unpin(t.id_str); 3015 pinnedTweet = null; 3016 tweet.remove(); 3017 let tweetTime = new Date(t.created_at).getTime(); 3018 let beforeTweet = Array.from(document.getElementsByClassName('tweet')).find(i => { 3019 let timestamp = +i.getElementsByClassName('tweet-time')[0].dataset.timestamp; 3020 return timestamp < tweetTime; 3021 }); 3022 if(beforeTweet) { 3023 appendTweet(t, timelineContainer, { after: beforeTweet, disableAfterReplyCounter: true }); 3024 } 3025 return; 3026 } else { 3027 await API.tweet.pin(t.id_str); 3028 pinnedTweet = t; 3029 let pinnedTweetElement = Array.from(document.getElementsByClassName('tweet')).find(i => { 3030 let topText = i.getElementsByClassName('tweet-top-text')[0]; 3031 return (topText && topText.className.includes('pinned')); 3032 }); 3033 if(pinnedTweetElement) { 3034 pinnedTweetElement.remove(); 3035 } 3036 tweet.remove(); 3037 appendTweet(t, timelineContainer, { 3038 prepend: true, 3039 top: { 3040 text: LOC.pinned_tweet.message, 3041 icon: "\uf003", 3042 color: "var(--link-color)", 3043 class: "pinned" 3044 } 3045 }); 3046 return; 3047 } 3048 }); 3049 } 3050 tweetInteractMoreMenuRefresh.addEventListener('click', async () => { 3051 let tweetData; 3052 try { 3053 tweetData = await API.tweet.getV2(t.id_str); 3054 } catch (e) { 3055 console.error(e); 3056 return; 3057 } 3058 if (!tweetData) { 3059 return; 3060 } 3061 if(typeof timeline !== 'undefined') { 3062 let tweetIndex = timeline.data.findIndex(tweet => tweet.id_str === t.id_str); 3063 if (tweetIndex !== -1) { 3064 timeline.data[tweetIndex] = tweetData; 3065 } 3066 } 3067 if (tweetInteractFavorite.className.includes('tweet-interact-favorited') && !tweetData.favorited) { 3068 tweetInteractFavorite.classList.remove('tweet-interact-favorited'); 3069 } 3070 if (tweetInteractRetweet.className.includes('tweet-interact-retweeted') && !tweetData.retweeted) { 3071 tweetInteractRetweet.classList.remove('tweet-interact-retweeted'); 3072 } 3073 if (!tweetInteractFavorite.className.includes('tweet-interact-favorited') && tweetData.favorited) { 3074 tweetInteractFavorite.classList.add('tweet-interact-favorited'); 3075 } 3076 if (!tweetInteractRetweet.className.includes('tweet-interact-retweeted') && tweetData.retweeted) { 3077 tweetInteractRetweet.classList.add('tweet-interact-retweeted'); 3078 } 3079 if(!options.mainTweet) { 3080 tweetInteractFavorite.innerText = tweetData.favorite_count; 3081 tweetInteractRetweet.innerText = tweetData.retweet_count; 3082 tweetInteractReply.innerText = tweetData.reply_count; 3083 } 3084 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 3085 }); 3086 tweetInteractMoreMenuMute.addEventListener('click', async () => { 3087 if(t.conversation_muted) { 3088 await API.tweet.unmute(t.id_str); 3089 toast.info(LOC.unmuted_convo.message); 3090 t.conversation_muted = false; 3091 tweetInteractMoreMenuMute.innerText = LOC.mute_convo.message; 3092 } else { 3093 await API.tweet.mute(t.id_str); 3094 toast.info(LOC.muted_convo.message); 3095 t.conversation_muted = true; 3096 tweetInteractMoreMenuMute.innerText = LOC.unmute_convo.message; 3097 } 3098 chrome.storage.local.set({tweetReplies: {}, tweetDetails: {}}, () => {}); 3099 }); 3100 let downloading = false; 3101 if (t.extended_entities && t.extended_entities.media.length > 0) { 3102 tweetInteractMoreMenuDownload.addEventListener('click', () => { 3103 if (downloading) return; 3104 downloading = true; 3105 t.extended_entities.media.forEach((item, index) => { 3106 let url = item.type === 'photo' ? item.media_url_https : item.video_info.variants[0].url; 3107 url = new URL(url); 3108 if (item.type === 'photo') { 3109 url.searchParams.set("name", "orig"); // force original resolution 3110 } 3111 fetch(url).then(res => res.blob()).then(blob => { 3112 downloading = false; 3113 let a = document.createElement('a'); 3114 a.href = URL.createObjectURL(blob); 3115 3116 let ts = new Date(t.created_at).toISOString().split("T")[0]; 3117 let extension = url.pathname.split('.').pop(); 3118 let _index = t.extended_entities.media.length > 1 ? "_"+(index+1) : ""; 3119 let filename = `${t.user.screen_name}_${ts}_${t.id_str}${_index}.${extension}`; 3120 a.download = filename; 3121 3122 a.click(); 3123 a.remove(); 3124 }).catch(e => { 3125 downloading = false; 3126 console.error(e); 3127 }); 3128 }); 3129 }); 3130 } 3131 if (t.extended_entities && t.extended_entities.media.some(m => m.type === 'animated_gif')) { 3132 tweetInteractMoreMenuDownloadGifs.forEach(dgb => dgb.addEventListener('click', e => { 3133 if (downloading) return; 3134 downloading = true; 3135 let n = parseInt(e.target.dataset.gifno)-1; 3136 let videos = Array.from(tweet.getElementsByClassName('tweet-media-gif')); 3137 let video = videos[n]; 3138 let canvas = document.createElement('canvas'); 3139 canvas.width = video.videoWidth; 3140 canvas.height = video.videoHeight; 3141 let ctx = canvas.getContext('2d'); 3142 if (video.duration > 10 && !confirm(LOC.long_vid.message)) { 3143 return downloading = false; 3144 } 3145 let mde = tweet.getElementsByClassName('tweet-media-data')[0]; 3146 mde.innerText = LOC.initialization.message + '...'; 3147 let gif = new GIF({ 3148 workers: 4, 3149 quality: 15, 3150 debug: true 3151 }); 3152 video.currentTime = 0; 3153 video.loop = false; 3154 let isFirst = true; 3155 let interval = setInterval(async () => { 3156 if(isFirst) { 3157 video.currentTime = 0; 3158 isFirst = false; 3159 await sleep(5); 3160 } 3161 mde.innerText = `${LOC.initialization.message}... (${Math.round(video.currentTime/video.duration*100|0)}%)`; 3162 if (video.currentTime+0.1 >= video.duration) { 3163 clearInterval(interval); 3164 gif.on('working', (frame, frames) => { 3165 mde.innerText = `${LOC.converting.message}... (${frame}/${frames})`; 3166 }); 3167 gif.on('finished', blob => { 3168 mde.innerText = ''; 3169 let a = document.createElement('a'); 3170 a.href = URL.createObjectURL(blob); 3171 a.download = `${t.id_str}.gif`; 3172 document.body.append(a); 3173 a.click(); 3174 a.remove(); 3175 downloading = false; 3176 video.loop = true; 3177 video.play(); 3178 }); 3179 gif.render(); 3180 return; 3181 } 3182 ctx.drawImage(video, 0, 0, canvas.width, canvas.height); 3183 let imgData = ctx.getImageData(0, 0, canvas.width, canvas.height); 3184 gif.addFrame(imgData, { delay: 100 }); 3185 }, 100); 3186 })); 3187 } 3188 if(tweetInteractMoreMenuFeedbacks) tweetInteractMoreMenuFeedbacks.forEach(feedbackButton => { 3189 let feedback = t.feedback[feedbackButton.dataset.index]; 3190 if (!feedback) return; 3191 feedbackButton.addEventListener('click', () => { 3192 chrome.storage.local.remove(["algoTimeline"], () => {}); 3193 if(feedback.richBehavior && feedback.richBehavior.markNotInterestedTopic) { 3194 fetch(`https://twitter.com/i/api/graphql/OiKldXdrDrSjh36WO9_3Xw/TopicNotInterested`, { 3195 method: 'post', 3196 headers: { 3197 'content-type': 'application/json', 3198 'authorization': OLDTWITTER_CONFIG.public_token, 3199 "x-twitter-active-user": 'yes', 3200 "x-csrf-token": OLDTWITTER_CONFIG.csrf, 3201 "x-twitter-auth-type": 'OAuth2Session', 3202 }, 3203 body: JSON.stringify({"variables":{"topicId": feedback.richBehavior.markNotInterestedTopic.topicId,"undo":false},"queryId":"OiKldXdrDrSjh36WO9_3Xw"}), 3204 credentials: 'include' 3205 }).then(i => i.json()).then(() => {}); 3206 } 3207 fetch(`https://twitter.com/i/api${feedback.feedbackUrl}`, { 3208 method: 'post', 3209 headers: { 3210 'content-type': 'application/x-www-form-urlencoded', 3211 'authorization': OLDTWITTER_CONFIG.public_token, 3212 "x-twitter-active-user": 'yes', 3213 "x-csrf-token": OLDTWITTER_CONFIG.csrf, 3214 "x-twitter-auth-type": 'OAuth2Session', 3215 }, 3216 body: `feedback_type=${feedback.feedbackType}&feedback_metadata=${t.feedbackMetadata}&undo=false`, 3217 credentials: 'include' 3218 }).then(i => i.json()).then(i => { 3219 alert(feedback.confirmation ? feedback.confirmation : LOC.feedback_thanks.message); 3220 tweet.remove(); 3221 }); 3222 }); 3223 }); 3224 3225 if(options.after) { 3226 options.after.after(tweet); 3227 } else if (options.before) { 3228 options.before.before(tweet); 3229 } else if (options.prepend) { 3230 timelineContainer.prepend(tweet); 3231 } else { 3232 timelineContainer.append(tweet); 3233 } 3234 if(vars.enableTwemoji) twemoji.parse(tweet); 3235 return tweet; 3236 } catch(e) { 3237 console.error(e); 3238 if(Date.now() - lastTweetErrorDate > 1000) { 3239 lastTweetErrorDate = Date.now(); 3240 createModal(` 3241 <div style="max-width:700px"> 3242 <span style="font-size:14px;color:var(--default-text-color)"> 3243 <h2 style="margin-top: 0">${LOC.something_went_wrong.message}</h2> 3244 ${LOC.tweet_error.message}<br> 3245 ${LOC.error_instructions.message.replace('$AT1$', "<a target='_blank' href='https://github.com/dimdenGD/OldTwitter/issues'>").replace(/\$AT2\$/g, '</a>').replace("$AT3$", "<a target='_blank' href='mailto:admin@dimden.dev'>")} 3246 </span> 3247 <div class="box" style="font-family:monospace;line-break: anywhere;padding:5px;margin-top:5px;background:rgba(255, 0, 0, 0.1);color:#ff4545"> 3248 ${escapeHTML(e.stack ? e.stack : String(e))} at ${t.id_str} (OldTwitter v${chrome.runtime.getManifest().version}) 3249 </div> 3250 </div> 3251 `); 3252 } 3253 return null; 3254 } 3255} 3256 3257function replaceAll(str, find, replace) { 3258 return str.split(find).join(replace); 3259}