Greasy Fork is available in English.

Sketchful Scripts Menu

Adds a menu with every Sketchful script from Greasy Fork

// ==UserScript==
// @name        Sketchful Scripts Menu
// @namespace   https://greasyfork.org/users/281093
// @match       *://sketchful.io/
// @grant       GM.xmlHttpRequest
// @version     1.1.2
// @author      Bell
// @license     MIT
// @copyright   2020, Bell
// @connect     greasyfork.org
// @run-at      document-body
// @description Adds a menu with every Sketchful script from Greasy Fork
// ==/UserScript==
/* jshint esversion: 8 */

// style by High
const css = `
	@import url(https://fonts.googleapis.com/css2?family=Londrina+Solid&display=swap);
	.content-wrapper {
		height: 490px
	}

	.dark .content-wrapper * {
		scrollbar-width: thin;
		scrollbar-color: rgba(200, 200, 200, .4) transparent;
	}

	.content-wrapper * {
		scrollbar-width: thin;
		scrollbar-color: rgba(10, 10, 10, .4) transparent;
	}

	.plugin-button-container {
		width: 100%;
		padding: 15px 30px;
		height: inherit;
		overflow-x: hidden;
		overflow-y: scroll
	}

	.dark .plugin-button {
		background-color: rgb(0 0 0 / .15)
	}

	.plugin-button {
		width: 100%;
		margin-bottom: 14px;
		height: 80px;
		background-color: rgb(152 191 216 / 24%);
		overflow: hidden;
		border-radius: 5px
	}

	.script-title-author {
		height: 100%;
		width: 230px;
		float: left
	}

	.script-title {
		width: 100%;
		height: 40px;
		text-align: center;
		float: left;
		line-height: 56px;
		font-size: 22px;
		font-family: 'Londrina Solid', cursive;
		overflow: hidden
	}

	.script-author {
		width: 100%;
		height: 40px;
		text-align: center;
		float: left;
		line-height: 30px
	}

	.script-info {
		padding: 10px 0;
		height: calc(100% - 10px);
		width: calc(100% - 340px);
		float: left;
		overflow: scroll;
		overflow-x: hidden;
		margin-top: 5px
	}

	.script-version {
		height: 100%;
		width: 100px;
		float: left;
		text-align: center;
		line-height: 80px;
		font-weight: 800
	}

	.content-wrapper ::-webkit-scrollbar-thumb {
		background-color: rgba(10, 10, 10, .4);
		border-radius: 4px;
	}

	.content-wrapper ::-webkit-scrollbar {
		width: 6px
	}

	.dark .content-wrapper ::-webkit-scrollbar-thumb {
		background-color: rgba(200, 200, 200, .4);
		border-radius: 4px
	}
`;

const scriptContainer = document.createElement('div');
const greasyForkURL = 'https://greasyfork.org/en/scripts/by-site/sketchful.io';

(function init() {
	addCSS(css);
	addMenuButton();
	addContainer();
	addScripts();
})();

async function addScripts() {
	let page = 1;
	do {
		await getScripts(`${greasyForkURL}?page=${page}`);
	} while (scriptContainer.childElementCount === 50 * page++);
}

function addMenuButton() {
	const menu = document.querySelector('#menu > div.menuNav > ul');
	const button = menu.lastChild.cloneNode(true);
	const linkTag = button.querySelector('a');
	const linkSpan = linkTag.lastChild;

	linkTag.href = '#menuScripts';
	linkSpan.textContent = 'Scripts';
	linkTag.firstChild.remove();

	const linkImage = document.createElement('img');
	linkImage.style.marginRight = '7px';
	linkImage.src = '';
	linkTag.insertBefore(linkImage, linkSpan);
	menu.appendChild(button);
}

function addContainer() {
	const contentWrapper = document.createElement('div');
	const menuScripts = document.createElement('div');

	menuScripts.id = 'menuScripts';
	contentWrapper.classList.add('content-wrapper');
	scriptContainer.classList.add('plugin-button-container');

	contentWrapper.appendChild(scriptContainer);
	menuScripts.appendChild(contentWrapper);
	document.querySelector('.menuTabs').appendChild(menuScripts);
}

async function getScripts(url) {
	const html = await getHTML(url);
	const scriptList = html.querySelector('.script-list');
	scriptList.childNodes.forEach(addScript);
}

function addScript(script) {
	const scriptObj = getScriptData(script);
	if (!scriptObj) return;

	const {
		scriptName,
		scriptDescription,
		scriptVersion,
		scriptLink,
		authorLink,
		authorNames
	} = scriptObj;

	const scriptDiv = document.createElement('div');
	scriptDiv.classList.add('plugin-button');
	scriptDiv.innerHTML = `
		<div class='script-title-author'>
			<a class='script-title' href="${scriptLink}" target="_blank">${scriptName}</a>
			<div class='script-author'>By: <a href="${authorLink}" target="_blank"><b>${authorNames}</b></a></div>
		</div>
		<div class='script-info'>${scriptDescription}</div>
		<div class='script-version'>v${scriptVersion}</div>
	`;
	scriptContainer.appendChild(scriptDiv);
}

function getScriptData(script) {
	const scriptData = script.dataset;
	if (!scriptData || !scriptData.hasOwnProperty('scriptId')) {return null;}

	const descriptionSpan = script.querySelector('.description');
	const authorsObject = JSON.parse(scriptData.scriptAuthors);

	scriptData.scriptDescription = descriptionSpan.textContent.trim();
	scriptData.scriptLink = `https://greasyfork.org/en/scripts/${scriptData.scriptId}`;
	scriptData.authorLink = `https://greasyfork.org/en/users/${Object.keys(authorsObject)[0]}`;
	scriptData.authorNames = Object.values(authorsObject).join(', ');

	return scriptData;
}

function getHTML(url) {
	return new Promise((resolve, reject) => {
		GM.xmlHttpRequest({
			method: 'GET',
			url: url,
			onload: (res) => {
				const html = document.createElement('html');
				html.innerHTML = res.responseText;
				resolve(html);
			},
			onerror: reject
		});
	});
}

function addCSS(style) {
	const stylesheet = document.createElement('style');
	stylesheet.type = 'text/css';
	stylesheet.innerText = style;
	document.head.appendChild(stylesheet);
}