URL 嗅探器

从 HTML 中嗅探 URL

  1. // ==UserScript==
  2. // @name URL Sniffer
  3. // @name:zh-CN URL 嗅探器
  4. // @namespace https://gera2ld.space/
  5. // @description Sniff URLs in HTML
  6. // @description:zh-CN 从 HTML 中嗅探 URL
  7. // @match *://*/*
  8. // @version 0.2.0
  9. // @author Gerald <gera2ld@live.com>
  10. // @require https://cdn.jsdelivr.net/combine/npm/@violentmonkey/dom@2,npm/@violentmonkey/ui@0.7
  11. // @supportURL https://github.com/intellilab/url-sniffer.user.js
  12. // @grant GM_addStyle
  13. // @grant GM_registerMenuCommand
  14. // @grant GM_setClipboard
  15. // @grant GM_unregisterMenuCommand
  16. // ==/UserScript==
  17.  
  18. (function () {
  19. 'use strict';
  20.  
  21. function _extends() {
  22. _extends = Object.assign ? Object.assign.bind() : function (target) {
  23. for (var i = 1; i < arguments.length; i++) {
  24. var source = arguments[i];
  25.  
  26. for (var key in source) {
  27. if (Object.prototype.hasOwnProperty.call(source, key)) {
  28. target[key] = source[key];
  29. }
  30. }
  31. }
  32.  
  33. return target;
  34. };
  35. return _extends.apply(this, arguments);
  36. }
  37.  
  38. function _objectWithoutPropertiesLoose(source, excluded) {
  39. if (source == null) return {};
  40. var target = {};
  41. var sourceKeys = Object.keys(source);
  42. var key, i;
  43.  
  44. for (i = 0; i < sourceKeys.length; i++) {
  45. key = sourceKeys[i];
  46. if (excluded.indexOf(key) >= 0) continue;
  47. target[key] = source[key];
  48. }
  49.  
  50. return target;
  51. }
  52.  
  53. var styles = {"root":"style-module_root__1vyw2","toast":"style-module_toast__OcS5G","image":"style-module_image__1P0bD"};
  54. var stylesheet=".style-module_root__1vyw2{background:#0008;inset:0;position:fixed;z-index:10000}.style-module_root__1vyw2:before{background:#0008;color:#bbb;content:\"Double click anywhere on the mask to exit\";font-size:12px;left:50%;padding:8px 16px;position:absolute;top:0;transform:translateX(-50%)}.style-module_root__1vyw2>*{border:2px solid;left:0;position:absolute;top:0}.style-module_toast__OcS5G{z-index:10001!important}.style-module_image__1P0bD{background:#0008;inset:80px;overflow:auto;position:absolute}.style-module_image__1P0bD>img{position:absolute;transform-origin:top left}";
  55.  
  56. const _excluded = ["elements", "getItem"];
  57. const STYLE_CURRENT = {
  58. stroke: '#0f08',
  59. fill: '#0f02'
  60. };
  61. const STYLE_SELECTION = {
  62. stroke: '#bbf8',
  63. fill: '#bbf2'
  64. };
  65. const STYLE_SELECTED = {
  66. stroke: '#ff08',
  67. fill: '#ff02'
  68. };
  69. const STYLE_TO_DESELECT = {
  70. stroke: '#88d8',
  71. fill: '#88d2'
  72. };
  73. const STYLE_TO_SELECT = {
  74. stroke: '#bb08',
  75. fill: '#bb02'
  76. };
  77. const MODE_SINGLE = 0;
  78. const MODE_MULTIPLE = 1;
  79. let rendering = false;
  80. const mask = VM.getHostElement(false);
  81. mask.addStyle(stylesheet);
  82. mask.root.className = styles.root;
  83. mask.root.addEventListener('mousedown', handleMouseDown);
  84. mask.root.addEventListener('mouseup', handleMouseUp);
  85. mask.root.addEventListener('mousemove', handleMouseMove);
  86. mask.root.addEventListener('click', handleClick);
  87. mask.root.addEventListener('dblclick', handleCallback);
  88. GM_registerMenuCommand('Sniff links', sniffLinks);
  89. GM_registerMenuCommand('Sniff images', sniffImages);
  90. let context;
  91.  
  92. function sniffLinks() {
  93. if (context) close();
  94. start({
  95. elements: document.querySelectorAll('a[href]'),
  96.  
  97. getItem(el) {
  98. const href = el.tagName.toLowerCase() === 'a' && el.getAttribute('href');
  99. if (href && !/^(?:#|javascript:)/.test(href)) return {
  100. el
  101. };
  102. },
  103.  
  104. mode: MODE_MULTIPLE,
  105.  
  106. callback(selectedItems) {
  107. copy(selectedItems);
  108. close();
  109. }
  110.  
  111. });
  112. }
  113.  
  114. function sniffImages() {
  115. if (context) close();
  116. const imageViewer = VM.hm("div", {
  117. className: styles.image,
  118. onClick: e => e.stopPropagation()
  119. });
  120.  
  121. const showImage = img => {
  122. mask.root.append(imageViewer);
  123. const {
  124. naturalWidth,
  125. naturalHeight
  126. } = img;
  127. const containerWidth = imageViewer.clientWidth;
  128. const containerHeight = imageViewer.clientHeight;
  129. const scale = Math.min(1, containerWidth / naturalWidth);
  130. const width = naturalWidth * scale;
  131. const height = naturalHeight * scale;
  132. const x = Math.max(0, (containerWidth - width) / 2);
  133. const y = Math.max(0, (containerHeight - height) / 2);
  134. imageViewer.innerHTML = '';
  135. imageViewer.append(img);
  136. img.style.transform = `scale(${scale}) translate(${x}px,${y}px)`;
  137. context.paused = true;
  138. mask.root.addEventListener('click', closeViewer);
  139. };
  140.  
  141. const closeViewer = () => {
  142. imageViewer.innerHTML = '';
  143. imageViewer.remove();
  144. context.paused = false;
  145. mask.root.removeEventListener('click', closeViewer);
  146. };
  147.  
  148. start({
  149. getItem(el) {
  150. let url;
  151.  
  152. if (el.tagName.toLowerCase() === 'img') {
  153. url = el.src;
  154. } else {
  155. const bgImg = el.style.backgroundImage.match(/^url\((['"]?)(.*?)\1\)/);
  156. url = bgImg == null ? void 0 : bgImg[2];
  157. }
  158.  
  159. return url && {
  160. el,
  161. url
  162. };
  163. },
  164.  
  165. mode: MODE_SINGLE,
  166.  
  167. callback([item]) {
  168. if (!item) return close();
  169. const img = new Image();
  170. img.src = item.url;
  171.  
  172. img.onload = () => {
  173. showImage(img);
  174. };
  175. }
  176.  
  177. });
  178. }
  179.  
  180. function start(opts) {
  181. if (context) throw new Error('Context already exists');
  182.  
  183. const {
  184. elements,
  185. getItem
  186. } = opts,
  187. rest = _objectWithoutPropertiesLoose(opts, _excluded);
  188.  
  189. const items = Array.from(elements || document.querySelectorAll('*')).map(getItem).filter(Boolean);
  190. context = _extends({}, rest, {
  191. items,
  192. index: -1,
  193. active: null,
  194. disconnect: VM.observe(document.body, mutations => {
  195. mutations.forEach(mut => {
  196. if (mut.type === 'childList') {
  197. const newItems = Array.from(mut.addedNodes).filter(el => !mask.root.contains(el)).map(getItem).filter(Boolean);
  198. context.items.push(...newItems);
  199. }
  200. });
  201. })
  202. });
  203. update();
  204. mask.show();
  205. document.addEventListener('scroll', update);
  206. document.addEventListener('resize', update);
  207. }
  208.  
  209. function close() {
  210. if (!context) return;
  211. context.disconnect == null ? void 0 : context.disconnect();
  212. mask.root.innerHTML = '';
  213. mask.hide();
  214. context = null;
  215. document.removeEventListener('scroll', update);
  216. document.removeEventListener('resize', update);
  217. GM_unregisterMenuCommand('Copy URLs');
  218. }
  219.  
  220. function update() {
  221. if (rendering) return;
  222. rendering = true;
  223. requestAnimationFrame(() => {
  224. context.items.forEach(item => {
  225. const rect = item.el.getBoundingClientRect();
  226. item.pos = {
  227. x: rect.left,
  228. y: rect.top,
  229. w: rect.width,
  230. h: rect.height
  231. };
  232. });
  233. render();
  234. rendering = false;
  235. });
  236. }
  237.  
  238. function render() {
  239. renderActive();
  240. renderSelected();
  241. }
  242.  
  243. function updateStyle(el, style) {
  244. el.style.borderColor = style.stroke;
  245. el.style.background = style.fill;
  246. }
  247.  
  248. function updatePosition(el, pos, padding = 2) {
  249. Object.assign(el.style, {
  250. width: `${pos.w + padding * 2}px`,
  251. height: `${pos.h + padding * 2}px`,
  252. transform: `translate(${pos.x - padding}px,${pos.y - padding}px)`
  253. });
  254. }
  255.  
  256. function renderActive() {
  257. const activeItem = !context.dragging && context.items[context.index];
  258.  
  259. if (!activeItem) {
  260. if (context.active) {
  261. context.active.remove();
  262. context.active = null;
  263. }
  264. } else {
  265. if (!context.active) {
  266. context.active = VM.hm(mask.id, null);
  267. updateStyle(context.active, STYLE_CURRENT);
  268. mask.root.append(context.active);
  269. }
  270.  
  271. updatePosition(context.active, activeItem.pos);
  272. }
  273. }
  274.  
  275. function renderSelected() {
  276. context.items.forEach(item => {
  277. if (item.rect) updatePosition(item.rect, item.pos);
  278. });
  279. }
  280.  
  281. function setItemRect(item, style) {
  282. if (style) {
  283. if (!item.rect) {
  284. item.rect = VM.hm(mask.id, null);
  285. mask.root.append(item.rect);
  286. }
  287.  
  288. updateStyle(item.rect, style);
  289. updatePosition(item.rect, item.pos);
  290. } else if (item.rect) {
  291. item.rect.remove();
  292. item.rect = null;
  293. }
  294. }
  295.  
  296. function handleClick() {
  297. if (context.paused) return;
  298. const activeItem = context.items[context.index];
  299.  
  300. if (activeItem) {
  301. if (context.mode === MODE_SINGLE) {
  302. context.callback([activeItem]);
  303. } else {
  304. activeItem.selected = !activeItem.selected;
  305. setItemRect(activeItem, activeItem.selected && STYLE_SELECTED);
  306. }
  307. }
  308. }
  309.  
  310. function handleMouseDown(e) {
  311. if (context.dragging || context.mode === MODE_SINGLE || context.paused) return;
  312. const x = e.clientX;
  313. const y = e.clientY;
  314. context.dragging = {
  315. x,
  316. y
  317. };
  318. }
  319.  
  320. function handleMouseMove(e) {
  321. if (context.paused) return;
  322. const x = e.clientX;
  323. const y = e.clientY;
  324.  
  325. if (context.dragging) {
  326. if (!context.dragging.rect) {
  327. const rect = VM.hm(mask.id, null);
  328. updateStyle(rect, STYLE_SELECTION);
  329. mask.root.append(rect);
  330. context.dragging.rect = rect;
  331. }
  332.  
  333. context.index = -1;
  334. let x0 = context.dragging.x;
  335. let y0 = context.dragging.y;
  336. const w = Math.abs(x - x0);
  337. const h = Math.abs(y - y0);
  338. x0 = Math.min(x0, x);
  339. y0 = Math.min(y0, y);
  340. updatePosition(context.dragging.rect, {
  341. x: x0,
  342. y: y0,
  343. w,
  344. h
  345. }, 0);
  346. context.items.forEach(item => {
  347. item.inSelection = item.pos.x >= x0 && item.pos.x + item.pos.w <= x0 + w && item.pos.y >= y0 && item.pos.y + item.pos.h <= y0 + h;
  348. const state = (item.inSelection ? 2 : 0) + (item.selected ? 1 : 0);
  349. setItemRect(item, {
  350. 1: STYLE_SELECTED,
  351. 2: STYLE_TO_SELECT,
  352. 3: STYLE_TO_DESELECT
  353. }[state]);
  354. });
  355. } else {
  356. context.index = context.items.findIndex(({
  357. pos
  358. }) => x >= pos.x && x <= pos.x + pos.w && y >= pos.y && y <= pos.y + pos.h);
  359. }
  360.  
  361. render();
  362. }
  363.  
  364. function handleMouseUp() {
  365. if (!context.dragging) return;
  366.  
  367. if (context.dragging.rect) {
  368. context.dragging.rect.remove();
  369. context.items.forEach(item => {
  370. if (item.inSelection) {
  371. item.inSelection = false;
  372. item.selected = !item.selected;
  373. setItemRect(item, item.selected && STYLE_SELECTED);
  374. }
  375. });
  376. }
  377.  
  378. context.dragging = null;
  379. }
  380.  
  381. function handleCallback() {
  382. const selectedItems = context.items.filter(item => item.selected);
  383. context.callback(selectedItems);
  384. }
  385.  
  386. function copy(selectedItems) {
  387. const urls = selectedItems.map(item => item.el.href);
  388. if (!urls.length) return;
  389. GM_setClipboard(urls.join('\r\n'));
  390. VM.showToast('URLs copied', {
  391. shadow: false,
  392. className: styles.toast
  393. });
  394. }
  395.  
  396. })();

QingJ © 2025

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