// ==UserScript== // @name bangumi-sort-index // @name:zh 排序首页条目 // @namespace https://trim21.me/ // @description Sort subjects on index page // @version 0.0.3 // @author Trim21 // @source https://github.com/Trim21/bangumi-sort-index // @supportURL https://github.com/Trim21/bangumi-sort-index/issues // @license MIT // @include /^https://(bangumi\.tv|bgm\.tv|chii\.in)/[^/]*/ // @require https://cdn.jsdelivr.net/npm/jquery@3.6.0/dist/jquery.min.js // @run-at document-end // @grant GM_addStyle // @downloadURL none // ==/UserScript== /******/ (function() { // webpackBootstrap /******/ "use strict"; var __webpack_exports__ = {}; ;// CONCATENATED MODULE: external "$" var external_$_namespaceObject = $; ;// CONCATENATED MODULE: ./src/style.css /* harmony default export */ var style = ("#prgManagerHeader ul#prgManagerOrder li a span {\n}\n\n#prgManagerHeader ul#prgManagerOrder li a.focus span {\n filter: invert();\n}\n\n/* https://www.flaticon.com/free-icons/sort */\n\n#prgManagerHeader ul#prgManagerOrder li a#switchSmartOrder span {\n /* https://icons8.com/icon/69918/sort-by-recently-viewed */\n background: url()\n no-repeat top;\n background-size: 20px 20px;\n}\n\n#prgManagerHeader ul#prgManagerOrder li a#switchNormalOrder span {\n /* https://icons8.com/icon/69920/sort-by-modified-date */\n background: url()\n no-repeat top;\n background-size: 20px 20px;\n}\n\n#prgManagerHeader ul#prgManagerOrder li a#switchUpdateOrder span {\n /* https://icons8.com/icon/69917/sort-by-follow-up-date */\n background: url()\n no-repeat top;\n background-size: 20px 20px;\n}\n\n#prgManagerHeader ul#prgManagerOrder li a span {\n display: block;\n text-indent: -9999px;\n height: 18px;\n width: 20px;\n}\n"); ;// CONCATENATED MODULE: ./src/index.ts GM_addStyle(style); const castKeyword = '首播'; const configKey = 'index-sort-order'; function bangumiSortIndex() { class Subject { constructor(el) { this.el = el; const titleEL = el.find('.epGird .tinyHeader .textTip').last(); this.title = titleEL.attr('data-subject-name-cn') ?? titleEL.attr('data-original-title') ?? titleEL.attr('data-subject-name') ?? 'title'; const nextWatch = el.find('li a:not(.epBtnWatched)').first(); const rel = nextWatch.attr('rel'); if (rel) { this.nextDate = getDate(rel); } else { this.nextDate = 0; } const lastWatched = el.find('li a:not(.epBtnWatched)').last(); const lastWatchedRel = lastWatched.attr('rel'); if (lastWatchedRel) { this.lastWatched = getDate(lastWatchedRel); } else { this.lastWatched = 0; } this.airing = el.find('li a.epBtnNA').length !== 0 ? -1 : 1; this.smartKey = this.airing; } } const container = external_$_namespaceObject('#cloumnSubjectInfo .infoWrapper_tv'); const originals = Array.from(external_$_namespaceObject('.infoWrapper_tv [id^=subjectPanel_]')).map(element => { return new Subject(external_$_namespaceObject(element)); }); const subjects = [...originals]; function getDate(rel) { var _document$querySelect, _document$querySelect2; const castDate = Array.from(((_document$querySelect = document.querySelector(rel)) === null || _document$querySelect === void 0 ? void 0 : (_document$querySelect2 = _document$querySelect.querySelector('span.tip')) === null || _document$querySelect2 === void 0 ? void 0 : _document$querySelect2.childNodes) ?? []).filter(e => e.nodeType === Node.TEXT_NODE).map(e => e.textContent ?? '').filter(t => t.includes(castKeyword)); if (castDate.length) { return new Date(castDate[0].replace(`${castKeyword}:`, '')).getTime(); } return 0; } function render(subjects) { subjects.forEach(s => { s.el.remove(); }); subjects.forEach((s, i) => { s.el.removeClass('odd'); s.el.removeClass('even'); if (i % 2) { s.el.addClass('even'); } else { s.el.addClass('odd'); } container.append(s.el); }); } function smart(subjects) { subjects.sort((a, b) => { if (a.airing === b.airing) { return (b.nextDate - a.nextDate) * a.airing; } return a.airing; }); render(subjects); } function update(subjects) { subjects.sort((a, b) => -a.lastWatched + b.lastWatched); render(subjects); } function normal() { render(originals); } function onLoad() { // prgManagerMain.classList.contains('tinyModeWrapper') const orderUI = external_$_namespaceObject(``); external_$_namespaceObject('#prgManagerHeader').append(orderUI[0]); if (!localStorage['index-sort-order']) { localStorage['index-sort-order'] = 'smart'; } const optionUIs = orderUI.find('li'); let mode = localStorage.getItem(configKey) ?? 'normal'; function click() { optionUIs.find('a').removeClass('focus'); if (this) { var _el$data; const el = external_$_namespaceObject(this); mode = (_el$data = el.data('mode')) === null || _el$data === void 0 ? void 0 : _el$data.toString(); localStorage.setItem(configKey, mode); } switch (mode) { case 'smart': smart(subjects); break; case 'update': update(subjects); break; case 'normal': normal(); break; default: mode = 'normal'; localStorage.setItem(configKey, mode); normal(); } external_$_namespaceObject(`#prgManagerOrder li[data-mode="${mode}"]`).find('a').addClass('focus'); } optionUIs.on('click', click); click(); } onLoad(); } bangumiSortIndex(); /******/ })() ;