- // ==UserScript==
- // @name Web选择
- // @name:en Web Select
- // @name:zh-TW Web捕獲
- // @namespace http://howardzhangdqs.eu.org/
- // @source https://github.com/Howardzhangdqs/web-select
- // @version 0.2.4
- // @description 由于Web选择太好用,微软就把他砍掉了。本脚本实现了Web选择的部分功能。按下Alt+S即可选择文本。
- // @description:en Due to the ease with which the "web select" was used, Microsoft cut it off. This script implements some functions of "web select". Press Alt+S to select text.
- // @description:zh-TW 由於Web選擇功能非常實用,微軟就將它砍掉了。本腳本實現了Web選擇的部分功能。按下Alt+S即可選擇文本。
- // @author HowardZhangdqs
- // @match *://*/*
- // @license MIT
- // @icon 
- // @grant none
- // ==/UserScript==
-
- (() => {
- var UnselectedBoxStyle = {
- border: "1px solid #fff",
- outline: "1px dashed #000",
- outlineOffset: "-1px",
- zIndex: "99999",
- };
- var SelectedBoxStyle = {
- border: "1px solid #fff",
- outline: "1px solid #000",
- outlineOffset: "0",
- zIndex: "99999",
- };
- var SelectBoxStyle = {
- position: "fixed",
- outline: "3px dashed white",
- zIndex: "99999",
- };
- var addStylesheetRules = function (dom, decls) {
- if (dom instanceof HTMLElement) dom = [dom];
- for (var i in decls) {
- for (var _i = 0, dom_1 = dom; _i < dom_1.length; _i++) {
- var j = dom_1[_i];
- j.style[i] = decls[i];
- }
- }
- };
-
- var makeButton = function (top, left) {
- var button = document.createElement("button");
- button.innerText = "Web选择";
- button.style.position = "fixed";
- if (top) button.style.top = "".concat(top, "px");
- else button.style.bottom = "10px";
- if (left) button.style.left = "".concat(left, "px");
- else button.style.right = "10px";
- button.style.zIndex = "99999";
- button.style.padding = "10px";
- button.style.borderRadius = "10px";
- button.style.background = "#fff";
- button.style.color = "#000";
- button.style.border = "1px solid #000";
- button.style.cursor = "pointer";
-
- document.body.appendChild(button);
- return button;
- };
-
- var getElementsPosition = function () {
- var elements = [];
- var allElements = document.querySelectorAll("*");
- allElements.forEach(function (element) {
- var _a = element.getBoundingClientRect(),
- top = _a.top,
- left = _a.left,
- width = _a.width,
- height = _a.height;
- if (top && left && width && height && element.innerText)
- elements.push({
- top: top,
- left: left,
- width: width,
- height: height,
- text: element.innerText,
- src: element,
- });
- });
- return elements;
- };
-
- var makeDiv = function (_a) {
- var top = _a.top,
- left = _a.left,
- width = _a.width,
- height = _a.height,
- text = _a.text,
- src = _a.src;
- var div = document.createElement("div");
- div.style.position = "fixed";
- div.style.top = "".concat(top, "px");
- div.style.left = "".concat(left, "px");
- div.style.width = "".concat(width, "px");
- div.style.height = "".concat(height, "px");
- addStylesheetRules(div, UnselectedBoxStyle);
- div.style.zIndex = "99999";
- document.body.appendChild(div);
- return {
- top: top,
- left: left,
- width: width,
- height: height,
- text: text,
- el: div,
- src: src,
- };
- };
-
- var dragSelect = function (allBoxes) {
- return new Promise(function (resolve, reject) {
- var div = document.createElement("div");
- addStylesheetRules(div, SelectBoxStyle);
- document.body.appendChild(div);
- var mask = [];
- var selectedBox = {
- _p1: [0, 0],
- _p2: [0, 0],
- set p1(val) {
- this._p1[0] = Math.floor(val[0]);
- this._p1[1] = Math.floor(val[1]);
- this.runwatch();
- },
- set p2(val) {
- this._p2[0] = Math.floor(val[0]);
- this._p2[1] = Math.floor(val[1]);
- this.runwatch();
- },
- get p1() {
- return this._p1;
- },
- get p2() {
- return this._p2;
- },
- get top() {
- return Math.min(this._p1[0], this._p2[0]);
- },
- get left() {
- return Math.min(this._p1[1], this._p2[1]);
- },
- get width() {
- return Math.abs(this._p1[1] - this._p2[1]);
- },
- get height() {
- return Math.abs(this._p1[0] - this._p2[0]);
- },
- watchfn: [],
- watch: function (callback) {
- this.watchfn.push(callback);
- },
- unwatch: function (callback) {
- this.watchfn = this.watchfn.filter(function (fn) {
- return fn !== callback;
- });
- },
- runwatch: function () {
- for (var _i = 0, _a = this.watchfn; _i < _a.length; _i++) {
- var fn = _a[_i];
- fn(this);
- }
- },
- };
-
- var makeMask = function () {
- var mask1 = document.createElement("div");
- var mask2 = document.createElement("div");
- var mask3 = document.createElement("div");
- var mask4 = document.createElement("div");
- addStylesheetRules([mask1, mask2, mask3, mask4], {
- position: "fixed",
- top: "0",
- left: "0",
- background: "#0007",
- zIndex: "99997",
- });
- selectedBox.watch(function () {
- addStylesheetRules(mask1, {
- top: "0",
- left: "0",
- width: "".concat(selectedBox.left, "px"),
- height: "100%",
- });
- addStylesheetRules(mask2, {
- top: "0",
- left: "".concat(selectedBox.left, "px"),
- width: "".concat(selectedBox.width, "px"),
- height: "".concat(selectedBox.top, "px"),
- });
- addStylesheetRules(mask3, {
- top: "".concat(
- selectedBox.top + selectedBox.height,
- "px",
- ),
- left: "".concat(selectedBox.left, "px"),
- width: "".concat(selectedBox.width, "px"),
- height: "calc(100% - ".concat(
- selectedBox.top + selectedBox.height,
- "px)",
- ),
- });
- addStylesheetRules(mask4, {
- top: "0px",
- left: "".concat(
- selectedBox.left + selectedBox.width,
- "px",
- ),
- width: "calc(100% - ".concat(
- selectedBox.left + selectedBox.width,
- "px)",
- ),
- height: "100vh",
- });
- });
- document.body.appendChild(mask1);
- document.body.appendChild(mask2);
- document.body.appendChild(mask3);
- document.body.appendChild(mask4);
- return [mask1, mask2, mask3, mask4];
- };
- var mouseDown = function (e) {
- e.preventDefault();
-
- mask.push.apply(mask, makeMask());
- div.style.display = "block";
- selectedBox.p1 = [e.clientY, e.clientX];
- selectedBox.p2 = [e.clientY, e.clientX];
- div.style.top = "".concat(selectedBox.top, "px");
- div.style.left = "".concat(selectedBox.left, "px");
- div.style.width = "0";
- div.style.height = "0";
- document.addEventListener("mousemove", mouseMove);
- document.addEventListener("mouseup", mouseUp);
- };
- var getSelectedElements = function () {
- return allBoxes.filter(function (el) {
- if (
- el.left < selectedBox.left ||
- el.top < selectedBox.top ||
- el.left + el.width >
- selectedBox.left + selectedBox.width ||
- el.top + el.height >
- selectedBox.top + selectedBox.height
- ) {
- return false;
- }
- return true;
- });
- };
- var mouseMove = function (e) {
- e.preventDefault();
- selectedBox.p2 = [e.clientY, e.clientX];
- div.style.top = "".concat(selectedBox.top, "px");
- div.style.left = "".concat(selectedBox.left, "px");
- div.style.width = "".concat(selectedBox.width, "px");
- div.style.height = "".concat(selectedBox.height, "px");
- var selectedElements = getSelectedElements();
- for (
- var _i = 0, allBoxes_1 = allBoxes;
- _i < allBoxes_1.length;
- _i++
- ) {
- var el = allBoxes_1[_i];
- if (selectedElements.includes(el)) {
- addStylesheetRules(el.el, SelectedBoxStyle);
- } else {
- addStylesheetRules(el.el, UnselectedBoxStyle);
- }
- }
- console.log(
- selectedElements
- .sort(function (a, b) {
- return b.left - a.left;
- })
- .sort(function (a, b) {
- return b.top - a.top;
- }),
- );
- };
- var mouseUp = function (e) {
- document.removeEventListener("mousemove", mouseMove);
- document.removeEventListener("mouseup", mouseUp);
- document.removeEventListener("mousedown", mouseDown);
- div.remove();
- for (var _i = 0, mask_1 = mask; _i < mask_1.length; _i++) {
- var i = mask_1[_i];
- i.remove();
- }
- addStylesheetRules(document.body, { pointerEvents: "" });
- resolve({
- top: parseInt(div.style.top),
- left: parseInt(div.style.left),
- width: parseInt(div.style.width),
- height: parseInt(div.style.height),
- el: getSelectedElements(),
- });
- };
- document.addEventListener("mousedown", mouseDown);
- document.addEventListener("mouseup", mouseUp);
- addStylesheetRules(document.body, { pointerEvents: "none" });
- });
- };
-
- var isChild = function (parent, child) {
- var node = child.parentNode;
- while (node !== null) {
- if (node === parent) return true;
- node = node.parentNode;
- }
- return false;
- };
-
- var __awaiter =
- (undefined && undefined.__awaiter) ||
- function (thisArg, _arguments, P, generator) {
- function adopt(value) {
- return value instanceof P
- ? value
- : new P(function (resolve) {
- resolve(value);
- });
- }
- return new (P || (P = Promise))(function (resolve, reject) {
- function fulfilled(value) {
- try {
- step(generator.next(value));
- } catch (e) {
- reject(e);
- }
- }
- function rejected(value) {
- try {
- step(generator["throw"](value));
- } catch (e) {
- reject(e);
- }
- }
- function step(result) {
- result.done
- ? resolve(result.value)
- : adopt(result.value).then(fulfilled, rejected);
- }
- step(
- (generator = generator.apply(
- thisArg,
- _arguments || [],
- )).next(),
- );
- });
- };
- var __generator =
- (undefined && undefined.__generator) ||
- function (thisArg, body) {
- var _ = {
- label: 0,
- sent: function () {
- if (t[0] & 1) throw t[1];
- return t[1];
- },
- trys: [],
- ops: [],
- },
- f,
- y,
- t,
- g;
- return (
- (g = { next: verb(0), throw: verb(1), return: verb(2) }),
- typeof Symbol === "function" &&
- (g[Symbol.iterator] = function () {
- return this;
- }),
- g
- );
- function verb(n) {
- return function (v) {
- return step([n, v]);
- };
- }
- function step(op) {
- if (f) throw new TypeError("Generator is already executing.");
- while ((g && ((g = 0), op[0] && (_ = 0)), _))
- try {
- if (
- ((f = 1),
- y &&
- (t =
- op[0] & 2
- ? y["return"]
- : op[0]
- ? y["throw"] ||
- ((t = y["return"]) && t.call(y), 0)
- : y.next) &&
- !(t = t.call(y, op[1])).done)
- )
- return t;
- if (((y = 0), t)) op = [op[0] & 2, t.value];
- switch (op[0]) {
- case 0:
- case 1:
- t = op;
- break;
- case 4:
- _.label++;
- return { value: op[1], done: false };
- case 5:
- _.label++;
- y = op[1];
- op = [0];
- continue;
- case 7:
- op = _.ops.pop();
- _.trys.pop();
- continue;
- default:
- if (
- !((t = _.trys),
- (t = t.length > 0 && t[t.length - 1])) &&
- (op[0] === 6 || op[0] === 2)
- ) {
- _ = 0;
- continue;
- }
- if (
- op[0] === 3 &&
- (!t || (op[1] > t[0] && op[1] < t[3]))
- ) {
- _.label = op[1];
- break;
- }
- if (op[0] === 6 && _.label < t[1]) {
- _.label = t[1];
- t = op;
- break;
- }
- if (t && _.label < t[2]) {
- _.label = t[2];
- _.ops.push(op);
- break;
- }
- if (t[2]) _.ops.pop();
- _.trys.pop();
- continue;
- }
- op = body.call(thisArg, _);
- } catch (e) {
- op = [6, e];
- y = 0;
- } finally {
- f = t = 0;
- }
- if (op[0] & 5) throw op[1];
- return { value: op[0] ? op[1] : void 0, done: true };
- }
- };
-
- var main = function () {
- return __awaiter(void 0, void 0, void 0, function () {
- var mask,
- quit,
- ElementPositions,
- allBoxes,
- _i,
- ElementPositions_1,
- position,
- selectedBox,
- _a,
- allBoxes_1,
- div,
- filtered;
- return __generator(this, function (_b) {
- switch (_b.label) {
- case 0:
- mask = document.createElement("div");
- addStylesheetRules(mask, {
- position: "fixed",
- top: "0",
- left: "0",
- inlineSize: "100%",
- blockSize: "100%",
- zIndex: "99998",
- background: "#0007",
- });
- quit = function () {};
- ElementPositions = getElementsPosition();
- console.log(ElementPositions);
- allBoxes = [];
- for (
- _i = 0, ElementPositions_1 = ElementPositions;
- _i < ElementPositions_1.length;
- _i++
- ) {
- position = ElementPositions_1[_i];
- allBoxes.push(makeDiv(position));
- }
- return [4, dragSelect(allBoxes)];
- case 1:
- selectedBox = _b.sent();
- for (
- _a = 0, allBoxes_1 = allBoxes;
- _a < allBoxes_1.length;
- _a++
- ) {
- div = allBoxes_1[_a];
- div.el.remove();
- }
- filtered = selectedBox.el
- .filter(function (val, index, arr) {
- console.log(val, index, arr);
- for (
- var _i = 0, arr_1 = arr;
- _i < arr_1.length;
- _i++
- ) {
- var i = arr_1[_i];
- if (isChild(i.src, val.src)) return false;
- }
- return true;
- })
- .sort(function (a, b) {
- return a.left - b.left;
- })
- .sort(function (a, b) {
- return a.top - b.top;
- });
- console.log(
- filtered.map(function (el) {
- return el.text;
- }),
- );
- navigator.clipboard.writeText(
- filtered
- .map(function (el) {
- return el.text;
- })
- .join("\n\n"),
- );
- return [2];
- }
- });
- });
- };
-
- window.addEventListener("keydown", function (e) {
- if (e.key === "s" && e.altKey && !e.ctrlKey && !e.shiftKey) main();
- });
- })();