// ==UserScript==
// @name Turkmaster (Mturk)
// @namespace https://greasyfork.org/users/3408
// @author DonovanM
// @description A page-monitoring web app for Mturk (Mechanical Turk) designed to make turking more efficient. Easily monitor mturk search pages and requesters and Auto-Accept the HITs you missed.
// @include https://www.mturk.com/mturk/*
// @version 1.4.1
// @require https://ajax.googleapis.com/ajax/libs/jquery/1.10.2/jquery.min.js
// @require https://ajax.googleapis.com/ajax/libs/webfont/1/webfont.js
// @grant GM_getValue
// @grant GM_setValue
// ==/UserScript==
var settings = (function() {
var LOCAL_STORAGE = "turkmaster_settings";
var pub = {
sound : true,
animation : true,
preloadHits : false,
volume : 50,
notifications : true,
alertOnly : false,
fontSize : 10,
typeface : "Oxygen",
desktopNotifications : false,
canHide : false,
stopOnCaptcha : true
}
_load();
function _setFontSize(val) {
if (val >= 5 && val <= 20) {
pub.fontSize = val;
$("#dispatcher div").css("font-size", val + "pt");
$(".notification_panel p").css("font-size", val + "pt");
$("#settingsDialog, #settingsDialog div, #settingsDialog li, #settingsDialog input, #settingsDialog button").css("font-size", val + "pt");
_save();
}
}
function _setDesktopNotifications(val, callback) {
if (val) {
requestDesktopNotifications(function(isPermitted) {
callback(isPermitted);
pub.desktopNotifications = isPermitted;
_save();
});
pub.desktopNotifications = false;
} else {
pub.desktopNotifications = false;
}
_save();
}
function _setVolume(val) {
if (val >= 0 && val <= 100) {
Sound.setVolume(val);
pub.volume = val;
_save();
}
}
function _save() {
// localStorage.setItem(LOCAL_STORAGE, JSON.stringify(pub));
GM_setValue(LOCAL_STORAGE, JSON.stringify(pub));
}
function _load() {
var values = GM_getValue(LOCAL_STORAGE);
if (typeof values === 'undefined')
values = localStorage.getItem(LOCAL_STORAGE);
if (values) {
values = JSON.parse(values);
for (i in values)
pub[i] = values[i];
}
}
pub.setFontSize = _setFontSize;
pub.setVolume = _setVolume;
pub.setDesktopNotifications = _setDesktopNotifications;
pub.save = _save;
return pub;
}());
var pageType = {
MAIN : true, // This is so remote watcher requests don't add new watchers to multiple pages and cause mturk errors.
DASHBOARD : false,
HIT : false,
REQUESTER : false,
SEARCH : false
};
var loadError = false;
var wasViewed = false;
var dispatch;
var notificationPanel;
if(!('contains' in String.prototype)) {
String.prototype.contains = function(str, startIndex) {
return -1 !== String.prototype.indexOf.call(this, str, startIndex);
};
}
$(document).ready(function(){
checkPageType();
loadFonts();
if (pageType.DASHBOARD) {
dispatch = new Dispatch();
DispatchUI.create(dispatch);
createDetailsPanel();
IgnoreList.init();
if (settings.preloadHits)
loadDefaultWatchers();
else
dispatch.load();
requestMain();
preloadImages();
addFormStyle();
}
if (pageType.HIT || pageType.REQUESTER || pageType.SEARCH)
addWatchButton();
notificationPanel = new NotificationPanel();
// Listen to messages
window.addEventListener('storage', onStorageEvent, false);
});
$(window).on('beforeunload', function() {
if (pageType.DASHBOARD && pageType.MAIN) {
dispatch.save();
}
});
function loadFonts() {
WebFont.load({
google: { families: [ 'Oxygen:400,700:latin', 'Droid Sans Mono' ] }
});
}
function onStorageEvent(event) {
if (event.key.substring(0,13) === "notifier_msg_")
onMessageReceived(event.key.substring(13), JSON.parse(event.newValue).content);
}
function checkPageType() {
// Dashboard, hit, requester, search
if (document.URL === "https://www.mturk.com/mturk/dashboard")
pageType.DASHBOARD = true;
else if (document.URL.match(/https:\/\/www.mturk.com\/mturk\/(preview|accept).+groupId=.*/) !== null)
pageType.HIT = true;
else if (document.URL.match(/requesterId=([A-Z0-9]+)/) !== null)
pageType.REQUESTER = true;
else if (document.URL.match(/(searchbar|findhits)/) !== null)
pageType.SEARCH = true;
}
function requestMain() {
sendMessage({ header: "request_main" });
}
function preloadImages() {
var images = [
'https://i.imgur.com/guRzYEL.png',
'https://i.imgur.com/5snaSxU.png',
'https://i.imgur.com/VTHXHI4.png',
'https://i.imgur.com/peEhuHZ.png'
];
$(images).each(function(){
$('<img>')[0].src = this;
});
}
var SettingsDialog = function() {
var DOMElement,
widgetGroups = [
{
id : 'soundSettings',
text : 'Sound',
type : 'toggle',
setting : 'sound',
items : [
{
id : 'volume',
text : 'Volume (0 - 100)',
type : 'text',
setting : 'volume',
handler : function (target, value) {
settings.setVolume(value);
}
}
]
},
{
id : 'notificationSettings',
text : 'Notifications',
type : 'toggle',
setting : 'notifications',
items : [
{
id : 'desktopNotifications',
text : 'Desktop Notifications',
type : 'toggle',
setting : 'desktopNotifications',
handler : _handleDesktopNotificationToggle
}
]
},
{
id : 'more',
text : 'Captchas',
items : [
{
id : 'stopOnCaptcha',
text : 'Stop on Captcha',
type : 'toggle',
setting : 'stopOnCaptcha'
}
]
},
{
id : 'fontSettings',
text : 'Font',
items : [
{
id : 'fontSize',
text : 'Size (pt)',
type : 'text',
setting : 'fontSize',
handler : function (target, value) {
settings.setFontSize(value);
}
}
]
},
{
id : 'uiSettings',
text : 'User Interface',
items : [
{
id : 'hideable',
text : 'Hideable',
type : 'toggle',
setting : 'canHide',
handler : function (target, value) {
settings.canHide = value;
setTimeout(function () { DispatchUI.setHide() }, 50);
}
}
]
},
{
id : 'backup',
text : 'Backup',
items : [
{
type : 'more',
id : 'export',
text : 'Export',
handler: _showExport
},
{
type : 'more',
id : 'import',
text : 'Import',
handler: _showImport
}
]
}
];
function _show() {
if (!DOMElement)
_createDOMElement()
_setCurrentSettings(widgetGroups);
DOMElement.show();
$(window).on('click', _handleWindowClick);
}
function _isVisible() {
if (DOMElement)
return DOMElement.is(":visible");
else
return false;
}
// Sets the widgets to reflect the current settings
function _setCurrentSettings(list) {
list.forEach(function (item) {
if (item.type)
_setSetting(item.id, item.type, item.setting);
if (item.items)
_setCurrentSettings(item.items);
})
}
function _setSetting(id, type, setting) {
var widget = DOMElement.find("#" + id),
value = settings[setting];
if (type === "toggle" && value) {
widget.children(".on_off").addClass("on");
} else if (type === "text") {
widget.children("input").val(value);
}
}
function _cancel() {
DOMElement.hide();
}
function _createDOMElement() {
_addStyle();
DOMElement = $('<div id="settingsDialog"><h2>Settings</h2></div>').append(_createGroups(widgetGroups));
_addHandlers();
$("body").append(DOMElement);
}
// Creates settings groups for each group item in groups
function _createGroups(groups) {
return groups.map(function (group) {
return _createGroup(group);
});
}
function _addDefaultHandlers(items) {
items.forEach(function (item) {
if (item.type && !item.handler)
item.handler = _setValue;
if (item.items)
_addDefaultHandlers(item.items);
})
}
function _setValue(target, value) {
settings[this.setting] = value;
}
function _createGroup(params) {
var widgetHTML = widgetHTML || "",
group,
list;
if (params.type === 'toggle')
widgetHTML = getToggleHTML();
group = $('<div id="' + params.id + '">' + widgetHTML + '<h3>' + params.text + '</h3><ul></ul></div>');
list = group.find('ul');
params.items.forEach(function (item) {
list.append(_createItem(item));
});
return group;
}
function _createItem(params) {
return (params.type === 'toggle') ? _createToggleItem(params.id, params.text) :
(params.type === 'text') ? _createTextItem(params.id, params.text) :
(params.type === 'more') ? _createMoreItem(params.id, params.text) :
undefined;
}
function _createTextItem(id, text) {
return $('<li id="' + id + '">' + text + '<input type="text" class="text" /></li>');
}
function _createToggleItem(id, text) {
return $('<li id="' + id + '">' + getToggleHTML() + text + '</li>');
}
function _createMoreItem(id, text) {
return $('<li id="' + id + '"><button class="more">...</button>' + text + '</li>');
}
function getToggleHTML() {
return '<button class="on_off"><span>ON</span><span>OFF</span></button>';
}
function _addHandlers() {
DOMElement.on('click', function(e) {
if (e.target.tagName === "BUTTON" || e.target.parentNode.tagName === "BUTTON")
_handleWidgetClick(e);
});
DOMElement.on('change', _handleInputChange);
_addDefaultHandlers(widgetGroups);
}
function _handleWindowClick(e) {
var target = e.target;
// Hides the settings dialog if user clicks outside of the dialog
if (!DOMElement.is(target) && DOMElement.has(target).length === 0 && $("#settings img").get(0) !== target) {
_cancel();
$(window).off('click', _handleWindowClick);
}
}
function _handleInputChange(e) {
var target = $(e.target),
value = target.val(),
id = target.parent().attr('id');
_setWidget(id, target, value);
}
function _handleWidgetClick(e) {
e.preventDefault();
// Chrome returns the span as the target while FF returns the button
var target = (e.target.tagName === "BUTTON") ? $(e.target) : $(e.target).parent(),
value = target.hasClass("on"),
id = target.parent().attr('id');
// Toggle widget unless it's for desktop notifications (because it's asynchronous)
if (id !== "desktopNotifications")
value = _toggle(target);
_setWidget(id, target, value);
}
function _setWidget(id, target, value) {
_findWidget(widgetGroups, id).handler(target, value);
settings.save();
}
function _toggle(element) {
if (element.hasClass("on")) {
element.removeClass("on");
return false;
} else {
element.addClass("on");
return true;
}
}
// Recursively looks for a widget with specified id
function _findWidget(widgets, id, i) {
return widgets.reduce(function (prev, curr) {
return (prev !== null) ? prev :
(curr.id === id) ? curr :
(curr.items) ? _findWidget(curr.items, id) :
null;
}, null);
}
function _handleDesktopNotificationToggle(target, value) {
if (value)
target.removeClass("on");
// Desktop notification requests require user action so we need a callback
// for when the user responds.
settings.setDesktopNotifications(!value, function(isPermitted) {
if (isPermitted) {
target.addClass("on");
} else {
target.removeClass("on");
console.error("Desktop notifications are blocked.");
}
});
}
function _showExport() {
var div = $('<div id="export-box" class="dialog-big"><h2>Export Watchers</h2><h3>Copy the text below. (Triple-click to highlight all)</h3><p>' + dispatch.exportWatchers() + '</p></h2></div>');
$('<button>Close</button>')
.click(function() { div.remove() })
.appendTo(div);
div.appendTo($("body"));
}
function _showImport() {
var div = $('<div id="import-box" class="dialog-big"><h2>Import Watchers</h2><h3>Paste the backup text to load watchers</h3><textarea></textarea></h2></div>');
$('<button>Save</button>')
.click(function() { dispatch.importWatchers($("#import-box textarea").val()); div.remove() })
.appendTo(div);
$('<button>Close</button>')
.click(function() { div.remove() })
.appendTo(div);
div.appendTo($("body"));
div.find("textarea").focus();
}
function _addStyle() {
addStyle("\
#settingsDialog {\
position: fixed;\
top: 16px;\
left: 249px;\
background-color: #fafafa;\
padding: 10px;\
width: 300px;\
font: " + settings.fontSize + "pt 'Oxygen', verdana, sans-serif;\
border-bottom: 1px solid #DDD;\
border-right: 1px solid #DDD;\
border-radius: 0.3em;\
color: #333;\
}\
#settingsDialog div, #settingsDialog li, #settingsDialog input, #settingsDialog button {\
font: " + settings.fontSize + "pt 'Oxygen', verdana, sans-serif;\
}\
#settingsDialog > div {\
margin: 0px 0px 0.5em;\
border: 1px solid #eee;\
padding: 0.35em 0.60em 0.05em;\
background-color: #fff;\
}\
#settingsDialog h2, #settingsDialog h3 {\
margin: 0 0 0.5em;\
font-size: 95%;\
letter-spacing: 0.5px;\
}\
#settingsDialog h2 {\
font-weight: 400;\
text-align: center;\
font-size: 110%;\
color: #333;\
}\
#settingsDialog button.on_off {\
background: none;\
border: none;\
padding: 0;\
outline: none;\
height: 1.3em;\
margin-top: 0em;\
}\
#settingsDialog .on_off span { color: #333; margin: 1px; font-size: 56%; font-weight: bold; border-radius: 1.6em; }\
#settingsDialog .on_off span:nth-child(2) { background-color: #aeaeae; color: #fff; padding: 0.4em 0.8em; }\
#settingsDialog .on_off.on span:nth-child(1) { background-color: #55b8ea; color: #fff; padding: 0.4em 0.8em; }\
#settingsDialog .on_off.on span:nth-child(2) { background-color: inherit; color: #333; padding: 0 0.8em 0 0; }\
#settingsDialog .on_off { margin-top: 6px; }\
#settingsDialog ul { margin: 0 0 0.2em; padding: 0 0 0 1.9em }\
#settingsDialog ul li { list-style: none; margin-bottom: 0.5em; }\
#settingsDialog li input, #settingsDialog li button { float: right; }\
#settingsDialog li input[type='text'] { width: 3em; font-size: 80%; margin-right: 0.8em; text-align: right; padding-right: 0.5em }\
#settingsDialog li .more { width: 24px; border: none; color: #808080; font: bold 160% inital; line-height: 0%; transform: rotate(90deg); background-color: transparent; position: relative; top: 2px; cursor: pointer; height: 0.7em; padding: 0 0 0.65em; }\
#settingsDialog li#typeface input { width: 8em }\
.dialog-big { position: fixed; top: 2em; left: 50%; width: 860px; margin-left: -430px; background-color: white; padding: 2%; border: 1px solid #ddd; font-family: 'Oxygen'; box-shadow: 3px 3px 3px rgba(0, 0, 0, 0.4); border-radius: 10px; }\
.dialog-big p { background-color: #f7f7f7; padding: 1.5em; height: 500px; overflow: scroll; }\
.dialog-big h2, #export-box h3 { font-weight: 400 }\
.dialog-big h2 { font-size: 170%; margin-top: 0 }\
.dialog-big button { background-color: #cecece; color: white; padding: 3px 10px; font-family: 'Oxygen'; border: none; border-radius: 3px; font-weight: bold; transition: background-color 0.3s; margin-right: 0.5em }\
.dialog-big button:hover { background-color: #55B8EA }\
.dialog-big textarea { display: block; width: 100%; height: 500px; margin: 1em 1em 1em 0; }\
");
}
return {
show: _show,
hide: _cancel,
isVisible: _isVisible
}
}();
function addWatchButton() {
var type = (pageType.HIT) ? 'hit' : (pageType.REQUESTER) ? 'requester' : (pageType.SEARCH) ? 'page' : '';
var button = $("<div>").addClass("watcher_button")
.append($("<a>")
.text("Watch this " + type + "?")
.attr('href', "javascript:void(0)")
.click(addWatcher)
);
function addWatcher() {
// Get current and default values
var time = 60,
auto = true,
alert = false,
name = "",
stopOnCatch = true;
// Find the name if available
if (pageType.REQUESTER) {
if ($(".title_orange_text_bold").length > 0) {
name = $(".title_orange_text_bold").text().match(/Created by '(.+)'/);
name = (typeof name !== 'undefined') ? name[1] : "";
} else if (document.URL.match(/prevRequester=/)) {
name = document.URL.match(/prevRequester=([^&]*)/)[1];
}
} else if (pageType.SEARCH) {
name = document.URL.match(/searchWords=([^&]*)/);
if (name !== null) {
name = name[1].replace('+', ' ');
name = name.charAt(0).toUpperCase() + name.slice(1); // Capitalize first letter
} else {
name = "";
}
} else if (pageType.HIT) {
name = $(".capsulelink_bold > div:nth-child(1)").text().trim();
}
// Pull up a Watcher Dialog with default values set
watcherDialog(
{
name: name,
time: time * 1000,
type: type,
option: {
auto : auto,
alert : alert,
stopOnCatch : stopOnCatch
}
},
function(values) {
var id = (document.URL.match(/groupId=([A-Z0-9]+)/) || document.URL.match(/requesterId=([A-Z0-9]+)/) || [,document.URL])[1],
watcher = {
id : id,
duration : values.time,
type : (type === "page") ? "url" : type,
name : values.name,
auto : values.auto,
alert : values.alert,
stopOnCatch : values.stopOnCatch,
muteBatch : values.muteBatch
};
sendMessage({
header : 'add_watcher',
content : watcher,
timestamp : true
});
}
);
}
var location; // Location to add the watch button
if (pageType.HIT) {
if ($(".message.success h6").length)
location = $(".message.success h6");
else if ($("#javascriptDependentFunctionality").length)
location = $("#javascriptDependentFunctionality");
else if ($("body > form:nth-child(7) > table:nth-child(9) > tbody:nth-child(1) > tr:nth-child(1) > td:nth-child(2) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(2) > td:nth-child(1)").length)
location = $("body > form:nth-child(7) > table:nth-child(9) > tbody:nth-child(1) > tr:nth-child(1) > td:nth-child(2) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(2) > td:nth-child(1)");
} else if (pageType.REQUESTER || pageType.SEARCH) {
if ($(".title_orange_text_bold").length)
location = $(".title_orange_text_bold");
else
location = $(".error_title");
}
location.append(button);
addFormStyle();
}
function addFormStyle() {
addStyle("\
#add_watcher_form {\
position: fixed;\
width: 600px;\
top: 50px;\
left: 50%;\
margin: 50px -300px;\
background-color: #fcfcfe;\
border: 1px solid #aaa;\
border-radius: 1px;\
text-align: center;\
}\
#add_watcher_form h3 {\
font: 12pt Verdana;\
margin: 0 0 15px;\
background-color: #def;\
background-color: rgba(230, 230, 230, 1);\
padding: 3px;\
color: #111;\
}\
#add_watcher_form input[type='text'] {\
font: 10pt Verdana;\
margin: 10px 20px 0 0;\
}\
#add_watcher_form input[type='button'] {\
margin-top: 20px;\
font: 9pt Verdana;\
color: #444;\
background-color: #eee;\
border: 1px solid #999;\
}\
#add_watcher_form input[type='button']:hover {\
background-color: #9df;\
}\
#add_watcher_form p {\
margin: 10px;\
font: 11pt Verdana;\
}\
#add_watcher_form .form_buttons input {\
margin: 5px;\
}\
.watcher_button { display: inline; }\
.watcher_button a {\
text-decoration: none;\
font-weight: normal;\
background-color: #CECECE;\
color: white;\
padding: 3px 10px;\
border-radius: 10px;\
font-family: 'Oxygen', verdana, sans-serif;\
transition: background-color 0.4s;\
}\
.watcher_button a:hover { background-color: #55B8EA }\
.error_title .watcher_button { display: block; margin: 15px }\
");
}
function addStyle(styleText) {
var style = '<style type="text/css">' + styleText + '</style>';
$("head").append(style);
}
function loadDefaultWatchers() {
// Add a few watchers. Won't be done like this in the future
dispatch.isLoading = true;
dispatch.add(new Watcher({
id: "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=survey&minReward=0.25&qualifiedFor=on&x=13&y=10",
time: 60000,
type: 'url',
name: "Surveys $0.25 and up"})); //$.25 surveys
dispatch.add(new Watcher({
id: "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=survey&minReward=0.75&qualifiedFor=on&x=13&y=10",
time: 60000,
type: 'url',
name: "Surveys $0.75 and up"})); //$.75 surveys
dispatch.add(new Watcher({
id: "A11L036EBWKONR",
time: 120000,
type: 'requester',
name: "Project Endor",
option: {alert:true}})); // Endor
dispatch.add(new Watcher({
id: "A6YG5FKV2TAVC",
time: 300000,
type: 'requester',
name: "Agent Agent",
option: {alert:true}})); // Agent Agent
dispatch.add(new Watcher({
id: "AKEBQYX32KM19",
time: 120000,
type: 'requester',
name: 'Crowdsurf Support'}));
dispatch.add(new Watcher({
id: "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=transcri&minReward=0.00&qualifiedFor=on&x=0&y=0",
time: 60000,
type: 'url',
name: "Transcription HITs"})); // Transcription HITs
dispatch.isLoading = false;
dispatch.save();
}
function onMessageReceived(header, message) {
if (pageType.DASHBOARD && pageType.MAIN) {
switch(header) {
case 'notification_viewed' :
wasViewed = true;
break;
case 'add_watcher' :
var msg = message;
dispatch.add(new Watcher({
id : msg.id,
time : msg.duration,
type : msg.type,
name : msg.name,
option : {
auto: msg.auto,
stopOnCatch: msg.stopOnCatch,
alert: msg.alert
}
})).start();
break;
case 'ignore_requester' :
IgnoreList.add(IgnoreList.REQUESTER, message.id);
break;
case 'mute_hit' :
var id = message.id;
IgnoreList.add(IgnoreList.HIT, id);
break;
case 'unmute_hit' :
var id = message.id;
IgnoreList.remove(IgnoreList.HIT, id);
break;
case 'request_main' :
sendMessage({ header: "request_denied" });
break;
case 'request_denied' :
dispatch.onRequestMainDenied();
break;
case 'show_main' :
alert("Showing the main dashboard. (Close this Mturk page to establish a notifier in a different tab or window)");
break;
}
} else if (!pageType.DASHBOARD || (pageType.DASHBOARD && !pageType.MAIN)) {
switch(header) {
case 'new_hits' :
var hits = message.hits;
// Re-create the hits so their methods can be used
for(var i = hits.length; i--;) hits[i] = new Hit(hits[i]);
// Show the hits and let the dashboard know it was seen
if (document.hasFocus())
sendMessage({ header: "notification_viewed" });
notificationPanel.add(new NotificationGroup({ title: message.title, hits: hits, url: message.url }));
break;
case 'captcha' :
if (document.hasFocus())
alert("Captcha Alert!");
break;
case 'turkopticon' :
// This needs a more elegant solution. If the servers start lagging we might be
// using addTO() for the wrong group. It won't show the TO for the wrong requester,
// though, so it's safe to use for now. It's just that some ratings could be missing.
notificationPanel.notifications[notificationPanel.notifications.length - 1].addTO(message);
break;
}
}
}
function sendMessage(message) {
var header = message.header;
var content = message.content || new Date().getTime(); // Make the content a timestamp when there's no actual content
var timestamp = message.timestamp && new Date().getTime(); // If wanted, adds a timestamp to the content so messages with the same content will still trigger the event consecutively
localStorage.setItem('notifier_msg_' + header, JSON.stringify({ content: content, timestamp: timestamp}));
}
function sendDesktopNotification(hits, watcher) {
// Let's check if the user is okay to get some notification
if (Notification.permission === "granted" && settings.desktopNotifications) {
// If the user isn't on a mturk page to receive a rich notification, then send a web notification
if (!wasViewed) {
var bodyText = "";
for (var i = 0, len = hits.length; i < len; i++)
bodyText += "\n" + hits[i].title.substring(0, 40) + ((hits[i].title.length > 40) ? "..." : "") + "\n" + hits[i].reward + "\n";
var notification = new Notification(
watcher.name,
{
body: bodyText,
icon: "http://halfelf.org/wp-content/uploads/sites/2/2012/06/amazon_icon.png"
}
);
notification.onclick = function() {
window.focus(); // Focus this window (dashboard)
this.close(); // Closes the notification
showDetailsPanel(watcher); // Opens the details panel for whatever watcher the notification was for
};
notification.onshow = function() { setTimeout(function() { notification.close() }, 5000) }; // Need to set a close time for Chrome
}
}
}
function requestDesktopNotifications(callback) {
// Let's check if the browser supports notifications
if (!("Notification" in window)) {
alert("This browser does not support desktop notification");
} else {
window.Notification.requestPermission(function (permission) {
// Whatever the user answers, we make sure Chrome stores the information
if(!('permission' in Notification))
window.Notification.permission = permission;
// If the user is okay, let's create a notification
if (permission === "granted") {
var notification = new window.Notification("Desktop notifications enabled.");
notification.onshow = function() { setTimeout(function() { notification.close() }, 5000) };
callback(true);
} else {
callback(false);
}
});
}
}
function Hit(attrs) {
attrs = attrs || {};
this.id = attrs.id;
this.uid = attrs.uid;
this.isAutoAccept = attrs.isAutoAccept || false;
this.requester = attrs.requester;
this.requesterID = attrs.requesterID;
this.url = attrs.url;
this.title = attrs.title;
this.reward = attrs.reward;
this.description = attrs.description;
this.available = attrs.available;
this.time = attrs.time;
this.isQualified = (typeof attrs.isQualified !== 'undefined') ? attrs.isQualified : true;
this.canPreview = (typeof attrs.canPreview !== 'undefined') ? attrs.canPreview : true;
}
Hit.prototype.getURL = function(type) {
switch(type) {
case 'preview':
return "https://www.mturk.com/mturk/preview?groupId=" + this.id;
case 'accept' :
return (this.isQualified) ? "https://www.mturk.com/mturk/previewandaccept?groupId=" + this.id : null;
case 'auto' :
return "https://www.mturk.com/mturk/previewandaccept?groupId=" + this.id + "&autoAcceptEnabled=true";
case 'view' :
return "https://www.mturk.com/mturk/continue?hitId=" + this.uid;
case 'return' :
// This will need to be changed. It's the same as 'view' until more testing is done on AMT's return functionality
return "https://www.mturk.com/mturk/preview?hitId=" + this.uid;
default:
return "";
}
};
// Returns the position of a hit in a hit array by its ID
Hit.indexOf = function(hitId, hits) {
for (var i = 0, len = hits.length; i < len; i++) {
if (hitId === hits[i].id)
return i;
}
return -1;
};
// Returns true if there are multiple hits in the array and all of the hits are from the same requester
Hit.isSameRequester = function(hits) {
if (hits.length > 1) {
var compareRequester = hits[0].requester;
for (var i = 1, len = hits.length; i < len; i++) {
if (compareRequester !== hits[i].requester)
return false;
}
return true;
} else {
return false;
}
};
// Returns a list of unique requester IDs from an array of hits
Hit.getUniqueReqeusters = function(hits) {
var ids = [];
for (var i = 0, len = hits.length; i < len; ++i) {
var id = hits[i].requesterID;
if (ids.indexOf(id) === -1)
ids.push(id)
}
return ids;
}
// Message object (Not used)
function Message() {
/* Status (changed): Unchanged, Added, Removed, Count
We should mark each Hit in the message with what has changed. The count change should be sent with this.
The message will also tell the client whether or not to pop-up the notification. */
}
// The details panel for each watcher
function createDetailsPanel() {
var div = $('<div>').attr('id', 'details_panel').addClass('notification_panel');
addStyle("#details_panel {\
background-color: #fff;\
position: absolute;\
top: 0px;\
margin-left: 1px;\
left: 270px;\
width: 500px;\
border: 1px solid #e3e3e3;\
border-radius: 0 0 3px 0;\
border-width: 0 1px 1px 0;\
transition: left 0.5s ease;\
display: none }\
#details_panel h4 { display: none }\
#details_panel.left { left: 30px }");
$(div).mouseleave(function() { $(this).hide() });
$("body").append(div);
}
var lastWatcher = "";
function showDetailsPanel(watcher) {
var panel = $("#details_panel");
var group;
// Only change the panel contents if it's a different watcher or the same one, but updated
if (watcher !== lastWatcher || (watcher === lastWatcher && watcher.isUpdated)) {
$("*", panel).remove();
if (watcher.lastHits.length > 0) {
group = new NotificationGroup({ hits: watcher.lastHits, isSticky: false, watcher: watcher });
$(panel).append((group).getDOMElement());
// This doesn't need a callback since the data will already be cached at this point
group.addTO(TO.get(Hit.getUniqueReqeusters(watcher.lastHits), _handleTOReceived));
} else {
$(panel).append($('<div>').append('<h2>').css('text-align', 'center').html("<br />There are no HITs available.<br /><br />"));
}
}
$(panel).show();
function _handleTOReceived(data) {
group.addTO(data);
}
lastWatcher = watcher;
}
var IgnoreList = (function() {
var _time = 60000,
_hits = [];
_requesters = [],
_HIT = 0,
_REQUESTER = 1;
function _init() {
// _clear();
_load();
_addListeners();
}
function _addListeners() {
$(window).on('unload', function() { _save(); })
}
function _save() {
localStorage.setItem('notifier_ignore', JSON.stringify(_hits));
localStorage.setItem('notifier_ignore_requesters', JSON.stringify(_requesters));
}
function _load() {
var storedHits = localStorage.getItem('notifier_ignore');
var storedRequesters = localStorage.getItem('notifier_ignore_requesters');
if (storedHits !== null) {
try {
_hits = JSON.parse(storedHits);
} catch (e) {
_clear();
_save();
console.error("Ignored hits couldn't be loaded correctly.");
}
} else {
console.log("No ignored hits found");
}
if (storedRequesters !== null) {
try {
_requesters = JSON.parse(storedRequesters);
} catch (e) {
_clear();
_save();
console.error("Ignored requesters couldn't be loaded correctly.");
}
} else {
console.log("No ignored requesters found");
}
}
function _clear() {
_hits = [];
_requesters = [];
localStorage.removeItem('notifier_ignore');
localStorage.removeItem('notifier_ignore_requesters');
}
function _contains(type, item) {
if (type === _HIT)
return (_hits.indexOf(item) !== -1);
else
return (_requesters.indexOf(item) !== -1);
}
function _isIgnored(requester) {
return (_requesters.indexOf(requester) !== -1);
}
function _isMuted(item) {
return (_hits.indexOf(item) !== -1);
}
function _filter(hits) {
var filteredHits = [];
for (var i = 0, len = hits.length; i < len; i++) {
var hit = hits[i];
if ((_hits.indexOf(hit.id) === -1) && (_requesters.indexOf(hit.requester) === -1))
filteredHits.push(hit);
}
return filteredHits;
}
function _add(type, id) {
if (type === _HIT) {
if (_hits.indexOf(id) === -1)
_hits.push(id);
} else if (type === _REQUESTER) {
if (_requesters.indexOf(id) === -1)
_requesters.push(id);
}
_save();
}
function _remove(type, id) {
if (type === _HIT) {
var index = _hits.indexOf(id);
if (index !== -1)
_hits.splice(index, 1);
} else if (type === _REQUESTER) {
var index = _requesters.indexOf(id);
if (index !== -1)
_requesters.splice(index, 1);
}
_save();
}
return {
init: _init,
add: _add,
remove: _remove,
filter: _filter,
isMuted: _isMuted,
isIgnored: _isIgnored,
HIT: _HIT,
REQUESTER: _REQUESTER
}
})();
function Evt() { /* Nothing */ };
Evt.ADD = 1;
Evt.REMOVE = 2;
Evt.START = 3;
Evt.STOP = 4;
Evt.CHANGE = 5;
Evt.UPDATE = 6;
Evt.HITS_CHANGE = 7;
Evt.DELETE = 8;
Evt.VIEW_DETAILS = 9;
Evt.prototype.addListener = function(type, callback) {
switch(type) {
case Evt.ADD:
this.listener.onadd.push(callback);
break;
case Evt.REMOVE:
this.listener.onremove.push(callback);
break;
case Evt.START:
this.listener.onstart.push(callback);
break;
case Evt.STOP:
this.listener.onstop.push(callback);
break;
case Evt.CHANGE:
this.listener.onchange.push(callback);
break;
case Evt.UPDATE:
this.listener.onupdate.push(callback);
break;
case Evt.HITS_CHANGE:
this.listener.onhitschange.push(callback);
break;
case Evt.DELETE:
this.listener.ondelete.push(callback);
break;
case Evt.VIEW_DETAILS:
this.listener.onviewdetails.push(callback);
break;
default:
console.error("Invalid Event type in addListener()");
}
}
Evt.prototype.notify = function(type, data) {
switch(type) {
case Evt.ADD:
this.callFunctionArray(this.listener.onadd, data);
break;
case Evt.REMOVE:
this.callFunctionArray(this.listener.onremove, data);
break;
case Evt.START:
this.callFunctionArray(this.listener.onstart, data);
break;
case Evt.STOP:
this.callFunctionArray(this.listener.onstop, data);
break;
case Evt.CHANGE:
this.callFunctionArray(this.listener.onchange, data);
break;
case Evt.UPDATE:
this.callFunctionArray(this.listener.onupdate, data);
break;
case Evt.HITS_CHANGE:
this.callFunctionArray(this.listener.onhitschange, data);
break;
case Evt.DELETE:
this.callFunctionArray(this.listener.ondelete, data);
break;
case Evt.VIEW_DETAILS:
this.callFunctionArray(this.listener.onviewdetails, data);
break;
default:
console.error("Unknown event type:", type);
}
}
Evt.prototype.callFunctionArray = function(functions, data) {
if (functions.length > 0)
for (var i = 0, len = functions.length; i < len; i++)
functions[i](data);
}
var DispatchUI = {
create: function(dispatch) {
DispatchUI.dispatch = dispatch;
DispatchUI.init();
DispatchUI.addStyle();
DispatchUI.addActions();
DispatchUI.addListeners();
DispatchUI.addDragAndDrop();
return DispatchUI.div;
},
init: function() {
var div = DispatchUI.div = $('<div id="dispatcher"><div id="controller"></div><div id="watcher_container"></div></div>');
DispatchUI.watchers = [];
// Move dashboard contents to the right and put the dispatch panel on the left
var pageElements = $("body > *");
$("body").html("");
$("body").append(
$("<div>")
.attr('id', "content_container")
.append($(pageElements))
);
$("body").css('margin', "0").prepend(div);
var ctrl = DispatchUI.ctrl = $("#controller", div);
var settingsBtn = $('<a id="settings" href="javascript:void(0)" title="Setting"><img /></a')
.click(function() {
if (!SettingsDialog.isVisible())
SettingsDialog.show();
else
SettingsDialog.hide();
});
ctrl.append(
settingsBtn,
'<div class="play_container">\
<div class="play_all" title="Start All"></div>\
<div class="play selected" title="Start Selected"></div>\
<div class="pause" title="Pause All"></div>\
</div>',
"Turkmaster"
);
// Adding the data URL inline wouldn't work for some reason, so I'm doing it this way.
// Image from http://latierrasenosestrecha.org/wp-content/themes/purity/img/icons/settings.png
$("img", settingsBtn)[0].src = "";
},
addActions: function() {
var dispatch = DispatchUI.dispatch,
ctrl = DispatchUI.div.find("#controller");
$("div.play_all", ctrl).mousedown(function() {
dispatch.start(true);
});
$("div.play", ctrl).mousedown(function() {
dispatch.start();
});
$("div.pause", ctrl).mousedown(function() {
dispatch.stop();
});
},
addListeners: function() {
var dispatch = DispatchUI.dispatch;
var div = DispatchUI.div;
dispatch.addListener(Evt.ADD, function(watcher) {
// This could be done on one line, but then we would lose access to the WatcherUI's internal Watcher object and functionality
var watcher = WatcherUI.create(watcher);
$("#watcher_container", DispatchUI.div).append(watcher.element);
DispatchUI.watchers.push(watcher);
// watchers.push(WatcherUI.create(watcher).appendTo($("#watcher_container", div)));
});
dispatch.addListener(Evt.REMOVE, function(watcher) {
// Remove watcher from array
var index = -1,
watchers = DispatchUI.watchers;
for (var i = 0, len = watchers.length; i < len; i++) {
if (watchers[i].watcher === watcher) {
index = i;
break;
}
}
if (index !== -1)
DispatchUI.watchers.splice(index, 1);
});
DispatchUI.setHide = function() {
if (settings.canHide) {
$(window).on('click', handleWindowClick);
} else {
$(window).off('click', handleWindowClick);
}
}
DispatchUI.setHide();
function handleWindowClick(e) {
if (!div.is(e.target) && div.has(e.target).length === 0 && $(".notification_panel").has(e.target).length === 0 && !$("#settingsDialog").is(e.target) && $("#settingsDialog").has(e.target).length === 0) {
hide();
$(window).off('click', handleWindowClick);
$(div).on('click', handleDivClick);
}
}
function handleDivClick(e) {
show();
$(div).off('click', handleDivClick);
if (settings.canHide)
$(window).on('click', handleWindowClick);
}
function hide() {
div.addClass("tm-hidden");
$("#content_container").addClass("full");
$("#details_panel").addClass("left");
}
function show() {
div.removeClass("tm-hidden");
$("#content_container").removeClass("full");
$("#details_panel").removeClass("left");
}
},
addStyle: function() {
addStyle("#dispatcher { background-color: #f5f5f5; position: fixed; top: 0px; float: left; left: 0; height: 100%; width: 270px; font-size: 8pt; margin-left: 0px; transition: left 0.5s ease; }\
#dispatcher.tm-hidden { left: -240px }\
#content_container { position: absolute; left: 270px; top: 0; right: 0; border-left: 2px solid #dadada; transition: left 0.5s ease; }\
#content_container.full { left: 30px }\
#dispatcher #controller { text-align: center; font: 160% Candara, sans-serif; color: #585858; position: relative; padding: 3px 5px; }\
#dispatcher #controller .on_off { margin: 6px 5px 0 0 }\
#dispatcher #controller .on_off a { font-size: 80% }\
#dispatcher #controller #settings { top: 2px; position: absolute; right: 5px; }\
#dispatcher #controller #settings img { width: 1.5em }\
#dispatcher #controller .play_container { position: absolute; left: 5px }\
#dispatcher #watcher_container { position: absolute; top: 30px; bottom: 0; overflow-y:auto; width: 100%;}\
#dispatcher #watcher_container p { margin: 30px 0px }\
#dispatcher #watcher_container .error_button a { text-decoration: none; color: #555; background-color: #fff; padding: 3px 10px; margin: 5px; border: 1px solid #aaa; border-radius: 2px }\
#dispatcher #watcher_container .error_button a:hover { background-color: #def; border-color: #aaa }\
#dispatcher div { font-size: 7pt }\
#dispatcher .watcher {\
box-sizing: border-box;\
margin: 3px 3px 0;\
background-color: #fff;\
position: relative;\
border-bottom: 1px solid #ddd;\
border-right: 1px solid #ddd;\
top: 0;\
transition: background-color 0.5s, top 0.1s;\
-moz-user-select: none;\
-webkit-touch-callout: none;\
-webkit-user-select: none;\
-khtml-user-select: none;\
}\
#dispatcher .watcher.dragging { cursor: grabbing; z-index: 100; opacity: 0.8; transition: background-color 0.5s, top 0s }\
#dispatcher .watcher div { font: " + settings.fontSize + "pt 'Oxygen', verdana, sans-serif }\
#dispatcher .watcher.running .details { background-color: #BADFF2 }\
#dispatcher .watcher.updated { background-color: rgba(218, 240, 251, 1); }\
#dispatcher .watcher .details { width: 25px; text-align: center; float: right; background-color: rgba(234, 234, 234, 1); position: absolute; top: 0; bottom: 0; right: 0; font-size: 90%; color: #fff; transition: background-color 0.5s }\
#dispatcher .watcher .details.updated { background-color: rgba(218, 240, 251, 1); background-color: #F0FF60; }\
#dispatcher .watcher .name { font-size 130%; color: black; text-decoration: none; display: inline-block; margin-top: -3px}\
#dispatcher .watcher .name:hover { text-decoration: underline }\
#dispatcher .watcher.dragging .name:hover { text-decoration: none }\
#dispatcher .watcher .time { display: block; float: left; font-size: 80% }\
.on_off { float: right; cursor: pointer }\
.on_off a { color: #333; margin: 1px; font-size: 56%; font-weight: bold }\
.on_off a:nth-child(2) { background-color: #aeaeae; color: #fff; border-radius: 12px; padding: 3px 6px; }\
.on_off.on a:nth-child(1) { background-color: #55b8ea; color: #fff; border-radius: 12px; padding: 3px 6px; }\
.on_off.on a:nth-child(2) { background-color: inherit; color: #333; border-radius: inherit; padding: inherit; }\
.watcher .on_off { }\
#dispatcher .watcher > .content { margin-right: 25px; padding: 5px 5px 5px 33px;}\
#dispatcher .watcher .bottom { margin: 0 0 -5px; color: #aaa }\
#dispatcher .watcher .bottom a:link { color: black; }\
#dispatcher .watcher .bottom a:hover { color: #cef; }\
#dispatcher .watcher .details { font-size: 150%; font-weight: bold }\
#dispatcher .watcher .last_updated { position: absolute; right: 30px; bottom: 4px; font-size: 80% }\
#dispatcher .watcher .icons { visibility: hidden; margin-left: 10px; bottom: 5px }\
#dispatcher .watcher:hover .icons { visibility: visible }\
#dispatcher .watcher .icons img { opacity: 0.2; height: 0.9em }\
#dispatcher .watcher .icons img:hover { opacity: 1 }\
#dispatcher .watcher .color_code { position: absolute; left: 0; top: 0; bottom: 0; width: 9px; cursor: grab;}\
#dispatcher .watcher .color_code div { position: absolute; left: 0; top: 0; bottom: 0; width: 5px; transition: width 0.15s; }\
#dispatcher .watcher.dragging .color_code { cursor: grabbing; }\
#dispatcher .watcher .color_code:hover div { width: 9px }\
#dispatcher .watcher .color_code.hit div { background-color: rgba(234, 111, 111, .7); }\
#dispatcher .watcher .color_code.requester div { background-color: rgba(51, 147, 255, .7); }\
#dispatcher .watcher .color_code.url div { background-color: rgba(57, 221, 122, .7); }\
.watcher .play_container {\
padding: 0px 0px 0px 12px;\
float: left;\
cursor: default;\
}\
.play, .pause, .play_all {\
width:20px;\
height: 20px;\
position: relative;\
display: block;\
}\
#controller .play, #controller .pause, #controller .play_all { float: left; }\
.play:before, .play_all:before {\
width: 0;\
height: 0;\
border-width: 8px 11px;\
border-style: solid;\
border-color: transparent transparent transparent #747474;\
position: absolute;\
content: '';\
top: 3px;\
left: 0px;\
}\
.play.selected:after {\
width: 6px;\
height: 6px;\
border-radius: 1px;\
position: absolute;\
content: '';\
background-color: #999;\
top: 13px;\
right: 9px;\
}\
.play_all:after {\
width: 0;\
height: 0;\
border-width: 8px 11px;\
border-style: solid;\
border-color: transparent transparent transparent #999;\
position: absolute;\
content: '';\
top: 3px;\
left: 5px;\
}\
.watcher.running .play:before, .watcher.running .play:after, .pause:before, .pause:after {\
width: 4px;\
height: 15px;\
background: #747474;\
position: absolute;\
content: '';\
top: 3px;\
}\
.watcher.running .play:before, .pause:before {\
left: 0px;\
border: none;\
}\
.watcher.running .play:after, .pause:after {\
left: 6px;\
}\
.play_select {\
width: 6px;\
height: 6px;\
border: 2px solid #cecece;\
border-radius: 2px;\
margin-top: 2px;\
}\
.watcher.selected .play_select { background-color: #55b8ea; border-color: #b4e6ff; }\
");
},
addDragAndDrop: function() {
// Drag watchers
var startY, currentBaseY, max, min, height,
dragDiv, nextDiv, prevDiv, startPos, endPos, isDragging,
slop = 7, currentWatcher, watchers = DispatchUI.watchers;
DispatchUI.div.on("mousedown", ".watcher", function(e) {
isDragging = false;
// Get the position of the watcher in the listing
startPos = endPos = $("#watcher_container .watcher").index(e.currentTarget);
// Get reference to the selected watcher
currentWatcher = watchers[startPos];
dragDiv = currentWatcher.element;
nextDiv = dragDiv.next();
prevDiv = dragDiv.prev();
// TODO Check target to prevent dragging from a component inside the watcher (i.e. buttons, links, etc.)
height = dragDiv.outerHeight(true);
startY = e.clientY;
currentBaseY = dragDiv.offset().top;
// max = Math.min($("#watcher_container").outerHeight(true), height * (DispatchUI.dispatch.watchers.length + .75)) - height;
min = watchers[0].element.offset().top;
max = Math.min($("#watcher_container").outerHeight(true), height * (watchers.length - 1) + watchers[0].element.offset().top);
$(window).on("mousemove", move);
$(window).on("mouseup", up);
});
function move(e) {
var offsetY = e.clientY - startY;
if (!isDragging && (Math.abs(offsetY) > slop)) {
// Start dragging
isDragging = true;
dragDiv.addClass("dragging");
}
if (isDragging) {
dragDiv.css('top', offsetY);
var diffY = dragDiv.offset().top - currentBaseY;
if (dragDiv.offset().top > max) {
dragDiv.offset({ top: max });
diffY = 0;
} else if (dragDiv.offset().top < min) {
dragDiv.offset({ top: min });
diffY = 0;
}
if (diffY > height / 2) {
// Move down one spot
nextDiv.offset({ 'top': nextDiv.offset().top - height });
nextDiv = nextDiv.nextAll(":not(.dragging)").first();
prevDiv = (prevDiv.length) ? prevDiv.nextAll(":not(.dragging)").first() : (dragDiv !== watchers[0].element) ? watchers[0].element : watchers[1].element;
currentBaseY += height;
endPos++;
} else if (-diffY > height / 2) {
// Move up one spot
prevDiv.offset({ 'top': prevDiv.offset().top + height });
prevDiv = prevDiv.prevAll(":not(.dragging)").first();
nextDiv = (nextDiv.length) ? nextDiv.prevAll(":not(.dragging)").first() : (dragDiv !== watchers[watchers.length - 1].element) ? watchers[watchers.length - 1].element : watchers[watchers.length - 2].element;
currentBaseY -= height;
endPos--;
}
}
}
function up(e) {
$(window).off("mousemove", move);
$(window).off("mouseup", up);
if (isDragging) {
e.preventDefault();
dragDiv.removeClass("dragging");
isDragging = false;
// $("div", colorCode).css('width', '');
dragDiv.css('cursor', '');
dragDiv.css('z-index', '');
dragDiv.css('opacity', '');
$(".name", dragDiv).removeClass("no_hover");
// Reset all watcher offsets
$("#watcher_container .watcher").css('transition', "background-color 0.5s, top 0s");
$("#watcher_container .watcher").css('top', '');
setTimeout(function() { $("#watcher_container .watcher").css('transition', ''); }, 600);
if (startPos !== endPos) {
if (endPos > startPos)
dragDiv.insertAfter($("#watcher_container .watcher")[endPos]);
else
dragDiv.insertBefore($("#watcher_container .watcher")[endPos]);
DispatchUI.dispatch.moveWatcher(startPos, endPos);
// Re-arrange our watchers array
watchers.splice(startPos, 1);
watchers.splice(endPos, 0, currentWatcher);
}
}
}
}
}
/** Dispatch object. Controls all of the watchers.
**/
function Dispatch() {
this.watchers = new Array();
this.isLoading = false;
// Listeners
this.listener = {
onadd: [],
onremove: [],
onstart: [],
onstop: [],
onchange: []
};
}
Dispatch.prototype = new Evt();
Dispatch.prototype.start = function(startAll) {
if (this.watchers.length > 0) {
var count = 0;
for (var i = 0, len = this.watchers.length; i < len; i++) {
// Don't start them all at the same time. There is a 2 second delay
// between each start. It had to be done in a self-executing function
// in order for the setTimeout to work properly.
if (this.watchers[i].state.isSelected || startAll) {
(function (watcher, x){
watcher.timer = setTimeout(function() { watcher.start(); }, x * 0000); // Let's try 0ms
})(this.watchers[i], count++);
}
}
}
this.notify(Evt.START, null);
}
Dispatch.prototype.stop = function() {
// Stop all Watchers
if (this.watchers.length > 0) {
for (var i = 0, len = this.watchers.length; i < len; i++)
this.watchers[i].stop();
}
this.interruptStart = true;
this.notify(Evt.STOP, null)
}
Dispatch.prototype.add = function(watcher) {
var self = this;
watcher.addListener(Evt.CHANGE, function() {
self.save();
})
this.watchers.push(watcher);
if (!this.isLoading) {
this.save();
}
this.notify(Evt.ADD, watcher);
// TODO Add a listener to save the watcher list after a watcher has been changed
return watcher;
}
Dispatch.prototype.save = function() {
if (!loadError) {
// localStorage.setItem('notifier_watchers', JSON.stringify(dispatch.watchers, Watcher.replacerArray));
GM_setValue('notifier_watchers', JSON.stringify(dispatch.watchers, Watcher.replacerArray));
var lastChecked = getLastChecked(dispatch.watchers);
if (lastChecked > 0)
localStorage.setItem('notifier_watchers_lastChecked', JSON.stringify(lastChecked));
}
function getLastChecked(watchers) {
var lastChecked = (watchers[0].date) ? watchers[0].date.getTime() : 0;
for (var i = 1, len = watchers.length; i < len; i++) {
if (watchers[i].isRunning)
return new Date.getTime();
if ((watchers[i].date) && (watchers[i].date.getTime() > lastChecked))
lastChecked = watchers[i].date.getTime();
}
return lastChecked;
}
}
Dispatch.prototype.load = function() {
this.isLoading = true;
var data;
var watchers,
lastChecked = localStorage.getItem('notifier_watchers_lastChecked');
data = GM_getValue('notifier_watchers');
if (typeof data === 'undefined')
data = localStorage.getItem('notifier_watchers');
if (data !== null) {
try {
watchers = JSON.parse(data);
try {
lastChecked = JSON.parse(lastChecked);
} catch(e) {
lastChecked = null;
}
var now = new Date().getTime(),
expTime = 180000, // 3 minutes
expired = (lastChecked !== null) ? now - lastChecked > expTime : false; // Expired if most recent watcher update happened more than x minutes before page was loaded
// Add the watchers. Clear last hits if past the expiration time
for(var i = 0; i < watchers.length; i++) {
if (expired)
watchers[i].lastHits = [];
this.add(new Watcher(watchers[i]));
}
} catch(e) {
loadError = true;
console.error("Error loading saved list", e);
}
} else {
loadDefaultWatchers();
}
this.isLoading = false;
}
Dispatch.prototype.remove = function(watcher) {
var index = this.watchers.indexOf(watcher);
if (index !== -1)
this.watchers.splice(index, 1);
watcher.delete();
this.save();
this.notify(Evt.REMOVE, watcher);
}
Dispatch.prototype.moveWatcher = function(from, to) {
if ((to >= 0 && to < this.watchers.length) && (from >= 0 && from < this.watchers.length)) {
var watcher = this.watchers.splice(from, 1);
this.watchers.splice(to, 0, watcher[0]);
this.save();
}
}
Dispatch.prototype.getWatcherByProperty = function(name, value) {
if (this.watchers.length > 0) {
for (var i = 0, len = this.watchers.length; i < len; i++) {
if (this.watchers[i][name] === value)
return this.watchers[i];
}
}
return null;
}
Dispatch.prototype.getWatcherIndex = function(watcher) {
return this.watchers.indexOf(watcher);
}
Dispatch.prototype.getWatcher = function(index) {
return this.watchers[index];
}
Dispatch.prototype.getWatcherCount = function() {
return this.watchers.length;
}
Dispatch.prototype.hideWatchers = function() {
$("#controller a").css('display', "none");
$("#watcher_container").html("");
$("#watcher_container")
.css('background-color', "#f9f9f9")
.css('color', "#ff6b6b")
.css('text-align', "center").append(
$("<p>").text("There is already a notifier running on a different page."),
$("<p>").addClass("error_button").append(
$("<a>")
.html("Close")
.attr('href', "javascript:void(0)")
.click(function() {
$("#dispatcher").css('display', "none");
$("#content_container").css('left', "0px");
}),
$("<a>")
.html("Show")
.attr('href', "javascript:void(0)")
.click(function() {
sendMessage({ header: 'show_main' });
})
));
}
Dispatch.prototype.onRequestMainDenied = function() {
pageType.MAIN = false;
this.hideWatchers();
}
Dispatch.prototype.exportWatchers = function() {
var watcherAttrs = ["id", "time", "type", "name", "option", "auto", "alert", "stopOnCatch", "state", "isSelected", "url"];
return JSON.stringify(this.watchers, watcherAttrs);
}
Dispatch.prototype.importWatchers = function(data) {
try {
data = JSON.parse(data);
dispatch.isLoading = true;
for (var i = 0, len = data.length; i < len; i++) {
var watcher = new Watcher(data[i]);
if (!this.getWatcherByProperty('id', watcher.id) && !this.getWatcherByProperty('name', watcher.name))
this.add(watcher);
}
dispatch.isLoading = false;
dispatch.save();
console.log("Watchers imported", dispatch.watchers);
} catch(e) {
console.error("Error importing watchers", e, data);
alert("Invalid input. Try disabling word wrap on your text editor and re-copy.")
}
}
function watcherDialog(watcher, callback) {
var dialog = $("<div>").attr('id', 'add_watcher_form').append(
$("<h3>").text("Add a watcher"),
$("<p>").append(
$("<label>").text("Name ").append(
$("<input>").attr('id', "watcherName").attr('type', "text").val(watcher.name)),
$("<label>").text(" Time ").append(
$("<input>").attr('id', "watcherDuration").attr('type', "text").val(watcher.time / 1000))
),
(watcher.type === "hit") ?
$("<p>").append(
$("<input>").attr('type', "checkbox").attr('id', "autoaccept").prop('checked', watcher.option.auto),
$("<label>").attr('for', "autoaccept").text("Auto-accept")
)
: "",
(watcher.type === "hit") ?
$("<p>").append(
$("<input>").attr('type', "checkbox").attr('id', "stopaccept").prop('checked', watcher.option.stopOnCatch),
$("<label>").attr('for', "stopaccept").text("Stop on accept")
)
: "",
$("<p>").append(
$("<input>").attr('type', "checkbox").attr('id', "alert").prop('checked', watcher.option.alert),
$("<label>").attr('for', "alert").text("Alert")
),
(watcher.type === "hit") ?
$("<p>").append(
$("<input>").attr('type', "checkbox").attr('id', "mutebatch").prop('checked', watcher.option.muteBatch),
$("<label>").attr('for', "mutebatch").text("Silence batches after first seen")
)
: "",
$("<p>").addClass("form_buttons").append(
$("<input>").attr('type', "button").attr('value', "Save"),
$("<input>").attr('type', "button").attr('value', "Cancel")
)
);
function save() {
callback({
name : $("#watcherName", dialog).val(),
time : parseInt($("#watcherDuration", dialog).val(), 10) * 1000,
alert : $("#alert", dialog).prop('checked'),
auto : $("#autoaccept", dialog).prop('checked'),
stopOnCatch : $("#stopaccept", dialog).prop('checked'),
muteBatch : $("#mutebatch", dialog).prop('checked')
})
hide();
};
function hide() {
dialog.hide();
dialog.remove();
dialog.empty();
}
$("input[value='Save']", dialog).click(save);
$("input[type='button']", dialog).click(hide);
$(dialog).keydown(function(e) {
switch(e.keyCode) {
case 13: // Enter
save();
break;
case 27: // Esc
hide();
break;
}
});
$("body").append(dialog);
if ($("#watcherName", dialog).val() === "")
$("#watcherName", dialog).focus().select();
else
$("#watcherDuration", dialog).focus().select();
}
function WatcherUI() { /* Nothing */ };
WatcherUI.create = function(watcher) {
// Create jQuery Element...
var div = $("<div>").addClass("watcher")
.html('<div class="details"> > </div>\
<div class="play_container"><div class="play"></div><div class="play_select"></div></div>\
<div class="content">\
<a class="name" href="' + watcher.getURL() + '" target="_blank">' + ((typeof watcher.name !== 'undefined') ? watcher.name : watcher.id) + '</a>\
<div class="bottom">\
<span class="time">' + (watcher.time / 1000) + ' seconds </span>\
<span class="icons">\
<a class="edit" href="javascript:void(0)"><img src="https://i.imgur.com/peEhuHZ.png" /></a>\
<a class="delete" href="javascript:void(0)"><img src="https://i.imgur.com/5snaSxU.png" /></a>\
</span>\
<div class="last_updated" title="Last checked: ' + ((typeof watcher.date !== 'undefined') ? watcher.date.toString() : "n/a") + '">' + ((typeof watcher.date !== 'undefined') ? watcher.getFormattedTime() : "n/a") + '</div>\
</div>\
<div class="color_code"><div></div></div>\
</div>');
if (watcher.state.isSelected)
div.addClass("selected");
// Add listeners
watcher.addListener(Evt.START, function() {
div.addClass("running");
});
watcher.addListener(Evt.STOP, function() {
div.removeClass("running");
});
watcher.addListener(Evt.UPDATE, function(e) {
$(".last_updated", div).text(watcher.getFormattedTime()).attr('title', "Last checked: " + watcher.date.toString());
div.addClass("updated");
setTimeout(function() { div.removeClass("updated") }, 1000);
});
watcher.addListener(Evt.CHANGE, function() {
$(".name", div).text(watcher.name).attr('href', watcher.url);
$(".time", div).text(watcher.time / 1000 + " seconds");
if (watcher.state.isSelected)
$(div).addClass("selected");
else
$(div).removeClass("selected");
});
watcher.addListener(Evt.HITS_CHANGE, function() {
$(".details", div).addClass("updated");
});
watcher.addListener(Evt.DELETE, function() {
div.remove();
});
watcher.addListener(Evt.VIEW_DETAILS, function() {
$(".details", div).removeClass("updated");
});
// Set actions
$(".edit", div).click(showWatcherDialog);
div.dblclick(showWatcherDialog);
function showWatcherDialog() {
watcherDialog(watcher, function(values) {
watcher.setValues({
name : values.name,
time : values.time,
alert : values.alert,
auto : values.auto,
stopOnCatch : values.stopOnCatch,
muteBatch : values.muteBatch
})
});
}
$(".delete", div).click(function() {
dispatch.remove(watcher);
});
$(".details", div).mouseover(function () {
showDetailsPanel(watcher);
$(this).removeClass("updated");
});
$("div.play_select", div).mousedown(function() {
watcher.toggleSelected();
});
$("div.play", div).mousedown(function() {
if (watcher.state.isRunning)
watcher.stop();
else
watcher.start();
});
// Add colors for watcher type
var colorCode = $(".color_code", div);
if (watcher.type === 'hit') {
colorCode.addClass("hit");
colorCode.attr('title', "HIT Watcher");
} else if (watcher.type === 'requester') {
colorCode.addClass("requester");
colorCode.attr('title', "Requester Watcher");
} else if (watcher.type === 'url') {
colorCode.addClass("url");
colorCode.attr('title', "URL Watcher");
}
colorCode.attr('title', colorCode.attr('title') + "\nClick and drag to re-order");
$(".delete img", div).hover(function() { $(this).attr('src', "https://i.imgur.com/guRzYEL.png")}, function() {$(this).attr('src', "https://i.imgur.com/5snaSxU.png")});
$(".edit img", div).hover(function() { $(this).attr('src', "https://i.imgur.com/VTHXHI4.png")}, function() {$(this).attr('src', "https://i.imgur.com/peEhuHZ.png")});
return { element: div, watcher: watcher };
}
/** The Watcher object. This is what controls the pages that are monitored and how often
Events:
onStart - The watcher has started to check the desired page with a time interval
onStop - The time interval has stopped
onUpdate - The watcher has just checked the page for hits
onChange - Attributes of the watcher changed, like name, interval time, etc.
onDelete - When a watcher has been deleted
onHitsChange - The watcher updated and found a different set of hits from last time
onCaptcha? - The watcher encounters a captcha. Not sure if this should be handled by the Watcher or Loader (maybe both)
**/
function Watcher(attrs) {
var DEFAULT_TIME = 60000;
this.interval = null; // For continuous interval
this.timer = null; // For initial setTimeout
this.newHits = [];
attrs = attrs || {};
// Default states
this.state = {};
state = attrs.state || {};
this.state.isRunning = (typeof state.isRunning !== 'undefined') ? state.isRunning : false;
this.state.isSelected = (typeof state.isSelected !== 'undefined') ? state.isSelected : false;
this.state.isUpdated = (typeof state.isUpdated !== 'undefined') ? state.isUpdated : false;
// TODO Erase these state overwrites once we implement resuming state after a page load
// Currently if a watcher is on when dispatch saves the watcher list, it'll still be marked as running even
// though it wouldn't be running on page load.
this.state.isRunning = false;
this.state.isUpdated = false;
// Required attributes
this.id = attrs.id;
this.time = attrs.time || DEFAULT_TIME;
this.type = attrs.type;
this.name = attrs.name || this.id;
this.lastHits = attrs.lastHits || [];
// Options
this.option = {};
option = attrs.option || {};
this.option.auto = (typeof option.auto !== 'undefined') ? option.auto : false;
this.option.alert = (typeof option.alert !== 'undefined') ? option.alert : false;
this.option.stopOnCatch = (typeof option.stopOnCatch !== 'undefined') ? option.stopOnCatch : true;
this.option.muteBatch = (typeof option.muteBatch !== 'undefined') ? option.muteBatch : false;
// Figure out the URL
this.url = attrs.url;
if (typeof this.url === 'undefined')
this.setUrl();
// Listeners
this.listener = {
onstart : [],
onstop : [],
onupdate : [],
onchange : [],
onhitschange : [],
ondelete : [],
onviewdetails : []
};
return this;
}
Watcher.prototype = new Evt();
Watcher.prototype.toString = function() {
return this.name;
}
Watcher.prototype.getHTML = function() {
this.DOMElement = $("<div>");
return $("<div>");
}
Watcher.prototype.getURL = function() {
return this.url;
}
Watcher.prototype.setUrl = function() {
switch(this.type) {
case 'hit':
this.url = "https://www.mturk.com/mturk/preview" + (this.option.auto ? "andaccept" : "") + "?groupId=" + this.id;
break;
case 'requester':
this.url = "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=" + this.id;
break;
case 'url':
if (typeof this.url === 'undefined')
this.url = this.id;
// URL watchers get a random id because of id requirements for CSS
this.id = "A" + Math.floor(Math.random() * 100000000);
break;
}
}
Watcher.prototype.setAuto = function(isAuto) {
this.option.auto = isAuto;
this.setUrl();
}
Watcher.prototype.isNewHit = function (hit) {
return (this.newHits.indexOf(hit) !== -1);
}
Watcher.prototype.onChanged = function(newHits) {
Messenger.sendHits(this, newHits);
this.isUpdated = true;
this.notify(Evt.HITS_CHANGE, newHits);
}
Watcher.prototype.start = function() {
if (!this.state.isRunning) {
var _this = this;
// Set the interval and start right away
this.interval = setInterval(function(){ _this.getData() }, this.time);
this.getData();
this.state.isRunning = true;
this.notify(Evt.START, null);
}
return this;
}
Watcher.prototype.stop = function() {
// Stop the interval object and the timer object
clearInterval(this.interval);
clearTimeout(this.timer);
this.state.isRunning = false;
this.notify(Evt.STOP, null);
}
Watcher.prototype.delete = function() {
this.notify(Evt.DELETE, this);
this.stop();
this.listener = null;
this.newHits = null;
this.lastHits = null;
}
Watcher.prototype.filterMessages = function(newHits) {
// Determine which hits, if any, the user should be notified of
// For now just showing new hits
var filteredHits;
if (typeof this.lastHits !== 'undefined' && this.lastHits.length > 0) {
filteredHits = [];
for (var i = 0, len = newHits.length; i < len; i++) {
for (var j = 0, len2 = this.lastHits.length; j < len2; j++) {
if (newHits[i].id === this.lastHits[j].id)
break;
// If we reach the end with no matches, add it to the changed hits array
if (j === len2 - 1 )
filteredHits.push(newHits[i]);
}
}
} else {
// If "last hits" doesn't exist, then all of the new hits should be considered new
filteredHits = newHits;
}
this.lastHits = newHits;
return filteredHits;
}
Watcher.prototype.toggleSelected = function() {
if (this.state.isSelected)
this.state.isSelected = false;
else
this.state.isSelected = true;
this.notify(Evt.CHANGE, null);
}
Watcher.prototype.markViewed = function () {
if (this.isUpdated) {
isUpdated = false;
this.notify(Evt.VIEW_DETAILS, null);
}
}
Watcher.prototype.updateWatcherPanel = function() {
this.date = new Date();
this.notify(Evt.UPDATE, null);
}
Watcher.prototype.setValues = function(values) {
var val = values || {};
this.name = val.name || this.name;
this.setAuto(val.auto);
this.option.stopOnCatch = val.stopOnCatch;
this.option.alert = val.alert;
this.option.muteBatch = val.muteBatch;
if (typeof val.time !== 'undefined' && this.time !== val.time) {
this.time = val.time;
if (this.state.isRunning) {
this.stop();
this.start();
}
}
this.notify(Evt.CHANGE, null);
}
Watcher.prototype.getFormattedTime = function() {
if (typeof this.date !== 'undefined') {
var time = this.date;
var str = "";
var hours = time.getHours();
var ampm = "am";
if (hours >= 12) {
if (hours > 12)
hours -= 12;
ampm = "pm";
} else if (hours === 0) {
hours = 12;
}
str += hours + ":"
+ ((time.getMinutes() < 10) ? "0" : "") + time.getMinutes() + ":"
+ ((time.getSeconds() < 10) ? "0" : "") + time.getSeconds()
+ ampm;
return str;
} else {
return "N/A";
}
}
Watcher.prototype.setHits = function(hits) {
if (typeof hits !== 'undefined') {
if (Object.prototype.toString.call(hits) !== '[object Array]')
hits = new Array(hits);
this.sendHits(hits);
}
this.updateWatcherPanel();
}
Watcher.prototype.sendHits = function(hits) {
// Only send the hits if there is actually something to send
// In the near future this will have to be changed to show when HITs go away completely
if (typeof hits !== 'undefined' && hits.length > 0) {
var newHits = this.newHits = this.filterMessages(hits);
if (newHits.length) {
this.onChanged(newHits);
} else if (this.option.auto && !this.option.stopOnCatch) {
this.onChanged(newHits); // Might add a different method for this case, but using onChanged for now
}
}
}
Watcher.prototype.getData = function() {
var _this = this;
Loader.load(this, this.url, function(data) { _this.onDataReceived($(data)); });
}
Watcher.prototype.onDataReceived = function(data) {
var error = $(".error_title", data);
if (error.length > 0) {
if (error.text().contains("You have exceeded")) {
console.error("Exceeded the maximum rate!");
return;
}
}
if (this.type === 'hit')
this.setHits(this.parseHitPage(data));
else
this.setHits(this.parseListing(data));
}
Watcher.prototype.parseListing = function(data) {
var hitCount = $("table:nth-child(3) > tbody:nth-child(1) > tr", data).length;
var hits = new Array();
var qryUrl = "td:nth-child(3) > span:nth-child(1) > a",
qryTitle = "td:nth-child(1) > a:nth-child(1)",
qryRequester = "td:nth-child(1) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(1) > td:nth-child(2) > a",
qryReward = "td:nth-child(3) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(1) > td:nth-child(2) > span:nth-child(1)",
qryAvailable = "td:nth-child(3) > table > tbody > tr:nth-child(2) > td:nth-child(2)",
qryTime = "td:nth-child(2) > table > tbody > tr:nth-child(2) > td:nth-child(2)";
for (var i = 0; i < hitCount; i++) {
// Get nearby ancestors so jQuery won't have to do a full search for each element (faster)
var base = $("table:nth-child(3) > tbody:nth-child(1) > tr:nth-child(" + (i+1) + ") > td:nth-child(1) > table:nth-child(1) > tbody:nth-child(1)", data),
topRow = $("tr:nth-child(2) > td:nth-child(1) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(1)", base),
content = $("tr:nth-child(3) > td:nth-child(3) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(1)", base);
var hit = new Hit();
hit.requester = $(qryRequester, content).text();
hit.requesterID = $(qryRequester, content).attr("href").match(/requesterId=([A-Z0-9]+)/)[1];
hit.title = $(qryTitle, topRow).text().trim();
hit.reward = $(qryReward, content).text().trim();
hit.available = $(qryAvailable, content).text().trim();
hit.time = $(qryTime, content).text().trim();
var urlData = $(qryUrl, topRow);
hit.url = urlData.attr("href");
var idMatch = hit.url.match(/(group|notqualified\?hit|requestqualification\?qualification)Id=([A-Z0-9]+)/);
if (idMatch !== null) {
hit.id = idMatch[2];
}
hit.canPreview = false;
// Check each link to see if user is qualified or can preview the HIT, etc.
urlData.each(function() {
if (typeof this.href !== 'undefined') {
if (this.href.contains("qual"))
hit.isQualified = false;
else if (this.href.contains("preview"))
hit.canPreview = true;
}
});
hits[i] = hit;
}
return hits;
}
Watcher.prototype.parseHitPage = function(data) {
var msgbox = $("#alertboxHeader", data);
var hasCaptcha = ($(data).length > 0) ? ($(data).text()).contains("In order to accept your next HIT") : false;
if ($(msgbox).length > 0 && ($(msgbox).text()).contains("There are no more available HITs in this group.")) {
// If there aren't any more available, keep checking. If they were just previously available
// then we should alert the user that it's gone.
} else {
// If it's newly available, alert the user and start auto-stacking if that's desired.
//TODO We need to test for "You are not qualified to accept this HIT."
if (hasCaptcha) {
console.log("Has captcha");
sendMessage({header: 'captcha'});
}
var uid = $("input[name='hitId']", data).attr("value");
var hit = new Hit({id: this.id, uid: uid, isAutoAccept: this.option.auto});
hit.requester = $("form:nth-child(7) > div:nth-child(9) > div:nth-child(1) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(3) > td:nth-child(3) > table:nth-child(1) > tbody:nth-child(1) > tr:nth-child(1) > td:nth-child(2)", data).text().trim();
hit.title = $(".capsulelink_bold > div:nth-child(1)", data).text().trim();
hit.reward = $("td.capsule_field_text:nth-child(5) > span:nth-child(1)", data).text().trim();
hit.available = $("td.capsule_field_text:nth-child(8)", data).text().trim();
hit.time = $("td.capsule_field_text:nth-child(11)", data).text().trim();
if (((hasCaptcha && settings.stopOnCaptcha) || (this.option.auto && this.option.stopOnCatch)) && this.state.isRunning)
// We should probably toggle off all auto-accept hits when we encounter a captcha. Maybe send a special message to all mturk windows while we're at it.
// The special message could be some kind of banner that says that no more hits can be accepted in the background until the captcha is entered. (It would
// be pretty cool if we could pull up the captcha image in the background and just show it and the form to enter it from another page).
this.stop();
return new Array(hit);
}
}
Watcher.replacerArray = ["id", "time", "type", "name", "option", "auto", "alert", "stopOnCatch", "state", "isRunning", "isSelected", "isUpdated", "url", "lastHits"];
var Messenger = function() {
var SEND_HITS = "new_hits";
var SEND_TO = "turkopticon";
var notificationGroup;
function _sendHits(watcher, hits) {
// Pass through ignore filters
hits = IgnoreList.filter(hits);
if (hits.length) {
var toData = TO.get(Hit.getUniqueReqeusters(hits), _handleTOReceived);
if (settings.notifications) {
// Set wasViewed to false to check if any receiving windows were focused when this was sent.
wasViewed = false;
// Send Hits
sendMessage({ header: SEND_HITS, content: { 'title': watcher.name, 'hits': hits, 'url': watcher.url } });
// Get TO and send it
if (toData)
sendMessage({ header: SEND_TO, content: toData });
// Attempt to send a browser notification after a brief period of time. If another mturk
// page was visible when it received the hits, this will cancel out.
if (!document.hasFocus())
setTimeout(function() { sendDesktopNotification(hits, watcher); }, 200);
}
// Show notification on dashboard, too
notificationGroup = notificationPanel.add(new NotificationGroup({ title: watcher.name, hits: hits, url: watcher.url }));
notificationGroup.addTO(toData);
// Sound alert for auto-accept HIT watchers and watchers that have the alert set on
if (watcher.option.auto || watcher.option.alert)
Sound.alert(watcher);
} else {
if (!watcher.option.muteBatch && watcher.option.auto && !watcher.option.stopOnCatch)
Sound.alert(watcher);
}
}
function _handleTOReceived(data) {
sendMessage({ header: SEND_TO, content: data });
if (data)
notificationGroup.addTO(data);
}
return {
sendHits: _sendHits
}
}();
/** Watcher Stack and Queue
Stack - Grab as many as possible right away
Limit - The number of HITs to stack at once
Stop - Stop after the limit is reached
Queue - Start queing after the limit is reached
Queue - Grab one at a time, paced about as fast as they can be done
**/
// Loader. This is what loads pages in the background. Page requests get added to a queue
// so we can load pages in moderation to avoid exceeding the maximum request rate.
//
// Public methods:
// - load(watcher, url, callback) is the only "public" method. The callback receives the data from
// the requested page.
var Loader = function() {
var queue = [],
pauseTime = 2000, // The amount of time to pause (in milliseconds)
intervalTime = 200, // The amount of time between page loads
count = 0,
paused = true,
maxLoad = 6; // The max number of pages to load without pausing
function _load(watcher, url, callback) {
if (!_isQueued(watcher)) {
queue.push({url: url, callback: callback, watcher: watcher});
// If queue length is now 1 and was paused, it means we should resume loading
if (queue.length === 1 && paused) {
paused = false;
_next();
}
}
}
// Checks to see if the watcher is already queued
function _isQueued(watcher) {
if (queue.length > 0) {
for (var i = 0, len = queue.length; i < len; i++)
if (queue[i].watcher === watcher)
return true;
}
return false;
}
// GETs thet next URL in the queue
function _next() {
if (queue.length > 0) {
var info = queue.shift();
_getData(info.url, info.callback);
} else {
paused = true;
}
}
function _getData(url, callback) {
$.get(url, function(data) {
callback(data);
if (++count < maxLoad) {
setTimeout(_next, intervalTime);
} else {
paused = true;
count = 0;
setTimeout(function() {
if (paused)
_next();
}, pauseTime);
}
})
}
return {
load: _load
}
}();
var TO = function() {
var URL_PREFIX = "https://turkopticon.ucsd.edu/api/multi-attrs.php?ids=",
cache = {};
setInterval(function() { cache = {}; }, 3600000); // Clear cache once per hour
function _get(ids, callback) {
var results = _getFromCache(ids);
// If not all requesters found in storage, fetch from server
if (results.missing.length > 0)
_fetchFromServer(URL_PREFIX + results.missing.join(','), callback);
return JSON.stringify(results.found);
}
function _getFromCache(ids) {
var sorted = { found: {}, missing: [] };
for (var i = 0, len = ids.length; i < len; i++) {
if (cache[ids[i]])
sorted.found[ids[i]] = cache[ids[i]];
else
sorted.missing.push(ids[i]);
}
return sorted;
}
function _fetchFromServer(url, callback) {
$.get(url, function(data) {
_cache(data);
if (typeof callback === 'function')
callback(data);
})
}
function _getCount(obj) {
var count = 0;
for (key in obj)
count++;
}
function _cache(data) {
var ratings = JSON.parse(data);
for (id in ratings) {
cache[id] = ratings[id];
}
}
return {
get: _get
}
}();
/** The NotificationPanel object. This holds and manipulates incoming notification groups
**/
function NotificationPanel() {
this.isHidden = true;
this.notifications = new Array();
this.createPanel();
this.isHovered = false;
this.timeout = null;
}
NotificationPanel.prototype.add = function(notification) {
var _this = this;
// Get rid of the leftover notification if there's one there
if (this.notifications.length > 0 && this.notifications[0].hasTimedOut && !this.notifications[0].isHovered) {
var oldNotification = this.notifications[0];
setTimeout(function() { _this.onTimeoutListener(oldNotification);}, 1500);
}
// Cancel delayed timeout from mouseout (so panel won't close right after a new
// notification comes in)
clearTimeout(this.timeout);
notification.onTimeout = function() { _this.onTimeoutListener(notification) };
this.notifications.push(notification);
this.addToPanel(notification);
if (this.isHidden) {
this.show();
}
return notification;
}
NotificationPanel.prototype.remove = function(notification) {
// Don't remove the notification if the user has their mouse hovering over it.
// The notification will trigger onTimeout later on mouseout which will call
// this method again for removal.
if (!notification.isHovered) {
this.removeFromPanel(notification);
var newArray = new Array();
for (var i = 0, len = this.notifications.length; i < len; i++)
if (this.notifications[i] !== notification)
newArray.push(this.notifications[i]);
this.notifications = newArray;
}
}
NotificationPanel.prototype.show = function() {
if (this.isHidden) {
this.getDOMElement().removeClass("tm-hidden");
this.isHidden = false;
}
}
NotificationPanel.prototype.hide = function() {
if (!this.isHidden && !this.isHovered) {
this.getDOMElement().addClass("tm-hidden");
this.isHidden = true;
}
}
NotificationPanel.prototype.createPanel = function() {
var _this = this;
var panel = $('<div class="notification_panel tm-hidden" id="receiver"></div>')
.hover(
function() {
clearTimeout(this.timeout);
_this.isHovered = true;
_this.show()
},
function(){
_this.isHovered = false;
this.timeout = setTimeout(function() { _this.hide() }, 1500); // Delay hiding the panel
}
)
$("body").append(panel);
this.DOMElement = panel;
addStyle("\
.notification_panel div, .notification_panel p { font: " + settings.fontSize + "pt 'Oxygen', verdana, sans-serif; }\
#receiver.notification_panel { \
position : fixed;\
width : 400px;\
bottom : 0px;\
right : 0px;\
background : rgba(255, 255, 255, 1);\
padding : 5px;\
border : 1px solid #d5d5d5;\
border-size : 1px 0 0 1px;\
overflow : auto;\
border-radius : 5px 0 0 0;\
border-right : 0;\
transition : right 0.2s;\
}\
#receiver.notification_panel.tm-hidden {\
right: -395px;\
}\
#receiver .notification_group {\
background : #fdfdfd;\
border : 1px solid #eaeaea;\
padding : 5px;\
margin : 10px 0;\
opacity : 1;\
overflow : hidden;\
transition : opacity 0.7s, max-height 0.2s ease-in-out 0.7s, margin 0.2s linear 0.7s, padding 0.2s linear 0.7s;\
border-right-color : #dedede;\
border-bottom-color : #dedede;\
}\
#receiver .notification_group.removed {\
opacity : 0;\
max-height : 0;\
padding : 0;\
margin : 0;\
}\
#receiver .notification_group h3 { margin: 3px; font-weight: normal }\
#receiver .notification_group h3 a:link, #receiver .notification_group h3 a:visited { color: #333 }\
#receiver .notification_group h4 a:link,\
#receiver .notification_group h4 a:visited { margin: 2px 0 0 4px; color: #222; }\
.notification_panel h2, #details_panel h2 { font-size: 100%; font-weight: normal; margin: 0px }\
.notification {\
padding : 5px 3px 0 5px;\
background-color : #fff;\
border-bottom : 1px solid #e9e9e9;\
position : relative;\
margin-left : 5px;\
}\
.notification:last-child { border: none; padding-bottom: 3px }\
.notification .mute {\
position : absolute;\
bottom : 6px;\
right : 5px;\
color : #999;\
cursor : pointer;\
font-size : 76%;\
}\
.notification a.requester:link, .notification a.requester:visited {\
margin-top : 2px;\
color : black;\
font-size : 80%;\
font-weight : bold;\
}\
#details_panel .notification.ignored {\
opacity: 0.4;\
}\
#receiver .notification.ignored {\
display: none;\
}\
.notification .ignore {\
font-size: 60%;\
color: #999;\
visibility: hidden;\
cursor: pointer;\
}\
.notification:hover .ignore {\
visibility: visible;\
}\
.notification .extra_info {\
font-style : italic;\
font-size : 80%;\
color : #505050;\
cursor : default;\
}\
.notification_panel a:link, .notification_panel a:visited {\
text-decoration : none;\
color : #6bf;\
}\
.notification_panel a.title:link, .notification_panel a.title:visited {\
display : block;\
white-space : nowrap;\
overflow : hidden;\
text-overflow : ellipsis;\
font-size : 102%;\
}\
.notification_panel .links {\
position : absolute;\
bottom : 6px;\
right : 35px;\
}\
.notification_panel a.hit_link {\
font-size : 70%;\
color : #fff;\
background : none repeat scroll 0% 0% #55B8EA;\
border-radius : 12px;\
display : inline;\
margin : 10px 5px 0px 0px;\
padding : 3px 9px;\
font-weight : bold;\
transition : background-color 0.25s;\
}\
.notification_panel a.hit_link:visited { background-color: #9df; }\
.notification_panel a.hit_link:hover { background: #8df; }\
.notification_panel p { margin: 3px 0 6px 0; font-size: 80%; cursor: default }\
.notification_panel .autoaccept {\
background-color : rgba(148, 236, 255, .3);\
background-color : rgba(214, 255, 91, 1);\
background-color : rgba(252, 255, 143, 1);\
}\
.notification.not_qualified { background-color: rgba(245, 244, 229, 1) }\
.notification_panel .new { background-color: rgba(220, 255, 228, 1); }\
.notification_panel .ratings-button {\
float: left;\
margin-right: 0.3em;\
height: 0.7em;\
width: 0.7em;\
background-color: #93C9FF;\
border-radius: 3px;\
font-size: 80%;\
position: relative;\
top: 0.6em;\
}\
.notification_panel .ratings.no-TO .ratings-button {\
background-color: #ccc;\
}\
.notification_panel .ratings-button > .ratings-chart {\
position: absolute;\
bottom: -3em;\
left: 0.4em;\
background-color: rgb(255, 255, 255);\
color: #444;\
visibility: hidden;\
padding: 0.3em;\
border: 1px solid #f0f0f0;\
z-index: 100;\
}\
.notification_panel .ratings-button:hover > .ratings-chart { visibility: visible; }\
.notification_panel .ratings.no-TO .ratings-button > .ratings-chart { bottom: -1em; }\
.notification_panel .ratings-chart table { border-collapse: collapse; }\
.notification_panel .ratings-chart td { font-family: 'Oxygen',verdana,sans-serif; font-size: 70%; color: #444; padding: 0 2em 0 0; cursor: default; vertical-align: center }\
.notification_panel .ratings-chart td:nth-child(3) { font-family: 'Droid Sans Mono',fixed-width; font-size: 60%; white-space: nowrap }\
.notification_panel .ratings-chart p { font-size: 80%; padding: 0 2em 0 0; margin: 0.5em 0 0; white-space: nowrap }\
.notification_panel .ratings-chart .light { opacity: 0.6 }\
.notification_panel .ratings.no-TO .ratings-chart { padding: 0.5em }\
.notification_panel .ratings-chart .rating { padding: 0.3em 0 0 }\
.notification_panel .ratings-chart .rating > div { background-color: #eee; height: 0.5em; width: 13em; margin: 0 0.5em 0 0; border-radius: 4px }\
.notification_panel .ratings-chart .rating > div > div { background-color: #55B8EA; height: 100%; border-radius: 4px }\
");
}
NotificationPanel.prototype.getDOMElement = function() {
return this.DOMElement;
}
NotificationPanel.prototype.addToPanel = function(notification) {
$(this.getDOMElement()).prepend(notification.getDOMElement());
}
NotificationPanel.prototype.removeFromPanel = function(notification) {
$(notification.getDOMElement()).remove();
}
NotificationPanel.prototype.onTimeoutListener = function(notification) {
if (this.notifications.length > 1) {
var _this = this;
if (document.hasFocus() && settings.animation) {
notification.fadeOut();
setTimeout(function() { _this.remove(notification) }, 905);
} else {
setTimeout(function() { _this.remove(notification) }, 705);
}
} else {
this.hide();
}
}
/** The NotificationGroup object. This holds groups of Notifications and interacts
directly with the NotificationPanel
**/
function NotificationGroup(obj) { // title, hits, isSticky, watcher, url
this.title = obj.title || null;
this.hits = obj.hits;
this.isSticky = (typeof obj.isSticky !== 'undefined') ? obj.isSticky : this.hasAutoAccept();
this.url = obj.url;
this.timeout = (this.isSticky) ? 15000 : 6000;
this.hasTimedOut = false;
this.isHovered = false;
if (typeof obj.watcher !== 'undefined') this.watcher = obj.watcher;
var _this = this;
setTimeout(function() {
if (typeof _this.onTimeout !== 'undefined' && _this.onTimeout !== null) {
_this.hasTimedOut = true;
if (!_this.isHovered)
_this.onTimeout(_this);
}
}, this.timeout);
if (typeof this.hits[0] === 'undefined')
console.error("Error, no hits for notification", document.URL, obj);
this.createDOMElement();
}
NotificationGroup.prototype.addTO = function(data) {
if (data !== "{}") {
var ratings = JSON.parse(data);
var group = this.getDOMElement();
var notifications = group.find(".notification"),
singleRequester = group.find("h4 .requester");
if (singleRequester.length > 0) {
var id
for (var i in ratings)
id = i;
this.appendRatings({ notification: singleRequester.parent(), id: id, ratings: ratings[id] });
}
for (var id in ratings) {
currentNotification = notifications.filter(function() { return $(this).data("requesterID") === id });
this.appendRatings({ notification: currentNotification, id: id, ratings: ratings[id] });
}
}
}
NotificationGroup.prototype.appendRatings = function(obj) {
var notification = obj.notification,
requesterID = obj.id,
ratings = obj.ratings,
attrs = ratings.attrs,
requesterEl = notification.find(".requester");
// Would be nice to have a chart-looking icon
var element = $('<div class="ratings"><div class="ratings-button" style="float: left"><div class="ratings-chart"></div></div></div>');
if (ratings) {
var html = '\
<table><tbody>\
<tr><td>Communicativity</td><td class="rating">' + bar(attrs.comm) + '</td><td class="light">' + attrs.comm + ' / 5</td></tr>\
<tr><td>Pay</td> <td class="rating">' + bar(attrs.pay ) + '</td><td class="light">' + attrs.pay + ' / 5</td></tr>\
<tr><td>Fairness</td> <td class="rating">' + bar(attrs.fair) + '</td><td class="light">' + attrs.fair + ' / 5</td></tr>\
<tr><td>Quickness</td> <td class="rating">' + bar(attrs.fast) + '</td><td class="light">' + attrs.fast + ' / 5</td></tr>\
</tbody></table>';
var count = ratings.reviews;
html += '<p>Scores based on <a href="http://turkopticon.ucsd.edu/' + requesterID + '" target="_blank">' + count + ' review' + ((count !== 1) ? "s" : "") + '</a>';
html += ' <span class="light">(' + ratings.tos_flags + ' TOS violation' + ((ratings.tos_flags !== 1) ? "s" : "") + ')</light> - <a href="http://turkopticon.ucsd.edu/report?requester[amzn_id]=' + requesterID + '&requester[amzn_name]=' + ratings.name +'" target="_blank">Add review</a></p>';
element.find(".ratings-chart").append(html);
element.find(".ratings-button").css('background-color', getHsl(avg(attrs) / 5 * 100));
} else {
var html = '<p>No ratings available.</p>';
html += '<p>Be the first to <a href="http://turkopticon.ucsd.edu/report?requester[amzn_id]=' + requesterID + '&requester[amzn_name]=' + ratings.name +'" target="_blank">review this requester</a>.';
element.find(".ratings-chart").append(html);
element.addClass("no-TO");
}
function bar(rating) {
var percent = rating / 5 * 100,
color = getHsl(percent);
return '<div><div style="width: ' + percent + '%; background-color: ' + color + '"> </div></div>';
}
function getHsl(percent) {
var hue = ((percent / 100 * 5) - 1) / 4 * 100; // Max hue = 100 (green)
return 'hsl(' + hue + ', 78%, 50%)';
}
function avg(attrs) {
var count = 0,
sum = 0,
comm = parseFloat(attrs.comm, 10),
pay = parseFloat(attrs.pay, 10),
fast = parseFloat(attrs.fast, 10),
fair = parseFloat(attrs.fair, 10);
if (comm !== 0) { sum += comm; count ++; }
if (pay !== 0) { sum += pay; count ++; }
if (fast !== 0) { sum += fast; count ++; }
if (fair !== 0) { sum += fair; count ++; }
return sum / count;
}
requesterEl.before(element);
}
NotificationGroup.prototype.createDOMElement = function() {
var _this = this,
REQUESTER_PREFIX = "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=",
hit = this.hits[0],
isSameReq = Hit.isSameRequester(this.hits),
self = this;
var div = $('<div>').addClass("notification_group")
.append((this.title !== null) ? $('<h3><a href="' + this.url + '" target="_blank">' + this.title + '</a></h3>') : "")
.append((isSameReq) ? $('<h4><a href="' + REQUESTER_PREFIX + hit.requesterID + '" target="_blank" class="requester">' + hit.requester + '</a></h4>') : "")
.hover(
function() { _this.isHovered = true },
function() {
_this.isHovered = false;
if (_this.hasTimedOut && typeof _this.onTimeout === 'function')
_this.onTimeout(_this);
}
);
// Sort the notifications (ignored go to the bottom)
if (pageType.DASHBOARD && pageType.MAIN)
this.hits.sort(function(a, b) { return (IgnoreList.isIgnored(a.requester)) ? 1 : 0 });
// Add the notifications
for (var i = 0, len = this.hits.length; i < len; i++) {
var notification = new NotificationHit(this.hits[i], isSameReq, (typeof this.watcher !== 'undefined') ? this.watcher : null);
notification.onIgnore = function(requesterID) {
// Remove all notifications within the group that match the requester ID
var notifications = self.DOMElement.find(".notification");
notifications.filter(function() { return $(this).data("requesterID") === requesterID }).addClass("ignored");
};
notification.onUnIgnore = function(requesterID) {
// Remove all notifications within the group that match the requester ID
var notifications = self.DOMElement.find(".notification");
notifications.filter(function() { return $(this).data("requesterID") === requesterID }).removeClass("ignored");
};
$(div).append(notification.getDOMElement());
}
if (this.hits[0].isAutoAccept)
div.addClass("autoaccept");
this.DOMElement = div;
// This is required to get the shrinking effect when notifications are removed
setTimeout(function() { div.css('max-height', div.height()); }, 1000);
}
NotificationGroup.prototype.getDOMElement = function() {
return this.DOMElement;
}
NotificationGroup.prototype.hasAutoAccept = function() {
var hasAutoAccept = false;
for (var i = 0, len = this.hits.length; i < len; i++)
if (this.hits[i].isAutoAccept) hasAutoAccept = true;
return hasAutoAccept;
}
NotificationGroup.prototype.fadeOut = function(duration) {
this.getDOMElement().addClass("removed");
}
/** The Notification object. This holds the notification data for individual hits
**/
function NotificationHit(hit, isSameReq, watcher) {
this.hit = hit;
this.isSameReq = isSameReq;
if (typeof watcher !== 'undefined') this.watcher = watcher;
this.createDOMElement();
}
NotificationHit.prototype.createDOMElement = function() {
var URL_PREFIX = "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=";
// Create notification
var hit = this.hit;
var notification = $('<div>').addClass("notification").append(
'<a class="title" target="_blank" href="' + hit.getURL('preview') + '" title="' + hit.title + '">' + hit.title + '</a>',
(!this.isSameReq) ? $('<a class="requester" href="' + URL_PREFIX + hit.requesterID + '" target="_blank">' + hit.requester + '</a> <a class="ignore">ignore</a>') : "",
'<p>' + hit.reward + " - " + hit.available + " rem. - " + hit.time.replace("minutes", "mins") + '</p>\
<div class="links"></div>\
<div><a class="mute"></a></div>'
).data("requesterID", hit.requesterID);
// Add links
if (typeof hit.id !== 'undefined' && hit.id !== "undefined" && hit.isQualified) {
if (this.hit.isAutoAccept) {
$(".links", notification).append('<a class="hit_link" href="' + hit.getURL('view') + '" target="_blank">VIEW</a>');
} else {
$(".links", notification).append('\
<a class="hit_link" target="_blank" href="' + hit.getURL('preview') + '">PREVIEW</a>\
<a class="hit_link" target="_blank" href="' + hit.getURL('accept') + '">ACCEPT</a>\
<a class="hit_link" target="_blank" href="' + hit.getURL('auto') + '">+AUTO</a>');
}
} else {
$(notification).addClass("not_qualified");
$(".links", notification).append(
(hit.canPreview) ? '<a class="hit_link" href="' + hit.getURL('preview') + '" target="_blank">PREVIEW</a>' : "",
'<span class="extra_info">Not Qualified </span>');
}
if (IgnoreList.isIgnored(hit.requester))
notification.addClass("ignored");
var id = hit.id;
var muteButton = $('a.mute', notification);
$(muteButton).text((typeof IgnoreList !== 'undefined' && IgnoreList.isMuted(id)) ? "muted" : "mute");
$(muteButton).click(function () {
if (!pageType.DASHBOARD || (pageType.DASHBOARD && !pageType.MAIN)) {
if ($(this).text() === "mute")
sendMessage({ header: "mute_hit", content: { id: id }, timestamp: true });
else
sendMessage({ header: "unmute_hit", content: { id: id }, timestamp: true });
} else {
if (!IgnoreList.isMuted(id))
IgnoreList.add(IgnoreList.HIT, id);
else
IgnoreList.remove(IgnoreList.HIT, id);
}
if ($(this).text() === "mute")
$(this).text("muted");
else
$(this).text("mute");
});
var ignoreButton = $("a.ignore", notification);
var self = this;
ignoreButton.click(function() {
if (!pageType.DASHBOARD || (pageType.DASHBOARD && !pageType.MAIN)) {
sendMessage({ header: "ignore_requester", content: { id: hit.requester } });
} else {
if (!notification.hasClass("ignored"))
IgnoreList.add(IgnoreList.REQUESTER, hit.requester);
else
IgnoreList.remove(IgnoreList.REQUESTER, hit.requester);
}
if (!notification.hasClass("ignored")) {
if (self.onIgnore && typeof self.onIgnore === 'function')
self.onIgnore(hit.requesterID);
} else {
if (self.onUnIgnore && typeof self.onUnIgnore === 'function')
self.onUnIgnore(hit.requesterID);
}
});
if (hit.isAutoAccept)
notification.addClass("autoaccept");
if (typeof this.watcher !== 'undefined' && this.watcher !== null && this.watcher.isNewHit(hit))
$(notification).addClass("new");
$(notification).append(muteButton);
this.DOMElement = notification;
}
NotificationHit.prototype.getDOMElement = function() {
return this.DOMElement;
}
var Sound = function() {
var sound = new Audio(),
altSound = new Audio();
if (sound.canPlayType('audio/ogg;codecs="vorbis"')) {
// Sound from http://rpg.hamsterrepublic.com/wiki-images/3/3e/Heal8-Bit.ogg
sound.src = "data:audio/ogg;base64," + "T2dnUwACAAAAAAAAAACiLgAAAAAAAI75y0sBHgF2b3JiaXMAAAAAAUSsAAD/////APQBAP////+4AU9nZ1MAAAAAAAAAAAAAoi4AAAEAAADUsB6EEC3//////////////////wYDdm9yYmlzHQAAAFhpcGguT3JnIGxpYlZvcmJpcyBJIDIwMDMwOTA5AAAAAAEFdm9yYmlzKUJDVgEACAAAADFMKMSA0JBVAAAQAAAgmDYQa6e11lprgqR2WmuqtdZaaya1tlprrbXWWmuttdZaa6211lpjIDRkFQAABABAKEoStGRSTEopZSBHjnLkOUjKJ6UoRwpi4jnoPfVka02mpORbTUopJQgNWQUAAAIAQAghhBBSSCGFFFJIIYUUYoghpphiyimnnHLKKccggwwyyCCDTDLppKOOOuqss846Cy200EIMscQSU2011tpzEMoopZRSSimllFJKKaWMMcYIQkNWAQAgAAAEQgYZZJBBCCGFFFKKKaaccgwy6IDQkFUAACAAgAAAAADHkBRJsRzL0RxP8iTPEi1REz3TM0XTNE3TNW1Vd3VVV3XVVnXVVmXTNW3TVmXTVXVXl3VXtnVd13Vd13Vd13Vd13Vd13Xdtm0gNGQVACABAKAjOZriKaJiGq7iOqoFhIasAgBkAAAEAKAJniEqoiZqouZpnud5nud5nud5nud5ngeEhqwCAAABAAQAAAAAAKBpmqZpmqZpmqZpmqZpmqZpmqZpmmZZlmVZlmVZlmVZlmVZlmVZlmVZlmVZlmVZlmVZlmVZlmVZlmVZQGjIKgBAAgBAx3EkR1IkRVIkx3IsBwgNWQUAyAAACABAUizFUjTHczxHdETHdExJlVTJlVzLtVwNCA1ZBQAAAgAIAAAAAABAEzTFUizHkixPMzVVUz1VVDXVUz3VVFVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVYHQkFUAAAQAAAGdZphqgAgzklkgNGQVAIAAAAAQgQxTDAgNWQUAAAQAAEiR5CSJkpNSSjkMksUkqZSTUkp5FJNHNckYlFJKKaWUUkoppZRSSikMkuUoqZSTUkpJjJLFKKlSk1JKeZSTJzXJ2JNSSimllFJKKaWUUkpZkJInLekalFJKSY6SBi3Z1JNSSolSlCg52Z6UUkoppZRSSimllFJK+aCUD0IppZRSSrnak2s9KaWUUkoZo5TwSSmllFJKKaWUUkoppZRSyghCQ1YBAEAAAIBx1iiHopPofHGGcqYpSCqUJnRvkqPkOcmttNycbsI5p5tTzvnknHOC0JBVAAAgAACEEFJIIYUUUkghhRRSiCGGGHLIKaegggoqqaSiiiqqrLLMMssss8wyyyyzzDLrrKOOOgsphJJCC63VGGuMsdXenLQ1RymdlFJKKaWUzjnnnCA0ZBUAAAIAQCBkkEEGGWUUUoghppxyyimopJIKCA1ZBQAAAgAIAAAAECXTMR3RERXRER3RER3RER3P8RxPEiXR8ixRMz1TNE3TVWVXlnXZlm1Xl3Vbl33bt3Xbtn3d2I3fOI7jOI7jOI7jOI7jOI5jCEJDVgEAIAAAAEIIIYQUUkghhZRiijHnoIMQQimB0JBVAAAgAIAAAAAARXEUx5EcSZIkS7IszdI0TdM0T/REz/RUzxVl0RZtz/Vs0fZcT/VUTxVVUzVd01Vd13Vd1VVlVXZt27Zt27Zt27Zt27Zt27ZlIDRkFQAgAQCgIzmSIimSIjmOIzmSBISGrAIAZAAABACgKIriOI7kWJIlaZIomZZquZrs6Z4u6qIOhIasAgAAAQAEAAAAAABgiIZoiI5oiZooiqIoiqIoiqIoiqIoiqIoiqIoiqIoiqIoiqIoiqIoiqIoiqIoiqIoip7neZ7neZ7neUBoyCoAQAIAQEdyJMdSLEVSJMVyLAcIDVkFAMgAAAgAwDEcQ1Ikx7IsS9M0z/M8T/REURRF01RNFQgNWQUAAAIACAAAAAAAQFEUy7EcSdIcTxIdURIl0RIlURM1URRFURRFURRFURRFURRFURRFURRFURRFURRFURRFURRFURRFURSB0JCVAAAZAAClxRgjhBCOoxZTTD1YzEEHLdRgQWqt5RaEpZRDjDkNGoTUSUm1d4w5xAyiIDoJGTQCeq69dtoQ5kH4IHKFJAhCQ1YEAFEAAIAxiDHDGHLOScmkRM4xCZ2UyDlHpZPSUSktphgzKSWmFGPknKPSScmkpBhLih2lEmOJrQAAgAAHAIAAC6HQkBUBQBQAAIIMUgophZRSzCnGkFKKMeUcUko5pphTzjkIHYSKMQadgxAppRxTzCnGHITMQeWcg9BBKAAAIMABACDAQig0ZEUAECcA4JAkz5M0SxQlSxNFTxRd1xNF15U0zRQ1UVRVyxNN1VRV2RZNVZYlTRNFS/RUUxNFVRVVU5ZNVbVlzzRt2VRV3xZV1bZl3RZ+V7Z93RNN2RZV1bZNVbV1WbaFYbZ1X5g0zTQ10VNVzRNV1VRV2zZV1bY1UVRVUVVlWVRVWVZdWRdWV/aNy/NU1TNN2RVV1ZVVWfVtVXZ931RVXVdl2fdVWTZ+29aF39aFpaiqtm66ri6ssqwLty7TdeE3Spommpooqqrmiapqqqptm6pq25YnqqqoqrbsmaZqq7Is7Kor274miqoqqqrsiqrqyqrs6roqu74uqqquq67s66bq+rru+9iy7iuj6uq6KsvCr8quLty+b9R13xg+05RtU1V131RV3bd1XVhuW1eWUVV9XZVlYVhlWRh24UcXhsKoqrquyq7vq7JsDLuvK8vtG8My6zrj9oXhuH1fWY5lyReOpWvbvjH7NuX2haWv/MowHEeeadqyqKq6baqu7cu6rSy37xvDqKq+rsoy4XRl3deNX1lu3TeOUXV1XZVlYVllWRh24VeWXfhxbZty+zpltn2lbxz5vjCUbVtoCz/l9n3lGJYh4xgSAAAw4AAAEGBCGSg0ZEUAECcAACnlnGIKQqUYhA5CSh2ElCrGIGSOSamYgxJKSS2EklrFGISKMQkZc1JCCS2FUlrqIKQUSmktlJJaai3GlFqLHYSUQikthVJaSy3FllqLtWIMQuaYhIw5KKGUlkIpqWXOSemck9I5J6WU1FopqbWKMSmZc1I65ySFUloqJbUWSmmtlNJaSaW11lqsrbVYQymphVJaKyW1llqqrbVWa8UYhMwxCRlzUEIpLYVSUqsYk9I5RyVzTkoppbVSSmqZc1I656R0zkkpqbRWUmktlNJaSSW2UEprrbVaU2qthlJaK6W0VlJprbVWa2utxg5CSqGU1kIpraXWakytxRhKaa2U0lpJqbXWYq2ttVpDCa2FUlorJbXWWqqxtRZrai3G1lqtLbZaY6wx11pzTinFmFqqsbVWa4stt1hrzh2ElEIprYVSWkutxZhaizWU0lIppbVQSmstthpTa7GGUlorpbRWUmqttVZri63GlFKMrbUaU2qxxlpzji22nFqLtbVWa2qt1lhrzrHGHAsAABhwAAAIMKEMFBqyEgCIAgAgCFHKOSkJQo45R6lBiDnnKFWOQSihtYo5KKG01jknoaUYO+egpBZjSamlGGstKbUWY60FAAAUOAAABNigKbE4QKEhKwGAKAAAxhiEGIPQGKMUYxAag5RiDEKkFGPOQYiUUsw5CBljzkEoJWPMOQilhBBKKCWlEEIopaRUAABAgQMAQIANmhKLAxQasiIAiAIAAIxBjCHGEHROSiclctBJ6aQ0EEJqnaXOUmqxxJhZKrGVGBsIHbWQWkatxFha7KiVGEtsBQCAHTgAgB1YCIWGrAQA8gAAEGSUYsw55xBCSjHmnHMIIaUYc845pRRjjjnnnFKKMeacc44x5phzEELIGGPOOQghdM45ByGEEDrnnIMQQgidc85BCCGEzjnnIIQQQgEAQAUOAAABNopsTjASVGjISgAgDwAAMEYp5yCU0ijFGIRSUmqUYgxCKSlVzkEoJaXWKucglJJSax2EUlJqrcYOQikptRZjKSWl1mKstZSSUosx1ppaii3WWnNOqcUYY605FwCAu+AAAHZgo8jmBCNBhYasBADyAAAQhJRijDHnkFJMMcaYc0gpxRhjzDnFFGOMOeecUowxxpxzjjHGmHPOOccYY8w555xjjDnnnHPOMcacc84555xzzjnnnHPOOeecc845AQBABQ4AAAE2imxOMBJUaMhKACAPAAAwBkIIIYQQQQghhBBCaCCEEEIIIYQQQgghhBBCCCGEEEIIIYQQQgghhBBCCCGEEEIIIYQQQgghhBBC55xzzjnnnHPOOeecc84555wTAOJ44QDoM2GjyOYEI0GFhqwEAFIBAABjEGJMQkqtNUw5ByGV2GJsFHMOQkkxthg5J6Gl1mLMtXJOSkqxxVpzJ6WlGGvOPfcOSmsx9pxzziWlGmvtPffeS2ut1pp77rmn1mrtPffee28txpxr7r333mrNtffee+891lhzz7333nsBACYRDgCICzasjnBSNBZYaMgqACAGAIAwxBiEEFJKKaWUYooxxhhjjDHGnHPOOeecc84555wTAACY4AAAEGAFuzJLqzaKmzrJiz4IfEJHbEaGXErFTE4EPVJDLVaCHVrBDV4AFhqyEgAgAwBAIMcee2stQsw5SSXGEiGlIJRaQ6WYclBiixlSRilnMXXMKcYYxVw6h5RBEEPoIGPGKEqptVI6hKC0mGNsmXIAAAAIAgAMRMhMIFAABQYyAOAAIUEKACgsMHQMFwEBuYSMAoPCMeGcdNoAAAQhMkMkIhaDxIRqoKiYDgAWFxjyASBDYyPt4gK6DHBBF3cdCCEIQQhicQAFJODghBueeMMTbnCCTlGpgwAAAAAAQACABwCAZAOIiIhmjqPD4wMkRGSEpMTkBCUAAAAAAIAA4AMAIEkBIiKimePo8PgACREZISkxOUEJAAAAAAAAAAAAAgICAAAAAAABAAAAAgJPZ2dTAAAAJQAAAAAAAKIuAAACAAAAK0CZ5ygBLzIyUlhX/1b/gf9C/z1ZUf85REdHRUVFU1hU/2lMS0tJTUtKWFT/ALwAO+U/NQdv6DO9g3Ngcj713GqvLqm0Nx+nOrnyP/p/L0PhCNl6AI4HOB+qFdoA1PikcU7l0v6Dyv/jRgVgHF9WPn9LzYSI53X6EaBNOg9CzyYvqho7vAudp3W3yrK3vAkU5dyuppqHphUXs+yecYBxPLnJn0+9DNtuFoPY2/vEdGij3hkiPht6idPO77JffKejD1QiodKrdPlAFMovjgzHYaD9FQGdrlfq5HD/Thoq77a59+32nO1+u9/79anKq/2HS/xyPT+fJ3xmcTxyZnRypnJ1vzgX9eOh/jv1c+nP5d/rIQDMZo0i8/Zj/Vm5yPkkB8gNV3ADe8EbUiTwZgAbhhIOgMHKv69ndsaz/FFz999XjIyiebtcBW7Sn1O0+wHTxEDGtdKOji5ME7j+RbeZ/HZ3yql24wLzNR4AZGaVKi2/VD0nnWTiiIjkALl8UoUh7DSAJw11AfhJCdkQR2HAb8VmuSGUBn2/9SgHJxTBtVyPSYQtUpesV5rbtZDEwn1tGAAgtxrA/36+C5Hf17WXyFkB2vl1N5ccbOHBZwQFTXScGb1XPf/+s9+Y54NnrmeZiwtXnbClBsACpEcAgFzWAABaC1iug4AobQcAwAgaIKoAIEjQ/u0XheqTq+frbKvmzFd7eHGu+ur7/uKzC4UbDF/Q0W80c25otlWpKpRK6rij4ydKCTOBRdFeA78AAPgOgCO+oGUnERrLajLkhYmOUmCDM7PXEMBUAGBN0zSLRcG7bkUBkJTgiabkFM/DMTxEjUlhGEEBwdRisebSdru7z1nep75rS06RQiwWCAEpHZtk87iItf2gh88WTQQqICftuUJevgcEWEFyyZoTcSx9+f+57ccmi1ym92VZ5SwwSshVXT8/hPixzDCAb7hhPh75LtOF0tF4ZmwIzAbFkodtfnXEGdfdn9OXDCPUXNVt/VBygOwW8YgkY77FYiJ1u+v9qBarsVoCBIF7/UcxPCN2uQcAAOiXCAF+2O0oK8h1Uf4Lr+MlxpYml+1vWyzFXLBk6UPChK8RsMcBA4ApAABglcOlB5QVAAAjv+cJQ8AuVQMsrCw92oY+rxYAggLsjwkA6DMBQFAAeLsdPlhvjO504nx7Rd9J2w32fvP/z/04mzmzqIK449W5DoVSafyUuJ8uOVQZ2CTrtwkxlIjn4QIYAIrAhTgEqNLM7Owfk/pdacDFIQSWUhEDgKuA/t67b324M5xkJylorz3YSZa+gXfhuPJvUikvKc+SiUZkRtq1gcjxq2FjVXHq1SIa7KgZttRgTCgSAA4AAIBUBK77fe5Pfb/fDwyplCJwwXEoxGCsI190J4QiAECVSoZSBHA/IkRtmEi4Pj59WgSTycWDB9v0//4HDi9MBEeutR/n+4ef9+j7tuYcNiIiIsyRx6OK8e8eVFRURGw7t/06zuNx3n1uNuPYtzVYDxjQ1VYd2BzV8o6+AhNDjswHe4E3RwxKfJBanuYnv2eBsdx5s4FkyBBz2y0Q5rzY+RheebWHBHU15a/3MdU0Jy6zeDkHWux3bY6lQHoCcEdCMr3ssWoJAID0rCYAgAHsFlURhNg/JwDgYXoEACBLUIBP/xgAAILiAbpXO2nDw7cS3t7mIHuyseykDdfaWdvB/VsJr+fVqvNTE3bGKfKZwZ1KrTmvVsOzrku8Ml3b63kAALrTWqnEFQqAmARSSRDWhanN65rR+3esaCmyNe5Rr5aYBHAgARAAAAAAQEs0n/ykKgEAAAAAAIrTHgAAAC5EIo4ENBs9Ymhs738TytbGyBy2DVBHxGK1GCyW0JAQQ0ht+u32YYLmjdDm++868Sx06DNDlBfBxKjJoafV7sWZTasNMyjnD7/tauRiJMKmn1dWxgQOtDQktWiDCgChIXZx2Vyf+kVNThZpJ3AboGm+KvMg0ufwwPHn/CQEAAAAwO3O6QBWuO10HYVr4IK3ttA3em741xALb3xsFreG+0Nxq57reuRs9UVIfwKYEJDSAW0PAJCeVQIAiBCE8GclAHIuawAAiaABAE/6RgAAEBQA9gY9h7fPs/uOifOdb//DKy86XhfZonN1v2ZSY3lxc/F/OT1ZKZXGteJtV0woenJjv5B7Vpm1tAOiAQCn1npCKIoSAAACcI3EqbxMq5U+ZR3rWAAUAAAAAKzjlI7ZjZ6EUBRFAAAAAHDdiyQAAACXU3HAmE1pTpqX50SvFp92LD0myTgwgJ1zROScc61t27b9OK+n86/M616czPU6e56Lf77DGZpl9iVywESPHJ+8U3sdwAm4w0nCOlbiPu+MwKMZdKgyS7sRyXSnHpfUIgpYTYKUg9g8PeehJEmF1D6YpIboigAFLjJGFM1nk7miseoB3GVv3Y3LV3dNCIV5OHEc9tXrJOB+QKwSW49hSBvuA7yceqtBkLhqNY52G2tV3u7JuxG3prRf5DKat5srzu7shM84YuF1SrPueInIFT5y3uSjBABAvrlPAQCcTT/s/PtUaVtVMdT3CLDbP1cAG5gDAASMU4AKRt38OnWt6Wy3i37VLxrdhag4U4EAgPBf3cmsmcXj8VW7UH3c/psNk/NXP/txHKDGcFtHViuyd+1qHCCeC9+S4IrT1keWvUnWRz/lvkTZL/VLbh+SifQGAACA2OkBAEDjyeyCYA8AAKQfAPvVoK936BrHs1mTNnn6vhYf1KenD1dn75USlabtfhPn62wTlIjFTsQmJGOhUChKBgAAdAEAKgAAWGtt1t/NrzjzsPeGj/HKVRY0FBIJRREEAViMCLPrpto6eqjPjUBuYLcJ813+emkPt3wU2leLAW1UMBNI3vSWLhFNRuw3CEWz5tSmvmya2zQcu/bOkxsSCcVJ4SYYeydNaNzSgfwYfqFfxsGbgoLwBxu3yMnxFPLO9UuTohGhi4wFphaxNbjza8QIEcquXDM5dTxIpQM4n/eRsUwEe8EU/IZjJyiQDOHHH4Gl4unjU/7eN9vGZLLOi/fvp1Mzl1ZbzZIlCMrcskfDPgDUOW83R/O2LiypYXGUoneSmdSvLF0BbADMJSCgd+B2l9pUHw4cGazlwPZFiWyaofDXowAAeHYQiFmTLmKeSlvorA9kAdQ5bz/9iDBCt2qIOGqjR4DV7wHsh7oEBIwLqCDZ1/yEWFkHRDnJZSGXIQDgoz38eZFE2PhFQxqdtLrIFgplw95T8EbiRWMHvD0/aPW2w9uajUWLoY6aALC8J7ABcCUgoPdRAT5X6EYNKOvyTEJpAAh1rILwOQQAuD6MZdk0GenWrwfV1rdDBs7jWRRGeXzsOW9O8dvOw4wBViaGGrZ3fZb+slwlsB9+Bgi4IsH9+bLlmec7Gb0XVckQArq0igTh8gAAitRKLh4Dzo96Iym1VziqPgC8OT9o+oaEslEziqMu6L3b+03H2KIS2A9XAALGDXDWu+YjKcumZFMpV1qNlbSeKhIobykAAK5vQROcRNnDa2SolGs4LVLMPW+/+5WfdchdKRJDjZoAhGVLYD8MBSBgPEAFwaIc25s5LYeHJgspUgrJYRjYcQ4A2D/sYzmFzdKvqoCYokq9it5cdRH0SW8efm8KOyh9YqjS76yD4TUubv87BmmYDWzYAQABdyvBsZK1a+aDT6FzQf4e7dKv1BkWlPmjREhLv25INk6JNmsROBXEyzQAgAkE8o/e09iiABRGj06yNpo2MxNH8fPP3nVtb2Tseb49Ac7WgFFJ1/FVoGcwc1T6TivFav8dvW1tJvuEHwLiqh1Pe6WYugnmx9Gy2FSLtFs6PHHublNzY/5kDwDQ4bwdIhkUXgVaHbXgvYs906Z4+G2WVVPn8qHkBvJu4QyQcNeA0ydwdFFhJqSDw0mP0l4ukkXeieT8fSssp66z4jo1cTG7NWmxFyUq0ZaTKdZU1BsVACA7BQDyF+anNE4MtrDzPRPNKSGz34Q+PilOcYx3mHeHevH5bdu2bUvPfa5uACwAwR4AoCDYAwDEEwCAMt0lAUCo7QAAGEEDdn9tQAkSVYVXfdrKF7wApgWbflX77+bqOny+HD48Hj08PDzM3zqd3dWy3gxDSSXgPsYDscWkvkyKGSOXVZC9AQDAbwdSKiRzJEBMbvAqVBiugLdhLSwVGVMBhS0AUFjrFA53n//F60U9lCABptqZb6PIjlVgCESchMJKz+xN3PJf73Eu12bSRpOoiIggdWSuhX5vXIHxFexpJ810kDdLz/OKN5KgpZDFVDJVWllj5qD/1+3f+zrlIUvWCsTazAdw2GMUzOz6O58GNV6T4jw7BGbckkkPTrX4jsOGSIka6WhK3Dfwn+brmSLs/KWR/ce/iappxJrGcEsi0CUC0Id2+2dXcp7b9ZJen7aLoSmp0yIqiopYsJItUc3Q1v4BAAAA+yhCAQDMWQt62VoV9HtPY49cW5AcAP7Awt6C/XyBowBPReDEkmy+t2KCsdBdiLAFZwAAAOEyAAA3dgCOj3ObfWx/UxK1kEPmegoBvgqkglMEvFEHpOrjM1V3YwcaPy45AMwlh35AdlBBtiF6GRh1EDvpKrJpQKz9yQhunHQXdpK/KMDPKKjcFnAALu4TAOt7llLkVT7JYgCiEgkAnFkHYvXaleKLOfboEMRBSXIAOC85GAOBQTKGoyLRKhAp2MXK1wKDY1cl7EzpXmlAjxLWFJw5twcA4GvLAAB5pQB8n+QQvsWXVhAbvFlLuPigwcuPnXxxsExyAHgPFlSHDeAeyDapLMQTCNle4mSGIilfCAGzswIAQKcZwPhVQ2v56B5EsSgIhNAvFr61fDZV4M8CANRZI1amlvXyH5OxRyeVOFwcnjnUU8BQcOgXOAMkXCANz++VsHjvI9GnCdLnMdnAJOS9u0CdJJr+TRjNkkz6GDLcigMAXHYaAAD3HAMAjFmPqJK1dqpWDAPXoWiTMMkB4AkO+xFOEnXSgSZg56X2d2RaRuegqSpwymu+yMF9EckkEA/xNMAB+MwIAACvaADST9qmON/es40AzFljrWTt8HOsOfbqCDR1KDlA4ytZMEWrkQ5UwHsnspkxKxsl6rQ6XMDPpgMAcNgSAABeUwD+nnZFfXj8XGYx4JkAU1FVf0YARAL8ZUumJ7Sp4madNbYwPOs4dlSHOoiSQ5/BAgNcJz8B2QIWC81r5/ih/v3Uk9IxdTp9s5aIFTYTQWP99l56bpHFVf/znf367+XfY8Nao3J8xngKAODCDwUAzF1v+u1FqS9WJa6V4VvM0feY9cD58AyAImEIfQ/gqg4QuOysXbD7n5874e+EYR58ojZ1+3BRLn716pQ8GxpwXMWMVxVHX1NEkYr7TDKTAYCkCQEAWogt2nvhPWggPqWEa4h80pjkKzPfz8fHed4wvmTYESc9AgCE6QEAoEwVEEWwAQCEaekJoD+cHRhBSt9ZAAAQFBNgo+cQcqs6gneiUJs4zi/NczPLaS+nfy6Kusj6M38NQAcAAN/7SsJCa10wzdCQBAYAlivZObTNydOwx9YVtIwAAAAAAABAqAUAAABqU0suWUNDQywWx9/LtUEsFsvEBAXWqsESYHUoR6qnEs60qdagKvfaP2sk21ATOfnqI/9SYRQNrCGINRJqtUQK1cVPtTeSlpSvqNp7V7we6kXDFqW6ts1ClRucotMUCc3qTo/zLkdRgwrZKKCCs33UDUgDT2dnUwABAE8AAAAAAACiLgAAAwAAAGbRf4YpMv9O/0z/N1lXWP9ZS0hHSEZJW1VW/zxMSkZGSEZIWFf/QkREREZCRUVs0Ji2MhVElIl48wv7Q/ULATGMDAkkfrkKo7xNQ/vO7FaXVN+NLZXKNCcAAG2GnC//Y1547da4C7mrMP7uwUGQH8pIvrHoJXRwrM7/MwUAAOh0AEB+5qsIAABWGURpewCAIC4LAM6rAjsAwJeoAkAqHeC4n16dUufP2uzppCUU6DkOl93o1er01fvbZ0fr9+4CsKB/O3l4brZUUkopIp0LtkQxhChlDmd8PCo+d3fFYjGRiEQC1jr9fre7e/fv8HU4XVIAYABgYRXKIN610n0s6LVACKH/e//43KKSoYhSAODYBRdCCP3l2+8nigAAAAAAAHhxAAAAIEe2s8MYLII2TXNqejts27TH+DFv+A7xrc5zn13n8vf9un0d+AR4aw/YAJFPobuE0s3M7L3xvFxnTiDA+PpO6AGEEyS/p16nIiIUmnxeN9XRZhXECKIiogSm58MxxFEVzHhzNCtdiT8i/xa9t72UoBKzJrXIdqnr+DJl5RZgEom47GMCwP/eND5I7Va/CcWNA+edaLRq3yvRNNj1osEIwIcDAOQK2N70BgAAwI1YA49LwbgQpwAAACVHAKBsAOgTCG6rT+f7Kfe258Ro1ET5dvZeKf3Xma/J5d2n8omeLl1YWJiXy+X5WWvb896bRLtdNE3TNE2TmMQyCPHe99q219veWjrfT5zz3nsftW3Uaz3AC6CgJfX056xpyoIuCi25XDUzkwt5eugMTRcmoAL0ol4bUd57DwLge/HfnZ733nsCAAAAWMAS0zR5m9Z8doHXD+IoyngTszRxSB0pjrBPTn/EuGqJsVgMFmteXnrGfd/kEmq1xAxDDXBmMkjJtfxLqCY5t1e+mtglBYbW10yJJqgRtDspCRbYf/ip9/PQEqse4ymu/iEjbC0dVVzkNUGeZ3+fBQriHJ/FYrGE5tSm9umUs4ZYrSEhlmCBmU/95BEGAAAWWO3GvED/B/V/LUV1bX7cN/Gc96einO1Y762MUcswFoD+5wkACBcgPQAA6HHQgVSWAAAEkH4A6EogNIgaHXmi4MtHTPPp4I2WiltuXbxa7bfUK/HfcBe+kjZKh5MzhxUzpZLKYxPixmWl5DgUAIDcxMT1sp7Ier82SWQByACA7QAAAAAgii2PW+3wOG/daadp15txJsbelMMKVAT5KdPpyT9t7cKTpsJtjhGTHTnn2rb9vNl041vZDk0qSUVVZXZgIvBGqFg2VCP2n6t6sT/T2g7wucjSD4Ohix8vIT5r+iopfyhA1MZ+mt5ffARMlImMzlSjAs8FGS96qsWqA5qnv9pzHhwX2sEthzBOgSPtJP9SQG8vrqnHWOuc/zhyROS21n6ucb/SlSQqIoAYP8USOI/05wkA3D1v2383jTw2WYSRPQHY1idAk4UNzBoAEHDnQAWJbgssHK9SEfC4lvXH9MeRUFyKnQmDCrgTcTMJAHjbzbupaarErvk75Ag3uVweUgeI3bTGJubF+S07oAccUpcm5tgmu9iDOArz4Z2zJ+cetnlJan0IF31ye1rEDVFsv7wUHna7lj2IG9Xl9vPDze9f4qN7plEm5/dsyWb+pwZBEPeaorb+fqpA+8sTJkxu/PEiBgDkXR1Ytma5nucs0kmtqe3wx8wjtoPJ1TIBEAMe/FsN51rw8G+dSrqB8e5vzxpXc28JJBNDfxyN1iRL8zO2dmU3Wgmdc7CSQO5jAAC4Ew8gj7mShDU9rzmskvdVBrcmQa3s3S3QnC40EfomyOv2nYbxk/26j73eQVm4Ji8ADOigrAAADADKGgAgXsDOqAYICDJkNQGAEWSAY+IAlCBhOR++a1LShAHXApOGp89c4+6qXlB37s2ZV69OHR8eHly9f68djUajomu6hJgkX2iG7pyfuRSuJmJK5bbAFQBAjqxKTQWfW2wbEWYAjgRC4k5mUhLq0MvJ0+k625h90jnoOJDMO0m1gZJAwIDHYRhCREqp0ej11t5ut1OEoggB3k5+1Pvj280mdCCZhiPnXNt2M/6e344vZaudNFIkoQjiqoZV0Km91pPeBKX3ty7DEoSEhuaUdez67eX/+/18P21Sqy0tocw5RUsoK4wgNLCQv1M6jXq0HkXE1GDGxyqcb+kgyHD3VCPjayomZiJEYV9b/nLEpGrBX/78ffZLQggEHNPrQ4vEYKfuURPAKgAAAAB9jTTkWSNYWguv/z3HXp1QGi8b3jXcHICXgsN+gDNAwgvIMKCtaCdUoxJs8JB5aonskVD6nSGgKe82fUcoC/HOBBxUAQDwOCcAADzTAAAMXgVUotqDzz2DvfJEqg7fTMcY8IVDv4IzQEKRIFvn+9974otD2CqR7LZahsuQYBTLIKWXfgNHXGXEhANIkgeAAnlEAABfAgBMWqVQoiICp05h34407gzvHOwIMJccVAJalRQ8y2NDgUkhU3dKKh4IVRojEEGFtT5TUj8oBzlTAQDA6hYAAO6JAKhdpBM1CLRZS6Rs7TPVK46dPWjKJDkAXMmCEeznAbJNmoCKkEpfkCptK9KPVRGwSg6AAkctA1ibuc+le1oe1Fgkl+KIbQB+l1xkBrwJAfRVRVK2OlJuhoy9OgnaBDKcuKqhBFzBQj9oFTJx8KvCo1cvKw0aCDubMmgjSd8ZSwCzCjCAq4IBAPJCAgBH0reYYBdFjoTESUugxAf1MjzWSdQUgeQAkAUr2DtJYBDZGjUBTYHSkgARw4rF5oSkHOwprwAd7MUBz0D2CADb7QM55/d6Z8UiADHzj6RAlBEA3GEdXrbcQe+z6CRoOmXDDQAwAXgDesD/TThbgKciADdAobHG9VeH356k11ccHxciovv/DQsCrPQwAAB3AADN06qZY+x0i7iyrnOToZ0/PjaKxSdnBR1isyyIARRmY87+OL32k1xNjj6hNjzaYku9X9oKkwGsEjeJllXpCBrfEunsxdStoUG//fg0vgtjmc/X9YvK33eOBjqPi7Vw381O/cruzwZN9ux3QvifAOALSwDcZRf6RtC/Vq2xgLgkkgNgZv83LKHhnwEHbw3uFQCNExSM7Ze7k0G7N1giyuPrkaAomc7kPAo6imHnsME62pLss8tfZwwAAHBTBwDnt+V9q2bqd/sOBNK37WpdhL34QvxVoW8yyd81BOiPs92M+3nZ3C3CKVMGCEhPAPk6IJkOgIiI4q3KrjYzAaDQFnC7mggA8mllAbD758SZCIIGAPpK3/0CACBI6FQuuGujzuTzKHoM/E9fTyte6LoqzVLuqQ3sk4pf0B/o53O5XG3P00SNx3WNSQcwXQ/PJ+ilPCtGgp6AQC7LOiwgJahW3d4emT6a8LhCk8MQYmbIiFno4KRurHVptQAASCllWF9cZzdVjdd+Zq0FAAB4fpYnCNQUAEUMWl+PRhFo+N/DKR5kQ3yQs4ks7AVuOYmM5IUMhEeLjmeeFkdsx7SVT+zLwv4iOlMK7nzl28RTRMUb23/kJYMAywTgScJE6idtfv8FW3VVUALEBRSa2UIgh1rUPZ45gBWe1yLbKQSRk7FCCszzDy0AAKRhP7A1WXSZ0iypknixSg4AuIYlgBcwgBWAq4RGhSAorljVNn88RlBljNxmk1WSwhJIvpwqABJAmLcMAADwmgDAa768HEJI8/9RVAzMXW87/fw41T5YqkzMHZ59qPUMAdewBFA9YQA7gEt2Ym3yG+ZnX5ViPEjtjS6ADeuU45BMc8yxkA5hkMP/DQEsgPCtYwAA4NZCABxal+bGwlNVMatIXGjJAQBXCVViDVdCoRcSVXU8rlyS2ESXQE+OtbkIjomThOtv6wACAEUeGAAAyCsJAHfrWEZRqX5IIRbkXW9sv19SrbaBWOxw4pewIwDXgAQPOCjAvQSAcksppyma7UGFKWC+FmmnvZ3QiepPgVgC6RCY3YqF9AB0XqYAAAC/KQAA9F2PTf/+3MpI7mpJLDv8rELPAN4HJDgHsQ/HhpUAwc0nkk/kUgn+ZNS8GSTgsSjZsVXOk5YiS0CSoxkQKOkAtD4KAAD8uRAAlGFvrf3GXQYhFE92kgMA/XMJSRzcS4BCXSSxKJ906Gs+AofNaalRfA0ReJaeuMsqDSAAkFuHBQAAuJAA8N6PP1mDluwMBuRhb8xNYdkfuokpR3IAoH8eAEyBnB02rARUAEduaCTenUZK19mZbMUkl1iTMCEuS4nBsAAA/B0OtympdT1fRQSXnuAS/JkAAMxhb3c2+r/Oj4o9GMBwknHo+83NrIObAfYcPgEHE2DpByoB8e212V3X00Xvj8F/9bd3ZnP73Oz9qyrlHndUleBM90i0Nk4TpqON571//59dKgkAQBojDACsNT/kbLV/LRAsFUPQBPAtzlOCYIC++mBcTQD5ddAScFpfzw/27398yWJzLdQajKfnf5m1Kv49Gb3ztNebbLHemKru5vN1DAB7WY9smojSz5snl/mDogFyJ+22cRWy4cavnTtnWb/em3jI4z5vKjJyb3ojymnJGAD7vQYAUgE6HURE/L1qM0Gr4yATyxIAgNGyAICjAnQd8Qw6urL/spe0mas07np26+j0UbG3nBYND53LRJUZNf7i6/PmarPqum5491ezEROKhBACYhICL3PnaKoIHyai3BTLvtra45/qup4jyPcagSkAAABYa53CpO8uTRO9bfw/5bQUxlRhF950+vdA2aNdc/tSWuBpUXP6yDJG23gWZrzNSveoZbwfpPt+MmFkqj1UEN5GSWkRDJBhZWs2S3AeE5mji1nSJI7XBC9vcHrOPE1sh9woVoPrc/BS6MGL1g9bw1LDsa9EGx0MZgbGSlclZ9Tk7Cn7ujKGsQ8OpTbaWYiIoJJoYhj/28dF3+UmSqjVIEwX4V0LciFWKUCAELjdAADMLW9/+hMUc1pQE4qhjtoJc0kUli2B/TAUgIDxAN4obYjquii0dSFL1DoFkc9cAuvNAQDSBd/b03y4dMaYzttpZcYYAcwtbz+m9TE6NL0WQ1WNAHILsB+yAAT0CipANx85y1CvBGEvuARE2QMAPq7H/qLVye3ycCGAbW/9xdbYHuZ9Q4Au814JxCk/sHniV5tT1kZIDFW+b55J9sbyADYAngAE9DrgJZhARfmry76zAkchgMwEeQT8XxwAAMR+1XYJt+cHNVIuIRdTKE28MT/4kOZvvOaV0iACVEcANwz74Qsg4O6BCjB8VyIsTWrRH3IIrL4EANgvf+flKJqZysMoUTgzduk1tPG7qp1uM90KCSsBvC0/ePtBzVqIoQ4bARYFsAHwJCBgXEACrGcvGWRCB8i/TAdSIA4AuO3WsywzQp+e5zANH2MUiqSJzSRXkgLKUewGxC0/aPcfjzXJXeNXMdSqT7hPtgZuCPbDXAICej8QnFN/xNsMXYnUxpHVNXmUZkg/cQAb6KHgajoAIIjHIQFMR4A9GAIA7C1vm/3r8UywFTFU9T9dBz2jW64A9kNdApIdpj/73WbxN1faMAT2JRdckDFxMBlCylkeC19KBwDcXuCcw1sUXLqnJNQAT2dnUwAAgHIAAAAAAACiLgAABAAAALdce4spVlNYVf9lR0pKRklcU1NZV/8+/2P/T0RBQ0ZEWVxV/2RHRUdGRkVHWVTsMW/ZFqLVS5MSQ2Y/uzrkuF201VfCAOrgHysT+90Av2sFw+8psr/YpH6U/75PD57df9Hs/G7RapWo/SC1bIfvujn0gT6uiQBwBkX9LJTC/4Z2KUEBAExC3SCO0u2mY4tGSA/vHLor7Bx4CaJjP1LBKOl4vCf/+wekydO1bm0Yxbvfnn27+KV/dqFqSkknRc2DTTcum+WFarGdefu+xtT66rVmuDYiBzQAvFljU7Z6B+3FTsRhCRtuAIBbvHXiY2gLwL8C2DC6gCYAbnQsvc7m1kUhyC5zwiwQ/5OQmxg65OrmyUohKD6KDADdcGhS/LW/zgSyvKcNpQ5P73PYCtwSAvRZGSxTswPfOWsdkqbT4WcOjnHqxbRahtvs2wg8YWgLLP4zIFvAe429HYnZT3dXihkavf0GFeVJb4jiS2grkR54ELYYld83s+13EJZAHswAAIAXAgBSx+WrHpmErey1lkhSZen7rGWqeJnn3OOI16Ossu23bdtW9j7n3QBYBSDYAwAM0WYyBQAIQF7AaqlBQKhddv+4cwDACDJATggAgoRvrjRMm/sf+Vcp1yzvvy740y2+7j08Xj8dHx4+PDzMd3Fzdz1fNILBoCiBrLc+vbj8uzGa3GR3kwNXAIIqhYZ6VT87kta9mpodAKcC9iIG5ZQnDM0MOKcQ4FcURJeP172ju39se4nhijEaykMhkP5+JVHzNfMQjyuSApRSSo1Gb/WZYGdCKEosRtJmwrAeFgwYULlqBx+JYjgi59r26/j/tN2/DEudtEiRhIoiWiaTxwz3Wur7Ts9I5FEoIraDcKV3zy7u++Nfu/zgKAs3QOpIiRoP16M9rfAlpBVAnDmXCVY5lHNTR6V6mveVhzwzr0WOvBWof2uUdtm7OK8RUp1HhQS5DEubee7fubBh/0/ncwCwAAAA/FqEALRRC6TMR/hxR9URikMmyQHgJVgwhX1GyTCeCkhybMarMe2u4GCFawJkkAYAwOOcAADwyACG4y0LW72LzjZiWckuYeonfEQHpFFLrMxHxutvOzYdoaYjkgPAFw7OAgYAN0C2TgWMcmS301Q5UBbqhoNAiAzfAaxGFO28tFaYBgBAniAA6jjWWfNT3TlADLxNCACUUUtapY+N11VzbDqOpo4kB4C5JFYA+3lAClAB1UWKpJ/KbkjyxUQRCSf9pzTNbDWXMAFc7AAAgBcCYDyPg3g33posFjzKQSA1AsxVi7i0tvDzqTl2OgRtFIYTH7YGXMmh7+EFKsg2iSO/qIiiKeBSyatT8ZhK7HMLKWSCQ6HiwzR4FJAqYQ8AwFHLAID/CwDEUR1W6aPsx+OTsVenQZsikgMsrmDBGEZVZJskYIWS8KlJnLFrybhaC+hlMQAArsoM4PJVt+iwPY4aGIHK5r1HchUShxPgX0MAtFkjXPajcrR90EnU9C4bbgCAL3YSTJIN7EiALOPoBUAGozsN2biJwTwnFuhftb32Z4+kPl+VWAB3sQYAAN8M4PgpMnreHw5tII7Doq6K+KE/kwJj8t1hbAGHkwE0SmMrrn/zXGyHyHYC1H57ZxWaCrEPHygkJOI2R8/8YqR8NYOXuuZNygIbPeooGSpK9m3FUFV4a/J+Ke3l4ofuzh4CAPjFAHCZpsuWWurwdR+uCfxdy/PnC8sUvk3ESTc862jPR+hN+9ewAgIqA4bgBsAXG1YAfLM/6o1xVbkuTT3pYtfXOk7BZdJfvfBXWGoHZ2ceIj+3I0uFOazs024CAABENQAAFGqjPvzxW62yV6H2spID6E32P4cVAH6fIDkA4OsFABIKw3MmbajKLZ1H7qY/O2PFAbUxociAwGFbQcbu53UGAADgkwGge3icBrW75n/TWAPbkXNO5WXTkQykYY+094tVZRsNAI6YTnIAG83+d1iBzMR7wxDyTQDPCwBUgLDtHH1gja9lFI7uuUZj4wZve0ZOfMh+AGeVL2ZNiC6yQswEAAB4xQDw3U+PrRPj9vRYMwGa1+2yTDD+Fx/fX1zdNGjJmN7E4+Gah/NTDtG3j8v1uI70BOCnDZUO0gMAQHqWAIABgHYx3EwaAEgHXukBygGw3yYMIIUgpa8KAAAECczziMzvzosT1dN3vL9DIFv3/BdfiQ5txG/vz/9sl4zKNeWip+iOLuKXnXFVkbGyMcU/0mmdtzOGNwEAvzUqFCLClBCaFjBVRiOLQr/XjhzV5DGi9ubfW7Ny5nsdEGuSAmSAhNeA1Oox/fX76zQnmbB2hIlixCwRs7LBfkpYmTx6SaprHIAAAAAAmJiAxwwzwkjg0DW+uA7bOw8GpU8aSaUGpipE2CK2iGQNDbVmzSlSpF1PR1ve9M1jcicZukQMMACoUzz+nMGuiZ5Ledq9iBF1TPIVWZWtPQCwAh9jjk5+vrKKTAGAARiAD4B86wMAAD5J7TSehRw2cEp06lW8Umy2g1XL3tc2wsENIoAFSAcAKP4YAADot+8IEKaD4w0AwlwOAK8GID7geJ0AwDp9DwAAWvibt9MHA9+VPmX75f345OTWoiO710fnD2rerJNvEsLntP2Zpa8Nx9ffdJ+V4+f3idva6GjDxVkUKyNkAl2SsRpvaIO4FzghcRRSkkyBdgz/R/u0PhszHZEYFaagCDNjzKzUOb/IL6ung4wjdaetPtMx+WiHQaEiRcChSqGkEf6KegHhuJVTkZQanrP7fss0hqQgDceqoXMfFOlnlp9Fzx3CAogFAQAAj61r3X+P27n/gnZHWlmxuYpafPvkk8VNHNTQHKeS0eeMcrM/OSkOEsWGJCdlAFaATGa/Tu+E77i9tqFNLwRDJGOxY+6Yr/8J1uOBKWYp2x47NrAaylwVPuAH++iRjZu8QIAo7bWx46Iq+js5AQCD/6/jGtL3xxEAABb37GaboP/Cx+w6x6xjGzkFABip1bLOCKB/PhIAyUWH9AIAAJDtMLtIP/DcFYDI0g8AXQkSVgKce3fW1X6+X1Q7fneX7OKrG92aYeSA2BxRFcoo5YmJ+5eZkuNYAABIvSj3m6IZ/ltKX+jGAsFnmLeuy4lMNm1fywYgA3CoU5icPJ5oMhIEAAAAIFIUV3xZbelyt/gf4rGIPrx0zd/Yjpd4bhfN/bSyiwcKI7DhZxMGQ6MXBJIc1hxGYLNXki1ZRdoS5MG+BbbtHFFzHdd52R2bZVe6oklREZvukNCtAmKap/fGI6xnuFMP5Q76+eXCoh87VuuCrS0X3pwsPcV+JF1iOe/pw43YCzVghQygEVq8DpR7bvFCmXJK3kI7Z1AFfUcDTpnjB2Hn3NEfYuacgUyrXSSiry6KluCcJH9zqWCkPjhSV60DSoEsBPjzBgD0IW9Z/f0TxmxsI4ZC7YRxBOmoAGA/PAkIGBdwk3v1JP3TyAozQYaIzMe1QWhPwwSyDsaBPIsAAHRKDxu9nsKw9XyGEtwZbzfG8WtPaWOxGCpsJ75vtrkhGFkAAnofApzXorKWVodgFREmmp12i0QDuJoGAMg4Q636NVRgG3BD66KS8RgJ1B0/MLWMpsIWu4qhDv6updALliuAkXGAZIfp06Eg+1GTg/QuKHEtzKLXWlhiwkm5316DlNIBAFfdeWx6LB7TAy2UAMQhP2j9Lw6Z54VEDBUaAVIANgCuBASMC0hAInsYZeKjhcc2d6SsTSCcINTK6SAA4HnbP5akWaTGocnQS5quUBgfMp7TEgGcJT9s+1ZJOy3ZnhgqGQHcGGwQvAME3AtQAY3yw4zsUqSoWwagKWUAwP1TPspexZS8HURri+xcZn9oH460a+AzciOXAJwlb7c7f3+NiU0khiwTwPX+pNokgQ1sIwBAwI0SNMGKe9nV4n6pTmDKC5347mthf/XxZPcxemQrtmje9mAC8VgHADh/STw1vYjbj/HqNqQk8LC7HqTY2p0ALBoXR2ofHNAhE2FI/paj69qgfkuvWzUdBjBBTtMA42MM7Lc/Jn+Lt8Y3NUu+6SuR28nvvx8hOV16Xf+46O+qrLpy4tzNPSmymw4kausiXLQ69y8ZkMvHqvMKBQA0Qg0tr3E/76j2ShwhO/x0aThs18FQcJIKkAOO4FVCNsSRnMuZeODjvvbOpz4IdDi8VsKrCXA16tblfBwTwUVvLKyziOeKZp+LddVBt9rRAODkpQYAMmflSga5+FvaU5Jd+qqJ0CkAwGhJMOEZNwB0WQUAGPAnQ1kBAOgDBMABbAI0QQj1mQRASJABoiIBCBLtJIDE2cp6RN3dv5ur31dXVxcXr9dPn6vtdhSpqkJZSsHcAPJmrq+P8VNv/tPpUe8BgBfg4HRG2m7ggxbt3IPb0GwfACVJCI4hJVHKcSAwHnRhcXYwDsrKbVUUaE1AxHp0bxi60xWvYyxQQIIQwtHo9FUellLY9zH762uNp53eSs9+rq/33/3kvxhrdy0DaYIvWa5u14DXvvv9YX6q6dKz7gkTNow3qCVRbprDwFUQqyYP+t19x3vsYqS3JcTY3jND280eto4LHp/JVub0YBfYGUkuqBt3Pq7X+P1duz5pipR7D2EnFPjuCahHt5Q9dpHXqGe5qB47PkndERETszLdIwiAD/RL/grX5kc0VympAwD/1J5RCt4iuPFQa9JJRAMDAADAXyIUAKxBHa30VzzF0mnsdCQaJ0+AYhzOXrRCCtAEXCvo/XWM/iL4j8sAZK9MIAD42ssAAPwQAN1inXuatetUEaNSTBxEE35pJWoSpD0XSemH5rJE0SFpmEoOYHmHnDVGdWSbVIBzRcLXuDTJN8jyohwIhxRAAGhmAPfpYy/aTEOvYE2moo9dUcr0Q4/An0IAfD3LWtmfkuv2Y+kUaiwyARjYgilaEDiagMOYZZoi8j1I7doagcRXKuAI8LEUAAD5yQC+xzTlNHZzG2GxbMmg2rQAzn8XrWKUOcus+MG4DnEse6Yx0wRgRLtwBq0gW6cJuJ1Jd88TUvdRbCYogLc54AjwuRgAfjZT6XLe1cGCGMsIeRVN+k2IYJhAfAQAlDmHSJmP8BJOhyKSIjkAPDjsOaA10vE0AW91mhX9FmUf0LAK3KMi/kBLYG8EeAfIQAAA7sAB2I1ySjHVpRaEsJPoSvhcAbQ1Y738QzadRURpvCA5AESJs6NVkK1wBvB0At8BVFpNhcd6D0KuhAAAXFwDwPYv1TbJk17EiAicYuaXmMiEiOsbAnGRAKQ9i2KZmu6yBTvRRkRyALiSBSWMqmTi6ANYVxw5FaNZn2+ahS4WrO+IAJIA0wEDAOQXAXD3q9LFyqv0Ymw9IPHU9LCy62wDxDldoUyt7YfZx14dV+tbcoDAr6FfY0fQywCE+AYygQqgAbX6jKuena/ZT1tjC/3vzo3v3y51528JKZvi8DHVKYqYvEyVq7MSABD2CwB8rMNYxzilbZoWsAGcRW92x4+ujIdQe9Pw0yl6amdmXAOCyQX7gBI8G1YCnKrJ2dG4eD+f+mrzHp70u0OX9+5l03NKp57h9DWNYV/Xj36NjDld6rbU5Zl6UrUmAPiXIQBPZ2dTAACAowAAAAAAAKIuAAAFAAAAbsxspSNTVv9b/23/Uf9Y/3r/bkNOTEU+WFdQ/xXy2p0jIyMnLzEuQrRVF/v+BuuuvTaIeys5AA76n8MCwPcZDvMt8K8EKJxBkk87AksOYa1Gtr65GxzjLVdbHRSltM7Kg94oyY1QIXRrwgUAALiQADD99YPUoC23uXUI3FWLnfePf3nFF68uM7LBnVWG56cAvMAGKgwhXwA8bFglQHh0Z5LOBgBTpbHB68umsJ7m9zNBb3ihONn1bAzCWBQphrE4C4AAnXgX72tbkMxy3MOtBwCahy1qI4z3xvgDznT+tqcAAGOLNnfMJlFWAAAqMQ/aBoA2DBMmJgcAMMQ7LR36fY4ACOJyAHbrygQpBBkA2KfEUgFAUABI0gAgsuRnH3X2xi7tIeXNgejtAoAbVNuZ6eityI585ArxhHNTGqFV4gq9++2uYYsdn27Rg4yRlgASlviATg2/MoSHXwM+FNeXNJicPEyfTZBS7irFIi26wXxqaiNx2AgYNeIAABBqCXI3e0w7/l9TmBoAAAAAACIGAAAAPP/Rx/46vOXLb9s+DO/79jI0m/3y/WW4+GHn3FUVnCI0BMllTPLXHEpoaIj/MiMY4BNmZdVV8aGubBzGnCZyRcl89lf1eHrRKo8mr85LA2V/tqkZqqB78naepskM0NLgJeDQ5aJobAdViVqDZvllXvFW1LmnuK41aGpMVqWYMZWZe/Wr2AyeVQ+ZSny/cEUcKRyeoAOAPJgO/kbt7noI5UUHdtp/nEGJ3VMAgCUb7+HAGJYeOAAaSAcAxNcrAABYGQiFduFmATACGACUA0A9BvIV8OkrAAA0gQQA3P93/nm8c+50qm3tV2r5jf+dzA0XS+V+zw5oRVyY7L+pdX2dkyaWJdfP3jCRJSadyxKOpgcF95dMyKfZdOY4EUsSkYKRjYmFsezoe8WHv6av4EihU5AxYqkUnb6c8Tf12f3681f7aUX6YYp+aTQ1Spg4daJwaCoinaRCkBmHyD5wpp4apU1WqmNPu2B2UdZ8HHrZ9xI3udwbVQiSILBSDUujXbanEIkDEQAA4L+sFelSKz/9snbDEJZRmbs5r23EPGne8ZK7t+O8MdiAD2wtCeRN8FM2zVCOPw061GsT1FsNZodwYKPT0wgmkIEsOMQcRzccMdZ37CKek6PtaplcicW5cYRYKeHS3gOHOS5AiG4Y6fqzn83aGp1gjAYGAADA/uN1V7inJwkAAP6YVMYSrquw7a5X1ATE/5XEEQiRRPc2Ih3DpAGAXhykFwAAgOwChLnVsgBuKgEknn4AeBqkGRiu3nYOeirrUcPAIL166fB7QXzG/b//Zqytor3hweJNNHSUTKRBR2RDKiq98Uq5P65EvFDAYSC5baVeDcom1mkkA3gAV6Vd5OCN4+WP5c2NA6xVs4tAtlSEVUyASoDCEoj9ml+jO5MKYYl4MLBJ0w3BMYp5UC5nFPIjcwamQEH+F7mMUx1cCuKdEToVTuVn8K4laAQqdpOlY/+46VPTtEWtCNDbfR2Gx07cvUkHOvhvyP3Q5s/Px5rAIxakKjFndB4EBeR4HRJ1GlOLWvBFkPhHNJ7r5ogyJwbKaknKk9HScjpFzIoh7CAHpI6DSeLzPSFmweh4Tumglr1B1gCsyYcNN54U0rzGzhjAsOhiNHy3rWZRvHcweI9ZAl44tEsHfSjlP3fOqtPKzHmlZC2i1vbeHCswqgP2qAIA4RCrDKQDm8dnVgMMwDMA8BxK6QkYAxCclg0ATw0Nvjw/Ua2Y2NyfpDopH1/j4zNP2WPxd9bIfejX9/t/392v7weGUiWlgMLTpGNQKBhLFCyrETsS0yUu0t/1VrEeTfdD9tgV8xq3p3Ry7E4e0JfBrRGko02yOX+/PlZKyeCaSLIlfk73k86KEIOxL38rjIkhsY6FlQBEKZJMnqeIx/0r0f6M2fPSLbzgIisMbI9wx+lLD69VRSfh/RsOMmtfiHk2fopt40TE1+sIRz6X6h/7BPReJ9S/oy1i264g2xVkTXpGOgfDZ3t+XG8Mw8IYI0CM6Oaz8ZIGbz1L5zVZBD5YzYP/bF+e+uRxQzPI+DQgK1XAf+v/v+xCxKZSgWKGE3TukZlqrefnDRREyvlTBaDeBYPwB0dbAAA++aODhmhIP+zKsdOh/cqWHduy3qw399JGC577KwHACMCSoGzo/g/OAAeAr2ECQPxZwxBcNwR8DYARgLbBEwCkEWQAtwGWAmbQv6+zIym4va85mxU958FG2qmHkfNn/u/506/J+2XBlVk9SmmTglvOCZQoHi8+aoZfn6/SqoX3bRRyuTwzM0zWlgiHqD/aOqk31ec//nlv3RiXBelZK82nxdVTv6eNvvJMsvKJdRpm5zbvz5qsgslYyIwwpdnNnzdKSl5qLYOj1nd4X3/42gLiPAla1hfoGmr3qXxmIzB89VibykfE4yvx6+mvHiIOClqzlGZetRJ0FNOK1hILCwJF0uhoxqhKVoHWzADxjkBP72xcE6bh3q2xBdKLJrH5mgvfUkYHkio6BnoE27JK8/BHl8W+9qbTeeueop0QOBkL1JU0szLivgfbqlSX1pqiDv7rq74BgeQ6VgFA/U5+nYSp5A5q1/bImeIbxKgHAM4kAIDmnO9/CvI9AJaHMwsHKaPJ1ki7P33kTXzM9z/q3WHneb4fVKMvhgk7iQCAAQ2UDV3/3bGAAZYqUDZA+iYwBLcQB5QAOEDZGktQd6tiJCCdYUAKqbADQSQo4KE1cT/Z+f6b9hdOn34HNZv3NqdT24XDf/Ei27WwE3bHxtqwKyUh7BBjUiSqh7bUP+XzEC8E0li65jB4q66VK9/Dp73fK6sVp6DaA6aGxtb4t1G6mqEVQokZHwu73jbns43VnJ3Umkk1PJ7dlNxu7LiINMUuM+XhTV2z38ZefA2mnGxcpInUmp1uVMRkJbHn8ev2ncuryQgXQiJXumeVXr4fuZo5t6yqXmodIAm96llFStUoYwIJBhSkJMaZyt04yAAAAADAueaHIln26yK3Bpi+GJFpHuTqSfa48MET05dAyhScK/rW3TMVyAe8VT1UlCjDwOXJz5w3o6EulBCoE7DCaIFqg00MMZgRbQEyO2kmpXuU8YMZAwGAkv5dvPGKqFaVmuxUxElKyAQAB6nW6QWM6wM/iSufGoNc2QkVkItqAC+fIwOt6KatJALGiwRTh/Qp7C2C6V3QD4xfRAMoAJQBl7fmgad+4NfRIGBuRaJ46p8Aam5YAngBA9IKwFVCowmgSV2qdFMbHkskYGGzvArAtQgD67sKAACENg2AHABwohEAfPG72atJeWhEBKTxFqd2f9y64NdhdQWceLKVHIALelgCKBI9IO0ArtIkqADIRR1oprgsQCkA0zKA3FMMHC72AAAgxBADEANAWBwAxof/tnQZ9uWVQACs9Y7I8Mebvfa12qiTCYArTQoaxz6FCqCsao+UxprETl+mGeyKj1hJUmanKUB+QNMGAMT2dus1HPk9We1bQl5EcowEFgDc4YZoyx9K87FJBP+tb9ganAYJ9mnUThzE6x3JfCcwrTAHjtjTErIHJ54GAAHto53sdRBQ/4EgDMm3k9IOALztCqH1leYrW5siVHIAcZ70z8OEEdwCh35Bcy8BalQ42AlOPje5k5wjs5hzJjsp5WynizT+kKpKapw+YjyOReEwcFVJlWoL4sbpAPDw8NwPNSSdv380JAM0Ctlo2UV/WKh0Aqz+um1QHmvEOrEuARWQ9rr21mPa351Xy2n/Nxb638O/heo6sbBJOb0SvP53rovKPLdeT/a8qWmoslNmm7+3AsDP61kOKUVeHJY1ugo0BkcPJJFPWPsPR2T09uDgl+A6uWVdITp6Sab8V4dn/2ps7r5//vg5dXdJuljdntMXH/eXpo9U205eXE3+3F6Unl7z7/fn8bIkMuhbjzZ9AvoH/Du7lEE2A2GdN/ExfHKa53me5/tX09mtOvYIBwDQFWAAkNut6yo5AAGLHm7PDydpRyfPm5fzqul+vxXd2YlkWCL3R94DRbX9tY9o+J3enUFWlg4henl/nobZ87I8r9GOZAunq9oRWV54e36Bsf3z796dgQe77Z47uTIkhJHo75mB13j//gp1vLs/Ex9DhCoD1fEukKXu8jgzNjQ80vsC3VFOANvj/x6ebEEcqVAguNedqlG7FYISPh5hJgK8/j2px89sKifq6JDlnhYjtRNQ/eXAxcPf7a0gGf4iRBlmarb6uQ+AwKUAaz4OPUw/nR8EHlVWsALQ3HATAraauSOtLF4hMT1V9mU5wHJdMTHvknCZAJ4m/PsaGwgfAMoaAGABursL0wrOAaAJAFhGtFc2+7suMFWiZyt8fR7aFGlcji6SQ8lWt1XF1eDNpbdDtv1nb32sz0kcxfHzLRKM9hFHshv+DqxnSwWb4nPMe/ZB0AF+tna93FtNQJKaX2tCTBpPrPasGVL067PLNUx72lTjiBBy8b4xOrx8wYi4XT9y9ZZP4tCIHZqg7DWk8D0et334O52378PnTIf5OxTTKzf1BJ3qpoVVdcw4NhGKdbIg41QEMse9BGOdTMbXZcEOJg3J30k7yUM6ilwBY0JELsNnEj+Q9Qzc94ZRCGVmBmWA0xiZDg8Anib89bFUXL7BhikAQMKOAwAwgkfR7u6cAgCaAACRh8HOe8+KjfPvX3mftJuhT5jogqeh503Ro/WvNdVKU5uerKI4LbdmWPUS5UY+poBw1/juuyYNi0aojFlSHa05Z1uTemMxUoBrewFxiSXk1GXaHoPSjeO0ix1302dU3dygGE/CdJRTI+iUMpnSDTDZdKpdpvgBxSEi9fE8wVcoF8f2QVx6kjnPu0Xs9+e4+WDeTmZhKvYPAYFzeeSq4HNtWMpIqWEWco+BOhjgHtKxnvqcEcoQIZRVNsEpBhIWFvznrUxhgA1TAACrHAjNAmQrGCPAAABAqwCA4vne/vvnl0zo3f3/X++KxuqwJFZboghffa/qN8F8zo9jP282u8Ppdv/4/Pr++f39//37+/p42a1jDQiKpcmXuyxqaCYQ1Uv6NkWEDeW4ZW5+mL4QTJoqN1pWt5pQJw8Fu+vG6y5EYAJciHAi3MSbju+dariek/B4NZDUGCSQFA81lAnf7sBfEONAAAgAWibQfb30JeVQjUga8Q/8B/4D1379AgCUCd/uwF8Q40AACABaJtB9vfQl5VCNSBrxD/wH/gPXfv0CAJQJ3+7AXxDjQAAIAFom0H299CXlUI1IGvEP/Af+A9d+/QIAlAnf7sBvENMIAINAywS6r5e+pJxVI1KKkD4NsX2HRnjzwpeFZOoBlA3fRwA1H7YjgBANtExAX+7qUHLWVJ3UEXbjoQnaXrPhhJemmD0LAc+2bbaJ6gTUBX93/kgql/aeEUBxoANc2nGcUq6qzirOuq3p9CaGHgCMMI+fsO3rO4+qgBdS+2cxXP2uASiOW7fef28DcJAF+r2b1cJJT+Zt+TB2ef7f7/Dlizu9czkUol3jS3Q9AewJ/3+FbcTyOhTFBTr/8YFlIIv5gk8vwLXZ9vei257/F2utqof4aWgsMq68Tv6tdqv5MB3c6e/D/6Al2mwqCMacDk9nZ1MABK+mAAAAAAAAoi4AAAYAAAD9Bq8DA1VBhtzhEgWC4lCbfzzqjIwcHR0d9YAGdfry+/P083B29s7iPYv3LB5aPGzZa7Nbm1o7umT/kv1L9opqO6vtrLOzzs46O+vsrLOzzs46237aPp/MD3qlNQDUCM/SMGn5cwaXzqh2EX3XaQ2eA27+OvW6/Dpe9FbTi/9fV09+da+vXsZB7VQs4zmxiJrTnv6pbUW1C+YczsRrBDoV/HWtXxDggMytu0EQBaMAAADA8/NzbQaazvyFHBRxEF1xDaBvWZidxykQZL3GPLz6YpH1qDza+WRlZT1OVlYWBrA75CV5BECgTBMA/ixkOr8ZspqZmfm7sQXGLTlVJQ4840Uwvsl6fgZ8X7MwiLgLsZgY4k70AsaL4PVMgF/M/E56AVUA";
sound.volume = settings.volume / 100;
}
if (altSound.canPlayType('audio/mp3')) {
// Sound from http://www.freesfx.co.uk (Multimedia System Alert 003)
altSound.src = "data:audio/mp3;base64," + "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VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVTEFNRTMuOf/7WGT/j/AAAGkAAAAIAAANIAAAAQAAAaQAAAAgAAA0gAAABDdVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVTEFNRTMuOf/7WGT/j/AAAGkAAAAIAAANIAAAAQAAAaQAAAAgAAA0gAAABDdVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVf/7WGT/j/AAAGkAAAAIAAANIAAAAQAAAaQAAAAgAAA0gAAABFVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQ==";
altSound.volume = settings.volume / 100;
}
function _alert(context) {
if (settings.sound) {
if (context instanceof Watcher) { // It's written this way in case we want sounds for other things in the future
var option = context.option;
if (!option.stopOnCatch && option.auto) {
altSound.currentTime = 0;
altSound.play();
} else if (option.auto || option.alert) {
sound.currentTime = 0;
sound.play();
}
}
}
}
function _setVolume(volume) {
sound.volume = altSound.volume = volume / 100;
}
return {
alert : _alert,
setVolume : _setVolume
}
}();