NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @name Github Gist Share // @namespace https://github.com/jerone/UserScripts/ // @description Share your GitHub Gist to Twitter, Dabblet, Bl.ocks & as userscript. // @author jerone // @copyright 2014+, jerone (https://github.com/jerone) // @license CC-BY-NC-SA-4.0; https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode // @license GPL-3.0-or-later; http://www.gnu.org/licenses/gpl-3.0.txt // @homepage https://github.com/jerone/UserScripts/tree/master/Github_Gist_Share // @homepageURL https://github.com/jerone/UserScripts/tree/master/Github_Gist_Share // @downloadURL https://github.com/jerone/UserScripts/raw/master/Github_Gist_Share/157850.user.js // @updateURL https://github.com/jerone/UserScripts/raw/master/Github_Gist_Share/157850.user.js // @supportURL https://github.com/jerone/UserScripts/issues // @contributionURL https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=VCYMHWQ7ZMBKW // @icon https://github.githubassets.com/pinned-octocat.svg // @include *://gist.github.com/* // @version 5.1 // @grant none // ==/UserScript== // cSpell:ignore Dabblet, Bl.ocks, itemprop, tweetbutton /* eslint security/detect-object-injection: "off" */ (function () { String.format = function (string) { const args = Array.prototype.slice.call(arguments, 1, arguments.length); return string.replace(/{(\d+)}/g, function (match, number) { return typeof args[number] !== "undefined" ? args[number] : match; }); }; function Menu(container) { const div$0$0 = document.createElement("div"); div$0$0.classList.add("file-navigation-option"); div$0$0.id = "Github_Gist_Share"; container.insertBefore(div$0$0, container.firstChild); const div$1$0 = document.createElement("div"); div$1$0.classList.add( "select-menu", "js-menu-container", "select-menu-modal-left", "js-select-menu", ); div$0$0.appendChild(div$1$0); const button$2$0 = document.createElement("button"); button$2$0.classList.add( "btn", "btn-sm", "select-menu-button", "icon-only", "js-menu-target", ); button$2$0.setAttribute("type", "button"); div$1$0.appendChild(button$2$0); const svg$3$0 = document.createElementNS( "http://www.w3.org/2000/svg", "svg", ); svg$3$0.classList.add("octicon", "octicon-link-external"); svg$3$0.setAttributeNS(null, "height", 16); svg$3$0.setAttributeNS(null, "version", "1.1"); svg$3$0.setAttributeNS(null, "viewBox", "0 0 12 16"); svg$3$0.setAttributeNS(null, "width", 12); button$2$0.appendChild(svg$3$0); const path$4$0 = document.createElementNS( "http://www.w3.org/2000/svg", "path", ); path$4$0.setAttributeNS( null, "d", "M11 10h1v3c0 .55-.45 1-1 1H1c-.55 0-1-.45-1-1V3c0-.55.45-1 1-1h3v1H1v10h10v-3zM6 2l2.25 2.25L5 7.5 6.5 9l3.25-3.25L12 8V2H6z", ); svg$3$0.appendChild(path$4$0); button$2$0.appendChild(document.createTextNode(" Share ")); const div$2$1 = document.createElement("div"); div$2$1.classList.add("select-menu-modal-holder"); div$1$0.appendChild(div$2$1); const div$3$0 = document.createElement("div"); div$3$0.classList.add( "select-menu-modal", "select-menu-modal", "js-menu-content", ); div$2$1.appendChild(div$3$0); const div$4$0 = document.createElement("div"); div$4$0.classList.add("select-menu-header"); div$3$0.appendChild(div$4$0); const svg$5$0 = document.createElementNS( "http://www.w3.org/2000/svg", "svg", ); svg$5$0.classList.add("octicon", "octicon-x", "js-menu-close"); svg$5$0.setAttributeNS(null, "height", 16); svg$5$0.setAttributeNS(null, "version", "1.1"); svg$5$0.setAttributeNS(null, "viewBox", "0 0 12 16"); svg$5$0.setAttributeNS(null, "width", 12); div$4$0.appendChild(svg$5$0); const path$6$0 = document.createElementNS( "http://www.w3.org/2000/svg", "path", ); path$6$0.setAttributeNS( null, "d", "M7.48 8l3.75 3.75-1.48 1.48L6 9.48l-3.75 3.75-1.48-1.48L4.52 8 .77 4.25l1.48-1.48L6 6.52l3.75-3.75 1.48 1.48z", ); svg$5$0.appendChild(path$6$0); const span$5$1 = document.createElement("span"); span$5$1.classList.add("select-menu-title"); div$4$0.appendChild(span$5$1); span$5$1.appendChild(document.createTextNode("Share Gist with…")); const div$4$1 = document.createElement("div"); div$4$1.classList.add("select-menu-list", "js-navigation-container"); div$3$0.appendChild(div$4$1); this.itemsContainer = div$4$1; } Menu.prototype.AddItem = function (text, title, href, icon, newTab) { const a = document.createElement("a"); a.classList.add("select-menu-item", "js-navigation-item"); a.setAttribute("href", href); if (title) a.setAttribute("title", title); if (newTab) a.setAttribute("target", "_blank"); this.itemsContainer.appendChild(a); const i = document.createElement("img"); i.classList.add("select-menu-item-icon"); i.setAttribute("src", icon); a.appendChild(i); const s = document.createElement("span"); s.classList.add("select-menu-item-text"); a.appendChild(s); s.appendChild(document.createTextNode(text)); }; function getValue(elm) { return elm ? elm.textContent.trim() : ""; } function getIntValue(elm) { return elm ? parseInt(elm.textContent.trim(), 10) : 0; } function addMenu() { const link = document.querySelector(".gist-header-title a"); const nav = document.querySelector(".file-navigation-options"); if (link && nav) { // Check if we're on an actual gist const data = { url: link.href, user: getValue( document.querySelector(".header-nav-current-user strong"), ), author: getValue( document.querySelector('.author [itemprop="author"]'), ), description: getValue( document.querySelector(".repository-meta-content") || link, ), files: document.querySelectorAll(".file").length, stars: getIntValue( document.querySelector( 'a[href$="/stargazers"] .counter, form[action$="/star"] .social-count', ), ), forks: getIntValue( document.querySelector( 'a[href$="/forks"] .counter, form[action$="/fork"] .social-count', ), ), revisions: getIntValue( document.querySelector('a[href$="/revisions"] .counter'), ), }; console.log(data); const menu = new Menu(nav); // Twitter // eslint-disable-next-line no-constant-condition if (true) { const stats = []; if (data.files > 1) { stats.push(data.files + " files"); } if (data.stars === 1) { stats.push(data.stars + " star"); } else if (data.stars > 1) { stats.push(data.stars + " stars"); } if (data.forks === 1) { stats.push(data.forks + " fork"); } else if (data.forks > 1) { stats.push(data.forks + " forks"); } if (data.revisions > 1) { stats.push(data.revisions + " revisions"); } const tweet = String.format( "Check out {0} #gist {1} on @github {2}", data.author === data.user ? "my" : data.author + "'s", data.description ? '"' + data.description + '"' : "", stats.length > 0 ? String.format("- {0} -", stats.join(", ")) : "-", ); const link = "https://twitter.com/intent/tweet" + "?original_referer=" + encodeURIComponent(data.url) + "&source=tweetbutton&url=" + encodeURIComponent(data.url) + "&text=" + encodeURIComponent(tweet); const icon = ""; menu.AddItem( "Twitter", tweet + " " + data.url, link, icon, true, ); } // Userscripts if ( document.querySelector( '.file .file-actions a[href$=".user.js" i]', ) ) { const icon = ""; const userscripts = document.querySelectorAll( '.file .file-actions a[href$=".user.js"]', ); Array.prototype.forEach.call( userscripts, function (userscript) { const text = String.format( 'Userscript "{0}"', userscript.href.split("/").pop(), ); menu.AddItem(text, null, userscript.href, icon, false); }, ); } // Dabblet if ( document.querySelector( ".file .type-css, .file .type-html, .file .type-javascript", ) ) { const link = "http://dabblet.com/gist/" + data.url.split("/").pop(); const icon = ""; menu.AddItem("Dabblet", link, link, icon, true); } // Bl.ocks if ( document.querySelector( '.file .file-actions a[href$="index.html" i], .file .file-actions a[href$="README.md" i]', ) ) { const link = data.url.replace( "https://gist.github.com/", "https://bl.ocks.org/", ); const icon = ""; menu.AddItem("Bl.ocks", link, link, icon, true); } } } // Init addMenu(); // Pjax document.addEventListener("pjax:end", addMenu); })();