Compare commits

...

11 commits

Author SHA1 Message Date
Chris Park 96b941b17a
Merge 348769a60f into 080fc043ea 2024-05-01 22:31:35 +02:00
jj 080fc043ea
best audio picking improvements for youtube, tiktok, and soundcloud (#476) 2024-04-30 09:39:32 +02:00
wukko 95925c9864
soundcloud: replace filter with find and clean up 2024-04-30 13:38:01 +06:00
wukko ed8af6ca96
tiktok & soundcloud: proper best audio picking
also improved tiktok audio file naming scheme. full audio now has the "_audio_original" tag. audio extracted from video is simply "_audio".
2024-04-30 13:22:29 +06:00
wukko 276caa011a
youtube: fall back to m4a audio if opus isn't available 2024-04-30 11:24:12 +06:00
chris 348769a60f
Merge branch 'wukko:current' into current 2024-01-06 15:33:34 +01:00
chris park 0b4a1f1f4e Update pl.json 2024-01-04 20:40:16 +01:00
chris 4a711a8cbc
Merge branch 'wukko:current' into current 2024-01-04 19:23:49 +01:00
chris 1bda27917a
reset pl to en 2023-12-30 02:15:25 +01:00
chris a582fa6095
Merge branch 'wukko:current' into current 2023-12-30 02:13:11 +01:00
chris 311c1ddf77 polish translation 2023-09-16 21:54:08 +00:00
6 changed files with 213 additions and 60 deletions

View file

@ -0,0 +1,164 @@
{
"name": "polski",
"substrings": {
"ContactLink": "sprawdź <a class=\"text-backdrop link\" href=\"{statusPage}\" target=\"_blank\">stronę statusu</a> lub <a class=\"text-backdrop link\" href=\"{repo}\" target=\"_blank\">stwórz zgłoszenie na github (GitHub Issues)</a>."
},
"strings": {
"AppTitleCobalt": "cobalt",
"LinkInput": "wklej tutaj link",
"AboutSummary": "cobalt to idealne miejsce do pobierania plików z platform społecznościowych i multimedialnych. bez reklam, śledzenia lub innych bzdur. wystarczy wkleić link udostępniania i gotowe!",
"EmbedBriefDescription": "zapisuj to co kochasz. bez reklam, śledzenia lub innych bzdur.",
"MadeWithLove": "zrobione z miłością przez wukko",
"AccessibilityInputArea": "obszar wprowadzenia linku",
"AccessibilityOpenAbout": "otwórz wyskakujące okienko informacji",
"AccessibilityDownloadButton": "przycisk pobierania",
"AccessibilityOpenSettings": "otwórz wyskakujące okienko z ustawieniami",
"AccessibilityOpenDonate": "otwórz wyskakujące okienko darowizn",
"TitlePopupAbout": "czym jest cobalt?",
"TitlePopupSettings": "ustawienia",
"TitlePopupError": "uh-oh...",
"TitlePopupChangelog": "co nowego?",
"TitlePopupDonate": "wesprzyj cobalt",
"TitlePopupDownload": "jak zapisać?",
"ErrorSomethingWentWrong": "coś poszło nie tak i nie byłem w stanie niczego znaleźć. spróbuj jeszcze raz, ale jeżeli ten problem nadal występuje, {ContactLink}.",
"ErrorUnsupported": "wygląda na to, że ten serwis nie jest jeszcze wspierany lub twój link jest źle wklejony. czy na pewno to dobry link został wklejony?",
"ErrorBrokenLink": "{s} jest wspierany, ale coś jest nie tak z twoim linkiem. może nie został dobrze skopiowany?",
"ErrorNoLink": "nie mogę zgadnąć co chcesz pobrać! proszę, podaj link :(",
"ErrorPageRenderFail": "jeśli to czytasz, to coś poszło nie tak podczas renderowania strony. proszę {ContactLink}. upewnij się, że domena, w której występuje ten błąd i aktualny github commit hash ({s}) zostanie podana przez ciebie. dziękuję z wyprzedzeniem :D",
"ErrorRateLimit": "wysyłasz zbyt wiele prób. spróbuj ponownie za minutę!",
"ErrorCouldntFetch": "nie mogłem znaleźć nic na temat tego linku. sprawdź, czy działa i spróbuj ponownie! niektóre treści mogą być ograniczone regionalnie, więc miej to na uwadze.",
"ErrorLengthLimit": "nie mogę przetwarzać filmów dłuższych niż {s} minut, zamiast tego wybierz coś krótszego!",
"ErrorBadFetch": "coś poszło nie tak, gdy próbowałem uzyskać informacje o twoim linku. czy na pewno działa? sprawdź, czy tak, i spróbuj ponownie.",
"ErrorNoInternet": "nie masz dostępu do internetu lub api cobalt jest tymczasowo niedostępne. sprawdź połączenie i spróbuj ponownie.",
"ErrorCantConnectToServiceAPI": "nie mogłem połączyć się z api usługi. być może nie działa lub cobalt został zablokowany. spróbuj ponownie, ale jeśli błąd będzie się powtarzał, {ContactLink}.",
"ErrorEmptyDownload": "nie widzę niczego, co mógłbym pobrać z twojego linku. spróbuj coś innego!",
"ErrorLiveVideo": "to jest wideo na żywo, jeszcze nie nauczyłem się patrzeć w przyszłość. poczekaj na zakończenie transmisji i spróbuj ponownie!",
"SettingsAppearanceSubtitle": "wygląd",
"SettingsThemeSubtitle": "motyw",
"SettingsFormatSubtitle": "format",
"SettingsQualitySubtitle": "jakość",
"SettingsThemeautomatyczny": "automatyczny",
"SettingsThemeLight": "jasny",
"SettingsThemeDark": "ciemny",
"SettingsKeepDownloadButton": "zostaw &gt;&gt; widoczny",
"AccessibilityKeepDownloadButton": "zostaw przycisk pobierania jako zawsze widoczny",
"SettingsEnableDownloadPopup": "pytaj jak zapisywać",
"AccessibilityEnableDownloadPopup": "pytaj co zrobić z pobranymi plikami",
"SettingsQualityDescription": "jeśli wybrana jakość nie jest dostępna, zamiast niej używana jest najbliższa.",
"NoScriptMessage": "cobalt używa javascript do żądań api i interaktywnego interfejsu. musisz zezwolić na javascript, aby korzystać z tej strony. nie ma żadnych szkodliwych skryptów, obietnica na mały paluszek.",
"DownloadPopupDescriptionIOS": "najłatwiejszy sposób na zapisywanie filmów na iOS:\n1. dodaj <a class=\"text-backdrop link\" href=\"{saveToGalleryShortcut}\" target=\"_blank\">ten skrót siri</a>.\n2. wybierz \"udostępnij\" wyżej, a pózniej wybierz \"zapisz do zdjęć\" w menu udostępniania, które się pojawi.\njeśli zostaniesz o to poproszony, przejrzyj prośbę o pozwolenie i naciśnij przycisk \"zawsze pozwól\".\n\nmetoda alternatywna:\nnaciśnij i przytrzymaj przycisk pobierania, ukryj podgląd wideo i wybierz opcję \"pobierz załączony plik\", aby pobrać.\nnastępnie otwórz pobrane w safari, wybierz pobrany plik, otwórz menu udostępniania, a na koniec naciśnij przycisk \"zapisz wideo\".",
"DownloadPopupDescription": "przycisk pobierania otwiera nową kartę z żądanym plikiem. możesz wyłączyć to wyskakujące okienko w ustawieniach.",
"ClickToCopy": "kliknij, aby skopiować",
"Download": "pobierz",
"CopyURL": "kopiuj",
"AboutTab": "o cobalt",
"ChangelogTab": "dziennik zmian",
"DonationsTab": "darowizny",
"SettingsVideoTab": "wideo",
"SettingsAudioTab": "audio",
"SettingsOtherTab": "inne",
"ChangelogLastMajor": "obecna wersja & github commit",
"AccessibilityModeToggle": "przełącz tryb pobierania",
"DonateLinksDescription": "to jest najlepszy sposób jeżeli chcesz, aby darowizna wpłyneła bezpośrednio do mnie.",
"SettingsAudioFormatBest": "najlepszy",
"SettingsAudioFormatDescription": "kiedy \"najlepszy\" format jest zaznaczony, otrzymasz audio w takiej postaci, w jakiej jest po stronie usługi. nie jest ponownie kodowane. wszystko inne zostanie ponownie zakodowane.",
"Keyphrase": "zapisuj to co kochasz",
"SettingsRemoveWatermark": "wyłącz znak wodny",
"ErrorPopupCloseButton": "rozumiem",
"ErrorLengthAudioConvert": "nie mogę przekonwertować audio dłuższego niż {s} minut. wybierz \"najlepszy\" format, jeżeli chcesz ominąć ograniczenia!",
"SettingsAudioFullTikTok": "pełne audio",
"SettingsAudioFullTikTokDescription": "pobiera oryginalny dźwięk użyty w wideo, bez zmian dokonanych przez autora wideo.",
"ErrorCantGetID": "nie mogłem uzyskać pełnych informacji ze skróconego linku. upewnij się, że działa lub wypróbuj pełny! jeśli problem będzie się powtarzał, {ContactLink}.",
"ErrorNoVideosInTweet": "nie mogłem znaleźć żadnych treści multimedialnych w tym tweecie. spróbuj innego!",
"ImagePickerTitle": "wybierz obrazy do pobrania",
"ImagePickerDownloadAudio": "pobierz audio",
"ImagePickerExplanationPC": "kliknij obraz prawym przyciskiem myszy, aby go zapisać.",
"ImagePickerExplanationPhone": "dotknij i przytrzymaj obraz, aby go zapisać.",
"ErrorNoUrlReturned": "nie otrzymałem linku do pobrania z serwera. to nigdy nie powinno się zdarzyć. spróbuj ponownie, ale jeśli nadal występują problemy, {ContactLink}.",
"ErrorUnknownStatus": "otrzymałem odpowiedź, której nie mogę przetworzyć. to nigdy nie powinno się zdarzyć. spróbuj ponownie, ale jeśli nadal występują problemy, {ContactLink}.",
"PasteFromClipboard": "wklej",
"ChangelogOlder": "poprzednie wesje",
"ChangelogPressToExpand": "rozwiń",
"Miscellaneous": "różne",
"ModeToggleautomatyczny": "automatyczne",
"ModeToggleAudio": "audio",
"SettingsDisableNotifications": "ukryj powiadomienia",
"MediaPickerTitle": "wybierz co zapisać",
"MediaPickerExplanationPC": "kliknij lub kliknij prawym przyciskiem myszy, aby pobrać to, co chcesz.",
"MediaPickerExplanationPhone": "dotknij lub dotknij i przytrzymaj, aby pobrać to, co chcesz.",
"TwitterSpaceWasntRecorded": "to twitter space nie zostało nagrane, więc nie ma co pobrać. spróbuj inne twitter space!",
"ErrorCantProcess": "nie mogłem przetworzyć twojego żądania :(\nmożesz spróbować ponownie, ale jeżeli problem nadal występuje, proszę {ContactLink}.",
"ChangelogPressToHide": "zwiń",
"Donate": "wesprzyj",
"DonateSub": "pomóż pozostawić cobalt online!",
"DonateExplanation": "cobalt nie wciska reklam w twarz i nie sprzedaje danych osobowych, a zatem jest <span class=\"text-backdrop\">całkowicie darmowy w użyciu dla każdego</span>. ale rozwój i utrzymanie serwisu multimedialnego, z którego korzysta ponad 350 tys. osób, jest dość kosztowne. zarówno pod względem czasu, jak i pieniędzy. jako studentowi trudno mi samodzielnie poradzić sobie z takimi wydatkami.\n\njeśli cobalt pomógł ci w przeszłości i chcesz, aby nadal się rozwijał i ewoluował, możesz to zrobić, przekazując darowiznę!\n\nprzekazując darowiznę pomagasz wszystkim, którzy korzystają z kobaltu: nauczycielom, studentom, muzykom, twórcom treści, artystom, wykładowcom i wielu, wielu innym!\n\nw ciągu ostatnich kilku miesięcy darowizny mi na to pozwoliły:\n*; zwiększyć stabilność i czas pracy do niemal 100%.\n*; przyspieszyć WSZYSTKIE pobierane pliki, zwłaszcza te cięższe.\n*; otworzyć api cobalt do bezpłatnego publicznego użytku.\n*; wytrzymać kilka ogromnych napływów użytkowników przy 0 przestojach.\n*; przejść do niezawodnego i godnego zaufania dostawcy infrastruktury chmurowej.\n*; oddzielić frontend i api dla odporności i przyszłej decentralizacji.\n\n<span class=\"text-backdrop\">każdy grosz ma znaczenie i jest niezwykle doceniany</span>, naprawdę możesz coś zmienić!",
"DonateVia": "wesprzyj poprzez",
"DonateHireMe": "...lub możesz <a class=\"text-backdrop link\" href=\"{s}\" target=\"_blank\">zatrudnić mnie</a> :)",
"SettingsVideoMute": "wycisz audio",
"SettingsVideoMuteExplanation": "usuwa dźwięk z pobieranych plików wideo, jeśli to możliwe.",
"ErrorSoundCloudNoClientId": "nie udało mi się uzyskać tymczasowego tokena, który jest wymagany do pobierania utworów z soundcloud. spróbuj ponownie, ale jeśli problem będzie się powtarzał, {ContactLink}.",
"CollapseServices": "wspierane serwisy",
"CollapseSupport": "pomoc & kod źródłowy",
"CollapsePrivacy": "polityka prywatności",
"ServicesNote": "ta lista nie jest ostateczna i z czasem się powiększa, dlatego warto ją sprawdzać od czasu do czasu!",
"FollowSupport": "pozostań w kontakcie z cobalt, aby uzyskać wsparcie, ankiety, wiadomości i nie tylko:",
"SupportNote": "należy pamiętać, że odpowiedź może chwilę potrwać, ponieważ wszystkim zarządza tylko jedna osoba.",
"SourceCode": "zgłaszaj błędy, eksploruj kod źródłowy, polub lub sklonuj repozytorium:",
"PrivacyPolicy": "polityka prywatności cobalt jest prosta: żadne dane o użytkowniku nie są nigdy gromadzone ani przechowywane. zero, zero, nada, nic.\nto, co pobierasz, jest wyłącznie twoją sprawą, a nie moją lub kogokolwiek innego.\n\njeśli pobieranie wymaga renderowania na żywo, niektóre nieśledzone dane są tymczasowo przechowywane w pamięci RAM serwera. jest to konieczne do działania tej funkcji.\n\nw tym przypadku informacje o żądanej zawartości są przechowywane do <span class=\"text-backdrop\">90 sekund</span> a następnie trwale usunięte.\nnikt (nawet ja) nie ma dostępu do tych danych. oficjalna baza kodu cobalt nie zapewnia sposobu na odczytanie ich poza funkcjami przetwarzania.\n\nmożesz sam sprawdzić <a class=\"text-backdrop link\" href=\"{repo}\" target=\"_blank\">kod źródłowy</a> cobalt i przekonać się, że wszystko jest zgodne z opisem.",
"ErrorYTUnavailable": "to wideo na youtube jest niedostępne, może mieć ograniczenia regionalne lub wiekowe. spróbuj innego!",
"ErrorYTTryOtherCodec": "nie mogłem znaleźć niczego do pobrania z twoimi ustawieniami. spróbuj innego kodeka lub jakości!\n\nczasami youtube api działa nieoczekiwanie. spróbuj ponownie lub wypróbuj inne ustawienia.",
"SettingsCodecSubtitle": "kodek youtube",
"SettingsCodecDescription": "h264: generalnie lepsza obsługa odtwarzacza, ale jakość osiąga szczyt przy 1080p.\nav1: niska obsługa odtwarzacza, ale obsługuje 8k i HDR.\nvp9: zwykle najwyższa szybkość transmisji, zachowuje najwięcej szczegółów. obsługuje 4k i HDR.\nwybierz h264, jeśli chcesz mieć najlepszą kompatybilność z edytorami/odtwarzaczami/mediami społecznościowymi.",
"SettingsAudioDub": "ścieżka dźwiękowa youtube",
"SettingsAudioDubDescription": "określa, która ścieżka dźwiękowa będzie używana. jeśli ścieżka dubbingowana nie jest dostępna, zamiast niej używany jest oryginalny język wideo.\n\noryginalny: używany jest oryginalny język wideo.\nautomatyczny: używany jest domyślny język przeglądarki (i cobalt).",
"SettingsDubDefault": "oryginalny",
"SettingsDubautomatyczny": "automatyczny",
"SettingsVimeoPrefer": "typ pobierania vimeo",
"SettingsVimeoPreferDescription": "progresywny: bezpośredni link do pliku cdn vimeo. maksymalna jakość to 1080p.\ndash: wideo i audio są łączone przez cobalt w jeden plik. maksymalna jakość to 4k.\n\nwybierz \"progresywny\", jeśli chcesz uzyskać najlepszą kompatybilność z edytorami/odtwarzaczami/mediami społecznościowymi. jeśli pobieranie progresywne nie jest dostępne, zamiast tego używany jest dash.",
"ShareURL": "udostępnij",
"ErrorTweetUnavailable": "nie mogłem znaleźć nic na temat tego tweeta. może to być spowodowane tym, że jego widoczność jest ograniczona. spróbuj inny tweet!",
"ErrorTwitterRIP": "twitter ograniczył dostęp do wszelkich treści dla nieuwierzytelnionych użytkowników. chociaż istnieje sposób na uzyskanie zwykłych tweetów, twitter spaces są niestety obecnie niemożliwe do uzyskania. szukam możliwych rozwiązań.",
"UrgentDonate": "cobalt potrzebuje twojej pomocy!",
"PopupCloseDone": "zrobione",
"Accessibility": "dostępność",
"SettingsReduceTransparency": "zredukuj przezroczystość",
"SettingsDisableAnimations": "wyłącz animacje",
"FeatureErrorGeneric": "twoja przeglądarka nie zezwala lub nie obsługuje tej funkcji. sprawdź, czy są dostępne aktualizacje i spróbuj ponownie!",
"ClipboardErrorFirefox": "używasz przeglądarki Firefox, w której wszystkie funkcje odczytu schowka są wyłączone.\n\nmożna to naprawić, wykonując czynności <a class=\"text-backdrop link\" href=\"{repo}/blob/current/docs/troubleshooting.md#how-to-fix-clipboard-pasting-in-firefox\" target=\"_blank\">stąd!</a>\n\n...lub zamiast tego możesz wkleić link ręcznie.",
"ClipboardErrorNoPermission": "cobalt nie może uzyskać dostępu do ostatniego elementu w schowku bez zgody użytkownika.\n\njeśli nie chcesz udzielać dostępu, po prostu wklej link ręcznie.\n\njeśli tak, przejdź do ustawień witryny i włącz uprawnienie schowka..",
"SupportSelfTroubleshooting": "występują jakieś problemy? wypróbuj najpierw jedno z tych rozwiązań::",
"AccessibilityGoBack": "wróć i zamknij wyskakujące okienko",
"CollapseKeyboard": "skróty klawiszowe",
"KeyboardShortcutsIntro": "korzystaj z cobalt jeszcze szybciej ze skrótami klawiszowymi:",
"KeyboardShortcutQuickPaste": "wklej link",
"KeyboardShortcutClear": "wyczyść obszar wprowadzania łącza",
"KeyboardShortcutClosePopup": "zamknij wszystkie okienka wyskakujące",
"CollapseLegal": "warunki i etyka",
"FairUse": "cobalt jest narzędziem internetowym, które ułatwia pobieranie treści z internetu i bierze <span class=\"text-backdrop\">zero odpowiedzialności</span>. serwery przetwarzania działają jak <span class=\"text-backdrop\">ograniczone proxy</span>, więc żadna zawartość multimedialna nie jest nigdy buforowana ani przechowywana.\n\nużytkownik końcowy jest odpowiedzialny za to, co pobiera, w jaki sposób wykorzystuje i rozpowszechnia te treści. należy zachować ostrożność podczas korzystania z treści innych osób i zawsze podawać dane twórców oryginalnych.\n\nw przypadku wykorzystania w celach edukacyjnych (wykład, praca domowa itp.) prosimy o załączenie linku źródłowego.\n\nuczciwe użytkowanie i kredyty przynoszą korzyści wszystkim.",
"UrgentFeatureUpdate71": "więcej wspieranych serwisów!",
"UrgentThanks": "dziękuję za wsparcie!",
"SettingsDisableMetadata": "nie dodawaj metadanych",
"UrgentNewDomain": "nowa domena, ten sam cobalt",
"NewDomainWelcomeTitle": "cześć!",
"NewDomainWelcome": "cobalt przenosi się! te same funkcje, ten sam właściciel, po prostu bardziej zapamiętywalna domena. i nadal bez reklam.\n\n<span class=\"text-backdrop\">cobalt.tools</span> to nowa domena główna, czyli miejsce, w którym jesteś teraz. pamiętaj, aby zaktualizować zakładki i ponownie zainstalować aplikację internetową!",
"DataTransferSuccess": "przy okazji, ustawienia zostały przeniesione automatycznie :)",
"DataTransferError": "coś poszło nie tak podczas przesyłania preferencji. będziesz musiał otworzyć ustawienia i skonfigurować cobalt ręcznie.",
"SupportNotAffiliated": "cobalt <span class=\"text-backdrop\">nie jest powiązany</span> z serwisami wymienionymi powyżej.",
"SponsoredBy": "sponsorowane przez",
"FilenameTitle": "styl nazwy pliku",
"FilenamePatternClassic": "klasyczny",
"FilenamePatternPretty": "ładny",
"FilenamePatternBasic": "zwykły",
"FilenamePatternNerdy": "nerdy",
"FilenameDescription": "classic: domyślny wzorzec nazwy pliku cobalt.\nzwykły: tytuł i podstawowe informacje w nawiasach.\nładny: tytuł i informacje w nawiasach.\nnerdy: tytuł i wszystkie informacje w nawiasach.\n\nniektóre usługi nie obsługują rozbudowanych nazw plików i zawsze używają stylu klasycznego.",
"Preview": "podgląd",
"FilenamePreviewVideoTitle": "Tytuł Wideo",
"FilenamePreviewAudioTitle": "Tytuł Audio",
"FilenamePreviewAudioAuthor": "Autor Audio",
"UrgentFilenameUpdate": "konfigurowalne nazwy plików!",
"UrgentTwitterPatch": "poprawki i łatwiejsze pobieranie",
"StatusPage": "strona statusu serwisu",
"TroubleshootingGuide": "instrukcja samodzielnego rozwiązywania problemów",
"UpdateNewYears": "noworoczne porządki"
}
}

View file

@ -137,40 +137,22 @@ export default function(r, host, userFormat, isAudioOnly, lang, isAudioMuted, di
audioFormat = "best"
}
const serviceBestAudio = r.bestAudio || services[host]["bestAudio"];
const isBestAudio = audioFormat === "best";
const isBestOrMp3 = audioFormat === "mp3" || isBestAudio;
const isBestAudioDefined = isBestAudio && services[host]["bestAudio"];
const isBestHostAudio = services[host]["bestAudio"] && (audioFormat === services[host]["bestAudio"]);
const isBestOrMp3 = isBestAudio || audioFormat === "mp3";
const isBestAudioDefined = isBestAudio && serviceBestAudio;
const isBestHostAudio = serviceBestAudio && (audioFormat === serviceBestAudio);
const isTikTok = host === "tiktok" || host === "douyin";
const isTumblrAudio = host === "tumblr" && !r.filename;
const isSoundCloud = host === "soundcloud";
if (isTikTok && services.tiktok.audioFormats.includes(audioFormat)) {
if (r.isMp3 && isBestOrMp3) {
audioFormat = "mp3";
processType = "bridge"
} else if (isBestAudio) {
audioFormat = "m4a";
processType = "bridge"
}
}
if (isSoundCloud && services.soundcloud.audioFormats.includes(audioFormat)) {
if (r.isMp3 && isBestOrMp3) {
audioFormat = "mp3";
processType = "render"
copy = true
} else if (isBestAudio || audioFormat === "opus") {
audioFormat = "opus";
processType = "render"
copy = true
}
}
if (isBestAudioDefined || isBestHostAudio) {
audioFormat = services[host]["bestAudio"];
audioFormat = serviceBestAudio;
processType = "bridge";
if (isSoundCloud) {
processType = "render"
copy = true
}
} else if (isBestAudio && !isSoundCloud) {
audioFormat = "m4a";
copy = true

View file

@ -4,7 +4,7 @@ import { cleanString } from "../../sub/utils.js";
const cachedID = {
version: '',
id: ''
};
}
async function findClientID() {
try {
@ -32,9 +32,7 @@ async function findClientID() {
cachedID.id = clientid;
return clientid;
} catch (e) {
return false;
}
} catch {}
}
export default async function(obj) {
@ -58,27 +56,30 @@ export default async function(obj) {
let json = await fetch(`https://api-v2.soundcloud.com/resolve?url=${link}&client_id=${clientId}`).then((r) => {
return r.status === 200 ? r.json() : false
}).catch(() => { return false });
}).catch(() => {});
if (!json) return { error: 'ErrorCouldntFetch' };
if (!json["media"]["transcodings"]) return { error: 'ErrorEmptyDownload' };
let isMp3,
selectedStream = json.media.transcodings.filter(v => v.preset === "opus_0_0")
let bestAudio = "opus",
selectedStream = json.media.transcodings.find(v => v.preset === "opus_0_0");
// fall back to mp3 if no opus is available
if (selectedStream.length === 0) {
selectedStream = json.media.transcodings.filter(v => v.preset === "mp3_0_0")
isMp3 = true
selectedStream = json.media.transcodings.find(v => v.preset === "mp3_0_0");
bestAudio = "mp3"
}
let fileUrlBase = selectedStream[0]["url"];
let fileUrlBase = selectedStream.url;
let fileUrl = `${fileUrlBase}${fileUrlBase.includes("?") ? "&" : "?"}client_id=${clientId}&track_authorization=${json.track_authorization}`;
if (fileUrl.substring(0, 54) !== "https://api-v2.soundcloud.com/media/soundcloud:tracks:") return { error: 'ErrorEmptyDownload' };
if (json.duration > maxVideoDuration) return { error: ['ErrorLengthAudioConvert', maxVideoDuration / 60000] };
if (json.duration > maxVideoDuration)
return { error: ['ErrorLengthAudioConvert', maxVideoDuration / 60000] };
let file = await fetch(fileUrl).then(async (r) => { return (await r.json()).url }).catch(() => { return false });
let file = await fetch(fileUrl).then(async (r) => { return (await r.json()).url }).catch(() => {});
if (!file) return { error: 'ErrorCouldntFetch' };
let fileMetadata = {
@ -94,7 +95,7 @@ export default async function(obj) {
title: fileMetadata.title,
author: fileMetadata.artist
},
isMp3,
bestAudio,
fileMetadata
}
}

View file

@ -41,8 +41,9 @@ export default async function(obj) {
detail = detail?.aweme_list?.find(v => v.aweme_id === postId);
if (!detail) return { error: 'ErrorCouldntFetch' };
let video, videoFilename, audioFilename, isMp3, audio, images,
filenameBase = `tiktok_${detail.author.unique_id}_${postId}`;
let video, videoFilename, audioFilename, audio, images,
filenameBase = `tiktok_${detail.author.unique_id}_${postId}`,
bestAudio = 'm4a';
images = detail.image_post_info?.images;
@ -56,12 +57,12 @@ export default async function(obj) {
} else {
let fallback = playAddr.url_list[0];
audio = fallback;
audioFilename = `${filenameBase}_audio_fv`; // fv - from video
audioFilename = `${filenameBase}_audio`;
if (obj.fullAudio || fallback.includes("music")) {
audio = detail.music.play_url.url_list[0]
audioFilename = `${filenameBase}_audio`
audioFilename = `${filenameBase}_audio_original`
}
if (audio.slice(-4) === ".mp3") isMp3 = true;
if (audio.slice(-4) === ".mp3") bestAudio = 'mp3';
}
if (video) return {
@ -72,7 +73,7 @@ export default async function(obj) {
urls: audio,
audioFilename: audioFilename,
isAudioOnly: true,
isMp3: isMp3
bestAudio
}
if (images) {
let imageLinks = [];
@ -86,13 +87,13 @@ export default async function(obj) {
urls: audio,
audioFilename: audioFilename,
isAudioOnly: true,
isMp3: isMp3
bestAudio
}
}
if (audio) return {
urls: audio,
audioFilename: audioFilename,
isAudioOnly: true,
isMp3: isMp3
bestAudio
}
}

View file

@ -4,7 +4,7 @@ import { cleanString } from '../../sub/utils.js';
const yt = await Innertube.create();
const c = {
const codecMatch = {
h264: {
codec: "avc1",
aCodec: "mp4a",
@ -23,8 +23,8 @@ const c = {
}
export default async function(o) {
let info, isDubbed,
quality = o.quality === "max" ? "9000" : o.quality; // 9000(p) - max quality
let info, isDubbed, format = o.format || "h264";
let quality = o.quality === "max" ? "9000" : o.quality; // 9000(p) - max quality
function qual(i) {
if (!i.quality_label) {
@ -56,10 +56,16 @@ export default async function(o) {
let bestQuality, hasAudio;
let adaptive_formats = info.streaming_data.adaptive_formats.filter(e =>
e.mime_type.includes(c[o.format].codec) || e.mime_type.includes(c[o.format].aCodec)
const filterByCodec = (formats) => formats.filter(e =>
e.mime_type.includes(codecMatch[format].codec) || e.mime_type.includes(codecMatch[format].aCodec)
).sort((a, b) => Number(b.bitrate) - Number(a.bitrate));
let adaptive_formats = filterByCodec(info.streaming_data.adaptive_formats);
if (adaptive_formats.length === 0 && format === "vp9") {
format = "h264"
adaptive_formats = filterByCodec(info.streaming_data.adaptive_formats)
}
bestQuality = adaptive_formats.find(i => i.has_video);
hasAudio = adaptive_formats.find(i => i.has_audio);
@ -105,14 +111,15 @@ export default async function(o) {
isAudioOnly: true,
urls: audio.decipher(yt.session.player),
filenameAttributes: filenameAttributes,
fileMetadata: fileMetadata
fileMetadata: fileMetadata,
bestAudio: format === "h264" ? 'm4a' : 'opus'
}
const matchingQuality = Number(quality) > Number(bestQuality) ? bestQuality : quality,
checkSingle = i => qual(i) === matchingQuality && i.mime_type.includes(c[o.format].codec),
checkSingle = i => qual(i) === matchingQuality && i.mime_type.includes(codecMatch[format].codec),
checkRender = i => qual(i) === matchingQuality && i.has_video && !i.has_audio;
let match, type, urls;
if (!o.isAudioOnly && !o.isAudioMuted && o.format === 'h264') {
if (!o.isAudioOnly && !o.isAudioMuted && format === 'h264') {
match = info.streaming_data.formats.find(checkSingle);
type = "bridge";
urls = match?.decipher(yt.session.player);
@ -128,8 +135,8 @@ export default async function(o) {
if (match) {
filenameAttributes.qualityLabel = match.quality_label;
filenameAttributes.resolution = `${match.width}x${match.height}`;
filenameAttributes.extension = c[o.format].container;
filenameAttributes.youtubeFormat = o.format;
filenameAttributes.extension = codecMatch[format].container;
filenameAttributes.youtubeFormat = format;
return {
type,
urls,

View file

@ -57,7 +57,6 @@
"alias": "tiktok videos, photos & audio",
"patterns": [":user/video/:postId", ":id", "t/:id", ":user/photo/:postId"],
"subdomains": ["vt", "vm"],
"audioFormats": ["best", "m4a", "mp3"],
"enabled": true
},
"douyin": {
@ -74,7 +73,6 @@
"soundcloud": {
"patterns": [":author/:song/s-:accessKey", ":author/:song", ":shortLink"],
"subdomains": ["on", "m"],
"audioFormats": ["best", "opus", "mp3"],
"enabled": true
},
"instagram": {