1fichier auto-clicker

Ce script automatise le téléchargement instantané ou différé par compte à rebours.

K instalaci tototo skriptu si budete muset nainstalovat rozšíření jako Tampermonkey, Greasemonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Violentmonkey.

K instalaci tohoto skriptu si budete muset nainstalovat rozšíření jako Tampermonkey nebo Userscripts.

You will need to install an extension such as Tampermonkey to install this script.

K instalaci tohoto skriptu si budete muset nainstalovat manažer uživatelských skriptů.

(Už mám manažer uživatelských skriptů, nechte mě ho nainstalovat!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(Už mám manažer uživatelských stylů, nechte mě ho nainstalovat!)

// ==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.7
// @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 = {
    "bypass_subscription_offer": false,
    "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="bypass-subscription-offer-container" title="bypass subscription offer">
                <input id="bypass-subscription-offer" type="checkbox">
                <label for="bypass-subscription-offer">Bypass Subscription Offer</label>
              </div>
              <div class="tab-auto-close-container" title="automatically close tab after download begins">
                <input id="tab-auto-close" type="checkbox">
                <label for="tab-auto-close">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 bypassSubscriptionOfferCheckbox = document.querySelector('#bypass-subscription-offer')
    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
    bypassSubscriptionOfferCheckbox.checked = config.bypass_subscription_offer
    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
    bypassSubscriptionOfferCheckbox.onchange = function () { config.bypass_subscription_offer = this.checked }
    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
      bypassSubscriptionOfferCheckbox.checked = defaultConfig.bypass_subscription_offer
      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 { bypass_subscription_offer ,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 immminent🚀",
      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
  }

  //---------------------------------- Bypass Subscription offer -------------------------------------

  if (bypass_subscription_offer) {
    // If a sub chart form is present use this function to get convert the time from the warning message in milliseconds (plus 1 extra minute) used for reloading the page and display remaining message on tab title
    // returned object: {ms: Number, time: Number, unit: String}
    function parseDelayInMs(t){
    const m=t.match(/(\d+)\s*([hms])/i);
    if(!m)return null;
    const v=+m[1],u=m[2].toLowerCase(),map={s:"secondes",m:"minutes",h:"heures"};
    return{ms:{s:1e3,m:6e4,h:36e5}[u]*v+6e4,time:v,unit:map[u]};
    }

    waitForElm('form#f1').then(form => {

      const spanWarn = document.querySelector("form .ct_warn span")
      if (spanWarn && /\d/.test(spanWarn.parentNode.textContent)) {
          const pageReloadDelay = parseDelayInMs(spanWarn.parentNode.textContent)
          let remainingTime = pageReloadDelay.time
          const updateTitle = (time, unit) => document.title = time > 0 ? `1fichier: ${time} ${unit} left` : `1fichier: download starting soon`
          updateTitle(remainingTime--, pageReloadDelay.unit)
          setInterval(() => updateTitle(remainingTime--, pageReloadDelay.unit), 60000)
          setTimeout(()=> location.reload(), pageReloadDelay.ms)
      } else if (document.querySelector('form .ct_warn span[style="color:red"]')) {
          let remainingMinutes = Math.round(userPageReloadDelay / 60000)
          let message = remainingMinutes > 0 ? `${remainingMinutes--} minutes` : 'Starting soon'
          document.title = `1fichier: Auto reload - ${message}`

          setInterval(() => {
            message = remainingMinutes > 0 ? `${remainingMinutes--} minutes` : 'Starting soon'
            document.title = `1fichier: Auto reload - ${message}`
          }, 60000)

          setTimeout(() => location.reload(), userPageReloadDelay)
      } else {
        form.submit()
      }
    })
  }
  //----------------------------------------------Code------------------------------------------------

  waitForElm('.ok.btn-general.btn-orange').then(dlBtn => {
    const spanWarn = document.querySelector("form .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.textContent)) {
        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}`
        setTimeout(() => 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(), 2500)
      }
    }
  })

  //------------------------------------------------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)
})()