diff --git a/src/addons/addons/sounds-newgrounds-button/_manifest_entry.js b/src/addons/addons/sounds-newgrounds-button/_manifest_entry.js new file mode 100644 index 00000000000..fa93d4a4f22 --- /dev/null +++ b/src/addons/addons/sounds-newgrounds-button/_manifest_entry.js @@ -0,0 +1,28 @@ +/* generated by pull.js */ +const manifest = { + "name": "Newgrounds Audio Import", + "description": "Import audio directly from Newgrounds into the sound editor." + "credits": [ + { + "name": "SharkPool", + "link": "https://github.com/SharkPool-SP/" + } + ], + "info": [ + { + "type": "notice", + "text": "Some audio on Newgrounds may not be licensed for public or commercial use. This addon will try to warn you or block downloads when restrictions apply.", + "id": "copyright-notice" + } + ], + "userscripts": [ + { + "url": "userscript.js" + } + ], + "tags": ["editor", "new"], + "enabledByDefault": true, + "dynamicEnable": true, + "dynamicDisable": false +}; +export default manifest; \ No newline at end of file diff --git a/src/addons/addons/sounds-newgrounds-button/_runtime_entry.js b/src/addons/addons/sounds-newgrounds-button/_runtime_entry.js new file mode 100644 index 00000000000..bc67131957e --- /dev/null +++ b/src/addons/addons/sounds-newgrounds-button/_runtime_entry.js @@ -0,0 +1,5 @@ +/* generated by pull.js */ +import _js from "./userscript.js"; +export const resources = { + "userscript.js": _js, +}; diff --git a/src/addons/addons/sounds-newgrounds-button/userscript.js b/src/addons/addons/sounds-newgrounds-button/userscript.js new file mode 100644 index 00000000000..0f220cacf6c --- /dev/null +++ b/src/addons/addons/sounds-newgrounds-button/userscript.js @@ -0,0 +1,227 @@ +// Newgrounds Audio Button +// By: SharkPool +// Thanks Tom Fulp! :) + +export default async function() { + const ngIcon = ""; + const safeIcon = ""; + const warnIcon = ""; + const unsafeIcon = ""; + + const proxy1 = "https://corsproxy.io?url="; + const proxy2 = "https://api.codetabs.com/v1/proxy?quest="; + + let ngButtonElement; + + async function safeFetch(url, respondType) { + const proxies = [proxy1, proxy2]; + for (const proxy of proxies) { + try { + const response = await fetch(proxy + url); + + if (response.ok) return await response[respondType](); + if (response.status === 400) return undefined; + continue; + } catch (e) { + console.warn(`Failed to fetch ${url} with proxy: ${proxy}`, e); + } + } + return undefined; + } + + async function addTrack2Editor(url, name) { + const buffer = await safeFetch(url, "arrayBuffer"); + if (!buffer) { + alert("Failed to Fetch Song!"); + return; + } + + const storage = vm.runtime.storage; + const asset = storage.createAsset( + storage.AssetType.Sound, storage.DataFormat.MP3, + new Uint8Array(buffer), null, true + ); + + try { + await vm.addSound( + { + asset, name, + md5: asset.assetId + "." + asset.dataFormat, + }, + vm.editingTarget.id + ); + } catch (e) { + console.warn(e); + } + } + + async function openNewgroundsPopup() { + let url, name, songURL; + let infoBox = undefined; + + /* ScratchBlocks is availiable when this is called */ + const modal = await ScratchBlocks.customPrompt( + { title: "Newgrounds Audio" }, { content: { width: "500px" } }, + [ + { + name: "Add Track", role: "ok", callback: () => { + if (url && name && songURL) addTrack2Editor(songURL, `${name} -- ${author}`); + } + }, + { name: "Cancel", role: "close", callback: () => {} } + ] + ); + + const okayButton = modal.parentNode.querySelector(`button[class^="prompt_ok-button"]`); + okayButton.style.filter = "brightness(70%)"; + okayButton.style.pointerEvents = "none"; + + const label = document.createElement("div"); + label.innerHTML = `Import Newgrounds audio directly into your Project.
Not all tracks are fully free-to-use, read the report after searching.`; + label.setAttribute("style", "text-align: center; font-size: .85rem;"); + + const idInputDiv = document.createElement("div"); + idInputDiv.setAttribute("style", "width: 100%; margin: 15px 0; padding: 10px 20px; border-radius: 15px; border: dashed 2px grey; text-align: center; font-size: .85rem;"); + + const idLabel = document.createElement("b"); + idLabel.textContent = "Track ID/URL: "; + const idInput = document.createElement("input"); + + idInput.setAttribute("style", "margin-left: 5px; width: 70%; height: 25px; text-align: center; background: #ffffff20; border-radius: 15px; border: solid grey 1px;"); + idInput.type = "text"; + idInput.placeholder = "https://www.newgrounds.com/audio/listen/1395716"; + idInput.value = "1395716"; + url = idInput.placeholder; + idInput.addEventListener("change", (e) => { + url = String(e.target.value); + if (!url.startsWith("https://www.newgrounds.com/audio/listen/")) url = "https://www.newgrounds.com/audio/listen/" + url; + e.stopPropagation(); + }); + + const searchBtn = document.createElement("button"); + searchBtn.setAttribute("style", "border: none; border-radius: 5px; padding: 10px 20px; margin: 10px 0 0; background: hsla(194, 100%, 50%, 1); cursor: pointer; font-weight: 600; font-size: 0.85rem; color: white;"); + searchBtn.textContent = "Search"; + searchBtn.addEventListener("click", async (e) => { + // unfortunately we have to scrape html here since the Newgrounds API is hidden + const htmlText = await safeFetch(url, "text"); + if (!htmlText) { + alert("Failed to Fetch Track URL!"); + return; + } + + /* extract info */ + author = htmlText.match(/"artist":"([^"]+)"/)?.[1] || ""; + name = htmlText.match(/(.*?)<\/title>/i)?.[1] || ""; + + let songMatch = htmlText.match( + /<meta property="og:audio" content="(https:\/\/audio\.ngfiles\.com\/[^"]+\.mp3\?[^"]+)">/ + ); + if (!songMatch) { + const regex = new RegExp( + `"params":\\{"filename":"(https:\\/\\/audio\\.ngfiles\\.com\\/[^"]+\\.mp3\\?[^"]+)"}` + ); + songMatch = htmlText.match(regex); + }; + songURL = songMatch ? songMatch[1].replace(/\\/g, "") : ""; + + /* song usage */ + // fun fact: we can check if a user is scouted if 'downloads' shows in their song! + const isScouted = htmlText.match(/<dt>\s*Listens\s*<\/dt>[\s\S]*?<dd>\d+<\/dd>[\s\S]*?<dt>\s*Downloads\s*<\/dt>[\s\S]*?<dd>(\d+)<\/dd>[\s\S]*?<dt>\s*Score\s*<\/dt>/i); + const ccLicense = htmlText.match(/<div class="pod-body creative-commons">[\s\S]*?<p>\s*([\s\S]*?)\s*<\/p>/i)?.[1] || ""; + const type = isScouted ? ccLicense2Rating(ccLicense) : "unwhitelisted"; + + if (infoBox) infoBox.remove(); + infoBox = genCopyrightInfoBox(type, name, author); + if (type === "bad" || type === "unwhitelisted") { + name = undefined; + okayButton.style.filter = "brightness(70%)"; + okayButton.style.pointerEvents = "none"; + } else { + okayButton.style.filter = ""; + okayButton.style.pointerEvents = ""; + } + modal.appendChild(infoBox); + e.stopPropagation(); + }); + + idInputDiv.append(idLabel, idInput, searchBtn); + modal.append(label, idInputDiv); + } + + function ccLicense2Rating(licence) { + licence = String(licence).toLowerCase().trim(); + const goodTexts = [ + `you may only use this piece for commercial purposes if your work is a web-based game or animation,`, + ]; + for (const text of goodTexts) { + if (licence.startsWith(text)) return "good"; + } + + const badTexts = [ + `you may not use this work for any purposes`, + ]; + for (const text of badTexts) { + if (licence.startsWith(text)) return "bad"; + } + + const warnTexts = [ + `you are free to copy, distribute and transmit this work under the following conditions:`, + `please contact me if you would like to use this in a project. we can discuss the details.`, + ]; + for (const text of warnTexts) { + if (licence.startsWith(text)) return "warn"; + } + return "warn"; // warn is the default + } + + function genCopyrightInfoBox(type, name, author) { + const color = type === "good" ? "#00ff00" : type === "bad" || type === "unwhitelisted" ? "#ff0000" : "#ffc400"; + const box = document.createElement("div"); + box.setAttribute("style", `display: flex; width: 100%; margin: 15px 0; padding: 10px 20px 10px 30px; border-radius: 15px; border: solid 2px ${color}; background: ${color}30; text-align: center; font-size: .9rem; font-weight: bold;`); + + const img = document.createElement("img"); + img.setAttribute("style", "width:35px; margin-right: 5px;"); + img.src = type === "good" ? safeIcon : type === "bad" || type === "unwhitelisted" ? unsafeIcon : warnIcon; + + const label = document.createElement("span"); + if (type === "good") label.innerHTML = `The Track: ${name} by ${author}, can freely be used for web-based games`; + else if (type === "bad") label.innerHTML = `The Track: ${name} by ${author}, is not allowed for use!`; + else if (type === "unwhitelisted") label.innerHTML = `The Track: ${name} by ${author}, is not allowed for use. ${author} is not scouted on Newgrounds!`; + else label.innerHTML = `The Track: ${name} by ${author}, can only be used for non-profit web-based games WITH credit. Further use requires permission from ${author}`; + + box.append(img, label); + return box; + } + + function addButtonNG() { + // TODO add a tooltip maybe + const itemDiv = document.querySelector(`div[class^="action-menu_menu-container"] div[class^="action-menu_more-buttons-outer"] div[class^="action-menu_more-buttons"]`); + + ngButtonElement = itemDiv.children[0].cloneNode(true); + const innerButton = ngButtonElement.firstChild; + innerButton.setAttribute("data-tip", "Newgrounds Sound"); + innerButton.setAttribute("aria-label", "Newgrounds Sound"); + /* cleanup */ + for (var i = 1; i < innerButton.children.length; i++) { + const child = innerButton.children[i]; + if (child) child.remove(); + } + innerButton.firstChild.src = ngIcon; + ngButtonElement.addEventListener("click", openNewgroundsPopup); + itemDiv.insertBefore(ngButtonElement, itemDiv.children[0]); + } + + function startListenerWorker() { + ReduxStore.subscribe(() => queueMicrotask(() => { + const reduxState = ReduxStore.getState().scratchGui; + /* sound tab */ + if (!reduxState.mode.isPlayerOnly && reduxState.editorTab.activeTabIndex === 2) { + if (!ngButtonElement) addButtonNG(); + } else { + ngButtonElement = undefined; + } + })); + } + + if (typeof scaffolding === "undefined") startListenerWorker(); +} diff --git a/src/addons/generated/addon-entries.js b/src/addons/generated/addon-entries.js index f34d93101e9..83218495021 100644 --- a/src/addons/generated/addon-entries.js +++ b/src/addons/generated/addon-entries.js @@ -77,4 +77,5 @@ export default { "multi-tab-code": () => import(/* webpackChunkName: "addon-default-entry" */ "../addons/multi-tab-code/_runtime_entry.js"), "editor-animations": () => import(/* webpackChunkName: "addon-default-entry" */ "../addons/editor-animations/_runtime_entry.js"), "reorder-custom-inputs": () => import(/* webpackChunkName: "addon-default-entry" */ "../addons/reorder-custom-inputs/_runtime_entry.js"), + "sounds-newgrounds-button": () => import(/* webpackChunkName: "addon-default-entry" */ "../addons/sounds-newgrounds-button/_runtime_entry.js"), }; diff --git a/src/addons/generated/addon-manifests.js b/src/addons/generated/addon-manifests.js index a9317d5d657..5b54eec8885 100644 --- a/src/addons/generated/addon-manifests.js +++ b/src/addons/generated/addon-manifests.js @@ -75,6 +75,7 @@ import _tw_disable_cloud_variables from "../addons/tw-disable-cloud-variables/_m import _multi_tab_code from "../addons/multi-tab-code/_manifest_entry.js"; import _editor_animations from "../addons/editor-animations/_manifest_entry.js"; import _reorder_custom_inputs from "../addons/reorder-custom-inputs/_manifest_entry.js"; +import _sounds_newgrounds_button from "../addons/sounds-newgrounds-button/_manifest_entry.js"; export default { "cat-blocks": _cat_blocks, @@ -154,4 +155,5 @@ export default { "multi-tab-code": _multi_tab_code, "editor-animations": _editor_animations, "reorder-custom-inputs": _reorder_custom_inputs, + "sounds-newgrounds-button": _sounds_newgrounds_button, };