// ==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

Enabled IDE list

Opening options

Made with by @zvizvi
`, (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 = `

    Options

    `; 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(); }