- // ==UserScript==
- // @name Open Multiple URLs
- // @name:zh-CN 批量打开链结
- // @name:zh-TW 批量打開鏈結
- // @version 2024-09-20
- // @description Open Multiple Text URLs,Open Multiple Element URLs.
- // @description:zh-CN 批量打开文本链结、批量选取链结后打开。
- // @description:zh-TW 批量打開文字鏈結、批量選取鏈結後打開。
- // @author tony0809
- // @match *://*/*
- // @icon 
- // @grant GM_openInTab
- // @grant GM_registerMenuCommand
- // @grant unsafeWindow
- // @license MIT
- // @namespace https://gf.qytechs.cn/users/20361
- // ==/UserScript==
-
- /***
-
- 一個用處不大的腳本,主要是我的另一個腳本圖片全載,需要手動確認網站存活時會用到。
-
- 注意一次性打開太多鏈結會造成瀏覽器嚴重卡頓,慎用!!!,在有需要的時候再開啟這個腳本。
-
- 說明
-
- 模式一:批量打開文字鏈結
-
- 創建一個文字輸入區,一行一個網址或域名,批量打開。
-
- 模式二:選取鏈結後打開
-
- 步驟1.透過腳本管理器選單或按快捷鍵Ctrl + Alt + U或雙擊頁面的空白處來注入事件。
- 步驟2.鼠標懸停要打開的鏈結進行標記和預讀,標記會對鏈結添加橙色的邊框,再次懸停會取消標記。
-
- Q:如何取消?
- A:按Esc鍵或點擊頁面的空白處。
-
- 步驟3.頁面的空白處按滑鼠右鍵或快捷鍵Ctrl + Alt + O,打開被標記的所有鏈結。
-
- 再次使用重複步驟1
-
- ***/
-
- (() => {
- 'use strict';
-
- const _unsafeWindow = unsafeWindow ?? window;
- const hasTouchEvents = (() => ("ontouchstart" in _unsafeWindow) || (_unsafeWindow.navigator.maxTouchPoints > 0) || (_unsafeWindow.navigator.msMaxTouchPoints > 0))();
-
- const language = _unsafeWindow.navigator.language;
-
- let scriptLanguage;
- switch (language) {
- case "zh-TW":
- case "zh-HK":
- case "zh-Hant-TW":
- case "zh-Hant-HK":
- scriptLanguage = "TW";
- break;
- case "zh":
- case "zh-CN":
- case "zh-Hans-CN":
- scriptLanguage = "CH";
- break;
- default:
- scriptLanguage = "EN";
- }
-
- let i18n;
- switch (scriptLanguage) {
- case "TW":
- i18n = {
- omu: "批量打開鏈結",
- ou: "批量打開",
- close: "關閉",
- otu: "批量打開文字鏈結",
- oeu: "選取鏈結後打開(Ctrl + Alt + U)"
- };
- break;
- case "CN":
- i18n = {
- omu: "批量打开链结",
- ou: "批量打开",
- close: "关闭",
- otu: "批量打开文本链结",
- oeu: "选取链结后打开(Ctrl + Alt + U)"
- };
- break;
- default:
- i18n = {
- omu: "Open Multiple URLs",
- ou: "Open URLs",
- close: "Close",
- otu: "Open Multiple Text URLs",
- oeu: "Open Multiple Element URLs(Ctrl + Alt + U)"
- };
- }
-
- const createFixedElement = () => {
-
- const mainHtml = '<div id="Batch_open_links" style="display: initial !important;position: fixed !important;z-index: 9999999 !important;"></div>';
- document.body.insertAdjacentHTML("beforeend", mainHtml);
-
- const mainElement = document.querySelector("#Batch_open_links");
- const shadow = mainElement.attachShadow({
- mode: "open"
- });
-
- const div = document.createElement("div");
-
- Object.assign(div.style, {
- left: "0",
- right: "0",
- top: "0",
- bottom: "0",
- width: "100vw",
- height: "100vh",
- margin: "auto",
- padding: "25px 10px 10px 10px",
- position: "fixed",
- opacity: "1",
- zIndex: "9999999",
- backgroundColor: "#eee",
- color: "#222",
- fontSize: "14px",
- overflow: "scroll",
- textAlign: "left"
- });
-
- const html = `
- <h3 style="font-size: 22px;overflow: hidden;text-overflow: ellipsis;white-space: nowrap;">${i18n.omu}</h3>
- <textarea id="links_textarea" style="display: block; margin: 10px 0 10px 0; white-space:pre; overflow:scroll; resize: revert; text-transform: initial;"></textarea>
- <button id="open" style="margin-right: 20px;">${i18n.ou}</button>
- <button id="close">${i18n.close}</button>
- `;
- div.innerHTML = html;
- shadow.appendChild(div);
-
- const reWidth = () => {
- shadow.querySelector("#links_textarea").style.width = hasTouchEvents ? (_unsafeWindow.innerWidth - 26) + "px": (_unsafeWindow.innerWidth - 40) + "px";
- shadow.querySelector("#links_textarea").style.height = hasTouchEvents ? (_unsafeWindow.innerHeight - 140) + "px": (_unsafeWindow.innerHeight - 200) + "px";
- };
- reWidth();
-
- _unsafeWindow.addEventListener("resize", reWidth);
-
- shadow.querySelector("#close").addEventListener("click", () => {
- mainElement.remove();
- _unsafeWindow.removeEventListener("resize", reWidth);
- });
-
- shadow.querySelector("#open").addEventListener("click", () => {
- const value = shadow.querySelector("#links_textarea").value;
-
- let links = value.split("\n").filter(e => e).map(url => {
- if (/^https?:\/\//.test(url)) {
- return url;
- } else {
- return "https://" + url;
- }
- });
- links = [...new Set(links)];
-
- for (const link of links) {
- let ok = true;
- try {
- new URL(link);
- } catch (error) {
- ok = false;
- console.error(link, error);
- }
- if (ok) GM_openInTab(link);
- }
-
- shadow.querySelector("#links_textarea").value = "";
- });
- };
-
- GM_registerMenuCommand(i18n.otu, () => createFixedElement());
-
- if (hasTouchEvents) return;
-
- const preloadLink = (url) => {
- if ([...document.getElementsByTagName("link")].some(link => link.href == url)) return;
- const preloadElement = document.createElement("link");
- preloadElement.rel = "prefetch";
- preloadElement.as = "document";
- preloadElement.href = url;
- document.head.appendChild(preloadElement);
- };
-
- const openElementLinks = () => {
-
- const contextmenuEvent = (event) => event.preventDefault();
- document.addEventListener("contextmenu", contextmenuEvent);
-
- const clickEvent = () => open_cb(0);
- document.addEventListener("click", clickEvent);
-
- const kEvent = (event) => {
- if (event.code === "Escape" || event.key === "Escape") {
- return open_cb(0);
- }
- if (event.ctrlKey && event.altKey && event.code === "KeyO") {
- open_cb();
- }
- };
- document.addEventListener("keydown", kEvent);
-
- const aElements = [...document.getElementsByTagName("a")];
-
- const aEvent = (event) => {
- if (event.target.getAttribute("select") == "true") {
- event.target.removeAttribute("select");
- event.target.removeAttribute("style");
- } else {
- event.target.setAttribute("select", "true");
- Object.assign(event.target.style, {
- paddingLeft: "4px",
- paddingRight: "4px",
- borderWidth: "2px",
- borderStyle: "solid",
- borderColor: "#ff9933"
- });
- preloadLink(event.target.href);
- }
- };
-
- const open_cb = (open = 1) => {
- const links = [...document.querySelectorAll("a[select=true]")];
- if (links.length > 0) {
- const urls = [];
- links.forEach(a => {
- try {
- if (open === 1 && !urls.includes(a.href)) {
- new URL(a.href);
- urls.push(a.href);
- GM_openInTab(a.href);
- }
- } catch (error) {
- console.error(a.href, error);
- }
- a.removeAttribute("select");
- a.removeAttribute("style");
- });
- }
- aElements.forEach(a => a.removeEventListener("mouseenter", aEvent));
- document.removeEventListener("keydown", kEvent);
- document.removeEventListener("click", clickEvent);
- setTimeout(() => {
- document.removeEventListener("contextmenu", contextmenuEvent);
- }, 1000);
- };
-
- aElements.forEach(a => a.addEventListener("mouseenter", aEvent));
-
- document.addEventListener("mousedown", (event) => {
- if (event.button == 2) {
- open_cb();
- }
- });
- };
-
- const kEvent = (event) => {
- if (event.ctrlKey && event.altKey && event.code === "KeyU") {
- openElementLinks();
- }
- };
-
- document.addEventListener("keydown", kEvent);
- document.addEventListener("dblclick", () => openElementLinks());
-
- GM_registerMenuCommand(i18n.oeu, () => openElementLinks());
-
-
- })();