您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Adds a button to your personal lists summary page (/lists) which appends the lists from other categories (/collaborations, /liked, /liked/official) for a consolidated view.
// ==UserScript== // @name Trakt.tv | Consolidated Lists View // @description Adds a button to your personal lists summary page (/lists) which appends the lists from other categories (/collaborations, /liked, /liked/official) for a consolidated view. // @version 1.0.1 // @namespace https://github.com/Fenn3c401 // @author Fenn3c401 // @license GPL-3.0-or-later // @homepageURL https://github.com/Fenn3c401/Trakt.tv-Userscript-Collection#readme // @supportURL https://github.com/Fenn3c401/Trakt.tv-Userscript-Collection/issues // @icon  // @match https://trakt.tv/* // @run-at document-start // @grant unsafeWindow // @grant GM_addStyle // ==/UserScript== /* ### General - Sorting, filtering and list actions (unlike, delete etc.) should work as usual. Also works on /lists pages of other users. - The [Trakt.tv | Bug Fixes and Optimizations](brzmp0a9.md) userscript contains an improved/fixed "Read more..." function which greatly speeds up the rendering of the appended lists. */ 'use strict'; addStyles(); document.addEventListener('turbo:load', () => { if (!/^\/users\/[^\/]+\/lists$/.test(location.pathname)) return; const $ = unsafeWindow.jQuery; if (!$) return; const $sortableGrid = $('#sortable-grid'), $spacer = $sortableGrid.children().length ? $(`<hr id="consolidated-lists-view-spacer">`).insertAfter($sortableGrid) : undefined, $btn = $(`<button id="consolidated-lists-view-btn" type="button">Consolidated Lists View</button>`).insertAfter($spacer ?? $sortableGrid); $btn.on('click', async () => { $btn.text('Loading...').prop('disabled', true); const fetchListElems = async (pathSuffix) => fetch(location.pathname + pathSuffix) .then((r) => r.text()) .then((r) => $(new DOMParser().parseFromString(r, 'text/html')).find('.personal-list')); let $fetchedLists = $((await Promise.all(['/collaborations', '/liked', '/liked/official'].map(fetchListElems))).flatMap(($listElems) => $listElems.get())); const $personalLists = $('.personal-list'), personalListsIds = $personalLists.map((_i, e) => $(e).attr('data-list-id')).get(); $fetchedLists = $fetchedLists.filter((_i, e) => !personalListsIds.includes($(e).attr('data-list-id'))); // duplicate removal because a user can like his own personal lists if (!$fetchedLists.length) { $btn.text('No other lists found.') return; } const rankOffset = +$personalLists.last().attr('data-rank'); $fetchedLists.each((i, e) => $(e).attr('data-rank', rankOffset + i + 1)); $fetchedLists .find('.btn-list-progress').click(function() { unsafeWindow.showLoading(); const dataListId = $(this).attr('data-list-id'); if(dataListId && unsafeWindow.userSettings?.user.vip) unsafeWindow.redirect(unsafeWindow.userURL('progress?list=' + dataListId)); else unsafeWindow.redirect('/vip/list-progress'); }) .end().find('.btn-list-subscribe').click(function() { unsafeWindow.showLoading(); const dataListId = $(this).attr('data-list-id'); if(dataListId && unsafeWindow.userSettings?.user.vip) { $.post(`/lists/${dataListId}/subscribe`, function(response) { unsafeWindow.redirect(response.url); }).fail(function() { unsafeWindow.hideLoading(); unsafeWindow.toastr.error('Doh! We ran into some sort of error.'); }); } else unsafeWindow.redirect('/vip/calendars'); }) .end().find('.collaborations-deny').on('ajax:success', function(_e, response) { $('#collaborations-deny-' + response.id).children().addClass('trakt-icon-delete-thick'); $('#collaborations-approve-' + response.id).addClass('off'); $('#collaborations-block-' + response.id).addClass('off'); }); const $btnListEditLists = $('#btn-list-edit-lists'); if ($btnListEditLists.hasClass('active')) $btnListEditLists.trigger('click'); $btnListEditLists.hide(); $sortableGrid.append($fetchedLists); $spacer?.remove(); $btn.remove(); unsafeWindow.genericTooltips(); unsafeWindow.vipTooltips(); unsafeWindow.shareIcons(); unsafeWindow.convertEmojis(); unsafeWindow.userscriptAddLinksToListPreviewPosters?.(); unsafeWindow.addOverlays(); unsafeWindow.$grid?.isotope('insert', $fetchedLists); // isotope instance is only initiliazed after first filtering/sorting unsafeWindow.updateListsCount(); unsafeWindow.lazyLoadImages(); // pre-filtering because the readmore plugin is a serious performance bottleneck // requires the renderReadmore() function override from the "Trakt.tv | Bug Fixes and Optimizations" userscript to actually speed things up requestAnimationFrame(() => { const $readmoreElemsToCollapse = $fetchedLists.find('.readmore').filter((_i, e) => $(e).height() > 300); unsafeWindow.renderReadmore($readmoreElemsToCollapse); }); }); }, { capture: true }); function addStyles() { GM_addStyle(` #consolidated-lists-view-btn { margin: 20px auto 0; padding: 8px 16px; border-radius: var(--btn-radius); border: 1px solid hsl(0deg 0% 20% / 65%); background-color: #fff; color: #333; font-size: 18px; font-weight: var(--headings-font-weight); font-family: var(--headings-font-family); transition: all 0.2s; } #consolidated-lists-view-btn:hover { color: var(--brand-primary); } #consolidated-lists-view-btn:active { background-color: #ccc; } body.dark-knight #consolidated-lists-view-btn { border: none; background-color: #333; color: #fff; } body.dark-knight #consolidated-lists-view-btn:hover { background-color: var(--brand-primary); } body.dark-knight #consolidated-lists-view-btn:active { background-color: #666; } @media (min-width: 768px) { body:has(> .bottom[id*="content-page"]) #consolidated-lists-view-btn { margin-bottom: -20px; } } :is(#consolidated-lists-view-btn, #consolidated-lists-view-spacer) { display: block !important; } body:has(#btn-list-edit-lists.active) :is(#consolidated-lists-view-btn, #consolidated-lists-view-spacer) { display: none !important; } `); }