// ==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
// @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);
})();