Greasy Fork is available in English.
Статистика карточных игр персонажа
- // ==UserScript==// @name GN_CardState// @namespace Gradient// @description Статистика карточных игр персонажа// @include /^https{0,1}:\/\/(www\.heroeswm\.ru|178\.248\.235\.15)\/pl_info\.php\?id=\d+/// @version 1.1.8// ==/UserScript=="use strict";//----------------------------------------------------------------------------//var script_name = 'GN_CardState'; // Enter your script name herevar script_version = '1.1.8';//----------------------------------------------------------------------------//(function(){ try{ // wrapper start//----------------------------------------------------------------------------//// UnifiedLibrary 1.7.0 start//----------------------------------------------------------------------------////----------------------------------------------------------------------------//// SysUtils//----------------------------------------------------------------------------//var GN_SysUtils = new SysUtils(script_name);var SU = GN_SysUtils;//----------------------------------------------------------------------------//function SysUtils(name){ // wrapper start//----------------------------------------------------------------------------//this.show_error = function(error_string, use_alert){if(use_alert)alert(error_string);throw new Error(error_string);};if(arguments.length != 1)this.show_error('Wrong SysUtils arguments');if(!arguments[0])this.show_error('Empty SysUtils argument');//----------------------------------------------------------------------------//this.compare = function(a, b){return (a == b) ? 0 : (a > b ? 1 : -1);};//----------------------------------------------------------------------------//this.send_get = function(url){var xhr = new XMLHttpRequest();xhr.open('GET', url, false);xhr.overrideMimeType('text/plain; charset=windows-1251');xhr.send(null);if(xhr.status == 200)return xhr.responseText;return null;};//----------------------------------------------------------------------------//this.save_value = function(desc, value){var div = document.getElementById('GN_GM_Handler');div.setAttribute('desc', desc);div.setAttribute('value', value);div.setAttribute('operation', 'save');div.click();if(div.getAttribute('state') != 'complete')this.show_error('Ошибка при сохранении значения');};//----------------------------------------------------------------------------//this.load_value = function(value, def){var div = document.getElementById('GN_GM_Handler');div.setAttribute('desc', value);div.setAttribute('operation', 'load');div.click();if(div.getAttribute('state') != 'complete')this.show_error('Ошибка при загрузке значения');return (div.getAttribute('is_null') == 'true' ? def : div.getAttribute('value'));};//----------------------------------------------------------------------------//this.remove_value = function(value){var div = document.getElementById('GN_GM_Handler');div.setAttribute('desc', value);div.setAttribute('operation', 'remove');div.click();if(div.getAttribute('state') != 'complete')this.show_error('Ошибка при удалении значения');};//----------------------------------------------------------------------------//var current_id = null;//----------------------------------------------------------------------------//function check_mandatory_scripts(alerter){var persistent_storage_sign = document.getElementById('GN_GM_Handler');var common_values_sign = document.getElementById('GN_CommonValuesSign');var alert_sign = document.getElementById('GN_AlertSign');if(!alert_sign){alert_sign = document.createElement('div');alert_sign.id = 'GN_AlertSign';alert_sign.setAttribute('alerted', 'false');document.body.appendChild(alert_sign);}var alerted = alert_sign.getAttribute('alerted') != 'false';if(!persistent_storage_sign){alert_sign.setAttribute('alerted', 'true');alerter('Скрипт ' + name + ' требует для своей работы скрипт управления данными (GN_PersistentStorage), который должен стоять первым в порядке выполнения скриптов.\n'+ 'Подробнее здесь: "https://greasyfork.org/ru/scripts/14049-Как-устанавливать-скрипты-читать-здесь"', !alerted);}if(!common_values_sign){alert_sign.setAttribute('alerted', 'true');alerter('Скрипт ' + name + ' требует для своей работы скрипт, хранящий данные (GN_CommonValuesFiller), который должен стоять вторым в порядке выполнения скриптов.\n'+ 'Подробнее здесь: "https://greasyfork.org/ru/scripts/14049-Как-устанавливать-скрипты-читать-здесь"', !alerted);}}this.check_login = function(){var re = /.*?pl_id=(\d+)[^\d]*?/gmi;var matches = re.exec(document.cookie.toString());if(matches){current_id = +matches[1];check_mandatory_scripts(this.show_error);}};//----------------------------------------------------------------------------//this.save_file = function(text, info){var res = 'data:text/csv;charset=utf-8,' + encodeURI(text);if(info)alert(info);window.open(res);};//----------------------------------------------------------------------------//this.string_to_date = function(str){var matches = /(\d{2})-(\d{2})-(\d{2})\s(\d{2}):(\d{2})/.exec(str);return new Date(2000 + +matches[3], +matches[2] - 1, +matches[1], +matches[4], +matches[5]);};//----------------------------------------------------------------------------//this.show_el = function(el, visible){el.style.display = visible ? '' : 'none';};//----------------------------------------------------------------------------//this.reload_page = function(){document.location.href = document.location.href;};//----------------------------------------------------------------------------//this.check_login();//----------------------------------------------------------------------------//} // wrapper end//----------------------------------------------------------------------------//// CommonValues//----------------------------------------------------------------------------//var GN_CommonValues = new CommonValues();//----------------------------------------------------------------------------//function CommonValues(){ // wrapper start//----------------------------------------------------------------------------//// Card types//----------------------------------------------------------------------------//this.enum_sct = { // sync?tavern: 0,tour_pvp: 1,tour_pve: 2};this.sorted_card_types = JSON.parse(SU.load_value('GN_CommonValues_SortedCardTypes', '[]'));this.card_types = JSON.parse(SU.load_value('GN_CommonValues_CardTypes', '[]'));//----------------------------------------------------------------------------//this.get_card_type = function(id){for(var i = 0; i < this.card_types.length; ++i)if(this.card_types[i].id == id)return this.card_types[i];return null;};//----------------------------------------------------------------------------//this.get_sorted_card_type = function(id){for(var i = 0; i < this.sorted_card_types.length; ++i)if(this.sorted_card_types[i].id == id)return this.sorted_card_types[i];return null;};//----------------------------------------------------------------------------//} // wrapper end//----------------------------------------------------------------------------//// UnifiedLibrary end//----------------------------------------------------------------------------//var show_error = SU.show_error;var load_value = SU.load_value;var save_value = SU.save_value;var remove_value = SU.remove_value;var send_get = SU.send_get;var compare = SU.compare;var save_file = SU.save_file;var show_el = SU.show_el;var string_to_date = SU.string_to_date;var reload_page = SU.reload_page;var CV = GN_CommonValues;var sorted_card_types = CV.sorted_card_types;var enum_sct = CV.enum_sct;//----------------------------------------------------------------------------//var card_states = load_states();var parser_info = load_parser_info();var is_parser_running = false;//----------------------------------------------------------------------------//var table_width = 0;start_work();//----------------------------------------------------------------------------//function start_work(){var prev_sibling = get_sibling();if(!prev_sibling)show_error('Не найден элемент привязки');table_width = prev_sibling.width;var header = draw_header_table(prev_sibling);draw_table(header);draw_expand_all(header);}//----------------------------------------------------------------------------//function get_sibling(){var transfer = document.querySelector('table > tbody > tr > td > a[href*="pl_transfers.php?id="]');return transfer ? transfer.parentNode.parentNode.parentNode.parentNode : null;}//----------------------------------------------------------------------------//function draw_expand_all(next_sibling){var table = document.createElement('table');table.className = 'wblight';table.width = table_width;table.align = 'center';next_sibling.parentNode.insertBefore(table, next_sibling);var tr = document.createElement('tr');table.appendChild(tr);var script_desc = 'Карточная статистика ' + script_version;var expander = document.createElement('td');expander.setAttribute('align', 'center');expander.setAttribute('colspan', '6');var is_expanded = load_value(script_name + 'Expand', 'false') == 'true';expander.setAttribute('expand', is_expanded ? 'true' : 'false');expander.textContent = script_desc + (is_expanded ? ' (скрыть)' : ' (показать)');expander.addEventListener('click', function(e){e.preventDefault();var expanded = expander.getAttribute('expand') == 'false';show_el(next_sibling, expanded);show_el(next_sibling.nextSibling, expanded);save_value(script_name + 'Expand', expanded ? 'true' : 'false');expander.setAttribute('expand', expanded ? 'true' : 'false');expander.textContent = expanded ? (script_desc + ' (скрыть)') : (script_desc + ' (показать)');});tr.appendChild(expander);show_el(next_sibling, is_expanded);show_el(next_sibling.nextSibling, is_expanded);}//----------------------------------------------------------------------------//function draw_header_table(prev_sibling){var table = document.createElement('table');table.className = 'wblight';table.width = table_width;table.align = 'center';table.id = script_name + 'ContentHeader';prev_sibling.parentNode.insertBefore(table, prev_sibling.nextSibling);var tr = document.createElement('tr');table.appendChild(tr);var td = document.createElement('td');td.id = script_name + 'RefreshDate';tr.appendChild(td);td.textContent = 'Дата последнего обновления: ' + (parser_info.refresh_date ? parser_info.refresh_date.toLocaleString() : 'еще не считывалось');td = document.createElement('td');td.align = 'right';tr.appendChild(td);var refresh_button = document.createElement('input');refresh_button.type = 'button';refresh_button.value = 'Обновить данные';refresh_button.id = script_name + 'Refresh';refresh_button.addEventListener('click', parse_data);td.appendChild(refresh_button);tr = document.createElement('tr');table.appendChild(tr);td = document.createElement('td');td.id = script_name + 'ParseDate';tr.appendChild(td);td.textContent = 'Дата последней считанной игры: ' + (parser_info.parse_date ? parser_info.parse_date.toLocaleString() : 'еще не считывалось');td = document.createElement('td');td.align = 'right';tr.appendChild(td);var export_button = document.createElement('input');export_button.type = 'button';export_button.value = 'Экспорт в файл';export_button.id = script_name + 'Export';export_button.addEventListener('click', export_to_file);td.appendChild(export_button);var remove_button = document.createElement('input');remove_button.type = 'button';remove_button.value = 'Очистить статистику';remove_button.id = script_name + 'Remove';remove_button.addEventListener('click', remove_data);td.appendChild(remove_button);return table;}//----------------------------------------------------------------------------//function draw_table(prev_sibling){var table = document.createElement('table');table.className = 'wblight';table.width = table_width;table.align = 'center';table.id = script_name + 'Content';prev_sibling.parentNode.insertBefore(table, prev_sibling.nextSibling);update_content(table, false);return table;}//----------------------------------------------------------------------------//function update_content(parent, remove_childs){if(remove_childs)while(parent.lastChild)parent.removeChild(parent.lastChild);if(card_states.length){draw_header(parent);sorted_card_types.forEach(function(current){current.win = current.loss = current.total = 0;});card_states.forEach(function(current){var card_type = CV.get_card_type(current.id);var sorted_card_type = CV.get_sorted_card_type(card_type.type);sorted_card_type.win += current.win;sorted_card_type.loss += current.loss;sorted_card_type.total += current.total;});sorted_card_types.sort(function(a, b){var a_pt = a.win*100/(a.win + a.loss);var b_pt = b.win*100/(b.win + b.loss);if(isNaN(a_pt) || isNaN(b_pt))return isNaN(a_pt) ? 1 : -1;if(a_pt == b_pt)return compare(b.win, a.win);return compare(b_pt, a_pt);});sorted_card_types.forEach(function(current){if(current.win + current.loss > 0)draw_sorted_row(parent, current);});draw_bottom_header(parent);}}//----------------------------------------------------------------------------//function draw_header(parent){var tr = document.createElement('tr');parent.appendChild(tr);['Тип игры', 'Баланс', 'Победы', 'Поражения', 'Процент побед', 'Итого'].forEach(function(current){var td = document.createElement('td');tr.appendChild(td);var b = document.createElement('b');td.appendChild(b);b.textContent = current;});}//----------------------------------------------------------------------------//function draw_sorted_row(parent, content){var tr = document.createElement('tr');tr.setAttribute('bgcolor', content.color);parent.appendChild(tr);var td = document.createElement('td');tr.appendChild(td);var a = document.createElement('a');a.setAttribute('expanded', 'false');a.addEventListener('click', function(){var expanded = a.getAttribute('expanded') == 'false';if(expanded){card_states.sort(function(a, b){var a_pt = a.win*100/(a.win + a.loss);var b_pt = b.win*100/(b.win + b.loss);if(a_pt == b_pt)return compare(a.win, b.win);return compare(a_pt, b_pt);});var count = 0;card_states.forEach(function(current){var card_type = CV.get_card_type(current.id);if(card_type.type == content.id){draw_row(tr, content.id, current, count % 2 === 0 ? '#ffffff' : '#eeeeee');++count;}});}elsewhile(tr.nextSibling && tr.nextSibling.getAttribute('id') && tr.nextSibling.getAttribute('id').indexOf(script_name + '_' + content.id + '_') != -1)parent.removeChild(tr.nextSibling);a.setAttribute('expanded', expanded ? 'true' : 'false');});td.appendChild(a);var type = CV.get_sorted_card_type(content.id);var u = document.createElement('u');u.textContent = type.desc;a.appendChild(u);var sum = content.win + content.loss;var percent = content.win*100/sum;[content.total, content.win, content.loss, percent.toFixed(2) + '%', sum].forEach(function(current){td = document.createElement('td');tr.appendChild(td);td.textContent = current;});}//----------------------------------------------------------------------------//function draw_row(prev_sibling, sibling_id, content, color){var tr = document.createElement('tr');tr.id = script_name + '_' + sibling_id + '_' + content.id;tr.setAttribute('bgcolor', color);prev_sibling.parentNode.insertBefore(tr, prev_sibling.nextSibling);var type = CV.get_card_type(content.id);var sum = content.win + content.loss;var percent = content.win*100/sum;[type.desc, content.total, content.win, content.loss, percent.toFixed(2) + '%', sum].forEach(function(current){var td = document.createElement('td');tr.appendChild(td);td.textContent = current;});}//----------------------------------------------------------------------------//function draw_bottom_header(parent){var tr = document.createElement('tr');tr.id = script_name + 'BottomHeader';parent.appendChild(tr);var win_sum = 0,loss_sum = 0,total_sum = 0;card_states.forEach(function(current){win_sum += current.win;loss_sum += current.loss;total_sum += current.total;});var sum = win_sum + loss_sum;var percent = win_sum*100/sum;['Все игры', total_sum, win_sum, loss_sum, percent.toFixed(2) + '%', sum].forEach(function(current){var td = document.createElement('td');tr.appendChild(td);var b = document.createElement('b');td.appendChild(b);b.textContent = current;});}//----------------------------------------------------------------------------//function export_to_file(){if(!card_states.length)return;var linebreak = '%0D%0A';var res = ['Тип игры', 'Баланс', 'Победы', 'Поражения', 'Процент побед', 'Итого'].join(';') + linebreak;card_states.forEach(function(current){var card_type = CV.get_card_type(current.id);var sum = current.win + current.loss;var percent = current.win*100/sum;res += [card_type.desc, current.total, current.win, current.loss, percent.toFixed(2) + '%', sum].join(';') + linebreak;});save_file(res, 'Сейчас будет предложено сохранить файл с результатами. Переименуйте его в формат .csv, разделитель - ";"');}//----------------------------------------------------------------------------//function parse_data(){if(is_parser_running)return;document.body.style.cursor = 'wait';['Refresh', 'Export', 'Remove'].forEach(function(current){var el = document.getElementById(script_name + current);el.setAttribute('disabled', '');});is_parser_running = true;var last_page = get_last_page();var counter = {current_page: last_page - parser_info.parse_page_count + (parser_info.parse_page_count === 0 ? 0 : 2),last_page: last_page};search_next(counter);}//----------------------------------------------------------------------------//function search_next(counter){var refresh_button = document.getElementById(script_name + 'Refresh');var diff = counter.last_page - counter.current_page + 1;refresh_button.value = 'Обработано ' + diff + '/' + (counter.last_page + 1) + ' страниц (' + Math.round(diff*100/(counter.last_page + 1)) + '%)';var url = '/pl_cardlog.php?id=' + get_id() + '&page=' + counter.current_page;send_async_get(url, counter);}//----------------------------------------------------------------------------//function send_async_get(url, counter){var xhr = new XMLHttpRequest();xhr.open('GET', url, true);xhr.overrideMimeType('text/plain; charset=windows-1251');xhr.onreadystatechange = function(){if(xhr.readyState == 4){if(xhr.status == 200){--counter.current_page;search_value(xhr.response);if(counter.current_page >= 0)search_next(counter);else{parser_info.refresh_date = new Date();parser_info.parse_page_count = counter.last_page + 1;var refresh_td = document.getElementById(script_name + 'RefreshDate');refresh_td.textContent = 'Дата последнего обновления: ' + parser_info.refresh_date.toLocaleString();var parse_td = document.getElementById(script_name + 'ParseDate');parse_td.textContent = 'Дата последней считанной игры: ' + (parser_info.parse_date ? parser_info.parse_date.toLocaleString() : 'еще не считывалось');var refresh_button = document.getElementById(script_name + 'Refresh');refresh_button.value = 'Обновить данные';var content_table = document.getElementById(script_name + 'Content');update_content(content_table, true);save_value(script_name + '_States' + get_id(), JSON.stringify(card_states));save_value(script_name + '_ParserInfo' + get_id(), JSON.stringify(parser_info));['Refresh', 'Export', 'Remove'].forEach(function(current){document.getElementById(script_name + current).removeAttribute('disabled');});document.body.style.cursor = 'default';is_parser_running = false;}}}};xhr.send(null);}//----------------------------------------------------------------------------//function search_value(response_){var re = /.*?<a href="cgame\.php\?gameid=(\d+)">(\d{2}-\d{2}-\d{2}\s\d{2}:\d{2})(.+?)([+\-0-9]*)<\/td><\/tr>.*?/gmi;var raw_data = [],matches = [];while(matches = re.exec(response_))raw_data.push({ id: +matches[1], game_date: string_to_date(matches[2]), game_str: matches[3], game_bet: matches[4] ? +matches[4] : 0});raw_data.sort(function(a, b){return compare(a.id, b.id);});raw_data = raw_data.filter(function(current){if(!parser_info.last_id)return true;return current.id > parser_info.last_id;});if(!raw_data.length)return;raw_data.forEach(function(current){re = /.*?arc_tour_hist\.php.*?/gmi;var type_id = (re.test(current.game_str) ? enum_sct.tour_pvp : enum_sct.tavern);if(type_id == enum_sct.tour_pvp){re = /.*?(pl_info\.php\?id=).*?/gmi;var count = 0;while(re.test(current.game_str))++count;type_id = count == 1 ? enum_sct.tour_pve : enum_sct.tour_pvp;}var state_id = null;switch(type_id){case enum_sct.tavern:state_id = "bet" + Math.abs(current.game_bet);break;case enum_sct.tour_pvp:case enum_sct.tour_pve:{var stage = 0;re = />1\/(\d+)</;matches = re.exec(current.game_str);if(matches)stage = +matches[1];if(/>Полуфинал</.test(current.game_str))stage = 2;if(/>Финал</.test(current.game_str))stage = 1;state_id = ((type_id == enum_sct.tour_pvp) ? "stage" : "bstage") + stage;}break;}var state = get_card_state(state_id);re = new RegExp('<a href="pl_info\\.php\\?id=' + get_id() + '" class=pi><b>');var win = re.test(current.game_str);win ? ++state.win : ++state.loss;state.total += (type_id == enum_sct.tavern && win ? current.game_bet*0.95 : current.game_bet);});parser_info.parse_date = raw_data[raw_data.length - 1].game_date;parser_info.last_id = raw_data[raw_data.length - 1].id;}//----------------------------------------------------------------------------//function load_states(){var states = load_value(script_name + '_States' + get_id());return states ? JSON.parse(states) : [];}//----------------------------------------------------------------------------//function load_parser_info(){var info = load_value(script_name + '_ParserInfo' + get_id());if(!info)return { refresh_date: null, last_id: null, parse_date: null, parse_page_count : 0 };info = JSON.parse(info);info.refresh_date = new Date(Date.parse(info.refresh_date));info.parse_date = new Date(Date.parse(info.parse_date));return info;}//----------------------------------------------------------------------------//function remove_data(){if(!confirm('Все данные по этому игроку будут удалены. Вы уверены?'))return;remove_value(script_name + '_ParserInfo' + get_id());remove_value(script_name + '_States' + get_id());reload_page();}//----------------------------------------------------------------------------//function get_last_page(){var url = '/pl_cardlog.php?id=' + get_id() + '&page=999999';var response = send_get(url);var page = /<\/a>\|<b><font color=red>(\d+?)<\/font><\/b><\/center>/gmi.exec(response);return page ? (+page[1] - 1) : 0;}//----------------------------------------------------------------------------//function get_id(){return /.+id=(\d+)/.exec(document.location)[1];}//----------------------------------------------------------------------------//function get_card_state(id){for(var i = 0; i < card_states.length; ++i)if(card_states[i].id == id)return card_states[i];var new_state = { id: id, total: 0, win: 0, loss: 0 };card_states.push(new_state);return new_state;}//----------------------------------------------------------------------------//} catch(e){alert('Ошибка в скрипте ' + script_name + ', обратитесь к разработчику:\n' + e);throw e;}}()); // wrapper end//----------------------------------------------------------------------------//