您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Visual indicator of page progress while scrolling
当前为
- // ==UserScript==
- // @license MIT
- // @name Scroll Page Progress
- // @namespace http://tampermonkey.net/
- // @version 1.6.2
- // @description Visual indicator of page progress while scrolling
- // @author You
- // @match *://*/*
- // @icon 
- // @grant none
- // ==/UserScript==
- (function() {
- 'use strict';
- if (window.trustedTypes && window.trustedTypes.createPolicy) {
- window.trustedTypes.createPolicy('default', {
- createHTML: (string, sink) => string
- });
- }
- let globalShadow
- function once(fn, context) {
- var result;
- return function() {
- if(fn) {
- result = fn.apply(context || this, arguments);
- fn = null;
- }
- return result;
- };
- }
- function insertCirculaProgressBarEl() {
- const shadowHost = document.createElement('div')
- const shadow = shadowHost.attachShadow({ mode: "closed" });
- const circularProgressBar = document.createElement('div')
- const contentWrapper = document.createElement('div')
- const closeOverlay = document.createElement('div')
- const title = document.createElement('div')
- const overlay = document.createElement('div')
- const leftSide = document.createElement('div')
- const rightSide = document.createElement('div')
- globalShadow = shadow
- circularProgressBar.classList.add('circular-progress-bar')
- title.classList.add('title');
- closeOverlay.classList.add('close-overlay');
- contentWrapper.classList.add('content-wrapper')
- overlay.classList.add('overlay');
- leftSide.classList.add('left-side');
- rightSide.classList.add('right-side');
- shadowHost.id = 'host-shwadow-circular-progress'
- title.innerText = '-%'
- if (window.trustedTypes) {
- closeOverlay.innerHTML = window.trustedTypes.defaultPolicy.createHTML('×')
- } else {
- closeOverlay.innerHTML = '×'
- }
- closeOverlay.addEventListener('click', function() {
- circularProgressBar.style.display = 'none'
- const path = window.location.pathname;
- let savedPaths = JSON.parse(localStorage.getItem('not-allowed-paths')) || [];
- console.log(path, savedPaths)
- if (!savedPaths.includes(path)) {
- savedPaths.push(path);
- localStorage.setItem('not-allowed-paths', JSON.stringify(savedPaths));
- }
- });
- ;[title, overlay, leftSide, rightSide].forEach(childEl => contentWrapper.appendChild(childEl))
- circularProgressBar.appendChild(closeOverlay)
- circularProgressBar.appendChild(contentWrapper)
- shadow.appendChild(circularProgressBar)
- document.body.appendChild(shadowHost)
- }
- function addCSS() {
- const styleSheet = document.createElement('style');
- styleSheet.textContent = `
- * {
- box-sizing: border-box;
- padding: 0;
- margin: 0;
- }
- .circular-progress-bar {
- --backgroundColor: #424242;
- --left-side-angle: 180deg;
- --barColor:orangered;
- width: 60px;
- height: 60px;
- color: #fff;
- border-radius: 50%;
- position: fixed;
- z-index: 2147483646;
- background: var(--backgroundColor);
- border: 5px solid white;
- box-shadow:
- 0 1px 1px hsl(0deg 0% 0% / 0.075),
- 0 2px 2px hsl(0deg 0% 0% / 0.075),
- 0 4px 4px hsl(0deg 0% 0% / 0.075),
- 0 8px 8px hsl(0deg 0% 0% / 0.075),
- 0 16px 16px hsl(0deg 0% 0% / 0.075);
- text-align: center;
- cursor: pointer;
- transition: opacity 0.2s ease;
- }
- .circular-progress-bar .overlay {
- width: 50%;
- height: 100%;
- position: absolute;
- top: 0;
- left: 0;
- z-index: 1;
- background-color: var(--backgroundColor);
- }
- .close-overlay {
- width: 20px;
- height: 20px;
- position: absolute;
- left: 100%;
- top: -30%;
- z-index: 2147483647;
- display: flex;
- justify-content: center;
- align-items: center;
- border-radius: 50%;
- background: orangered;
- font-size: 19px;
- text-align: center;
- opacity: 0;
- }
- .close-overlay:hover {
- font-weight: bold;
- }
- .content-wrapper {
- overflow: hidden;
- height: 100%;
- width: 100%;
- border-radius: 50%;
- position: relative;
- }
- .circular-progress-bar:hover .close-overlay {
- opacity:1;
- }
- .circular-progress-bar .title {
- font-size: 15px;
- font-weight: bold;
- position:relative;
- height: 100%;
- display:flex;
- justify-content:center;
- align-items: center;
- z-index: 100;
- }
- .circular-progress-bar .left-side,
- .circular-progress-bar .right-side {
- width: 50%;
- height: 100%;
- position: absolute;
- top: 0;
- left: 0;
- border: 5px solid var(--barColor);
- border-radius: 100px 0px 0px 100px;
- border-right: 0;
- transform-origin: right;
- }
- .circular-progress-bar .left-side {
- transform: rotate(var(--left-side-angle));
- z-index: var(--zindex);
- }
- .circular-progress-bar .right-side {
- transform: rotate(var(--right-side-angle));
- }
- `
- globalShadow.appendChild(styleSheet)
- }
- insertCirculaProgressBarEl()
- addCSS()
- const currentState = {
- deg: 0,
- progress: 0,
- zIndex: 0
- }
- function setAngle(deg) {
- const progressBar = globalShadow.querySelector('.circular-progress-bar')
- const leftSide = globalShadow.querySelector('.left-side')
- const rightSide = globalShadow.querySelector('.right-side')
- leftSide.style.opacity = "0"
- const zIndex = deg > 180 ? 100 : 0
- const rightSideAngle = deg < 180 ? deg : 180
- const leftSideAngle = deg
- const zIndexChangedToPositive = currentState.zIndex === 0 && zIndex === 100
- if (deg > 180) {
- leftSide.style.opacity = "1"
- progressBar.style.setProperty('--left-side-angle', `${leftSideAngle}deg`);
- progressBar.style.setProperty('--zindex', zIndex);
- }
- progressBar.style.setProperty('--right-side-angle', `${rightSideAngle}deg`);
- currentState.deg = deg
- /*if (zIndex === 100 && zIndexChangedToPositive) {
- setUnsetOpacity()
- }*/
- }
- function percentageToAngle (percentageNumber) {
- if (percentageNumber > 100) {
- return 360
- }
- if (percentageNumber < 0) {
- return 0
- }
- return (360 * percentageNumber)/100
- }
- function setPercentage(percentageNumber) {
- const angle = percentageToAngle(percentageNumber)
- setAngle(angle)
- }
- function debounce(callback, wait) {
- let timerId;
- return (...args) => {
- clearTimeout(timerId);
- timerId = setTimeout(() => {
- callback(...args);
- }, wait);
- };
- }
- const progressBar = globalShadow.querySelector('.circular-progress-bar')
- const body = document.body;
- let offsetX = 0, offsetY = 0, isDragging = false;
- progressBar.addEventListener("mousedown", (e) => {
- e.preventDefault()
- offsetX = e.clientX - progressBar.offsetLeft;
- offsetY = e.clientY - progressBar.offsetTop;
- isDragging = true;
- progressBar.style.cursor = "grabbing";
- progressBar.style.opacity = "0.3";
- });
- document.addEventListener("mousemove", (e) => {
- if (isDragging) {
- e.preventDefault();
- const left = e.clientX - offsetX;
- const top = e.clientY - offsetY;
- progressBar.style.left = `${left}px`;
- progressBar.style.top = `${top}px`;
- savePosition(left, top); // Guardar la posición cada vez que se mueve
- }
- });
- document.addEventListener("mouseup", () => {
- isDragging = false;
- progressBar.style.cursor = "grab";
- progressBar.style.opacity = "1";
- });
- function savePosition(left, top) {
- const position = { left, top };
- localStorage.setItem("elementPosition", JSON.stringify(position));
- }
- function loadPosition() {
- const savedPosition = localStorage.getItem("elementPosition");
- if (savedPosition) {
- const { left, top } = JSON.parse(savedPosition);
- progressBar.style.left = `${left}px`;
- progressBar.style.top = `${top}px`;
- } else {
- progressBar.style.right = `10px`;
- progressBar.style.top = `10px`;
- }
- }
- function getCurrentScrollProgress() {
- const winScroll = document.body.scrollTop || document.documentElement.scrollTop;
- const height = document.documentElement.scrollHeight - document.documentElement.clientHeight;
- const progress = (winScroll / height) * 100;
- return Math.trunc(progress);
- }
- const progressBarTitle = globalShadow.querySelector('.title')
- document.onreadystatechange = function () {
- if (document.readyState == "complete") {
- if (window.trustedTypes && window.trustedTypes.createPolicy && !window.trustedTypes.defaultPolicy) {
- window.trustedTypes.createPolicy('default', {
- createHTML: (string, sink) => string
- });
- }
- const currentPath = window.location.pathname;
- const savedPaths = JSON.parse(localStorage.getItem('not-allowed-paths')) || [];
- if (savedPaths.includes(currentPath)) {
- progressBar.style.display = 'none';
- return;
- }
- loadPosition()
- document.addEventListener('scroll', debounce(() => {
- setPercentage(getCurrentScrollProgress())
- progressBarTitle.innerText = getCurrentScrollProgress() + '%'
- console.log(progressBarTitle, getCurrentScrollProgress())
- currentState.progress = getCurrentScrollProgress()
- currentState.deg = percentageToAngle(getCurrentScrollProgress())
- }, 50))
- }
- }
- })();