MyDealz Reactions Viewer 2025-04

zeigt für Kommentare erhaltene like,funny,helpful

此腳本不應該直接安裝,它是一個供其他腳本使用的函式庫。欲使用本函式庫,請在腳本 metadata 寫上: // @require https://update.gf.qytechs.cn/scripts/528580/1564442/MyDealz%20Reactions%20Viewer%202025-04.js

  1. function createProgressDialog(username, maxPages) {
  2. const dialog = document.createElement('div');
  3. dialog.style.cssText = 'position:fixed; top:50%; left:50%; transform:translate(-50%,-50%); z-index:9999; background:white; padding:20px; border-radius:8px; box-shadow:0 0 10px rgba(0,0,0,0.3); width:450px;';
  4. dialog.innerHTML = `
  5. <h2 style="margin-top:0; border-bottom:1px solid #ddd; padding-bottom:10px;">Analysiere Kommentare von ${username}</h2>
  6. <div style="margin:10px 0;">
  7. <div id="progressText">Untersuche Aktivitätsseite auf Kommentare (1/${maxPages})</div>
  8. <div style="background:#eee; height:20px; border-radius:4px; margin:10px 0;">
  9. <div id="progressBar" style="background:#4CAF50; height:100%; width:0; border-radius:4px;"></div>
  10. </div>
  11. <div id="progressDetails" style="font-family:monospace; min-height:20px;"></div>
  12. </div>
  13. <div style="text-align:right; margin-top:15px;">
  14. <button id="cancelButton" style="background-color:#5cb85c; border:1px solid #4cae4c; color:white; padding:6px 12px; border-radius:4px; cursor:pointer;">Auswertung abbrechen</button>
  15. </div>
  16. `;
  17. document.body.appendChild(dialog);
  18. document.getElementById('cancelButton').addEventListener('click', () => {
  19. dialog.remove();
  20. window.stop();
  21. location.reload(true);
  22. return false;
  23. });
  24. let currentPageDots = 0;
  25. return {
  26. dialog,
  27. updateProgress: (currentPage) => {
  28. const percent = Math.min(((currentPage) / (maxPages + 1)) * 100, 100);
  29. document.getElementById('progressBar').style.width = `${percent}%`;
  30. document.getElementById('progressText').textContent = `Untersuche Aktivitätsseite auf Kommentare (${currentPage}/${maxPages})`;
  31. // Zurücksetzen der Punkte bei neuer Seite
  32. document.getElementById('progressDetails').textContent = '';
  33. currentPageDots = 0;
  34. },
  35. addProgressDetail: (type) => {
  36. if (currentPageDots >= 20) return; // Maximal 20 Punkte pro Seite
  37. const details = document.getElementById('progressDetails');
  38. details.textContent += type === 'wait' ? 'W' : '.';
  39. currentPageDots++;
  40. }
  41. };
  42. }
  43.  
  44. async function viewReactions(username) {
  45. function checkPopups() {
  46. const popup = window.open(null, '_blank', 'width=100,height=100');
  47. if (!popup || popup.closed) {
  48. alert("Bitte PopUps für diese Seite erlauben und erneut versuchen.");
  49. return false;
  50. }
  51. popup.close();
  52. return true;
  53. }
  54. if (!checkPopups()) return;
  55. let stopProcessing = false;
  56. const YEAR_IN_MS = 365 * 864e5;
  57. const n = username;
  58. let p = 0;
  59. const results = [];
  60. async function fetchWithRetry(url, options, maxRetries = 5) {
  61. for (let i = 0; i < maxRetries; i++) {
  62. if (stopProcessing) return null;
  63. try {
  64. const response = await fetch(url, options);
  65. if (response.status === 429) {
  66. progressUI.addProgressDetail('wait');
  67. if (i === maxRetries - 1) {
  68. alert("Der Server ist zur Zeit überlastet. Bitte später erneut versuchen.");
  69. progressUI.dialog.remove();
  70. window.stop();
  71. location.reload(true);
  72. return null;
  73. }
  74. await new Promise(t => setTimeout(t, 5000));
  75. continue;
  76. }
  77. return response;
  78. } catch (e) {
  79. if (i === maxRetries - 1) {
  80. progressUI.dialog.remove();
  81. throw e;
  82. }
  83. }
  84. }
  85. return null;
  86. }
  87. const firstPageResponse = await fetchWithRetry(`https://www.mydealz.de/profile/${n}?page=1`);
  88. if (!firstPageResponse) return;
  89. const firstPageHtml = await firstPageResponse.text();
  90. if (p === 0) {
  91. const match = firstPageHtml.match(/window\.__INITIAL_STATE__.*?"lastPage":(\d+)/);
  92. p = match ? parseInt(match[1]) : 1;
  93. }
  94. const progressUI = createProgressDialog(username, p);
  95. for (let i = 1; i <= p; i++) {
  96. if (stopProcessing) break;
  97. const pageResponse = i === 1 ?
  98. { text: () => firstPageHtml } :
  99. await fetchWithRetry(`https://www.mydealz.de/profile/${n}?page=${i}`);
  100. if (!pageResponse) break;
  101. const pageText = await pageResponse.text();
  102. progressUI.updateProgress(i);
  103. const matches = [...pageText.matchAll(/href=(https:\/\/www\.mydealz\.de\/.*?-(\d+)#(?:comment|reply)-(\d+))/g)];
  104. const wrapper = document.createElement('div');
  105. wrapper.innerHTML = pageText;
  106. const titles = [];
  107. const titleElements = wrapper.querySelectorAll('.size--all-xs.color--text-TranslucentSecondary.space--mt-1');
  108. titleElements.forEach(element => {
  109. titles.push(element.textContent);
  110. });
  111. for (let j = 0; j < matches.length; j++) {
  112. if (stopProcessing) break;
  113. const match = matches[j];
  114. try {
  115. const reactionsResponse = await fetchWithRetry("https://www.mydealz.de/graphql", {
  116. method: 'POST',
  117. headers: { 'Content-Type': 'application/json' },
  118. body: JSON.stringify({
  119. query: `query commentReactions($commentId:ID!){commentReactions(commentId:$commentId){counts{type count}}}`,
  120. variables: { commentId: match[3] }
  121. })
  122. });
  123. if (!reactionsResponse) break;
  124. progressUI.addProgressDetail('comment');
  125. const reactionsData = await reactionsResponse.json();
  126. const counts = reactionsData.data.commentReactions.counts;
  127. const likes = counts.find(x => x.type === "LIKE")?.count || 0;
  128. const funny = counts.find(x => x.type === "FUNNY")?.count || 0;
  129. const helpful = counts.find(x => x.type === "HELPFUL")?.count || 0;
  130. const commentDateResponse = await fetchWithRetry("https://www.mydealz.de/graphql", {
  131. method: 'POST',
  132. headers: { 'Content-Type': 'application/json' },
  133. body: JSON.stringify({
  134. query: 'query comment($id:ID!){comment(id:$id){createdAt}}',
  135. variables: { id: match[3] }
  136. })
  137. });
  138. if (!commentDateResponse) break;
  139. const createdAt = (await commentDateResponse.json()).data.comment.createdAt;
  140. if (new Date(createdAt).getTime() < Date.now() - YEAR_IN_MS) {
  141. progressUI.dialog.remove();
  142. const resultWindow = window.open("", "", "width=800,height=600");
  143. resultWindow.document.write(`
  144. <style>
  145. body { font-family: Arial, sans-serif; font-size: 14px; color: #333; background-color: #f5f5f5; }
  146. .result-container { background-color: white; padding: 20px; border-radius: 8px; box-shadow: 0 2px 4px rgba(0,0,0,0.1); }
  147. .result-item { border-bottom: 1px solid #ddd; padding: 10px 0; display: flex; justify-content: space-between; }
  148. .result-item:last-child { border-bottom: none; }
  149. .reactions { min-width: 100px; }
  150. .date { min-width: 150px; }
  151. a { color: #2c7cbc; text-decoration: none; }
  152. a:hover { text-decoration: underline; }
  153. h2 { color: #2c7cbc; }
  154. </style>
  155. <div class="result-container">
  156. <h2>Auswertung der Reactions für ${n}</h2>
  157. ${results.map(x => `
  158. <div class="result-item">
  159. <span class="reactions">L:${x.l} F:${x.f} H:${x.h}${x.h > 2 ? "*" : ""}</span>
  160. <span class="date">${x.date}</span>
  161. <a href="${x.url}" target="_blank">${x.title}</a>
  162. </div>
  163. `).join('')}
  164. <h3>Zusammenfassung:</h3>
  165. <p>${results[results.length-1].date} - ${results[0].date}</p>
  166. <p>${results.length} Kommentare davon ${results.filter(x => x.l + x.f + x.h > 0).length} mit mindestens einer Reaction</p>
  167. <p>L:${results.reduce((a,c) => a + c.l, 0)} F:${results.reduce((a,c) => a + c.f, 0)} H:${results.reduce((a,c) => a + c.h, 0)} davon ${results.filter(x => x.h > 2).length} mit * (mindestens 3 Hilfreich-Bewertungen)</p>
  168. </div>
  169. `);
  170. return;
  171. }
  172. const elementType = match[1].includes('#reply-') ? 'reply' : 'comment';
  173. const url = `https://www.mydealz.de/${match[2]}#${elementType}-${match[3]}`;
  174. results.push({
  175. url: url,
  176. id: match[3],
  177. l: likes,
  178. f: funny,
  179. h: helpful,
  180. date: createdAt,
  181. title: titles[j] || '',
  182. isReply: elementType === 'reply'
  183. });
  184. } catch (e) {
  185. if (e.response?.status === 429) {
  186. await new Promise(resolve => setTimeout(resolve, 5000));
  187. j--;
  188. continue;
  189. }
  190. console.error(e);
  191. }
  192. }
  193. if (i < p) await new Promise(resolve => setTimeout(resolve, 5000));
  194. }
  195. progressUI.dialog.remove();
  196. const resultWindow = window.open("", "", "width=800,height=600");
  197. resultWindow.document.write(`<style>.hidden{display:none;}.comment-line{line-height:1;}</style><script>function toggleReactionsOnly(){const c=document.getElementById('reactionsOnly');Array.from(document.getElementsByClassName('comment-line')).forEach(e=>{e.classList.toggle('hidden',c.checked&&e.getAttribute('data-reactions')==0);})}</script><pre><div style="position:sticky;top:0;background:white;padding:5px;border-bottom:1px solid #ccc">Auswertung der Reactions für ${n}<label><input type="checkbox" id="reactionsOnly" onclick="toggleReactionsOnly()"> Nur Kommentare mit Reactions anzeigen</label></div>${results.map(x=>`<div class="comment-line" data-reactions="${x.l+x.f+x.h}"><a href="${x.url}" target="_blank">${x.url}</a>${x.isReply?' ':''} L:${x.l} F:${x.f} H:${x.h}${x.h>2?"*":""} ${x.date} ${x.title}</div>`).join('')}\n\nZusammenfassung:\n${results[results.length-1].date} - ${results[0].date}\n${results.length} Kommentare davon ${results.filter(x=>x.l+x.f+x.h>0).length} mit mindestens einer Reaction\nL:${results.reduce((a,c)=>a+c.l,0)} F:${results.reduce((a,c)=>a+c.f,0)} H:${results.reduce((a,c)=>a+c.h,0)} davon ${results.filter(x=>x.h>2).length} mit * (mindestens 3 Hilfreich-Bewertungen)</pre>`);
  198. }
  199.  

QingJ © 2025

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