// ==UserScript==
// @name 1fichier auto-clicker
// @name:en 1fichier auto-clicker
// @namespace Violentmonkey Scripts
// @icon 
// @match https://1fichier.com/*
// @grant window.close
// @grant GM_addStyle
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_registerMenuCommand
// @version 1.2.0
// @license GNU GPLv3
// @author Dummy_mole
// @description Ce script automatise le téléchargement instantané ou différé par compte à rebours.
// @description:en This script automate instant or delayed by countdown download.
// ==/UserScript==
(() => {
'use strict'
//===================================script config panel part====================================
const defaultConfig = {
"tab_auto_close": true,
"page_reload_delay": 5,
"update_time_delay": 10
}
function createConfigPanel() {
// Script config panel DOM creation
const configPanel = `
<div class="config-panel-container slide-in-top">
<h1 class="config-panel-title">1fichier configuration</h1>
<span class="separator"></span>
<div class="panel">
<div class="setting-elements">
<div class="tab-auto-close-container" title="automatically close tab after download begins">
<input id="tab-auto-close" type="checkbox">
<label for="auto-sort">Tab Auto Close</label>
</div>
<div class="page-reload-delay-container"
title="delay in minutes (min: 1, max: 60) between page reloads when a download is already in progress">
<label for="page-reload-delay">Page Reload Delay (minutes)</label>
<input id="page-reload-delay" type="number" inputmode="numeric" min="1" max="60">
</div>
<div class="update-time-delay-container"
title="refresh time in seconds (min: 1, max: 300) between each update of new values">
<label for="update-time-delay">Update Time Delay (seconds)</label>
<input id="update-time-delay" type="number" inputmode="numeric" min="1" max="300">
</div>
</div>
<span class="separator"></span>
<div class="validation-buton-container">
<button class="save-btn">save</button>
<button class="reset-btn">reset</button>
<button class="cancel-btn">cancel</button>
</div>
</div>
</div>
`
document.body.insertAdjacentHTML('afterbegin', configPanel)
//---------------------------------------------------------------------------------------------
function slideOutAndRemove(element, classOut, classIn) {
element.classList.remove(classOut)
element.classList.add(classIn)
setTimeout(() => element.remove(), 1000)
}
const configPanelContainer = document.querySelector('.config-panel-container')
const tabAutoCloseCheckbox = document.querySelector('#tab-auto-close')
const pageReloadDelayInput = document.querySelector('#page-reload-delay')
const updateTimeDelayInput = document.querySelector('#update-time-delay')
// Preventing the user from typing letters in numerical inputs, I didn't say I didn't trust you but you know you'll try if I don't stop you from doing it.
for (const [index, input] of [updateTimeDelayInput, pageReloadDelayInput].entries()) {
input.addEventListener('input', () => {
const maxLength = index === 1 ? 2 : 3;
const maxValue = index === 1 ? 60 : 300;
input.value = input.value.replace(/\D/g, '').slice(0, maxLength);
if (Number(input.value) > maxValue) input.value = maxValue;
})
}
const saveBtn = document.querySelector('.save-btn')
const resetBtn = document.querySelector('.reset-btn')
const cancelBtn = document.querySelector('.cancel-btn')
// Load configuration values from script's local storage or use default values
let config = JSON.parse(GM_getValue("1fichier_cfg", defaultConfig))
// Updates user interface elements with current configuration values
tabAutoCloseCheckbox.checked = config.tab_auto_close
pageReloadDelayInput.value = config.page_reload_delay
updateTimeDelayInput.value = config.update_time_delay
// Updates configuration when user interface elements change
tabAutoCloseCheckbox.onchange = function () { config.tab_auto_close = this.checked }
pageReloadDelayInput.onchange = function () { config.page_reload_delay = this.value }
updateTimeDelayInput.onchange = function () { config.update_time_delay = this.value }
// Saves configuration to script's local storage when user clicks "Save"
saveBtn.onclick = function () {
GM_setValue("1fichier_cfg", JSON.stringify(config))
location.reload()
}
// Resets the configuration to default values when the user clicks on "Reset".
resetBtn.onclick = function () {
config = defaultConfig
tabAutoCloseCheckbox.checked = defaultConfig.tab_auto_close
pageReloadDelayInput.value = defaultConfig.page_reload_delay
updateTimeDelayInput.value = defaultConfig.update_time_delay
GM_setValue("1fichier_cfg", JSON.stringify(defaultConfig))
location.reload()
}
// Cancels changes made to the configuration when the user clicks on "Cancel".
cancelBtn.onclick = function () {
slideOutAndRemove(configPanelContainer, "slide-in-top", "slide-out-top")
}
//------------------------------------config panel css style-----------------------------------
const configPanelStyle = `.config-panel-container{position:fixed!important;top:10px;right:10px;z-index:10001;display:flex;flex-direction:column;min-width:250px!important;max-width:300px!important;background:linear-gradient(0deg,#141617 0,rgba(27,27,28,.9) 50%,rgba(33,34,36,.9) 100%)!important;box-shadow:6px 6px 5px #00000069!important;padding:5px 15px!important;border:1px solid #282828!important;border-radius:5px!important;font-family:inherit;backdrop-filter:blur(15px);width:300px}.config-panel-title{margin:10px auto;padding-bottom:5px;color:#159cff!important;font-size:20px!important;font-weight:600}#file-viewing-method,.panel{color:#aaa!important}.page-reload-delay-container,.tab-auto-close-container,.update-time-delay-container{margin:10px auto}.page-reload-delay-container,.update-time-delay-container{display:flex;justify-content:space-between}#page-reload-delay,#update-time-delay{border-radius:5px!important;color:#f5f5f5!important;background:#262626!important;border:1px solid #4d4b4b!important;padding:2px 4px 2px 8px !important}.separator{display:block;width:98%;height:1px;background:#fdfdfd0d}.validation-buton-container{display:flex;justify-content:space-evenly;margin:10px auto}.validation-buton-container>button{display:block;font-size:16px!important;color:#f5f5f5!important;background:#0087ff;border:transparent!important;border-radius:3px!important;height:30px!important;padding:0 10px!important;width:66px;box-shadow:unset!important}.validation-buton-container>button:hover{opacity:.8}.validation-buton-container>button:active{opacity:1}.cancel-btn{background:#4a4949!important}.slide-in-top{-webkit-animation:.9s cubic-bezier(.25,.46,.45,.94) both slide-in-top;animation:.9s cubic-bezier(.25,.46,.45,.94) both slide-in-top}.slide-out-top{-webkit-animation:.9s cubic-bezier(.55,.085,.68,.53) both slide-out-top;animation:.9s cubic-bezier(.55,.085,.68,.53) both slide-out-top}@-webkit-keyframes slide-in-top{0%{-webkit-transform:translateY(-1000px);transform:translateY(-1000px);opacity:0}75%{-webkit-transform:translateY(20px);transform:translateY(20px)}100%{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}}@keyframes slide-in-top{0%{-webkit-transform:translateY(-1000px);transform:translateY(-1000px);opacity:0}75%{-webkit-transform:translateY(20px);transform:translateY(20px)}100%{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}}@-webkit-keyframes slide-out-top{0%{-webkit-transform:translateY(0);transform:translateY(0);opacity:1}25%{-webkit-transform:translateY(20px);transform:translateY(20px)}100%{-webkit-transform:translateY(-1000px);transform:translateY(-1000px);opacity:0}}@keyframes slide-out-top{0%{-webkit-transform:translateY(0);transform:translateY(0)}25%{-webkit-transform:translateY(20px);transform:translateY(20px)}100%{-webkit-transform:translateY(-1000px);transform:translateY(-1000px)}}`
GM_addStyle(configPanelStyle)
}
// Script config panel initialization
GM_registerMenuCommand("1fichier configuration", createConfigPanel)
if (!GM_getValue("1fichier_cfg")) GM_setValue("1fichier_cfg", JSON.stringify(defaultConfig))
const { tab_auto_close, page_reload_delay, update_time_delay } = JSON.parse(GM_getValue("1fichier_cfg"))
//================================================================================================
//----------------------------------------General Variables---------------------------------------
const localLangage = (navigator.language || navigator.userLanguage)?.split('-')[0] === "fr" ? "fr" : "en"
const userIntervalDelay = +update_time_delay * 1000
const userPageReloadDelay = +page_reload_delay * 60000
const messages = {
fr: {
downloadStart: "Votre téléchargement commencera automatiquement vers ",
reloadMessage: "Un téléchargement est déjà en cours - Prochain rechargement à ",
autoStartMessage: "Le téléchargement commencera automatiquement à la fin du compte à rebours",
startSoonMessage: "🚀Téléchargement imminent🚀",
remainingMinutes: "🚀",
minutesLeft: "min restantes",
lessThanFive:"quelques"
},
en: {
downloadStart: "Your download will start automatically around ",
reloadMessage: "A download is already in progress - next Reload at ",
autoStartMessage: "The download will start automatically at the end of the countdown",
startSoonMessage: "🚀The download is about to start🚀",
remainingMinutes: "🚀",
minutesLeft: "min left",
lessThanFive:"just a few"
}
}
//------------------------------------------Functions----------------------------------------------
// Replaces the native confirm dialog, which appears after prolonged inactivity, with a custom one that auto-validates to prevent blocking script automation purpose.
window.confirm = function (message) {
console.log(message)
return true
}
function getLocalizedMessage(key) {
return messages[localLangage][key]
}
function waitForElm(selector, all = false) {
const elementDirect = all
? document.querySelectorAll(selector)
: document.querySelector(selector)
return new Promise(resolve => {
if (elementDirect) {
return resolve(elementDirect)
}
const observer = new MutationObserver(mutations => {
const elementObserved = all
? document.querySelectorAll(selector)
: document.querySelector(selector)
if (elementObserved) {
resolve(elementObserved)
observer.disconnect()
}
})
observer.observe(document.body, { childList: true, subtree: true })
})
}
function moveMessage(parent, endTargetElement) {
const newDiv = document.createElement('div')
newDiv.className = 'moved-message'
while (parent.firstChild !== endTargetElement) newDiv.appendChild(parent.firstChild)
parent.prepend(newDiv)
const lastNode = [...newDiv.childNodes].at(-1)
const text = lastNode.nodeValue
const minutes = text.match(/\d+/)[0]
const beforeMinutes = text.match(/(.*?)\d+/)[1]
const afterMinutes = text.match(/\d+(.*)/)[1]
const formattedTimeNode = `
<div class="time-container">
<div class="time-node">${beforeMinutes}<span class="time-left">${minutes}</span>${afterMinutes}</div>
<div class="time-node">${getLocalizedMessage('downloadStart')}<span class="dl-start-time"></span></div>
</div>`
newDiv.removeChild(lastNode)
newDiv.insertAdjacentHTML('beforeend', formattedTimeNode)
return newDiv
}
function calculateEndTime(timeInMinutes, locale) {
const date = new Date(Date.now() + timeInMinutes * 60000)
const options = locale === 'en' ? { hour: 'numeric', minute: 'numeric', hour12: true } : { hour: '2-digit', minute: '2-digit' }
return date.toLocaleTimeString(locale, options)
}
function updateTabTitle(time) {
document.title = time >= 2
? `1fichier - ⏳ ${time} ${getLocalizedMessage('minutesLeft')}`
: getLocalizedMessage('startSoonMessage')
}
function updateValues(newValues, firstEl, lastEl) {
const { remainingMinutes, endTime } = newValues
updateTabTitle(remainingMinutes)
firstEl.textContent = remainingMinutes >= 5 ? remainingMinutes : getLocalizedMessage('lessThanFive')
lastEl.textContent = endTime
}
//----------------------------------------------Code------------------------------------------------
waitForElm('.ok.btn-general.btn-orange').then(dlBtn => {
const spanWarn = document.querySelector(".ct_warn span")
// If "!" appears in the warning, this means that the clock is running or a download is already in progress.
if (spanWarn && /!/gi.test(spanWarn.textContent)) {
const spanWarnParent = spanWarn.parentNode
// If there is a warning with no number in it, that means this is a "downloas in progress" message.
if (!/\d/.test(spanWarnParent.innerText)) {
const delayDisplayed = userPageReloadDelay / 60000
const message = getLocalizedMessage('reloadMessage')
const endTime = calculateEndTime(delayDisplayed, localLangage)
dlBtn.value = `${message} ${endTime}`
dlBtn.disabled = true
console.log("A download is already in progress")
document.title = `1fichier: Auto reload - ${endTime}`
setInterval(() => location.reload(), userPageReloadDelay)
} else {
dlBtn.disabled = true
waitForElm('.clock.flip-clock-wrapper').then(clockWrapper => {
// Move all nodes before clockWrapper to a new div for convenience.
const timeMessageContainer = moveMessage(spanWarnParent, clockWrapper)
const timeToWaitEl = timeMessageContainer.querySelector('.time-left')
const dlStartTimeEl = timeMessageContainer.querySelector('.dl-start-time')
// Time information initialization part
const initialTimeToWait = +timeToWaitEl.textContent
const startTime = Date.now()
const calculateRemainingMinutes = (startTime, initialTime) => Math.floor((startTime + (initialTime + 1) * 60000 - Date.now()) / 60000)
const endTime = calculateEndTime(initialTimeToWait, localLangage)
let remainingMinutes = calculateRemainingMinutes(startTime, initialTimeToWait)
updateValues({ remainingMinutes, endTime }, timeToWaitEl, dlStartTimeEl)
dlBtn.value = getLocalizedMessage('autoStartMessage')
let allInn = clockWrapper.querySelectorAll('.inn')
const allInLength = allInn.length
const totalDisplayedNumber = allInLength / 4
// INTERVAL
const countDownChecker = setInterval(() => {
// Update remaining time message every (n) seconds
remainingMinutes = calculateRemainingMinutes(startTime, initialTimeToWait)
const innTextContentArray = []
for (const inn of allInn) {
innTextContentArray.push(+inn.textContent)
}
const sum = innTextContentArray.reduce((acc, value) => acc + value, 0)
if (+sum === totalDisplayedNumber * 2) {
clearInterval(countDownChecker)
dlBtn.disabled = false
dlBtn.click()
}
// refresh new values
allInn = clockWrapper.querySelectorAll('.inn')
innTextContentArray.length = 0
updateValues({ remainingMinutes, endTime }, timeToWaitEl, dlStartTimeEl)
console.log( `next check in ${update_time_delay }s`)
}, userIntervalDelay)
})
}
} else {
// The element below only appears on the first page, allowing us to differentiate page 1 from page 2 and to know when to automatically close the tab/window.
const firstPageRefElement = document.querySelector('.CKBL')
dlBtn.click()
if (!firstPageRefElement && tab_auto_close) {
setTimeout(() => window.close(), 1000)
}
}
})
//------------------------------------------------CSS-----------------------------------------------
const style = `#dlw,#dlw1{display:none!important}#dlb{display:block!important;width:unset!important}.time-container{display:flex;margin:auto;justify-content:center}.dl-start-time,.time-left{color:red;text-decoration:underline;font-style:normal}`
GM_addStyle(style)
})()