Greasy Fork is available in English.
鼠标指向链接标识图标预览链接网页
当前为
// ==UserScript==
// @name 链接预览
// @name:en Link Previewer
// @namespace http://greasyfork.icu/zh-CN/users/1073-hzhbest
// @version 1.0
// @description 鼠标指向链接标识图标预览链接网页
// @description:en Hovering to preview a link
// @author hzhbest
// @match *://*/*
// @grant none
// @license MIT
// ==/UserScript==
(function () {
'use strict';
const ixonimg = "";
const inonimg = "";
const loadimg = "";
const domainregex = /(?<=:\/\/)[^\/]+/;
const minHgap = 20, minVgap = 10; // 小窗距鼠标位置的右、下最大距离:像素
var winWidth = 700, winHeight = 550; // 小窗宽、高:像素
const scale = 0.6; // 小窗内页面放大率:(0~1]
const animationtime = 0.5; // 动画时长:秒
const iconsize = 20; // 图标放大后大小:像素
const icontrpr = 0.7; // 图标放大后不透明度:(0~1)
const pre = "░░░░░░ "; // 拖动手柄前缀:字符串
const id = "__link__prev";
const css = `
/* 链接样式 */
a.___prevlink>img.___previcon {
width: 16px !important; height: 16px !important;
opacity: 0%; transition: opacity 0.5s ease-out, background 0s;
position: absolute;
display: inline-block !important; margin: 0 !important;
}
a.___prevlink.__pr {
position: relative;
}
*:hover>a.___prevlink>img.___previcon {
opacity: 30%;
}
a.___prevlink:hover>img.___previcon {
opacity: ${icontrpr}; background: white; border: 1px solid #8d8d8db1;
transition: opacity 0.5s linear; z-index: 10008;
transform: scale(${iconsize / 16});
}
a.___prevlink>img.___previcon:hover {
background: #3e3ed3 !important; transition: background 1s !important;
}
a.__link__preved {
outline: 3px solid #3e3ed3;
}
/* 小窗样式 */
div#__link__prev_win {
opacity: 0; position: absolute; z-index: 10009; box-shadow: 1px 1px 7px 1px #717171;
padding: 0; margin: 0; background-color: white; transition: ${animationtime}s ease;
}
div#__link__prev_win.__close {
transition: 0s !important; display: none;
}
div#__link__prev_win.__loading {
background: url(${loadimg}) no-repeat center center; background-color: #e7e7e7;
}
div#__link__prev_win.__link__visible {
opacity: 1; transition: 0.2s !important;
}
div#__link__prev_win.visible .__link__prev_ifr {
border: none; transform: scale(${scale});
}
div#__link__prev_win.__ondrag {
transition: 0s !important; opacity: 0.7;
}
/* 关闭按钮样式 */
@keyframes rotating_button {
from {
transform: rotate(0deg);
}
to {
transform: rotate(360deg);
}
}
div#__link__prev_win>.__link__clos_btn {
position: absolute; top: -10px; right: -10px; z-index: 10010;
line-height: 24px; width: 24px; height: 24px;
color: white; font-family: Consolas; font-size: 11pt; text-align: center;
background: #df2020; border: 1px solid black; border-radius: 50%;
box-shadow: 1px 1px 5px #333; cursor: default; padding: 0 !important; margin: 0;
}
div#__link__prev_win>.__link__clos_btn.__loading {
animation: 3s linear 0s infinite rotating_button; background: #d76565;
}
div#__link__prev_win>.__link__clos_btn.__loading:hover {
animation: none; background: #df2020;
}
div#__link__prev_win>.__link__clos_btn:hover {
outline: 3px solid #be3737a2;
}
div#__link__prev_win>.__link__clos_btn.__close {
display: none;
}
/* 拖动手柄样式 */
div#__link__prev_win>div.__link__prev_han {
position: absolute; top: -5px; left: 0px; z-index: 10010; color: #d7d7d7;
height: 6px; width: 30px; border: 1px solid #242424; background: #3e3e3e; cursor: move;
border-radius: 3px; overflow: hidden; max-width: ${winWidth}px;
}
div#__link__prev_win>div.__link__prev_han>* {
display: inline-block; width: fit-content;
}
div#__link__prev_win>div.__link__prev_han:hover {
color: #e0a52e; height: 20px; top: -20px; width: fit-content;
}
div#__link__prev_win.__ondrag>div.__link__prev_han {
background: #6f6122; height: 50px; top: -20px;
}
div#__link__prev_win>.__link__prev_ttl {
font-size: 14px; font-family: Arial; white-space: nowrap;
}
div#__link__prev_win>.__link__prev_btn {
height: 18px; width: 18px; border: 1px solid #616161; color: #9a9276; font-size: 13px;
position: absolute; right: 0; bottom: 0; z-index: 10011; background: #ccd0d3af;
}
div#__link__prev_win>.__link__prev_btn:hover {
background: #2c96e24f; border-color: #2c96e2af;
}
div#__link__prev_win.__ondrag>.__link__prev_btn {
height: 40px; width: 40px;
}
`;
var previewwin, pre_ir, timer, linkprev, closebtn, draghand, titlebox, ondrag, rsbtn, rsdrag;
addCSS(css, id);
makeprevwin();
var site = location.host;
document.addEventListener('mouseover', (evt) => {
const t = evt.target;
if (t.tagName == "A" && ispagelink(t)) {
addIconTo([t]);
} else {
var links = getLinksInThreeLayer(t);
addIconTo(links);
}
});
setTimeout(function () { // 延时设置监视器,若标题变化则重新加载
var watch = document.querySelector('title');
new (window.MutationObserver || window.WebKitMutationObserver)(function (mutations) {
// console.log('标题变了', document.title);
if (!document.querySelector("#" + id)) {
addCSS(css, id);
}
if (!document.querySelector("#" + "__link__prev_win")) {
makeprevwin();
}
site = location.host;
}).observe(watch, { childList: true, subtree: true, characterData: true });
}, 1000);
function addIconTo(links) {
var lnkl = links.length;
var exl = 0;
// console.log(lnkl);
for (let i = 0; i < lnkl; i++) {
const lnk = links[i];
if (lnk.classList.contains("___prevlink")) {
continue;
}
lnk.classList.add("___prevlink");
if (window.getComputedStyle(lnk).position == "static") {
lnk.classList.add("__pr");
}
var lnkposit = getTrueSize(lnk);
var textWidth = lnk.textContent.trim().length * parseInt(window.getComputedStyle(lnk).fontSize)
textWidth = Math.min(lnkposit.w, textWidth);
var img;
img = creaElemIn('img', lnk);
img.className = '___previcon';
img.style.bottom = '0.1em';
var adj = (lnkposit.r >= (4 + iconsize)) ? textWidth + 4 : Math.min(textWidth + 4, lnkposit.w - 5 - iconsize);
img.style.left = adj + "px";
if (domainregex.exec(lnk.href)[0] !== site) {
img.src = ixonimg;
exl += 1;
} else {
img.src = inonimg;
}
img.addEventListener('mouseover', (evt) => {
//clearTimeout(timer);
setTimeout(() => {
if (previewwin.classList.contains("__close")) {
setWinPosit(evt.x, evt.y);
}
}, 1000);
timer = setTimeout(previewlink, 1000, evt);
});
img.addEventListener('mouseleave', () => {
clearTimeout(timer);
});
}
//console.log("external link count:" + exl);
window.addEventListener('mousedown', (e) => {
var target = e.target;
// 隐藏预览
if (isprevvisual() && !previewwin.contains(target)) {
hidepreview(e);
}
}, true);
}
function makeprevwin() {
previewwin = creaElemIn('div', document.body);
previewwin.id = "__link__prev_win";
previewwin.style.width = '100px'; // 显示前小窗大小
previewwin.style.height = '100px';
previewwin.classList.add("__close");
draghand = creaElemIn('div', previewwin);
draghand.classList.add("__link__prev_han");
draghand.innerHTML = pre;
rsbtn = creaElemIn('div', previewwin);
rsbtn.className = "__link__prev_btn";
rsbtn.innerHTML = "┘";
titlebox = creaElemIn('div', draghand);
titlebox.className = "__link__prev_ttl";
pre_ir = creaElemIn('iframe', previewwin);
pre_ir.className = "__link__prev_ifr";
pre_ir.style.border = 0;
closebtn = creaElemIn('div', previewwin);
closebtn.classList.add("__link__clos_btn");
closebtn.innerHTML = "X";
closebtn.addEventListener('click', hidepreview);
ondrag = endrag(previewwin, { x: 'left', y: 'top' }, draghand);
ondrag.hook('__drag_begin', () => {
previewwin.classList.add("__ondrag");
});
ondrag.hook('__drag_end', () => {
previewwin.classList.remove("__ondrag");
});
rsdrag = endrag(previewwin, { x: 'width', y: 'height' }, rsbtn);
rsdrag.hook('__dragging', () => {
if (!!rsdrag.position) { // 未开始拖动前该对象不存在,判断避免编译出错
winWidth = rsdrag.position._x;
winHeight = rsdrag.position._y;
setIfrWinSize(winWidth, winHeight);
}
});
rsdrag.hook('__drag_begin', () => {
previewwin.classList.add("__ondrag");
});
rsdrag.hook('__drag_end', () => {
previewwin.classList.remove("__ondrag");
});
}
function previewlink(evt) {
if (!!linkprev) {
linkprev.classList.remove("__link__preved");
}
linkprev = evt.target.parentNode;
linkprev.classList.add("__link__preved");
pre_ir.src = linkprev.href;
pre_ir.style.transform = "scale(" + scale + ")";
previewwin.classList.remove("__close");
previewwin.classList.add("__loading");
closebtn.classList.remove("__close");
closebtn.classList.add("__loading");
titlebox.innerHTML = linkprev.textContent;
setTimeout(() => {
previewwin.classList.add("__link__visible");
let l = Math.max(minHgap, Math.min(window.innerWidth - winWidth - minHgap, evt.x));
let t = Math.max(minVgap, Math.min(window.innerHeight - winHeight - minVgap, evt.y));
setWinPosit(l, t);
setPrevWinSize(winWidth, winHeight);
}, 100);
pre_ir.onload = () => {
closebtn.classList.remove("__loading");
previewwin.classList.remove("__loading");
};
}
function setWinPosit(l, t) { // 根据屏幕位置设置网页位置
previewwin.style.left = l + window.scrollX + 'px';
previewwin.style.top = t + window.scrollY + 'px';
}
function setPrevWinSize(w,h) { // 设置预览窗口大小
winWidth = w;
winHeight = h;
previewwin.style.width = w + 'px';
previewwin.style.height = h + 'px';
setIfrWinSize(w, h);
}
function setIfrWinSize(w, h) { // 根据放大率设置iframe窗口大小
var f = (1 - scale) / 2 * -1 / scale;
pre_ir.width = w / scale;
pre_ir.height = h / scale;
pre_ir.style.marginLeft = (w * f) + 'px';
pre_ir.style.marginTop = (h * f) + 'px';
}
function hidepreview(evt) {
setPrevWinSize(100, 100); // 小窗缩小效果
pre_ir.src = '';
setWinPosit(evt.x, evt.y); // 在鼠标位置消失
previewwin.classList.remove('__link__visible');
linkprev.classList.remove('__link__preved');
closebtn.classList.add("__close");
setTimeout(() => {
previewwin.classList.add("__close");
}, animationtime * 1000 + 10);
}
function isprevvisual() {
return previewwin && previewwin.classList.contains('__link__visible');
}
function getLinksInThreeLayer(elem) {
var links = [];
elem.childNodes.forEach(cnode => {
if (ispagelink(cnode)) {
links.push(cnode);
} else {
cnode.childNodes.forEach(ccnode => {
if (ispagelink(ccnode)) {
links.push(ccnode);
} else {
ccnode.childNodes.forEach(cccnode => {
if (ispagelink(cccnode)) {
links.push(cccnode);
}
});
}
});
}
});
return links;
}
function ispagelink(node) { // 若节点是A且指向实网址(非js非锚点)返回true
if (node.tagName !== "A" || !node.href) {
return false;
}
const h = node.href, l = location.href;
if (h.indexOf('javascript:') == 0 || h.replace(l.split('#')[0], "").indexOf("#") == 0) {
return false;
} else {
return true;
}
}
function creaElemIn(tagname, destin) { //在 destin 内末尾创建元素 tagname
let theElem = destin.appendChild(document.createElement(tagname));
return theElem;
}
function addCSS(css, cssid) {
let stylenode = creaElemIn('style', document.getElementsByTagName('head')[0]);
stylenode.textContent = css;
stylenode.type = 'text/css';
stylenode.id = cssid || '';
}
// 输入元素,返回对象{元素可见的宽、高、顶、左、一层父元素右侧余量}
function getTrueSize(elem, posiz) {
if (!posiz) {
var p = elem.getBoundingClientRect();
posiz = {
w: p.width,
h: p.height,
t: p.top,
l: p.left,
r: 0
};
}
var pp = elem.parentNode.getBoundingClientRect();
if (posiz.r == 0) posiz.r = (pp.left + pp.width) - (posiz.l + posiz.w); // 父元素对当前元素的右侧余量
if (pp.width >= posiz.w && pp.height >= posiz.h && pp.top <= posiz.t && pp.left <= posiz.l) {
return posiz;
} else {
return getTrueSize(elem.parentNode, {
w: Math.min(posiz.w, pp.width),
h: Math.min(posiz.h, pp.height),
t: Math.max(posiz.t, pp.top),
l: Math.max(posiz.l, pp.left),
r: posiz.r
});
}
}
// 对target拖动handle时,实现拖动的功能
// 输入:目标元素target,拖动位置参考系opt,拖动手柄handle
// 输入opt:形如【{x:'right',y:'bottom'}】,或者width、height(右下角拖动)
function endrag(target, opt, handle) {
var p_x, p_y, isDragging;
endrag = function (target, opt, handle) {
return new endrag.proto(target, opt || {}, handle);
}
endrag.proto = function (target, opt, handle) {
var self = this;
this.target = target;
this.style = target.style;
this.handle = handle;
var _x = opt.x !== 'right';
var _y = opt.y !== 'bottom';
this.x = opt.x; //_x ? 'left' : 'right';
this.y = opt.y; //_y ? 'top' : 'bottom';
// p_x = this.x;
// p_y = this.y;
this.xd = _x ? -1 : 1;
this.yd = _y ? -1 : 1;
this.computed_style = document.defaultView.getComputedStyle(target, '');
this.drag_begin = function (e) { self.__drag_begin(e); };
this.handle.addEventListener('mousedown', this.drag_begin, false); //only drag on handler
this.dragging = function (e) { self.__dragging(e); };
document.addEventListener('mousemove', this.dragging, false);
this.drag_end = function (e) { self.__drag_end(e); };
document.addEventListener('mouseup', this.drag_end, false);
};
endrag.proto.prototype = {
__drag_begin: function (e) {
if (e.button == 0) {
var _c = this.computed_style;
this.isDragging = isDragging = true;
this.position = {
_x: parseFloat(_c[this.x]),
_y: parseFloat(_c[this.y]),
x: e.pageX,
y: e.pageY
};
e.preventDefault();
}
},
__dragging: function (e) {
if (!this.isDragging) return;
var x = Math.floor(e.pageX), y = Math.floor(e.pageY), p = this.position;
// prevent moving out of window
var x_border = window.innerWidth - 40, y_border = window.innerHeight - 20;
if (x - window.scrollX > x_border) x = window.scrollX + x_border;
if (y - window.scrollY > y_border) y = window.scrollY + y_border;
p._x = p._x + (p.x - x) * this.xd;
p._y = p._y + (p.y - y) * this.yd;
this.style[this.x] = p._x + 'px';
this.style[this.y] = p._y + 'px';
p.x = x;
p.y = y;
},
__drag_end: function (e) {
if (e.button == 0) {
if (this.isDragging) {
this.isDragging = isDragging = false;
}
}
},
hook: function (method, func) {
if (typeof this[method] === 'function') {
var o = this[method];
this[method] = function () {
if (func.apply(this, arguments) === false) {
return;
}
o.apply(this, arguments);
};
}
}
};
return endrag(target, opt, handle);
}
})();