NOTICE: By continued use of this site you understand and agree to the binding Terms of Service and Privacy Policy.
// ==UserScript== // @id TattsPro // @name Tatts Pro // @namespace http://goslotto.ru/viewtopic.php?f=35&t=3830 // @version 2.0.2 // @author flobik <info@northweb.ru> // @description Заполняет билеты в лотереях Tatts.com комбинациями из текстового файла. // @include https://thelott.com/tattersalls/buy-lotto/purchase-ticket?product=* // @updateURL https://openuserjs.org/install/flobik/Tatts_Pro.user.js // @downloadURL https://openuserjs.org/install/flobik/Tatts_Pro.user.js // @run-at document-end // @grant none // @icon  // ==/UserScript== /* CHANGELOG 2.0.2 (01.06.2016) - Фикс для нового домена (спасибо Barsik) 2.0.1 (10.10.2015) - Фикс позиции блока с корзиной 2.0 (15.01.2015) - Скрипт переписан с нуля - Новый интерфейс - Новый принцип отправки билетов на сервер - Исправлены ошибки - Поддержка всех игр Tatts - Поддержка Tampermonkey и Greasemonkey - Автообновление 1.3 (05.02.2012) - Возможность не ставить цифру 0 перед числом, меньше 10 (при условии разделения чисел пробелами) 1.2 (05.02.2012) - Возможность использования комб с пробелами-разделителями 1.1 (21.12.2011) - Добавлена поддержка браузера Google Chrome - Исправлено несколько опечаток в сообщениях 1.0 (19.12.2011) - Первый релиз */ //Запуск скрипта в режиме IIFE (Immediately-Invoked Function Expression) (function($, window, document){ 'use strict'; //Константы var maxTickets = 20, //Максимум билетов в корзине maxCombsPerTicket = 50, //Максимум комб в билете minCombsPerTicket = 4, //Минимум комб в билете dotsTimerTimeout = 1000; //Задержка между изменением многоточия //Переменные var elTattsPro, //Ссылка на форму elCombsInput, //Поле выбора файла game, //Текущая игра dots = [], //Ссылки на точки в многоточии curDot = -1, //Текущий шаг многоточия dotsTimerEnabled = false, //Включен ли таймер обновления многоточия maxSelection = 0; //Максимальное число, которое можно выбрать в текущей лотерее //Сохраняющиеся на протяжении ввода всех комб переменные var data = { combs: [], //Готовые комбинации combsBreak: 0, //Комба, после которой нужно закончить ввод билета, чтобы в следующем билете было минимум 4 комбы curComb: 0, //Индекс вводимой комбинации draw: 0, //Тираж system: 0 //Система }; //Обработка ошибки function Error(text) { alert(text); elCombsInput.val(''); return false; } //Функции для работы с localStorage function StorageDelete(name) { localStorage.removeItem(name); } function StorageGet(name, defaultValue) { var value = localStorage[name]; return value == null ? defaultValue : JSON.parse(value); } function StorageSet(name, value) { localStorage[name] = JSON.stringify(value); } //Импортирует CSS function InjectCSS(css) { var head, style; head = document.getElementsByTagName('head')[0]; if (!head) { return; } style = document.createElement('style'); style.type = 'text/css'; style.innerHTML = css; head.appendChild(style); } //Изменение отображаемой панели интерфейса function ShowGUI(panel) { //Скрытие все панелей $('#TattsPro .Panel').hide(); //Отображение необходимой панели $('#TattsPro #'+panel+'.Panel').show(); //Запуск таймера многоточия if (panel == 'Process') { dotsTimerEnabled = true; DotsTimerIteration(); } else { dotsTimerEnabled = false; } } //Обновление информации в GUI function UpdateGUI(panel) { switch (panel) { case 'Start': $('#Start span[data-tp="Draw"]').html(data.draw); $('#Start span[data-tp="System"]').html(data.system); $('#Start span[data-tp="CombsCount"]').html(data.combs.length); $('#Start span[data-tp="TicketsCount"]').html(Math.ceil(data.combs.length / maxCombsPerTicket)); $('#Start span[data-tp="StepsCount"]').html(Math.ceil(data.combs.length / (maxCombsPerTicket * maxTickets))); break; case 'Process': $('#Process span[data-tp="CompletePercentage"]').html(Math.ceil((data.curComb / data.combs.length) * 100)); $('#Process span[data-tp="CombsRemain"]').html(data.combs.length - data.curComb); $('#Process span[data-tp="CombsCount"]').html(data.combs.length); $('#Process span[data-tp="Step"]').html(Math.floor(data.curComb / (maxCombsPerTicket * maxTickets)) + 1); $('#Process span[data-tp="StepsCount"]').html(Math.ceil(data.combs.length / (maxCombsPerTicket * maxTickets))); break; case 'Continue': $('#Continue span[data-tp="CompletePercentage"]').html(Math.ceil((data.curComb / data.combs.length) * 100)); $('#Continue span[data-tp="CombsRemain"]').html(data.combs.length - data.curComb); $('#Continue span[data-tp="CombsCount"]').html(data.combs.length); $('#Continue span[data-tp="Step"]').html(Math.floor(data.curComb / (maxCombsPerTicket * maxTickets)) + 1); $('#Continue span[data-tp="StepsCount"]').html(Math.ceil(data.combs.length / (maxCombsPerTicket * maxTickets))); break; } } //Таймер многоточия function DotsTimerIteration() { if (++curDot == 3) curDot = -1; for (var i = 0; i < 3; i++) dots[i].css('visibility', (i <= curDot ? 'visible' : 'hidden')); if (dotsTimerEnabled) setTimeout(DotsTimerIteration, dotsTimerTimeout); } //Выбран файл с комбинациями function FileLoad() { //Файл var file = elCombsInput[0].files[0]; //Проверка типа файла if (!file.type.match(/text/)) return Error('Выбранный тип файла не поддерживается.'); //Чтение содержимого файла var reader = new FileReader(); reader.onload = function(e) { //Парсинг комбинаций var parseSuccess = ParseCombs(reader.result); //Проверка комбинаций (если парсинг удался) var checkSuccess = (parseSuccess ? CheckCombs() : false); //Подготовка к вводу билетов и отображение GUI с информацией if (parseSuccess && checkSuccess) { PrepareTicketsEnter(); UpdateGUI('Start'); ShowGUI('Start'); } } reader.readAsText(file); } //Парсинг комбинаций из текста function ParseCombs(text) { //Разрешены только цифры и пробельные символы if (!text.match(/^[\d|\s]+$/)) return Error('Неправильный формат комбинаций.'); //trim() text = text.replace(/^\s\s*/, '').replace(/\s\s*$/, ''); //Добавление цифры 0 перед числами < 10 text = text.replace(/^(\d) /gm, '0$&'); text = text.replace(/ (\d)(?!\d)/gm, ' 0$1'); //Разбор комбинаций по строкам data.combs = []; var lines = text.split('\n'); for (var i = 0; i < lines.length; i++) { //Удаление всех пробельных символов var line = lines[i].replace(/\s/g, ''); //Количество чисел в комбинации var numCount = line.length / 2; //Временный массив с текущей комбинацией var curComb = []; //Парсинг комбинации for (var j = 0; j < numCount; j++) curComb[j] = parseInt(line.substr(j * 2, 2), 10); //Сохранение комбинации data.combs.push(curComb); } return true; } //Проверка комбинаций function CheckCombs() { //Минимум 4 комбинации if (data.combs.length < minCombsPerTicket) return Error('Необходимо минимум '+minCombsPerTicket+' комбинации.'); //Минимальное количество чисел в комбе var minCombsCount = $('.gameSelectionsDiv:first input').length + (game == 'Powerball' ? 1 : 0); //Количество чисел в первой комбе var firstCombNumCount = data.combs[0].length; //Все комбы одного формата, и есть минимальное кол-во чисел for (var i = 0; i < data.combs.length; i++) { if (data.combs[i].length != firstCombNumCount) return Error('Одна или несколько комбинаций отличаются по длине от первой комбинации.'); if (data.combs[i].length < minCombsCount) return Error('Одна или несколько комбинаций содержит менее '+minCombsCount+' чисел.'); } return true; } //Подготовка к вводу билетов function PrepareTicketsEnter() { //Количество комб в последнем билете var remainCombs = data.combs.length % maxCombsPerTicket; //Если их меньше минимального кол-ва, рассчитываем номер комбы, на которой закончится предпоследний/последний билет data.combsBreak = (remainCombs > 0 && remainCombs < minCombsPerTicket ? data.combs.length - minCombsPerTicket - 1 : data.combs.length - 1); //Индекс вводимой комбинации data.curComb = 0; //Выбранный тираж data.draw = parseInt($('.startDraw.selected').attr('startdraw'), 10); //Система data.system = (game == 'Powerball' ? data.combs[0].length - 1 : data.combs[0].length); } //Начало ввода билетов function EnterTickets() { //Интерфейс ввода билетов UpdateGUI('Process'); ShowGUI('Process'); //Запуск итерации ввода билетов EnterIteration(); } //Ввод билета function EnterIteration(AJAXData) { //Функция вызвана возвратом ответа от сервера if (typeof AJAXData != 'undefined') { UpdateGUI('Process'); //Обновление корзины trolley_generate(AJAXData); if (data.curComb == data.combs.length) { //Кончились комбы? StorageDelete('data'); ShowGUI('Complete'); return; } else if(AJAXData.trolley.freeItemCount == 0) { //Кончилось место в тележке StorageSet('data', data); ShowGUI('NeedPay'); return; } } //Индекс последней комбы в текущем билете if (data.curComb + maxCombsPerTicket <= data.combsBreak) var ticketLastCombIndex = data.curComb + maxCombsPerTicket - 1; //Максимум else if (data.curComb > data.combsBreak) var ticketLastCombIndex = data.combs.length - 1; //То, что осталось else var ticketLastCombIndex = data.combsBreak; //До линии разреза билета //Количество комб в билете //var curTicketCombsCount = ticketLastCombIndex - data.curComb + 1; //Подготовка данных билета var ticketData = { gameAutoPicks: [], //False, False, False... gameSelections: [], //Кодированные комбинации gamePowerballs: [] //Шары powerball }; //Разбор комбинаций for (;data.curComb <= ticketLastCombIndex; data.curComb++) { //Все билеты введены вручную... ticketData.gameAutoPicks[ticketData.gameAutoPicks.length] = 'False'; if (game != 'Powerball') { //Комбинация ticketData.gameSelections[ticketData.gameSelections.length] = EncodeComb(data.combs[data.curComb]); } else { //Комбинация ticketData.gameSelections[ticketData.gameSelections.length] = EncodeComb(data.combs[data.curComb].slice(0, data.combs[data.curComb].length - 1)); //Powerball ticketData.gamePowerballs[ticketData.gamePowerballs.length] = data.combs[data.curComb][data.combs[data.curComb].length - 1] } } //Формирование параметров AJAX запроса var queryData = { product: game, drawCount: 1, drawNumbers: data.draw, gameCount: ticketData.gameAutoPicks.length, gameAutoPicks: ticketData.gameAutoPicks.join(','), gameSelections: ticketData.gameSelections.join('_'), gamePowerballs: ticketData.gamePowerballs.join(','), powerhit: false, systemNumber: data.system, favouriteName: '', nswDayOfWeek: 0, autoPlayDrawCount: '', autoPlayJackpotAmount: '' }; //Отправка билета $.ajax({ url: 'https://thelott.com/CallProcedure.ashx?procedure=addTicketsToTrolley&nocache=' + Math.random(), type: "POST", data: queryData, success: EnterIteration }); } //Кодирует комбинацию (копия функции getGameSelectionMask(c)) function EncodeComb(c) { var a = []; for (var d = 0; d <= maxSelection; d++) a[d] = 0; while (a.length % 8 != 0) a.push(0); for (d = 0; d < c.length; d++) a[c[d]] = 1; var e = ""; for (d = 0; d < a.length / 8; d++) { var b = 0; for (j = 0; j < 8; j++) b += a[d * 8 + j] << j; e = StrPadLeft(b.toString(16), "0", 2) + e; } return e; } //Дополняет строку str символами pad слева до длинны length (копия функции string_padLeft(d, b, c)) function StrPadLeft(str, pad, length) { if (str.length >= length) return str; var e = []; for (var a = 1; a <= length - str.length; a++) e.push(pad); e.push(str); return e.join(""); } //Функция отмены ввода билетов function CancelTicketsEnter() { if (confirm("Вы уверены, что хотите отменить ввод билетов?")) { StorageDelete('data'); ShowGUI('Canceled'); } } //Инициализация скрипта function Init() { //Проверка совместимости if (window.FileReader === undefined) return Error('Ваш браузер не поддерживает HTML5 File API. Попробуйте последнюю версию Chrome или Firefox.'); //Выбранная игра game = product.name; //window.location.href.match(/product=(.+?)(?:&|$)/)[1]; //Максимальное число, которое можно выбрать в текущей лотерее maxSelection = product.maxSelection; //Сдвигаем корзину вниз $('#shopping-trolley').css('top', '310px'); //Добавляем CSS InjectCSS(resources.css); //Добавление формы скрипта $('#InputDiv').append(resources.html); //Ссылка на форму и её элементы elTattsPro = $('#TattsPro'); elCombsInput = $('#CombsInput'); for (var i = 0; i < 3; i++) dots[i] = $('.process .dot' + (i + 1)); //Стилизация заголовка CufonReplace('#TattsPro h2', { fontFamily: 'TheSans-ExtraBold', textShadow: '#FFFFFF 2px 0px', color: '-linear-gradient(#2f5ea9, .25=#4782ca,.50=#6bb8fd,.50=#6bb8fd)' }); //Проверка на наличие места в телеге if (trolleyData.trolley.freeItemCount == 0) { ShowGUI('NeedPay'); return; } //Продолжение ввода билетов var tmpData = StorageGet('data', false); if (tmpData !== false) { data = tmpData; UpdateGUI('Continue'); ShowGUI('Continue'); } //Событие выбора файла elCombsInput.change(function() { FileLoad(); }); //Событие начала ввода билетов, нажатие кнопки - продолжить ввод билетов $('#TattsProEnter, #TattsProContinue').click(function() { EnterTickets(); }); //Отмена ввода билетов $('#TattsProCancel').click(function() { CancelTicketsEnter(); }); } //Ресурсы var resources = {}; //Иконка OK resources.okIcon = ''; //Иконка отмены resources.cancelIcon = ''; //Иконка визы resources.visaIcon = ''; //HTML resources.html = '\ <div id="TattsPro">\ <h2>Tatts Pro</h2>\ <div class="Panel" id="FileChoose">\ Выберите файл с комбинациями\ <div>\ <div class="TattsProButton Big">Открыть</div>\ <input id="CombsInput" type="file">\ </div>\ </div>\ <div class="Panel" id="Start">\ <table>\ <tr>\ <td>\ <strong>Комбинаций:</strong> <span data-tp="CombsCount">0</span><br>\ <strong>Билетов:</strong> <span data-tp="TicketsCount">0</span><br>\ <strong>Заходов:</strong> <span data-tp="StepsCount">0</span>\ </td>\ <td>\ <strong>Draw:</strong> <span data-tp="Draw">0000</span><br>\ <strong>Система:</strong> <span data-tp="System">0</span>\ </td>\ </tr>\ </table>\ <div id="TattsProEnter" class="TattsProButton Big">Начать ввод</div>\ </div>\ <div class="Panel" id="Process">\ <div class="data">\ <strong>Завершено:</strong> <span data-tp="CompletePercentage">0</span>%<br>\ <strong>Осталось комб.:</strong> <span data-tp="CombsRemain">0</span> из <span data-tp="CombsCount">0</span><br>\ <strong>Заход:</strong> <span data-tp="Step">0</span> из <span data-tp="StepsCount">0</span><br>\ </div>\ <div class="process">Идет ввод билетов<span class="dot dot1">.</span><span class="dot dot2">.</span><span class="dot dot3">.</span></div>\ </div>\ <div class="Panel" id="Continue">\ <div class="data">\ <strong>Завершено:</strong> <span data-tp="CompletePercentage">0</span>%<br>\ <strong>Осталось комб.:</strong> <span data-tp="CombsRemain">0</span> из <span data-tp="CombsCount">0</span><br>\ <strong>Заход:</strong> <span data-tp="Step">0</span> из <span data-tp="StepsCount">0</span><br>\ </div>\ Продолжить ввод билетов?<br>\ <div id="TattsProContinue" class="TattsProButton Small">Да</div> <div id="TattsProCancel" class="TattsProButton Small Red">Нет</div>\ </div>\ <div class="Panel" id="Complete">\ <img src="'+resources.okIcon+'"/><br>Ввод билетов завершен!\ </div>\ <div class="Panel" id="NeedPay">\ <img src="'+resources.visaIcon+'"/><br>Для продолжения необходимо оплатить уже введенные билеты.\ </div>\ <div class="Panel" id="Canceled">\ <img src="'+resources.cancelIcon+'"/><br>Ввод билетов отменен.\ </div>\ </div>\ '; //CSS resources.css = '\ #TattsPro {\ position: absolute;\ left: 660px;\ top: 0px;\ width: 235px;\ height: 150px;\ background-color: rgba(252, 253, 254, 0.29);\ padding: 10px;\ border-radius: 10px;\ }\ #TattsPro h2 {\ margin: 0px 0 5px 0px;\ }\ #FileChoose {\ text-align: center;\ padding-top: 26px;\ }\ #FileChoose > div {\ position: relative;\ overflow: hidden;\ margin: 10px;\ }\ .TattsProButton {\ background: #f0d300;\ border: 1px solid #E3A800;\ background: linear-gradient(to bottom, #f0d300 0%,#efab00 100%);\ color: #FFF;\ font-weight: bold;\ text-shadow: 0px 2px 0px #E59D00;\ cursor: pointer;\ }\ .TattsProButton.Big {\ border-radius: 9px;\ font-size: 18px;\ padding: 7px 0;\ }\ .TattsProButton.Small {\ border-radius: 6px;\ padding: 2px 0;\ width: 100px;\ display: inline-block;\ margin: 6px 2px 0 2px;\ }\ .TattsProButton.Red {\ background: #da0000;\ border: 1px solid #fc0101;\ background: linear-gradient(to bottom, #FF4C4C 0%,#DD0E0E 100%);\ color: #FFF;\ font-weight: bold;\ text-shadow: 0px 2px 0px rgba(0, 0, 0, 0.18);\ }\ #CombsInput {\ position: absolute;\ top: 0;\ right: 0;\ margin: 0;\ padding: 0;\ font-size: 60px;\ cursor: pointer;\ opacity: 0;\ filter: alpha(opacity=0);\ }\ #Start, #Process, #Continue, #Complete, #NeedPay, #Canceled {\ display: none;\ text-align: center;\ }\ #Start table {\ width: 100%;\ white-space: nowrap;\ margin: 18px 0 19px 0;\ }\ #Start table td {\ text-align: left;\ }\ #Start table td:first-child {\ padding-right: 5px;\ }\ #Process .data {\ text-align: left;\ margin: 18px 0 19px 0;\ }\ #Process .process {\ text-align: center;\ font-weight: bold;\ background: linear-gradient(to bottom, rgba(255, 241, 142, 0.65) 0%, rgba(255, 216, 116, 0.65) 100%);\ color: #C78F00;\ padding: 3px 0px;\ border-radius: 4px;\ }\ #Complete {\ text-align: center;\ margin-top: 10px;\ }\ #Complete img {\ margin-bottom: 11px;\ }\ #NeedPay {\ text-align: center;\ margin-top: -2px;\ line-height: 18px;\ }\ #Canceled {\ text-align: center;\ margin-top: 10px;\ }\ #Canceled img {\ margin-bottom: 11px;\ }\ #Continue .data {\ text-align: left;\ margin: 9px 0 9px 0;\ }\ '; //Запуск инициализации после загрузки страницы $(function() { Init(); }); })($, window, document);