flobik / Tatts Pro

// ==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);