Skip to content

Newgrounds Audio Button Addon #153

New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Merged
merged 5 commits into from
Aug 17, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
28 changes: 28 additions & 0 deletions src/addons/addons/sounds-newgrounds-button/_manifest_entry.js
Original file line number Diff line number Diff line change
@@ -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;
5 changes: 5 additions & 0 deletions src/addons/addons/sounds-newgrounds-button/_runtime_entry.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
/* generated by pull.js */
import _js from "./userscript.js";
export const resources = {
"userscript.js": _js,
};
227 changes: 227 additions & 0 deletions src/addons/addons/sounds-newgrounds-button/userscript.js
Original file line number Diff line number Diff line change
@@ -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 <a href="https://www.newgrounds.com/audio" target="_blank">Newgrounds</a> audio directly into your Project.<br><b>Not all tracks are fully free-to-use, read the report after searching.</b>`;
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>(.*?)<\/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();
}
1 change: 1 addition & 0 deletions src/addons/generated/addon-entries.js

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions src/addons/generated/addon-manifests.js
Original file line number Diff line number Diff line change
Expand Up @@ -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,
Expand Down Expand Up @@ -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,
};
Loading