NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Twitter HQ Photo Downloader // @version 1.0 // @description Click button to download HQ photo from twitter.com. // @author navchandar // @homepage https://github.com/navchandar // @copyright 2020, navchandar (https://openuserjs.org/users/navchandar) // @grant GM_openInTab // @grant GM_download // @grant GM_addStyle // @match *://*.twitter.com/* // @run-at document-end // @require http://code.jquery.com/jquery-3.4.1.min.js // @updateURL https://openuserjs.org/meta/navchandar/Twitter_HQ_Photo_Downloader.meta.js // @downloadURL https://openuserjs.org/install/navchandar/Twitter_HQ_Photo_Downloader.user.js // @icon  // @license MIT // ==/UserScript== function getElementsByXPath(xpath, parent) { let results = []; let query = document.evaluate(xpath, parent || document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (let i = 0, length = query.snapshotLength; i < length; ++i) { results.push(query.snapshotItem(i)); } return results }; function has(String, search) { try { if (String.indexOf(search) > -1) { return true; } } catch (err) {} return false; } function saveStory(zEvent) { let parent = document.elementFromPoint(innerWidth / 2, innerHeight / 2).parentNode.parentNode.parentNode.parentNode, video = parent.querySelector("video source"), image = parent.querySelector("img"); if (!video && !image) { parent = document.elementFromPoint(innerWidth / 2, innerHeight / 2).parentNode.parentNode.parentNode.parentNode.parentNode.parentNode; video = parent.querySelector("video source"); image = parent.querySelector("img"); } if (video) { GM_download(video.getAttribute("src"), "Twitter_vid.mp4"); } else if (image) { GM_download(image.getAttribute("src").replace('format=jpg', 'format=png').replace('name=small', 'name=large'), "Twitter_img.png"); } else { console.log("Nothing img/video found to save.") } } function openStory(zEvent) { let parent = document.elementFromPoint(innerWidth / 2, innerHeight / 2).parentNode.parentNode.parentNode.parentNode, video = parent.querySelector("video source"), image = parent.querySelector("img"); if (!video && !image) { parent = document.elementFromPoint(innerWidth / 2, innerHeight / 2).parentNode.parentNode.parentNode.parentNode.parentNode.parentNode; video = parent.querySelector("video source"); image = parent.querySelector("img"); } if (video) { GM_openInTab(video.getAttribute("src"), "Twitter_vid.mp4"); } else if (image) { GM_openInTab(image.getAttribute("src").replace('format=jpg', 'format=png').replace('name=small', 'name=large'), "Twitter_img.png"); } else { console.log("Nothing img/video found to open.") } } function AddButton() { // Add a button element on div var zNode = document.createElement('div'); zNode.innerHTML = '<button id="openButton" title="Click to open this photo" type="button">Open</button>'; zNode.setAttribute('id', 'myContainer1'); document.body.appendChild(zNode); zNode = document.createElement('div'); zNode.innerHTML = '<button id="saveButton" title="Click to download this photo" type="button">Save</button>'; zNode.setAttribute('id', 'myContainer2'); document.body.appendChild(zNode); //--- Activate the newly added button. document.getElementById("saveButton").addEventListener("click", saveStory, false); document.getElementById("openButton").addEventListener("click", openStory, false); //--- Style our newly added element using CSS. GM_addStyle(` #myContainer1{position:fixed;bottom:.25em;left:51%;right:48%;margin:0;width:5%;height:6%;text-align:center} #myContainer2{position:fixed;bottom:.25em;left:47%;right:48%;margin:0;width:5%;height:6%;text-align:center} #openButton{opacity:.55;cursor:pointer;background-color:#262626;color:White;font-size:12px;border-radius:7px;padding:2.5px} #saveButton{opacity:.55;cursor:pointer;background-color:#262626;color:White;font-size:12px;border-radius:7px;padding:2.5px} #saveButton:hover,#openButton:hover{color:White;opacity:1} `); } function HideBtn() { document.getElementById('openButton').style.visibility = 'hidden'; document.getElementById('saveButton').style.visibility = 'hidden'; } function UnHideBtn() { document.getElementById('openButton').style.visibility = 'visible'; document.getElementById('saveButton').style.visibility = 'visible'; } (function () { 'use strict'; AddButton(); HideBtn(); // call this every 2 seconds to update according to page load setInterval(function () { var hostURL = window.location.href; if (has(hostURL, 'twitter.com') && has(hostURL, 'photo')) { UnHideBtn(); } else { HideBtn(); } }, 2000); })();