// ==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
// @updateURL https://raw.githubusercontent.com/Fenn3c401/Trakt.tv-Userscript-Collection/main/userscripts/meta/p2o98x5r.meta.js
// @downloadURL https://raw.githubusercontent.com/Fenn3c401/Trakt.tv-Userscript-Collection/main/userscripts/dist/p2o98x5r.user.js
// @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 ? $(`
`).insertAfter($sortableGrid) : undefined,
$btn = $(``).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;
}
`);
}