// ==UserScript==
// @name Github Web IDE
// @name:zh-CN Github Web IDE
// @name:zh-TW Github Web IDE
// @name:en Github Web IDE
// @version 2.0.3
// @author zvizvi (migrated by Cesaryuan)
// @description ⚡ Open GitHub repositories in online web IDE (Migrated from https://github.com/zvizvi/GitHub-Web-IDE)
// @description:zh-CN ⚡ Open GitHub repositories in online web IDE (Migrated from https://github.com/zvizvi/GitHub-Web-IDE)
// @description:zh-TW ⚡ Open GitHub repositories in online web IDE (Migrated from https://github.com/zvizvi/GitHub-Web-IDE)
// @description:en ⚡ Open GitHub repositories in online web IDE (Migrated from https://github.com/zvizvi/GitHub-Web-IDE)
// @match *://github.com/*
// @match *://gitlab.com/*
// @resource css https://unpkg.com/@primer/css@^20.2.4/dist/primer.css
// @icon 
// @grant GM_registerMenuCommand
// @grant GM_getResourceText
// @grant GM_getValue
// @grant GM_setValue
// @license none
// @run-at document-end
// @namespace https://github.com/zvizvi/GitHub-Web-IDE
// @supportURL https://github.com/zvizvi/GitHub-Web-IDE
// @homepageURL https://github.com/zvizvi/GitHub-Web-IDE
// @downloadURL https://update.greasyfork.icu/scripts/454972/Github%20Web%20IDE.user.js
// @updateURL https://update.greasyfork.icu/scripts/454972/Github%20Web%20IDE.meta.js
// ==/UserScript==
(function () {
GM_registerMenuCommand("Options", () => {
const dialogId = "open-in-web-ide-options";
const dialog = document.getElementById(dialogId) || addFrameDialog(dialogId, `
Options
`, (iframe) => {
// other methods run javascript in iframe
// 1. iframe.contentWindow.eval;
// 2. inline script in html
// however, both of them are blocked by CSP with policy "script-src github.githubassets.com;"
optionFrameJS.call(iframe.contentWindow)
});
dialog.showModal();
});
const platform = location.host.split(".")[0];
const ideWebsitesList = [
{
title: "GitHub Dev",
name: "gitHubDev",
baseurl: "https://github.dev/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "VSCode Dev",
name: "vsCodeDev",
baseurl: "https://vscode.dev/github/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "CodeSandbox",
name: "codeSandbox",
baseurl: `https://codesandbox.io/s/${platform}/`,
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "GitHub1s",
name: "gitHub1s",
baseurl: "https://github1s.com/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "GitLab1s",
name: "gitLab1s",
baseurl: "https://gitlab1s.com/",
platforms: ["gitlab"],
openInNewTab: true,
icon: '',
},
{
title: "Repl.it",
name: "replit",
baseurl: `https://repl.it/${platform}/`,
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "StackBlitz",
name: "stackBlitz",
baseurl: "https://githubblitz.com/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "Gitpod",
name: "gitpod",
baseurl: `https://gitpod.io/#https://${platform}.com/`,
platforms: ["github", "gitlab"],
openInNewTab: true,
icon: '',
},
{
title: "Glitch",
name: "glitch",
baseurl: "https://glitch.com/edit/#!/import/github/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "Sourcegraph",
name: "sourcegraph",
baseurl: `https://sourcegraph.com/${platform}.com/`,
platforms: ["github", "gitlab"],
openInNewTab: true,
icon: '',
},
{
title: "Active Forks",
name: "activeForks",
baseurl: "https://techgaun.github.io/active-forks/index.html#",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "Gitpop2",
name: "gitpop2",
baseurl: "http://gitpop2.herokuapp.com/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "GitHub Memory",
name: "gitHubMemory",
baseurl: "https://githubmemory.com/repo/",
platforms: ["github"],
openInNewTab: true,
icon: '',
},
{
title: "Clone in VSCode",
name: "cloneInVSCode",
baseurl: `vscode://vscode.git/clone?url=https://${platform}.com/`,
platforms: ["github"],
icon: '',
class: "d-none d-md-block",
},
// {
// title: 'VSCode Remote Repositories',
// name: 'vsCodeRemoteRepositories',
// platforms: ['github'],
// baseurl: 'vscode://GitHub.remotehub/open?url=https://github.com/',
// icon: '',
// class: 'd-none d-md-block'
// }
];
const defaultOptions = {
gitHubDev: true,
vsCodeDev: true,
codeSandbox: true,
gitHub1s: true,
replit: true,
stackBlitz: true,
gitLab1s: true,
gitpod: true,
glitch: true,
sourcegraph: true,
activeForks: false,
gitpop2: false,
gitHubMemory: false,
cloneInVSCode: true,
vsCodeRemoteRepositories: false,
openInNewTab: true,
};
let repoUrlPath = getRepoUrlPath();
const hasPackageJson = [
...document.querySelectorAll(
".Details > .js-active-navigation-container > .Box-row a.js-navigation-open"
),
].some((el) => el.innerText === "package.json");
let options;
function init() {
const storage = GM_getValue("options");
options = storage || defaultOptions;
switch (platform) {
case "github": {
const itemWithBorderOnTop = ideWebsitesList.find(
// first item only
(item) =>
(item.name === "cloneInVSCode" ||
item.name === "vsCodeRemoteRepositories") &&
filterItems(item)
);
if (itemWithBorderOnTop) {
itemWithBorderOnTop.class += " border-top";
}
addGitHubSelectMenu();
document.addEventListener("soft-nav:success", () => {
console.log("soft-nav:success");
document.getElementById("open-in-web-ide")?.remove();
repoUrlPath = getRepoUrlPath();
addGitHubSelectMenu();
});
break;
}
case "gitlab": {
addGitLabSelectMenu();
localStorage.setItem("gl-web-ide-button-selected", "webide");
break;
}
default: {
break;
}
}
}
function getRepoUrlPath() {
return location.pathname.split("/").slice(1, 3).join("/");
}
function filterItems(item) {
if (item.title === "StackBlitz" && !hasPackageJson) {
return false;
}
return (!options || options[item.name]) && item.platforms.includes(platform);
}
function addGitHubSelectMenu() {
const menuElement = document.querySelector("#repo-content-turbo-frame .file-navigation");
if (!menuElement || menuElement.querySelector("#open-in-web-ide")) {
return;
}
const githubHtml = `
Open In Web IDE
IDE
`;
const detailsElement = document.createElement("details");
detailsElement.setAttribute(
"class",
"details-overlay details-reset position-relative d-block"
);
detailsElement.setAttribute("id", "open-in-web-ide");
detailsElement.innerHTML = githubHtml;
menuElement.appendChild(detailsElement);
}
function addGitLabSelectMenu() {
const webIDEDropdown = document.querySelector(
".tree-controls .gl-new-dropdown .dropdown-menu .gl-new-dropdown-contents"
);
if (!webIDEDropdown || document.querySelector("#open-in-web-ide")) {
return;
}
const gitLabHtml = `${ideWebsitesList
.filter(filterItems)
.map(
(item) =>
`
${item.icon}
${item.title}
`
)
.join("")}`;
webIDEDropdown.setAttribute("id", "open-in-web-ide");
webIDEDropdown.setAttribute("style", "--color-fg-default: currentColor");
webIDEDropdown.innerHTML = gitLabHtml;
}
window.addEventListener("load", () => {
init();
});
init();
})();
function addFrameDialog(dialogId, html, frameOnload) {
const dialog = document.createElement("dialog");
dialog.setAttribute("id", dialogId);
dialog.setAttribute(
"style",
"width: 60%; background-color: #fff; border: 1px solid #ccc; border-radius: 5px; padding: 10px; box-sizing: border-box;"
);
dialog.innerHTML = `
`;
const iframe = document.createElement("iframe");
iframe.setAttribute("style", "width: 100%; height: 950px; border: none; overflow: hidden;");
dialog.appendChild(iframe);
document.body.appendChild(dialog);
iframe.autofocus = true;
iframe.contentWindow.document.open();
iframe.contentWindow.document.write(html);
iframe.contentWindow.document.close();
frameOnload(iframe);
let closeBtn = dialog.querySelector("#dialog-close-gwi");
closeBtn.addEventListener("click", () => {
dialog.close();
});
// close dialog when click outside
dialog.addEventListener("click", (e) => {
if (e.target === dialog) {
dialog.close();
}
});
return dialog;
}
function optionFrameJS() {
let document = this.document;
console.log("optionFrameJS", document);
const defaultOptions = {
gitHubDev: true,
vsCodeDev: true,
codeSandbox: true,
gitHub1s: true,
gitLab1s: true,
replit: true,
stackBlitz: true,
gitpod: true,
glitch: true,
sourcegraph: true,
activeForks: false,
gitpop2: false,
gitHubMemory: false,
cloneInVSCode: true,
vsCodeRemoteRepositories: false,
openInNewTab: true,
};
let options = { ...defaultOptions };
// Load options from storage
const load = function () {
const storage = GM_getValue("options");
options = storage || defaultOptions;
show();
};
// Save options to storage
const save = function (object = options) {
if (!Object.keys(object).some((key) => key !== "openInNewTab" && options[key])) {
object = {
gitHubDev: true,
openInNewTab: object.openInNewTab,
};
}
GM_setValue("options", object);
load();
};
// Show options
const show = function () {
for (const key in defaultOptions) {
const ele = document.getElementById(key);
if (ele) {
switch (typeof defaultOptions[key]) {
case "boolean": {
ele.checked = options[key];
break;
}
case "string": {
ele.value = options[key];
break;
}
}
}
}
};
// On checkbox changed
document.addEventListener("input", (event) => {
if (event.target.type === "checkbox") {
for (const key in defaultOptions) {
const ele = document.getElementById(key);
if(!ele)continue;
switch (typeof defaultOptions[key]) {
case "boolean": {
options[key] = !!ele.checked;
break;
}
case "string": {
options[key] = ele.value;
break;
}
}
}
save(options);
}
});
// Reset to defaults
const reset = function () {
save(defaultOptions);
};
// On reset button clicked
document.addEventListener("click", (event) => {
if (event.target.id === "reset-button") {
reset();
}
});
load();
}