IMDB works#

Shows number of credited works after names on IMDB site

  1. // ==UserScript==//
  2. // @name IMDB works#
  3. // @description Shows number of credited works after names on IMDB site
  4. // @match *://*.imdb.com/*
  5. // @version 1.0.7
  6. // @author wOxxOm
  7. // @namespace wOxxOm.scripts
  8. // @license MIT License
  9. // @grant GM_addStyle
  10. // @run-at document-start
  11. // @require https://gf.qytechs.cn/scripts/12228/code/setMutationHandler.js
  12. // ==/UserScript==
  13.  
  14. var SELECTOR = 'a[href^="/name/nm"]';
  15. var CACHE_FRESH_DURATION = 30 * 24 * 3600 * 1000; // 1 month
  16. var CACHE_STALE_DURATION = CACHE_FRESH_DURATION * 2; // keep stale data another 2 months, then kill
  17.  
  18. GM_addStyle(
  19. '.number-of-works, .number-of-works span { opacity: 0.5; transition: opacity .25s ease-in-out .25s; }' +
  20. '.number-of-works span:before { content: "/"; }' +
  21. '.number-of-works:hover { opacity: 1.0; }' +
  22. '.number-of-works:hover span { opacity: 1.0; }' +
  23. '.number-of-works:before { content: " ["; opacity: 0.5; }' +
  24. '.number-of-works:after { content: "]"; opacity: 0.5; }'
  25. );
  26.  
  27. process(document.querySelectorAll(SELECTOR));
  28. setMutationHandler(document, SELECTOR, process);
  29.  
  30. if (location.pathname.match(/\/name\/nm\d+\/?$/)) {
  31. document.addEventListener('DOMContentLoaded', function() {
  32. parseNamePage(document, {cacheKey: 'works#' + location.pathname.match(/\/name\/nm(\d+)\/?$/)[1]});
  33. });
  34. }
  35.  
  36. function process(links) {
  37. var now = Date.now();
  38. for (var link, i = 0; (link = links[i++]); ) {
  39. if (link.querySelector('img') ||
  40. !link.textContent.trim() ||
  41. link.children[0] && link.textContent.trim() != link.children[0].textContent.trim() ||
  42. link.nextElementSibling && link.nextElementSibling.matches('.number-of-works'))
  43. continue;
  44. var id = (link.pathname.match(/\/name\/nm(\d+)\/?$/) || [])[1];
  45. if (!id)
  46. continue;
  47. var cacheKey = 'works#' + id;
  48. var cache = (localStorage[cacheKey] || '').split('\t');
  49. if (cache.length == 2 && +cache[0] && cache[1]) {
  50. showWorksNum(link, cache[1]);
  51. var isFresh = +cache[0] > now;
  52. if (isFresh)
  53. continue;
  54. }
  55. doXHR({
  56. url: link.pathname,
  57. link: link,
  58. cacheKey: cacheKey,
  59. onload: parseNamePage,
  60. });
  61. }
  62. }
  63.  
  64. function showWorksNum(link, num) {
  65. num = num.toString().replace(/\/(\d+)/, '<span>$1</span>');
  66. if (link.nextElementSibling && link.nextElementSibling.matches('.number-of-works')) {
  67. link.nextElementSibling.innerHTML = num;
  68. } else {
  69. link.insertAdjacentHTML('afterend', '<span class="number-of-works">' + num + '</span>');
  70. }
  71. }
  72.  
  73. function parseNamePage(doc, options) {
  74. var credits = [].map.call(doc.querySelectorAll('#filmography .head'), function(e) {
  75. return {
  76. num: +(e.textContent.match(/\((\d+) credits?\)/i) || [])[1],
  77. type: (e.querySelector('a[name]') || {name: ''}).name,
  78. };
  79. });
  80. if (!credits.length)
  81. return;
  82. var creditsSum = credits.reduce(function(sum, e) { return sum + e.num; }, 0);
  83. var rxIgnore = /^(self|archive_footage|thanks)$/;
  84. var creditsSorted = credits.sort(function(a, b) {
  85. if (rxIgnore.test(a.type))
  86. return 1;
  87. else if (rxIgnore.test(b.type))
  88. return -1;
  89. else
  90. return b.num - a.num || (a.type == 'actor' ? -1 : b.type == 'actor' ? 1 : 0);
  91. });
  92. var worksNum = rxIgnore.test(creditsSorted[0].type) ? creditsSum : creditsSorted[0].num + (credits.length > 1 ? '/' + creditsSum : '');
  93. localStorage[options.cacheKey] = '' + (Date.now() + CACHE_FRESH_DURATION) + '\t' + worksNum;
  94. if (options.link)
  95. showWorksNum(options.link, worksNum);
  96. }
  97.  
  98. function doXHR(options) {
  99. if (document.readyState == 'complete') {
  100. sendRequest(options);
  101. return;
  102. }
  103. if (!doXHR.queue)
  104. initQueue();
  105. if (!isDupe()) {
  106. doXHR.queue.push(options);
  107. doXHR.queuedUrl[options.url] = options;
  108. }
  109.  
  110. function sendRequest(options) {
  111. var xhr = new XMLHttpRequest();
  112. xhr.open('GET', options.url);
  113. xhr.responseType = 'document';
  114. xhr.onload = function(e) {
  115. options.onload(xhr.response, options);
  116. doXHR.activeRequests--;
  117. poolQueue();
  118. };
  119. doXHR.activeRequests++;
  120. xhr.send();
  121. }
  122.  
  123. function initQueue() {
  124. doXHR.queue = [];
  125. doXHR.queuedUrl = {};
  126. doXHR.activeRequests = 0;
  127. document.addEventListener('DOMContentLoaded', function() {
  128. cleanupStorage();
  129. poolQueue();
  130. });
  131. }
  132.  
  133. function isDupe() {
  134. var dupe = doXHR.queuedUrl[options.url];
  135. if (!dupe)
  136. return false;
  137. if (dupe.link == options.link)
  138. return true;
  139. // this request's link element will use the will-be-cached data from the earlier queued request
  140. options.url = '';
  141. var _onload = dupe.onload;
  142. dupe.onload = function() {
  143. _onload.apply(null, arguments);
  144. showWorksNum(options.link, localStorage[options.cacheKey].split('\t')[1]);
  145. };
  146. return true;
  147. }
  148.  
  149. function poolQueue() {
  150. while (doXHR.queue.length && doXHR.activeRequests < 16) {
  151. sendRequest(doXHR.queue.shift());
  152. }
  153. }
  154. }
  155.  
  156. function cleanupStorage() {
  157. setTimeout(function doCleanup() {
  158. var now = Date.now(), i = 0;
  159. for (var k in localStorage) {
  160. if (k.lastIndexOf('works#', 0) === 0 && (+localStorage[k].split('\t')[0]) + CACHE_STALE_DURATION < now) {
  161. delete localStorage[k];
  162. }
  163. if (++i % 100 === 0 && Date.now() - now > 10) {
  164. setTimeout(doCleanup, 1000);
  165. return;
  166. }
  167. }
  168. }, 1000);
  169. }

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址