mirror of
https://github.com/th-ch/youtube-music.git
synced 2026-01-17 13:12:07 +00:00
114 lines
3.0 KiB
TypeScript
114 lines
3.0 KiB
TypeScript
import { createSignal } from 'solid-js';
|
|
|
|
import { render } from 'solid-js/web';
|
|
|
|
import { defaultConfig } from '@/config/defaults';
|
|
import { getSongMenu } from '@/providers/dom-elements';
|
|
import { getSongInfo } from '@/providers/song-info-front';
|
|
import { t } from '@/i18n';
|
|
import {
|
|
isAlbumOrPlaylist,
|
|
isMusicOrVideoTrack,
|
|
} from '@/plugins/utils/renderer/check';
|
|
|
|
import { DownloadButton } from './templates/download';
|
|
|
|
import type { RendererContext } from '@/types/contexts';
|
|
import type { DownloaderPluginConfig } from './index';
|
|
|
|
let download: () => void;
|
|
|
|
const [downloadButtonText, setDownloadButtonText] = createSignal<string>('');
|
|
|
|
let buttonContainer: HTMLDivElement | null = null;
|
|
|
|
const menuObserver = new MutationObserver(() => {
|
|
const menu = getSongMenu();
|
|
|
|
if (
|
|
!menu ||
|
|
menu.contains(buttonContainer) ||
|
|
!(isMusicOrVideoTrack() || isAlbumOrPlaylist()) ||
|
|
!buttonContainer
|
|
) {
|
|
return;
|
|
}
|
|
|
|
menu.prepend(buttonContainer);
|
|
});
|
|
|
|
export const onRendererLoad = ({
|
|
ipc,
|
|
}: RendererContext<DownloaderPluginConfig>) => {
|
|
download = () => {
|
|
const songMenu = getSongMenu();
|
|
|
|
let videoUrl = songMenu
|
|
?.querySelector(
|
|
'ytmusic-menu-navigation-item-renderer[tabindex="0"] #navigation-endpoint',
|
|
)
|
|
?.getAttribute('href');
|
|
|
|
if (!videoUrl && songMenu) {
|
|
for (const it of songMenu.querySelectorAll(
|
|
'ytmusic-menu-navigation-item-renderer[tabindex="-1"] #navigation-endpoint',
|
|
)) {
|
|
if (it.getAttribute('href')?.includes('podcast/')) {
|
|
videoUrl = it.getAttribute('href');
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (videoUrl) {
|
|
if (videoUrl.startsWith('watch?')) {
|
|
videoUrl = defaultConfig.url + '/' + videoUrl;
|
|
}
|
|
|
|
if (videoUrl.startsWith('podcast/')) {
|
|
videoUrl =
|
|
defaultConfig.url + '/watch?' + videoUrl.replace('podcast/', 'v=');
|
|
}
|
|
|
|
if (videoUrl.includes('?playlist=')) {
|
|
ipc.invoke('download-playlist-request', videoUrl);
|
|
return;
|
|
}
|
|
} else {
|
|
videoUrl = getSongInfo().url || window.location.href;
|
|
}
|
|
|
|
ipc.invoke('download-song', videoUrl);
|
|
};
|
|
|
|
ipc.on('downloader-feedback', (feedback: string) => {
|
|
const targetHtml = feedback || t('plugins.downloader.templates.button');
|
|
setDownloadButtonText(targetHtml);
|
|
});
|
|
};
|
|
|
|
export const onPlayerApiReady = () => {
|
|
setDownloadButtonText(t('plugins.downloader.templates.button'));
|
|
|
|
buttonContainer = document.createElement('div');
|
|
buttonContainer.classList.add(
|
|
'style-scope',
|
|
'menu-item',
|
|
'ytmusic-menu-popup-renderer',
|
|
);
|
|
buttonContainer.setAttribute('aria-disabled', 'false');
|
|
buttonContainer.setAttribute('aria-selected', 'false');
|
|
buttonContainer.setAttribute('role', 'option');
|
|
buttonContainer.setAttribute('tabindex', '-1');
|
|
|
|
render(
|
|
() => <DownloadButton onClick={download} text={downloadButtonText()} />,
|
|
buttonContainer,
|
|
);
|
|
|
|
menuObserver.observe(document.querySelector('ytmusic-popup-container')!, {
|
|
childList: true,
|
|
subtree: true,
|
|
});
|
|
};
|