// ==UserScript==
// @name Spotify Enhancer (Full-Sized Cover Art Downloader)
// @description Integrates an overlay button in Spotify Web Player to view and download full-sized (2000px) album cover art.
// @icon https://raw.githubusercontent.com/exyezed/spotify-enhancer/refs/heads/main/extras/spotify-enhancer.png
// @version 1.6
// @author exyezed
// @namespace https://github.com/exyezed/spotify-enhancer/
// @supportURL https://github.com/exyezed/spotify-enhancer/issues
// @license MIT
// @match *://open.spotify.com/*
// @grant GM_addStyle
// @require https://cdn.jsdelivr.net/npm/@iconify/iconify@3.1.1/dist/iconify.min.js
// ==/UserScript==
(function() {
'use strict';
const styles = `
.preview-modal-title {
color: white;
font-size: 16px;
margin-bottom: 16px;
text-align: center;
max-width: 90vw;
word-wrap: break-word;
}
.custom-overlay-button {
position: absolute;
top: 10px;
right: 10px;
z-index: 9999 !important;
background: rgba(0, 0, 0, 0.7);
border: none;
border-radius: 50%;
width: 30px;
height: 30px;
cursor: pointer;
display: flex;
align-items: center;
justify-content: center;
color: white;
transition: all 0.3s ease;
pointer-events: all !important;
}
.custom-overlay-button:hover {
background: rgba(0, 0, 0, 0.9);
}
.custom-overlay-button svg {
width: 18px;
height: 18px;
opacity: 1;
transition: opacity 0.3s ease;
}
.custom-overlay-button .icon-normal {
position: absolute;
}
.custom-overlay-button .icon-hover {
position: absolute;
opacity: 0;
color: #1ed760;
}
.custom-overlay-button:hover .icon-normal {
opacity: 0;
}
.custom-overlay-button:hover .icon-hover {
opacity: 1;
}
.custom-overlay-button * {
pointer-events: none;
}
.preview-modal {
position: fixed;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: rgba(0, 0, 0, 0.85);
display: flex;
justify-content: center;
align-items: center;
z-index: 10000;
opacity: 0;
visibility: hidden;
transition: opacity 0.1s ease, visibility 0.1s ease;
}
.preview-modal.active {
opacity: 1;
visibility: visible;
}
.preview-modal-content {
position: relative;
max-width: 90vw;
max-height: 90vh;
display: flex;
flex-direction: column;
align-items: center;
transform: scale(0.95);
transition: transform 0.1s ease;
}
.preview-modal.active .preview-modal-content {
transform: scale(1);
}
.preview-modal-content.loading .preview-actions {
opacity: 0;
visibility: hidden;
}
.preview-modal-content .preview-actions {
opacity: 1;
visibility: visible;
transition: opacity 0.1s ease, visibility 0.1s ease;
}
.preview-modal img {
max-width: 100%;
max-height: 80vh;
object-fit: contain;
cursor: pointer;
border-radius: 8px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.5);
}
.preview-actions {
margin-top: 16px;
}
.fetch-button {
position: relative;
background: none;
border: none;
color: white;
cursor: pointer;
padding: 8px;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.1s ease;
gap: 6px;
}
.fetch-button:hover {
opacity: 1;
color: #1ed760;
}
.fetch-button svg {
width: 18px !important;
height: 18px !important;
}
.fetch-button span {
font-size: 14px;
font-weight: 600;
text-transform: uppercase;
}
`;
GM_addStyle(styles);
const modal = document.createElement('div');
modal.className = 'preview-modal';
document.body.appendChild(modal);
async function downloadImage(url, filename) {
try {
const response = await fetch(url);
const blob = await response.blob();
const blobUrl = URL.createObjectURL(blob);
const link = document.createElement('a');
link.href = blobUrl;
link.download = filename + '.jpg';
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
URL.revokeObjectURL(blobUrl);
} catch (error) {
console.error('Download failed:', error);
}
}
function createModal(imageUrl, title) {
const content = document.createElement('div');
content.className = 'preview-modal-content loading';
const titleDiv = document.createElement('div');
titleDiv.className = 'preview-modal-title';
const img = document.createElement('img');
img.onload = () => {
content.classList.remove('loading');
titleDiv.textContent = `${title} (${img.naturalWidth} x ${img.naturalHeight})`;
if (window.Iconify) {
window.Iconify.scan(content);
}
};
img.src = imageUrl;
img.alt = title;
img.onclick = () => {
window.open(imageUrl, '_blank');
};
const actionsDiv = document.createElement('div');
actionsDiv.className = 'preview-actions';
const fetchButton = document.createElement('button');
fetchButton.className = 'fetch-button';
const downloadIcon = document.createElement('span');
downloadIcon.className = 'iconify';
downloadIcon.setAttribute('data-icon', 'mynaui:fat-arrow-down-solid');
downloadIcon.setAttribute('data-width', '18');
downloadIcon.setAttribute('data-height', '18');
const downloadText = document.createElement('span');
downloadText.textContent = 'DOWNLOAD';
fetchButton.appendChild(downloadIcon);
fetchButton.appendChild(downloadText);
fetchButton.onclick = (e) => {
e.stopPropagation();
downloadImage(imageUrl, title || 'spotify-cover');
};
actionsDiv.appendChild(fetchButton);
content.appendChild(titleDiv);
content.appendChild(img);
content.appendChild(actionsDiv);
while (modal.firstChild) {
modal.removeChild(modal.firstChild);
}
modal.appendChild(content);
modal.classList.add('active');
modal.onclick = (e) => {
if (e.target === modal) {
modal.classList.remove('active');
}
};
document.addEventListener('keydown', function(e) {
if (e.key === 'Escape') {
modal.classList.remove('active');
}
});
}
function getTitleFromElement(element) {
const playButton = element.querySelector('button[aria-label^="Play"]');
if (playButton) {
const ariaLabel = playButton.getAttribute('aria-label');
if (ariaLabel) {
return ariaLabel.replace('Play ', '');
}
}
const entityTitle = element.querySelector('[data-testid="entityTitle"] h1');
if (entityTitle) {
return entityTitle.textContent.trim();
}
const img = element.querySelector('img[alt]:not([alt=""])');
if (img && img.alt) {
return img.alt;
}
return 'spotify-cover';
}
function getFullsizeUrl(originalUrl) {
const resolutionPatterns = {
'ab67616d00004851': 'ab67616d000082c1',
'ab67616d0000b273': 'ab67616d000082c1',
'ab67616d00001e02': 'ab67616d000082c1',
'ab67616100005174': 'ab6761610000e5eb',
'ab6761610000f174': 'ab6761610000e5eb',
'ab676161000051748': 'ab6761610000e5eb',
'ab67616100000000': 'ab6761610000e5eb'
};
let newUrl = originalUrl;
for (const [pattern, replacement] of Object.entries(resolutionPatterns)) {
if (originalUrl.includes(pattern)) {
newUrl = originalUrl.replace(pattern, replacement);
break;
}
}
return newUrl;
}
function addOverlayButton(cardElement) {
if (cardElement && !cardElement.querySelector('.custom-overlay-button')) {
const imgElement = cardElement.querySelector('img');
if (!imgElement) return;
const button = document.createElement('button');
button.className = 'custom-overlay-button';
button.setAttribute('tabindex', '0');
const iconNormal = document.createElement('span');
iconNormal.className = 'iconify icon-normal';
iconNormal.setAttribute('data-icon', 'mdi:image-size-select-large');
iconNormal.setAttribute('data-width', '18');
iconNormal.setAttribute('data-height', '18');
const iconHover = document.createElement('span');
iconHover.className = 'iconify icon-hover';
iconHover.setAttribute('data-icon', 'mdi:image-size-select-actual');
iconHover.setAttribute('data-width', '18');
iconHover.setAttribute('data-height', '18');
button.appendChild(iconNormal);
button.appendChild(iconHover);
const handleClick = (e) => {
e.preventDefault();
e.stopPropagation();
e.stopImmediatePropagation();
const title = getTitleFromElement(cardElement);
const fullsizeUrl = getFullsizeUrl(imgElement.src);
createModal(fullsizeUrl, title);
return false;
};
button.addEventListener('click', handleClick, true);
button.addEventListener('mousedown', (e) => e.stopPropagation(), true);
button.addEventListener('mouseup', (e) => e.stopPropagation(), true);
button.addEventListener('touchstart', (e) => e.stopPropagation(), true);
button.addEventListener('touchend', (e) => e.stopPropagation(), true);
const buttonWrapper = document.createElement('div');
buttonWrapper.style.cssText = `
position: absolute;
top: 0;
right: 0;
z-index: 9999;
pointer-events: none;
padding: 5px;
`;
buttonWrapper.appendChild(button);
cardElement.style.position = 'relative';
cardElement.appendChild(buttonWrapper);
}
}
const observer = new MutationObserver((mutations) => {
mutations.forEach((mutation) => {
mutation.addedNodes.forEach((node) => {
if (node.nodeType === 1) {
const cards = [
...Array.from(node.classList?.contains('xBV4XgMq0gC5lQICFWY_') ? [node] : node.querySelectorAll('.xBV4XgMq0gC5lQICFWY_')),
...Array.from(node.classList?.contains('CmkY1Ag0tJDfnFXbGgju') ? [node] : node.querySelectorAll('.CmkY1Ag0tJDfnFXbGgju'))
];
cards.forEach(addOverlayButton);
}
});
});
});
function initialize() {
const existingCards = document.querySelectorAll('.xBV4XgMq0gC5lQICFWY_, .CmkY1Ag0tJDfnFXbGgju');
existingCards.forEach(addOverlayButton);
observer.observe(document.body, {
childList: true,
subtree: true
});
}
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', initialize);
} else {
initialize();
}
console.log('Spotify Enhancer (Full-Sized Cover Art Downloader) is running');
})();