Any Hackernews Link

Check if current page has been posted to Hacker News

  1. // ==UserScript==
  2. // @name Any Hackernews Link
  3. // @namespace http://tampermonkey.net/
  4. // @version 0.1.8
  5. // @description Check if current page has been posted to Hacker News
  6. // @author RoCry
  7. // @icon https://news.ycombinator.com/favicon.ico
  8. // @match https://*/*
  9. // @exclude https://news.ycombinator.com/*
  10. // @exclude https://hn.algolia.com/*
  11. // @exclude https://*.google.com/*
  12. // @exclude https://mail.yahoo.com/*
  13. // @exclude https://outlook.com/*
  14. // @exclude https://proton.me/*
  15. // @exclude https://localhost/*
  16. // @exclude https://127.0.0.1/*
  17. // @exclude https://192.168.*.*/*
  18. // @exclude https://10.*.*.*/*
  19. // @exclude https://172.16.*.*/*
  20. // @exclude https://web.whatsapp.com/*
  21. // @exclude https://*.facebook.com/messages/*
  22. // @exclude https://*.twitter.com/messages/*
  23. // @exclude https://*.linkedin.com/messaging/*
  24. // @grant GM_xmlhttpRequest
  25. // @connect hn.algolia.com
  26. // @grant GM_addStyle
  27. // @grant GM_getValue
  28. // @grant GM_setValue
  29. // @require https://update.gf.qytechs.cn/scripts/524693/1525919/Any%20Hackernews%20Link%20Utils.js
  30. // @license MIT
  31. // ==/UserScript==
  32.  
  33. (function() {
  34. 'use strict';
  35.  
  36. // Fallback implementation for Safari
  37. if (typeof GM_addStyle === 'undefined') {
  38. window.GM_addStyle = function(css) {
  39. const style = document.createElement('style');
  40. style.textContent = css;
  41. document.head.appendChild(style);
  42. return style;
  43. };
  44. }
  45.  
  46. // Fallback implementations for GM storage functions
  47. if (typeof GM_getValue === 'undefined') {
  48. window.GM_getValue = function(key, defaultValue) {
  49. const value = localStorage.getItem('GM_' + key);
  50. return value === null ? defaultValue : JSON.parse(value);
  51. };
  52. }
  53.  
  54. if (typeof GM_setValue === 'undefined') {
  55. window.GM_setValue = function(key, value) {
  56. localStorage.setItem('GM_' + key, JSON.stringify(value));
  57. };
  58. }
  59.  
  60. /**
  61. * Constants
  62. */
  63. const POSITIONS = {
  64. BOTTOM_LEFT: { bottom: '20px', left: '20px', top: 'auto', right: 'auto' },
  65. BOTTOM_RIGHT: { bottom: '20px', right: '20px', top: 'auto', left: 'auto' },
  66. TOP_LEFT: { top: '20px', left: '20px', bottom: 'auto', right: 'auto' },
  67. TOP_RIGHT: { top: '20px', right: '20px', bottom: 'auto', left: 'auto' }
  68. };
  69.  
  70. /**
  71. * Styles
  72. */
  73. const STYLES = `
  74. @keyframes fadeIn {
  75. 0% { opacity: 0; transform: translateY(10px); }
  76. 100% { opacity: 1; transform: translateY(0); }
  77. }
  78. @keyframes pulse {
  79. 0% { opacity: 1; }
  80. 50% { opacity: 0.6; }
  81. 100% { opacity: 1; }
  82. }
  83. #hn-float {
  84. position: fixed;
  85. bottom: 20px;
  86. left: 20px;
  87. z-index: 9999;
  88. font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif;
  89. display: flex;
  90. align-items: center;
  91. gap: 12px;
  92. background: rgba(255, 255, 255, 0.98);
  93. padding: 8px 12px;
  94. border-radius: 12px;
  95. box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1), 0 0 0 1px rgba(0, 0, 0, 0.05);
  96. cursor: move;
  97. user-select: none;
  98. transition: all 0.2s ease;
  99. max-width: 50px;
  100. overflow: hidden;
  101. opacity: 0.95;
  102. height: 40px;
  103. backdrop-filter: blur(8px);
  104. -webkit-backdrop-filter: blur(8px);
  105. animation: fadeIn 0.3s ease forwards;
  106. will-change: transform, max-width, box-shadow;
  107. color: #111827;
  108. display: flex;
  109. align-items: center;
  110. height: 40px;
  111. box-sizing: border-box;
  112. }
  113. #hn-float:hover {
  114. max-width: 600px;
  115. opacity: 1;
  116. transform: translateY(-2px);
  117. box-shadow: 0 4px 16px rgba(0, 0, 0, 0.12), 0 0 0 1px rgba(0, 0, 0, 0.05);
  118. }
  119. #hn-float .hn-icon {
  120. min-width: 24px;
  121. width: 24px;
  122. height: 24px;
  123. background: linear-gradient(135deg, #ff6600, #ff7f33);
  124. color: white;
  125. display: flex;
  126. align-items: center;
  127. justify-content: center;
  128. font-weight: bold;
  129. border-radius: 6px;
  130. flex-shrink: 0;
  131. position: relative;
  132. font-size: 13px;
  133. text-shadow: 0 1px 1px rgba(0, 0, 0, 0.1);
  134. transition: transform 0.2s ease;
  135. line-height: 1;
  136. padding-bottom: 1px;
  137. }
  138. #hn-float:hover .hn-icon {
  139. transform: scale(1.05);
  140. }
  141. #hn-float .hn-icon.not-found {
  142. background: #9ca3af;
  143. }
  144. #hn-float .hn-icon.found {
  145. background: linear-gradient(135deg, #ff6600, #ff7f33);
  146. }
  147. #hn-float .hn-icon.loading {
  148. background: #6b7280;
  149. animation: pulse 1.5s infinite;
  150. }
  151. #hn-float .hn-icon .badge {
  152. position: absolute;
  153. top: -4px;
  154. right: -4px;
  155. background: linear-gradient(135deg, #3b82f6, #2563eb);
  156. color: white;
  157. border-radius: 8px;
  158. min-width: 14px;
  159. height: 14px;
  160. font-size: 10px;
  161. display: flex;
  162. align-items: center;
  163. justify-content: center;
  164. padding: 0 3px;
  165. font-weight: 600;
  166. box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
  167. border: 1.5px solid white;
  168. }
  169. #hn-float .hn-info {
  170. white-space: nowrap;
  171. overflow: hidden;
  172. text-overflow: ellipsis;
  173. line-height: 1.4;
  174. font-size: 13px;
  175. opacity: 0;
  176. transition: opacity 0.2s ease;
  177. width: 0;
  178. flex: 0;
  179. }
  180. #hn-float:hover .hn-info {
  181. opacity: 1;
  182. width: auto;
  183. flex: 1;
  184. }
  185. #hn-float .hn-info a {
  186. color: inherit;
  187. font-weight: 500;
  188. text-decoration: none;
  189. }
  190. #hn-float .hn-info a:hover {
  191. text-decoration: underline;
  192. }
  193. #hn-float .hn-stats {
  194. color: #6b7280;
  195. font-size: 12px;
  196. margin-top: 2px;
  197. }
  198. @media (prefers-color-scheme: dark) {
  199. #hn-float {
  200. background: rgba(17, 24, 39, 0.95);
  201. color: #e5e7eb;
  202. box-shadow: 0 2px 12px rgba(0, 0, 0, 0.2), 0 0 0 1px rgba(255, 255, 255, 0.1);
  203. }
  204. #hn-float:hover {
  205. box-shadow: 0 4px 16px rgba(0, 0, 0, 0.3), 0 0 0 1px rgba(255, 255, 255, 0.1);
  206. }
  207. #hn-float .hn-stats {
  208. color: #9ca3af;
  209. }
  210. #hn-float .hn-icon .badge {
  211. border-color: rgba(17, 24, 39, 0.95);
  212. }
  213. }
  214. `;
  215.  
  216. /**
  217. * UI Component
  218. */
  219. const UI = {
  220. /**
  221. * Create and append the floating element to the page
  222. * @returns {HTMLElement} - The created element
  223. */
  224. createFloatingElement() {
  225. const div = document.createElement('div');
  226. div.id = 'hn-float';
  227. // Create icon element
  228. const iconDiv = document.createElement('div');
  229. iconDiv.className = 'hn-icon loading';
  230. iconDiv.textContent = 'Y';
  231. // Create info element
  232. const infoDiv = document.createElement('div');
  233. infoDiv.className = 'hn-info';
  234. infoDiv.textContent = 'Checking HN...';
  235. // Append children
  236. div.appendChild(iconDiv);
  237. div.appendChild(infoDiv);
  238. document.body.appendChild(div);
  239.  
  240. // Apply saved position
  241. const savedPosition = GM_getValue('hnPosition', 'BOTTOM_LEFT');
  242. this.applyPosition(div, POSITIONS[savedPosition]);
  243.  
  244. // Add drag functionality
  245. this.addDragHandlers(div);
  246.  
  247. return div;
  248. },
  249.  
  250. /**
  251. * Update the floating element with HN data
  252. * @param {Object|null} data - HN post data or null if not found
  253. */
  254. applyPosition(element, position) {
  255. Object.assign(element.style, position);
  256. },
  257.  
  258. getClosestPosition(x, y) {
  259. const viewportWidth = window.innerWidth;
  260. const viewportHeight = window.innerHeight;
  261. const isTop = y < viewportHeight / 2;
  262. const isLeft = x < viewportWidth / 2;
  263.  
  264. if (isTop) {
  265. return isLeft ? 'TOP_LEFT' : 'TOP_RIGHT';
  266. } else {
  267. return isLeft ? 'BOTTOM_LEFT' : 'BOTTOM_RIGHT';
  268. }
  269. },
  270.  
  271. addDragHandlers(element) {
  272. let isDragging = false;
  273. let currentX;
  274. let currentY;
  275. let initialX;
  276. let initialY;
  277.  
  278. element.addEventListener('mousedown', e => {
  279. if (e.target.tagName === 'A') return; // Don't drag when clicking links
  280. isDragging = true;
  281. element.style.transition = 'none';
  282. initialX = e.clientX - element.offsetLeft;
  283. initialY = e.clientY - element.offsetTop;
  284. });
  285.  
  286. document.addEventListener('mousemove', e => {
  287. if (!isDragging) return;
  288.  
  289. e.preventDefault();
  290. currentX = e.clientX - initialX;
  291. currentY = e.clientY - initialY;
  292.  
  293. // Keep the element within viewport bounds
  294. currentX = Math.max(0, Math.min(currentX, window.innerWidth - element.offsetWidth));
  295. currentY = Math.max(0, Math.min(currentY, window.innerHeight - element.offsetHeight));
  296.  
  297. element.style.left = `${currentX}px`;
  298. element.style.top = `${currentY}px`;
  299. element.style.bottom = 'auto';
  300. element.style.right = 'auto';
  301. });
  302.  
  303. document.addEventListener('mouseup', e => {
  304. if (!isDragging) return;
  305. isDragging = false;
  306. element.style.transition = 'all 0.2s ease';
  307.  
  308. const position = this.getClosestPosition(currentX + element.offsetWidth / 2, currentY + element.offsetHeight / 2);
  309. this.applyPosition(element, POSITIONS[position]);
  310. // Save position
  311. GM_setValue('hnPosition', position);
  312. });
  313. },
  314.  
  315. updateFloatingElement(data) {
  316. const iconDiv = document.querySelector('#hn-float .hn-icon');
  317. const infoDiv = document.querySelector('#hn-float .hn-info');
  318. iconDiv.classList.remove('loading');
  319. if (!data) {
  320. iconDiv.classList.add('not-found');
  321. iconDiv.classList.remove('found');
  322. iconDiv.textContent = 'Y';
  323. infoDiv.textContent = 'Not found on HN';
  324. return;
  325. }
  326. iconDiv.classList.remove('not-found');
  327. iconDiv.classList.add('found');
  328. // Clear existing content
  329. iconDiv.textContent = 'Y';
  330. // Make icon clickable
  331. iconDiv.style.cursor = 'pointer';
  332. iconDiv.onclick = (e) => {
  333. e.stopPropagation();
  334. window.open(data.link, '_blank');
  335. };
  336. // Add badge if there are comments
  337. if (data.comments > 0) {
  338. const badge = document.createElement('span');
  339. badge.className = 'badge';
  340. badge.textContent = data.comments > 999 ? '999+' : data.comments.toString();
  341. iconDiv.appendChild(badge);
  342. }
  343. // Clear and rebuild info content
  344. infoDiv.textContent = '';
  345. const titleDiv = document.createElement('div');
  346. const titleLink = document.createElement('a');
  347. titleLink.href = data.link;
  348. titleLink.target = '_blank';
  349. titleLink.textContent = data.title;
  350. titleDiv.appendChild(titleLink);
  351. const statsDiv = document.createElement('div');
  352. statsDiv.className = 'hn-stats';
  353. statsDiv.textContent = `${data.points} points | ${data.comments} comments | ${data.posted}`;
  354. infoDiv.appendChild(titleDiv);
  355. infoDiv.appendChild(statsDiv);
  356. }
  357. };
  358.  
  359. /**
  360. * Initialize the script
  361. */
  362. function init() {
  363. // Skip if we're in an iframe
  364. if (window.top !== window.self) {
  365. console.log('📌 Skipping execution in iframe');
  366. return;
  367. }
  368.  
  369. // Skip if document is hidden (like background tabs or invisible frames)
  370. if (document.hidden) {
  371. console.log('📌 Skipping execution in hidden document');
  372. // Add listener for when the tab becomes visible
  373. document.addEventListener('visibilitychange', function onVisible() {
  374. if (!document.hidden) {
  375. init();
  376. document.removeEventListener('visibilitychange', onVisible);
  377. }
  378. });
  379. return;
  380. }
  381.  
  382. const currentUrl = window.location.href;
  383. // Check if the floating element already exists
  384. if (document.getElementById('hn-float')) {
  385. console.log('📌 HN float already exists, skipping');
  386. return;
  387. }
  388.  
  389. if (URLUtils.shouldIgnoreUrl(currentUrl)) {
  390. console.log('🚫 Ignored URL:', currentUrl);
  391. return;
  392. }
  393.  
  394. // Check if content is primarily English
  395. if (!ContentUtils.isEnglishContent()) {
  396. console.log('🈂️ Non-English content detected, skipping');
  397. return;
  398. }
  399.  
  400. GM_addStyle(STYLES);
  401. const normalizedUrl = URLUtils.normalizeUrl(currentUrl);
  402. console.log('🔗 Normalized URL:', normalizedUrl);
  403. UI.createFloatingElement();
  404. HNApi.checkHackerNews(normalizedUrl, UI.updateFloatingElement);
  405. }
  406.  
  407. // Start the script
  408. init();
  409. })();

QingJ © 2025

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