Greasy Fork is available in English.

Access Browser Time

Blocking the browser after the time has expired

// ==UserScript==
// @name         Access Browser Time
// @namespace    https://github.com/AlekPet/Access-Browser-Time/
// @version      1.0
// @description  Blocking the browser after the time has expired
// @author       AlekPet
// @copyright    AlekPet
// @license      MIT; https://raw.githubusercontent.com/AlekPet/Access-Browser-Time/master/LICENSE
// @icon         https://raw.githubusercontent.com/AlekPet/Access-Browser-Time/master/Access-Browser-By-Time-userscript.png
// @match        http*://*/*
// @run-at document-idle
// @grant GM_setValue
// @grant GM_getValue
// @grant GM_addStyle
// @grant GM_addValueChangeListener
// @require https://code.jquery.com/jquery-3.1.0.min.js
// ==/UserScript==

GM_addStyle(`
.countDown_Panel {
position: fixed;
max-width: 400px;
max-height: 450px;
bottom: 45px;
left: 5px;
text-align: center;
z-index: 99999999999999999999;
}
.info_time{
font: bold 10pt monospace;
text-align: left;
margin: 5px 10%;
}
.cTime {
color: #4d3ea2;
}
.upTime{
color: #910808;
}
.countDown_Panel .cd_form_box{
width: 400px;
display: none;
border: 1px solid silver;
background: #fff;
font-family: cursive;
opacity:0.8;
box-shadow: 0 3px 5px silver;
}

.cd_title {
background: linear-gradient(#250280,#0089ff);
padding: 10px;
color: #fff;
}
.cd_body {
padding: 5px;
}

.cd_title div {
display: inline-block;
float: right;
cursor: pointer;
transition: 1s font-size;
}

.cd_title div:hover {
font-size: 1.2em;
transition: 1s font-size;
}

.cD_Buttons {
width: 100px;
border: 1px solid silver;
padding: 5px;
background: linear-gradient(#b2b8bb,#88898c);
color: white;
user-select: none;
cursor: pointer;
transition: 1s font-size;
}

.cD_Buttons:hover{
font-size: 1.2em;
transform: rotateZ(-360deg);
transition: 1s font-size;
}

.countDown_Setting {
background: linear-gradient(#00a4ff,#0826ff);
margin: 5px auto;
display:none;
}
.controls {
margin: 0 auto;
}
.box_timer_table{
display: table;
margin: 0px auto;
width: 80%;
}
.box_timer_table > div {
display:table-row;
}
.box_timer_table > div div{
display:table-cell;
}
.box_timer_table .head_caption{
font-weight: bold;
}
.box_timer_table input[type='number']{
text-align:center;
width: 60px;
margin-left: 5px;
}
.buttons {
color: white;
border: 1px solid silver;
border-radius: 3px;
box-shadow: 0 3px 6px;
margin: 0 auto;
padding: 5px;
cursor:pointer;
font-famaly: monospace;
}
.button_start{
background: linear-gradient(limegreen, green);
width: 100px;
}
.button_start:hover{
background: linear-gradient(limegreen, #00FF50);
}
.countDown_Time {
margin: 0 20px;
font-size: 8pt;
min-height: 12px;
}
.controls .fieldset_all{
margin: 5px auto;
width: 90%;
}
.controls input#fon_, .controls #text_block  {
width: 100%;
color: grey;
}
.controls #text_block{
height: 70px;
}
.cont_blocks {
display: table;
width: 100%;
border-spacing: 3px;
}
.table_row {
display: table-row;
}
.table_cell {
display: table-cell;
vertical-align: top;
text-align: left;
}
.blockContainer{
margin-top: 10%;
text-align: center;
}
.block_message{
font: bold 3em monospace;
}

.input_pass_box {
position: absolute;
top: 40%;
left: 50%;
width: 235px;
transform: translate(-50%,-50%);
background: silver;
font: normal 12pt monospace;
opacity: 0.9;
box-shadow: 2px 2px 5px 3px #c0c0c082;
text-align: center;
display:none;
}
.input_pass_box > div {
padding: 7px 6px;
}
.input_pass_box > div:first-child {
background: darkcyan;
color: white;
}
.input_pass_box > div span:last-child {
float: right;
cursor:pointer;
}
.input_pass_box > div span:last-child:hover {
color: cyan;
}
.unblock{color: green;font:bold 12pt monospace;}
.unblock:hover{color: cyan;}
.info_pass_unblock{
background: white;
margin: 3px;
font-style: italic;
font-size: 9pt;
}
`);

(function() {
    'use strict';
    const $ = window.jQuery,
          debug = false,
          defaultBlockImage = ""

    var ObjTimer = null,
        mtint = null,
        curInterTimer = null

    function loadStorage(){
        let ObjTimer_tmp = GM_getValue('ObjTimer');

        ObjTimer = (ObjTimer_tmp) ? JSON.parse(GM_getValue('ObjTimer')) : {
            options: {
                enabled: false,
                date: null,
                block_data: {text:"", img: defaultBlockImage},
                password: null,
                settings_inputs: null
            }
        }

        if(debug) console.log(ObjTimer)
    }

    function saveToStorage(){
        try{
            var save_data = JSON.stringify(ObjTimer);

            if(save_data.length>0 && save_data !== null && save_data !=="" && save_data !== undefined){
                GM_setValue('ObjTimer', save_data);
                if(debug) console.log("Сохраненно: ",ObjTimer);
            }
        }catch(e){
            console.log(e);
        }
    }

    function getCurrentTime(set_time = null){
        let curTime = set_time ? new Date(set_time) : new Date(),
            year = curTime.getFullYear(),
            month = curTime.getMonth()+1,
            date = curTime.getDate(),
            h = curTime.getHours(),
            m = curTime.getMinutes(),
            s = curTime.getSeconds()

        return `${(date<10?"0"+date:date)+"."+(month<10?"0"+month:month)+"."+year+" "+(h<10?"0"+h:h)+":"+(m<10?"0"+m:m)+":"+(s<10?"0"+s:s)}`
    }

    $.fn.makeForm = function(){
        let DivForm = $(`
<div class="countDown_Panel">
<div class="cD_Buttons countDown_Setting">Настройки</div>
<div class="countDown_Time">Timer: --д. --:--:--</div>
</div>
`);
        this.append(DivForm);

        $(".countDown_Setting").click(checkParole); // showHide

        $(".countDown_Time").on("click", function(){
            $(".countDown_Setting").fadeToggle('slow')
        })

    };

    function showHide(){
        let $cd_form_box = $(".cd_form_box")

        clearInterval(curInterTimer)
        curInterTimer = setInterval(function(){
            let cTime = $(".cTime"),
                upTime = $(".upTime").text(getCurrentTime(returnNewDate())),
                curTime = getCurrentTime()

            if(cTime.length) $(".cTime").text(curTime)
        },1000)


        if(!$cd_form_box.length){
            $cd_form_box = $(`<div class="cd_form_box">
<div class="cd_title"><span>Timer Setting...</span><div>X</div></div>
<div class="cd_body">
<div class="controls">
<fieldset class="fieldset_all">
<legend>Установка времени</legend>
<div class="box_timer_table">
<div>
<div class="head_caption">Дней</div>
<div class="head_caption">Часов</div>
<div class="head_caption">Минут</div>
<div class="head_caption">Секунд</div>
</div>
<div>
<div><input type="number" min="0" title="Дней" id="days_" value="0"></div>
<div><input type="number" min="0" title="Часов" id="hours_"  value="0"></div>
<div><input type="number" min="0" title="Минут" id="mins_" value="0"></div>
<div><input type="number" min="0" title="Секунд" id="secs_"  value="0"></div>
</div>
</div>
<div class="info_time">
<div>Текущее время:</div>
<div class="cTime">00.00.0000 00:00:00</div>
<div>Время блокировки:</div>
<div class="upTime">00.00.0000 00:00:00</div>
</div>
</fieldset>
<fieldset class="fieldset_all">
<legend>Фон при блокировке</legend>
<div class="cont_blocks">
<div class="table_row">
<div class="cont_img_block table_cell">
<div class="img_container"><img src="${ObjTimer.options.block_data.img}" id="image_block" width="128" title="Фон" /></div>
</div>
<div class="cont_block_setting table_cell">
<div class="link_div">
<div>Ссылка:</div>
<input type="text" id="fon_" value="${ObjTimer.options.block_data.img}">
</div>
<div class="textarea_div">
<div>Текст при блокировке:</div>
<textarea id="text_block">Заблокировано ФБР!</textarea>
</div>
</div>
</div>
</div>
</fieldset>
<div class="buttons button_start">Start</div>
</div>
</div>
</div>
</div>`);


            $cd_form_box.find(".cd_title div").off().click(showHide);
            $cd_form_box.find("input").each(function(){
                $(this).on("input", function(){
                    if(checkinputs()){
                        let getTimeUp = getCurrentTime(returnNewDate())
                        $(".upTime").text(getTimeUp)
                    }
                })
            })

            $cd_form_box.find(".button_start").click(startTimer)

            $cd_form_box.find("input#fon_").on("input", imageURLCheck.bind("",$cd_form_box.find("img#image_block")))

            $(".countDown_Panel").append($cd_form_box);
        }

        $cd_form_box.animate({
            width: [ "toggle", "swing" ],
            height: [ "toggle", "swing" ],
            opacity: "toggle"
        }, 1500, "linear", function(){
            $(".countDown_Setting").animate({
                opacity: "hide",
                height: [ "hide", "swing" ]
            },'slow', function(){
                if(!$(".cd_form_box").is(":visible")){
                    $(".countDown_Panel").find(".cd_form_box").remove();
                }
            });
        });

    }

    function imageURLCheck(img,event){
        let imgInput = event.target

        img.one("load",function(){
            // loaded
        })
            .one("error", function(){
            this.src = ""
        })
            .attr("src",imgInput.value)
    }

    function returnNewDate(saveS = false){
        let date = new Date(),
            days_ = parseInt($("#days_").val()),
            hours_ = parseInt($("#hours_").val()),
            mins_ = parseInt($("#mins_").val()),
            secs_ = parseInt($("#secs_").val()),
            totalms = days_*(1000*60*60*24)+hours_*(1000*60*60)+mins_*(1000*60)+secs_*1000,

            newDate = new Date(date.getTime()+totalms)

        if(saveS){
            ObjTimer.options.enabled = true
            ObjTimer.options.date = newDate.getTime()
            ObjTimer.options.settings_inputs = {days:days_, hours: hours_, mins: mins_, sec: secs_}
            ObjTimer.options.block_data.text = $("#text_block").val()
            ObjTimer.options.block_data.image = $("#fon_").val()
        }

        return newDate.getTime()
    }

    function startTimer(){
        if(checkinputs()){

            if(mtint) clearInterval(mtint);
            $(".countDown_Time").empty();
            timer_run(returnNewDate(true))
            showHide()
            saveToStorage()

        } else {
            alert("В полях были ошибки исправьте их!")
        }
    }

    function checkinputs(){
        let input_ok = true
        $(".controls .box_timer_table input").each(function(indx, el){

            let id = el.id,
                val = el.value,
                title = el.title

            if(/^\s?$/.test(val)){
                alert("Поле '"+title+"' не может быть пустым!")
                el.value = 0
                input_ok = false
            }

            if(!/^[0-9]+$/.test(val)){
                let searchspchar = val.match(/[^0-9]/g)
                alert("Поле '"+title+"' содержит не цифры!\n"+(searchspchar && searchspchar.length? "\nСимволы в поле: "+val.match(/[^0-9]/g).join(","):""))
                el.value = val = parseInt(val) || 0
                input_ok = false
            }

            if(val < 0){
                alert("Поле '"+title+"' не может быть меньше 0!")
                el.value = 0
                input_ok = false
            }

            /*if(id == "days_"){
                el.value = val > 365 ? 365 : val
            } else if(id == "hours_"){
                el.value = val > 23 ? 23 : val
            } else if(id == "mins_" || id == "secs_"){
                el.value = val > 59 ? 59 : val
            }*/
        })
        return input_ok
    }

    function makeParol(){
        const p = prompt("Введите пароль, для защиты","")
        if(/^\s?$/.test(p)){
            alert("Пароль не может быть без символов, или состоять из одних пробелов!")
            if(confirm("Попробовать снова?")) makeParol(); else return;
        }
        ObjTimer.options.password = p
        saveToStorage()
        //checkParole()
        return p
    }

    function checkParole(){
        let p_cor = ObjTimer.options.password

        if(p_cor == null || /^\s?$/.test(p_cor)){
            p_cor = makeParol()
        } else {
            p_cor = ObjTimer.options.password.toString()
        }

        if(p_cor.length){
            const p = prompt("Введите пароль:","")
            if(p !== p_cor){
                alert("Пароль неверный!")
                $(".countDown_Setting").hide()
                return;
            } else {
                alert("Пароль верный!")
                showHide()
            }
        }
    }

    function timer_run(date){
        let d_t = document.title,
            yd = date,
            countText = $(".countDown_Time").get(0)

        mtint = setInterval(function(){
            let nowDateMsec = new Date().getTime(),
                msec = yd-nowDateMsec,
                sec = Math.floor(msec/1000),
                min = Math.floor(sec/60),
                hour = Math.floor(min/60),
                day = Math.floor(hour/24)

            sec %= 60
            min %= 60
            hour %= 24

            if (min<10){
                min='0'+min
            }
            if (sec<10){
                sec='0'+sec
            }
            if (hour<10){
                hour='0'+hour
            }
            if(msec>0){
                let timeOut = "Timer: "+day+"д. "+hour+":"+min+":"+sec
                countText.innerHTML = ((sec % 2 === 0 && msec<60000)?"":msec>60000?timeOut:"<span style='color:red'>"+timeOut+"</span>")
                document.title = d_t+" - "+ countText.innerText.replace("Timer: ","")
            } else {
                countText.innerText ="Время истекло!"
                clearInterval(mtint)
                blockBrowser()
            }
        }, 1000);
    }

    function blockBrowser(){
        let canvas = $("<canvas>").get(0),
            ctx = canvas.getContext("2d")
        let img = $("<img>").on("load", function(){
            let html = `<head><title>${ObjTimer.options.block_data.text}</title></head><body></body>`,
                containerCanvas = $("<div class='blockContainer'>"),
                messageBlock = $("<div class='block_message'>").text(ObjTimer.options.block_data.text),
                unBlock = $("<span class='unblock'>").text("Отключить блокировку").css("cursor","pointer").click(function(){
                    $(".input_pass_box").fadeToggle('slow', function(){
                        if(input_password.is(":visible")){
                            input_password.focus()
                        }
                    })
                }),
                popUPEnterPaswword = $("<div class='input_pass_box'><div><span>Enter password</span><span title='Close' onclick='$(\".unblock\").trigger(\"click\")'>X</span></div><div class='input_pass_body'></div></div>"),
                input_password = $("<input>").attr({"type":"password","title":"Field password input...","id":"input_password"}),
                button_check = $("<button></button>").text("Enter").click(function(){
                    const p_cor = ObjTimer.options.password.toString(),
                          user_input = input_password.val()

                    if(p_cor === user_input){
                        input_password.css('border-color','limegreen')
                        info_bar.css('color','green').text("Пароль верный!")

                        let funct = function(time){
                            var timelef = time+1
                            return function(){
                                timelef-= 1
                                info_bar.text("Разблокировка через: " + timelef +' сек.')
                                if(timelef <=0 ){
                                    ObjTimer.options.enabled = false;
                                    saveToStorage()
                                    location.reload()
                                }
                            }
                        }
                        setInterval(funct.call("",3), 1000)

                    } else {
                        input_password.css('border-color','red')
                        info_bar.text("Пароль не верный!")
                    }
                }),
                info_bar = $("<div class='info_pass_unblock'>")

            popUPEnterPaswword.find(".input_pass_body").append(input_password,button_check,info_bar)

            canvas.width = this.naturalWidth
            canvas.height = this.naturalHeight
            ctx.drawImage(this, 0, 0)

            containerCanvas.append(canvas,messageBlock,unBlock)

            const tmStyles = $("style").filter(function(){
                if(/^[0-9a-f-]+$/.test(this.id)) return this
            }),

                  jquery = $("<script>")
            jquery.attr({"src":"https://code.jquery.com/jquery-3.1.0.min.js","type":"text/javascript"})

            $(document).children("html").html(html)
            $("head").append(tmStyles,jquery)
            $("body").append(containerCanvas,popUPEnterPaswword)

        }).on("error", function(){
            $("body").empty()
        }).attr("src", defaultBlockImage)
        }

    function timerIsset(){
        if(ObjTimer.hasOwnProperty("options")){
            if(ObjTimer.options.hasOwnProperty("date") && ObjTimer.options.hasOwnProperty("enabled") && ObjTimer.options.enabled && ObjTimer.options.date){
                if((ObjTimer.options.date - new Date().getTime())>0){
                    timer_run(ObjTimer.options.date)
                }
                else blockBrowser()
            }
        }
    }

    function init(){
        loadStorage()
        $("body").makeForm()
        timerIsset()
    }

    if (window.top === window.self) {
        init();
    }
})();