// ==UserScript== // @name Twitter Media Downloader // @name:ja Twitter Media Downloader // @name:zh-cn Twitter 媒体下载 // @name:zh-tw Twitter 媒體下載 // @description Save Video/Photo by One-Click. // @description:ja ワンクリックで動画・画像を保存する。 // @description:zh-cn 一键保存视频/图片 // @description:zh-tw 一鍵保存視頻/圖片 // @version 0.51 // @author AMANE // @namespace none // @match https://twitter.com/* // @grant GM_setValue // @grant GM_getValue // @grant GM_download // @downloadURL none // ==/UserScript== /* jshint esversion: 8 */ (function () { 'use strict'; const preset_filename = 'twitter_{user-name}(@{user-id})_{date-time}_{status-id}_{file-type}'; const language = { en: {download: 'Download', completed: 'Download Completed', settings: 'Download Settings', save: 'Save', confirm: 'Confirm Save As Dialog', record: 'Remember Download History', clear: '(Clear)', clear_confirm: 'Clear download history?', pattern: 'File Name Pattern'}, ja: {download: 'ダウンロード', completed: 'ダウンロード完了', settings: 'ダウンロード設定', save: '保存', confirm: '保存場所を確認する', record: 'ダウンロード履歴を保存する', clear: '(クリア)', clear_confirm: 'ダウンロード履歴を削除する?', pattern: 'ファイル名パターン'}, zh: {download: '下载', completed: '下载完成', settings: '下载设置', save: '保存', confirm: '确认文件名和保存位置', record: '保存下载记录', clear: '(清除)', clear_confirm: '确认要清除下载记录?', pattern: '文件名格式'}, 'zh-Hant': {download: '下載', completed: '下載完成', settings: '下載設置', save: '保存', confirm: '確認文件名和保存位置', record: '保存下載記錄', clear: '(清除)', clear_confirm: '確認要清除下載記錄?', pattern: '文件名規則'}, }; const str = language[document.querySelector('html').lang] || language.en; const svg = ` `; const css = ``; let history = storage('history'); document.head.insertAdjacentHTML('beforeend', css); new MutationObserver(mutations => mutations.forEach(mutation => { mutation.addedNodes.forEach(node => { btn_inject(node.tagName == 'DIV' && node.querySelector('article')); }); })).observe(document.body, {childList: true, subtree: true}); function btn_inject(article) { if (article && article.querySelector('div[role="progressbar"], div[data-testid="playButton"], a[href*="/photo/1"], a[href="/settings/safety"]')) { let status_id = article.querySelector('a[href*="/status/"]').href.split('/status/').pop().split('/').shift(); let is_exist = history.indexOf(status_id) >= 0; let group = article.querySelector('div[role="group"]'); let btn = group.querySelector(':scope>:last-child').cloneNode(true); btn.querySelector('svg').innerHTML = svg; group.appendChild(btn); btn_status(btn, is_exist ? 'completed' : 'download', is_exist ? str.completed : str.download); btn.onclick = () => btn_click(btn, status_id, is_exist); btn.oncontextmenu = e => { e.preventDefault(); down_settings(); }; } } async function btn_click(btn, status_id, is_exist) { if (btn.classList.contains('loading')) return; btn_status(btn, 'loading'); let filename = (await GM_getValue('filename', preset_filename)).split('\n').join(''); let confirm = await GM_getValue('confirm', false); let record = await GM_getValue('record', true); let json = await fetch_json(status_id); let tweet = json.globalObjects.tweets[status_id]; let user = json.globalObjects.users[tweet.user_id_str]; let info = { 'status-id': status_id, 'user-name': user.name, 'user-id': user.screen_name, 'date-time': formatDate(tweet.created_at, 'YYYYMMDD-hhmmss') }; let medias = tweet.extended_entities && tweet.extended_entities.media; if (medias) { let tasks = medias.length; medias.forEach((media, i) => { info.url = media.type == 'photo' ? media.media_url + ':orig' : media.video_info.variants.filter(n => n.content_type == 'video/mp4').sort((a, b) => b.bitrate - a.bitrate)[0].url; info.file = info.url.split('/').pop().split(/[:?]/).shift(); info['file-name'] = info.file.split('.').shift(); info['file-ext'] = info.file.split('.').pop(); info['file-type'] = media.type.replace('animated_', ''); info.out = (filename.replace(/\.?{file-ext}/, '') + (medias.length > 1 && !filename.match('{file-name}') ? '-' + i : '') + '.{file-ext}').replace(/{([^{}]+)}/g, (match, name) => info[name]); GM_download({ url: info.url, name: info.out, // saveAs: confirm, onload: () => { tasks -= 1; if (tasks === 0) { btn_status(btn, 'completed', str.completed); if (record && !is_exist) { history.push(status_id); storage('history', status_id); } } }, onerror: result => { tasks = - 1; btn_status(btn, 'failed', result.details.current); } }); }); } else { btn_status(btn, 'failed', 'MEDIA_NOT_FOUND'); } } function btn_status(btn, css, title) { btn.classList.remove('tmd-down', 'download', 'completed', 'loading', 'failed'); btn.classList.add('tmd-down', css); if (title) btn.title = title; } async function down_settings() { const $element = (parent, tag, style, content, css) => { let el = document.createElement(tag); if (style) el.style.cssText = style; if (typeof content !== 'undefined') { if (tag == 'input') { if (content == 'checkbox') el.type = content; else el.value = content; } else el.innerHTML = content; } if (css) css.split(' ').forEach(c => el.classList.add(c)); parent.appendChild(el); return el; }; let wapper = $element(document.body, 'div', 'position: fixed; left: 0px; top: 0px; width: 100%; height: 100%; background-color: #0009; z-index: 10;'); let wapper_close; wapper.onmousedown = e => { wapper_close = e.target == wapper; }; wapper.onmouseup = e => { if (wapper_close && e.target == wapper) wapper.remove(); }; let dialog = $element(wapper, 'div', 'position: absolute; left: 50%; top: 50%; transform: translateX(-50%) translateY(-50%); width: fit-content; width: -moz-fit-content; background-color: #f3f3f3; border: 1px solid #ccc; border-radius: 10px;'); let title = $element(dialog, 'h3', 'margin: 10px 20px;', str.settings); let options = $element(dialog, 'div', 'margin: 10px; border: 1px solid #ccc; border-radius: 5px;'); // let confirm_input = $element($element(options, 'label', 'display: block; margin: 10px;', str.confirm), 'input', 'float: left;', 'checkbox'); // confirm_input.checked = await GM_getValue('confirm', false); // confirm_input.onchange = () => GM_setValue('confirm', confirm_input.checked); let record_label = $element(options, 'label', 'display: block; margin: 10px;', str.record); let record_input = $element(record_label, 'input', 'float: left;', 'checkbox'); record_input.checked = await GM_getValue('history', true); record_input.onchange = () => GM_setValue('history', record_input.checked); $element(record_label, 'label', 'margin: 10px; color: blue;', str.clear).onclick = () => { if (confirm(str.clear_confirm)) { history = []; localStorage.removeItem('history'); } }; let filename_div = $element(dialog, 'div', 'margin: 10px; border: 1px solid #ccc; border-radius: 5px;'); let filename_label = $element(filename_div, 'label', 'display: block; margin: 10px 15px;', str.pattern); let filename_input = $element(filename_label, 'textarea', 'display: block; min-width: 500px; max-width: 500px; min-height: 100px; font-size: inherit;', await GM_getValue('filename', preset_filename)); let filename_tags = $element(filename_div, 'label', 'display: table; margin: 10px;', ` {user-name} {user-id} {status-id} {date-time}
{file-type} {file-name} {file-ext} `); filename_input.selectionStart = filename_input.value.length; filename_tags.querySelectorAll('.tmd-tag').forEach(tag => { tag.onclick = () => { let ss = filename_input.selectionStart; let se = filename_input.selectionEnd; filename_input.value = filename_input.value.substring(0, ss) + tag.innerText + filename_input.value.substring(se); filename_input.selectionStart = ss + tag.innerText.length; filename_input.selectionEnd = ss + tag.innerText.length; filename_input.focus(); }; }); let btn_save = $element(title, 'label', 'float: right;', str.save, 'tmd-btn'); btn_save.onclick = async() => { await GM_setValue('filename', filename_input.value); wapper.remove(); }; } function storage(name, value) { let data = JSON.parse(localStorage.getItem(name) || '[]'); if (value) data.push(value); else return data; localStorage.setItem(name, JSON.stringify(data)); } async function fetch_json(status_id) { let url = 'https://twitter.com/i/api/2/timeline/conversation/' + status_id + '.json?tweet_mode=extended&include_entities=false&include_user_entities=false'; let cookies = getCookie(); let headers = { 'authorization': 'Bearer AAAAAAAAAAAAAAAAAAAAANRILgAAAAAAnNwIzUejRCOuH5E6I8xnZz4puTs%3D1Zv7ttfk8LF81IUq16cHjhLTvJu4FA33AGWWjCpTnA', 'x-twitter-active-user': 'yes', 'x-twitter-client-language': cookies.lang, 'x-csrf-token': cookies.ct0 }; if (cookies.ct0.length == 32) headers['x-guest-token'] = cookies.gt; return await fetch(url, {headers: headers}).then(result => result.json()); } function getCookie(name) { let cookies = {}; document.cookie.split(';').filter(n => n.indexOf('=') > 0).forEach(n => { n.replace(/^([^=]+)=(.+)$/, (match, name, value) => { cookies[name.trim()] = value.trim(); }); }); return name ? cookies[name] : cookies; } function formatDate(i, o) { let d = new Date(i); let v = { YYYY: d.getUTCFullYear().toString(), YY: d.getUTCFullYear().toString(), MM: '0' + (d.getUTCMonth() + 1), DD: '0' + d.getUTCDate(), hh: '0' + d.getUTCHours(), mm: '0' + d.getUTCMinutes(), ss: '0' + d.getUTCSeconds() }; return o.replace(/(YY(YY)?|MM|DD|hh|mm|ss)/g, n => v[n].substr( - n.length)); } })();