Greasy Fork is available in English.

Flickr - AUTO More (Merged) all "Auto" at once

in Photo page : Auto Load More Comments / Groups / Galleries / Video Play and Replay

Ekde 2025/03/16. Vidu La ĝisdata versio.

  1. // ==UserScript==
  2. // @name Flickr - AUTO More (Merged) all "Auto" at once
  3. // @version 1.01
  4. // @description in Photo page : Auto Load More Comments / Groups / Galleries / Video Play and Replay
  5. // @author decembre
  6. // @icon https://external-content.duckduckgo.com/ip3/blog.flickr.net.ico
  7. // @namespace https://greasyfork.org/fr/users/8-decembre
  8. // @include http*://www.flickr.com/photos/*
  9. // @exclude http*://*flickr.com/photos/*/map*
  10. // @exclude http*://*flickr.com/photos/*/page*
  11. // @exclude http*://*flickr.com/groups/*/pool/*
  12. // @exclude http*://*flickr.com/photos/*/favorites*
  13. // @exclude http*://*flickr.com/photos/*/albums*
  14. // @require https://ajax.googleapis.com/ajax/libs/jquery/3.6.0/jquery.min.js
  15. // @grant GM_addStyle
  16. // @run-at document-end
  17. // ==/UserScript==
  18.  
  19. (function () {
  20. // Use jQuery directly instead of $
  21. const $ = window.jQuery;
  22.  
  23. // 0 - Fo GM "Utags" - Function to check if the modal is open
  24. function isModalOpen() {
  25. return $('.browser_extension_settings_container:visible, .utags_modal:visible').length > 0;
  26. }
  27.  
  28. function waitForKeyElements(
  29. selectorTxt,
  30. actionFunction,
  31. bWaitOnce,
  32. iframeSelector
  33. ) {
  34. if (isModalOpen()) {
  35. setTimeout(function() {
  36. waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector);
  37. }, 100);
  38. return;
  39. }
  40.  
  41. var targetNodes, btargetsFound;
  42.  
  43. if (typeof iframeSelector === "undefined") {
  44. targetNodes = $(selectorTxt);
  45. } else {
  46. targetNodes = $(iframeSelector).contents().find(selectorTxt);
  47. }
  48.  
  49. if (targetNodes && targetNodes.length > 0) {
  50. btargetsFound = true;
  51. targetNodes.each(function () {
  52. var jThis = $(this);
  53. var alreadyFound = jThis.data('alreadyFound') || false;
  54.  
  55. if (!alreadyFound) {
  56. var cancelFound = actionFunction(jThis);
  57. if (cancelFound) {
  58. btargetsFound = false;
  59. } else {
  60. jThis.data('alreadyFound', true);
  61. }
  62. }
  63. });
  64. } else {
  65. btargetsFound = false;
  66. }
  67.  
  68. var controlObj = waitForKeyElements.controlObj || {};
  69. var controlKey = selectorTxt.replace(/[^\w]/g, "_");
  70. var timeControl = controlObj[controlKey];
  71.  
  72. if (btargetsFound && bWaitOnce && timeControl) {
  73. clearInterval(timeControl);
  74. delete controlObj[controlKey];
  75. } else {
  76. if (!timeControl) {
  77. timeControl = setInterval(function () {
  78. waitForKeyElements(selectorTxt, actionFunction, bWaitOnce, iframeSelector);
  79. }, 300);
  80. controlObj[controlKey] = timeControl;
  81. }
  82. }
  83. waitForKeyElements.controlObj = controlObj;
  84. }
  85.  
  86.  
  87.  
  88. // 1 - PHOTO pages - AUTO MORE Comments
  89. function actionMoreCommnents(node){
  90. if (isModalOpen()) {
  91. return false;
  92. }
  93. console.log ("Found More Comments Button. Clicking it!");
  94. var clickEvent = document.createEvent ('MouseEvents');
  95. clickEvent.initEvent ('click', true, true);
  96. node[0].dispatchEvent (clickEvent);
  97. return true;
  98. }
  99. console.log ("Waiting for More Comments Button");
  100. waitForKeyElements(".sub-photo-content-container .photo-comments.with-emoji-picker a.load-more-button:not(.hidden)", actionMoreCommnents);
  101.  
  102.  
  103. // 2 - PHOTO pages - AUTO MORE Groups
  104. function actionMorePools(node){
  105. if (isModalOpen()) {
  106. return false;
  107. }
  108. console.log ("Found More Pools Button. Clicking it!");
  109. var clickEvent = document.createEvent ('MouseEvents');
  110. clickEvent.initEvent ('click', true, true);
  111. node[0].dispatchEvent (clickEvent);
  112. return true;
  113. }
  114. console.log ("Waiting for More Pools Button");
  115. waitForKeyElements(".sub-photo-contexts-view .sub-photo-context.sub-photo-context-groups .view-all-contexts-of-type a", actionMorePools);
  116.  
  117.  
  118.  
  119. // 3 - PHOTO pages - AUTO MORE Galleries
  120. function actionMoreGall(node){
  121. if (isModalOpen()) {
  122. return false;
  123. }
  124. console.log ("Found More Galleries Button. Clicking it!");
  125. var clickEvent = document.createEvent ('MouseEvents');
  126. clickEvent.initEvent ('click', true, true);
  127. node[0].dispatchEvent (clickEvent);
  128. return true;
  129. }
  130. console.log ("Waiting for More Galleries Button");
  131. waitForKeyElements(".sub-photo-galleries-view:not(.empty) .sub-photo-context.sub-photo-context-galleries .view-all-contexts-of-type a", actionMoreGall);
  132.  
  133.  
  134. // 4 - PHOTO pages - AUTO MORE Previews in Mini Thumbnail
  135. function actionNEXT(node) {
  136. if (isModalOpen()) {
  137. return false;
  138. }
  139. console.log("Found More thumbnail next arrow. Clicking it!");
  140. var clickEvent = document.createEvent('MouseEvents');
  141. clickEvent.initEvent('click', true, true);
  142. node[0].dispatchEvent(clickEvent);
  143. return true;
  144. }
  145.  
  146. console.log("Waiting for More thumbnail next arrow");
  147. waitForKeyElements(".context-slider-scrappy-view.hover .context-slider.big-slider > .nav-r", actionNEXT);
  148.  
  149. // ONLY PREV arrow
  150. function actionOnlyPrev(node) {
  151. if (isModalOpen()) {
  152. return false;
  153. }
  154. console.log("Found More thumbnail only PREV arrow. Clicking it!");
  155. var clickEvent = document.createEvent('MouseEvents');
  156. clickEvent.initEvent('click', true, true);
  157. node[0].dispatchEvent(clickEvent);
  158. return true;
  159. }
  160.  
  161. console.log("Waiting for More thumbnail ONLY PREV arrow");
  162. waitForKeyElements(".context-slider-scrappy-view .context-slider.hover:has(.nav-l):has(.nav-r[style='display: none;']):has(.nav-r[hidden]) > .nav.nav-l span", actionOnlyPrev);
  163.  
  164. // 4 - AUTO Video Play - PHOTO pages
  165. function loadScript(url, callback) {
  166. var script = document.createElement('script');
  167. script.src = url;
  168. script.onload = callback;
  169. document.head.appendChild(script);
  170. }
  171.  
  172. loadScript('https://code.jquery.com/jquery-3.6.0.min.js', function() {
  173. (function($) {
  174. 'use strict';
  175.  
  176. console.log('Script loaded');
  177.  
  178. // Wait for the video element to be loaded
  179. var videoElement = document.querySelector('.fluid.html-photo-page-scrappy-view video#video_1_html5_api');
  180. if (videoElement) {
  181. console.log('Video element found');
  182. // Set the video to auto play and loop
  183. videoElement.autoplay = true;
  184. videoElement.loop = true;
  185. videoElement.muted = true; // Mute the video
  186. videoElement.volume = 0.5;
  187.  
  188. console.log('Auto-play and loop properties set');
  189.  
  190. // Add an event listener to play the video when the user interacts with it
  191. videoElement.addEventListener('click', function() {
  192. if (!isModalOpen()) {
  193. this.play();
  194. }
  195. });
  196.  
  197. // Add an event listener to replay the video when it ends
  198. videoElement.addEventListener('ended', function() {
  199. if (!isModalOpen()) {
  200. this.play();
  201. }
  202. });
  203.  
  204. // Add an event listener to handle volume control
  205. videoElement.addEventListener('volumechange', function() {
  206. this.muted = false;
  207. });
  208.  
  209. // Play the video
  210. if (!isModalOpen()) {
  211. videoElement.play();
  212. }
  213.  
  214. console.log('Video played');
  215. } else {
  216. console.log('Video element not found');
  217.  
  218. // If the video element is not found, wait for it to be loaded
  219. var intervalId = setInterval(function() {
  220. if (isModalOpen()) {
  221. return;
  222. }
  223. var videoElement = document.querySelector('.fluid.html-photo-page-scrappy-view video#video_1_html5_api');
  224. if (videoElement) {
  225. console.log('Video element found after interval');
  226.  
  227. // Set the video to auto play and loop
  228. videoElement.autoplay = true;
  229. videoElement.loop = true;
  230. videoElement.muted = true; // Mute the video
  231. videoElement.volume = 0.5;
  232.  
  233. console.log('Auto-play and loop properties set after interval');
  234.  
  235. // Add an event listener to play the video when the user interacts with it
  236. videoElement.addEventListener('click', function() {
  237. if (!isModalOpen()) {
  238. this.play();
  239. }
  240. });
  241.  
  242. // Add an event listener to replay the video when it ends
  243. videoElement.addEventListener('ended', function() {
  244. if (!isModalOpen()) {
  245. this.play();
  246. }
  247. });
  248.  
  249. // Add an event listener to handle volume control
  250. videoElement.addEventListener('volumechange', function() {
  251. this.muted = false;
  252. });
  253.  
  254. // Play the video
  255. if (!isModalOpen()) {
  256. videoElement.play();
  257. }
  258.  
  259. console.log('Video played after interval');
  260.  
  261. // Clear the interval
  262. clearInterval(intervalId);
  263. }
  264. }, 100);
  265. }
  266. })(null);
  267. });
  268. })();
  269.