Greasy Fork 还支持 简体中文。

HIT Scraper WITH EXPORT

Snag HITs.

目前為 2015-08-21 提交的版本,檢視 最新版本

  1. // ==UserScript==
  2. // @name HIT Scraper WITH EXPORT
  3. // @author Kerek, Tjololo, clickhappier, and feihtality
  4. // @description Snag HITs.
  5. // Based in part on code from mmmturkeybacon Export Mturk History and mmmturkeybacon Color Coded Search with Checkpoints
  6. // @namespace https://greasyfork.org/users/710
  7. // @match https://www.mturk.com/mturk/findhits?*hit_scraper
  8. // @version 2.4.7.5
  9. // @grant GM_xmlhttpRequest
  10. // @grant GM_setClipboard
  11. // @require http://code.jquery.com/jquery-latest.min.js
  12. // @require https://greasyfork.org/libraries/jscolor/1.4.2/jscolor.js
  13. // ==/UserScript==
  14.  
  15. // fixes jscolor directory detection to work with greasyfork hosted library
  16. jscolor.getDir = function() {return 'https://greasyfork.org/libraries/jscolor/1.4.2/'};
  17.  
  18. //Fixes it so as not to rely on GM's storage schemes
  19. this.GM_getValue=function (key,def) {
  20. return localStorage[key] || def;
  21. };
  22. this.GM_setValue=function (key,value) {
  23. return localStorage[key]=value;
  24. };
  25. this.GM_deleteValue=function (key) {
  26. return localStorage.removeItem(key);
  27. };
  28.  
  29. var default_list = ["oscar smith", "Diamond Tip Research LLC", "jonathon weber", "jerry torres", "Crowdsource", "we-pay-you-fast", "turk experiment", "jon brelig"];
  30. var ignore_list = default_list;
  31. if (GM_getValue("scraper_ignore_list"))
  32. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  33. else
  34. GM_setValue("scraper_ignore_list", default_list.join('^'));
  35.  
  36. var include_list = [];
  37. if (!GM_getValue("scraper_include_list"))
  38. GM_setValue("scraper_include_list","nothing includelisted yet");
  39. if (GM_getValue("scraper_include_list"))
  40. include_list = GM_getValue("scraper_include_list").split('^');
  41.  
  42. //This is to update the hit export symbol
  43. var symbol = "☭";
  44.  
  45. //this searches extra pages if you skip too much, helps fill out results if you hit a chunk of ignored HITs. Change to true for this behavior.
  46. var correct_for_skips = true;
  47.  
  48. //This will hide masters if selected to "hide"
  49. var hide_masters = false;
  50.  
  51. //This is to test sorting by TO
  52. var sort_TO = false; // by pay
  53. var sort_TO2 = false; // by overall user-weighted ratings
  54.  
  55. // this fixes mismatched names due to Turkopticon using an older value after the requester changed it on mturk.
  56. var fix_names = true;
  57.  
  58. var SPACER_TEXT = " | "; // spacing for the control panel
  59.  
  60. //Used for theming
  61. var rcolors = {}; // random
  62. var wcolors = { "highlight" :'#1F3847', "background" :'#232A2F', "accent" :'#00ffff', "bodytable" :'#AFCCDE', "cpBackground" :'#394752',
  63. "highTO" :'#009DFF', "goodTO" :'#40B6FF', "averageTO" :'#7ACCFF', "lowTO" :'#B5E3FF', "poorTO" :'#DEF1FC',
  64. "hitDB" :'#CADA95', "nohitDB" :'#DA95A8', "unqualified" :'#808080', "reqmaster" :'#C1E1F6', "nomaster" :'#D6C1F6',
  65. "defaultText" :'#AFCCDE', "inputText" :'#98D6D6', "secondText" :'#808080', "link" :'#7B8F8F', "vlink" :'#40F0F0',
  66. "noTO" :'#AFCCDE', "export" :'#86939C', "hover" :'#1E303B'
  67. }; // whisper default
  68. var sdcolors = { "highlight" :'#657b83', "background" :'#002b36', "accent" :'#b58900', "bodytable" :'#839496', "cpBackground" :'#073642',
  69. "highTO" :'#859900', "goodTO" :'#A2BA00', "averageTO" :'#b58900', "lowTO" :'#cb4b16', "poorTO" :'#dc322f',
  70. "hitDB" :'#82D336', "nohitDB" :'#D33682', "unqualified" :'#9F9F9F', "reqmaster" :'#B58900', "nomaster" :'#839496',
  71. "defaultText" :'#839496', "inputText" :'#eee8d5', "secondText" :'#93a1a1', "link" :'#000000', "vlink" :'#6c71c4',
  72. "noTO" :'#839496', "export" :'#CCC6B4', "hover" :'#122A30'
  73. }; // solarium dark default
  74. var slcolors = { "highlight" :'#657b83', "background" :'#fdf6e3', "accent" :'#b58900', "bodytable" :'#657b83', "cpBackground" :'#eee8d5',
  75. "highTO" :'#859900', "goodTO" :'#A2BA00', "averageTO" :'#b58900', "lowTO" :'#cb4b16', "poorTO" :'#dc322f',
  76. "hitDB" :'#82D336', "nohitDB" :'#36D0D3', "unqualified" :'#9F9F9F', "reqmaster" :'#B58900', "nomaster" :'#6C71C4',
  77. "defaultText" :'#657b83', "inputText" :'#6FA3A3', "secondText" :'#A6BABA', "link" :'#000000', "vlink" :'#6c71c4',
  78. "noTO" :'#657b83', "export" :'#000000', "hover" :'#C7D2D6'
  79. }; // solarium light default
  80. var clcolors = { "highlight" :'#30302F', "background" :'#131313', "accent" :'#94704D', "bodytable" :'#000000', "cpBackground" :'#131313',
  81. "highTO" :'#66CC66', "goodTO" :'#ADFF2F', "averageTO" :'#FFD700', "lowTO" :'#FF9900', "poorTO" :'#FF3030',
  82. "hitDB" :'#66CC66', "nohitDB" :'#FF3030', "unqualified" :'#9F9F9F', "reqmaster" :'#551A8B', "nomaster" :'#0066CC',
  83. "defaultText" :'#94704D', "inputText" :'#000000', "secondText" :'#997553', "link" :'#0000FF', "vlink" :'#800080',
  84. "noTO" :'#d3d3d3', "export" :'#000000', "hover" :'#21211F'
  85. }; // classic default
  86. var cucolors = { "highlight" :'#1F3847', "background" :'#434e56', "accent" :'#fbde2d', "bodytable" :'#f8f8f8', "cpBackground" :'#384147',
  87. "highTO" :'#6FFA3C', "goodTO" :'#D9FC35', "averageTO" :'#fbde2d', "lowTO" :'#FAB050', "poorTO" :'#FA6F50',
  88. "hitDB" :'#d8fa3c', "nohitDB" :'#DA95A8', "unqualified" :'#ADC6EE', "reqmaster" :'#BFADEE', "nomaster" :'#ADEEDF',
  89. "defaultText" :'#f8f8f8', "inputText" :'#D8FA3C', "secondText" :'#ADC6EE', "link" :'#BFADEE', "vlink" :'#DCEEAD',
  90. "noTO" :'#697048', "export" :'#ADC6EE', "hover" :'#426075'
  91. }; // custom default
  92.  
  93. var uwcolors = {}, usdcolors = {}, uslcolors = {}, uclcolors = {}, ucucolors = {};
  94.  
  95. // Random themes are still in testing stages! It can, on occasion, cause very minor lag on initial page load.
  96. // NOTE: The issue is on the inital page initialization; it does not affect searching/scraping.
  97. // Comment out the next line to disable randomization.
  98. randomizeScheme();
  99.  
  100. //display your hitdb records if applicable
  101. var check_hitDB = true;
  102.  
  103. //default text size
  104. var default_text_size=11;
  105.  
  106. //set to "true" to override checkbox setting and ding on new hits
  107. var newHitDing = false;
  108.  
  109. //DO NOT EDIT ANYTHING BELOW THIS LINE UNLESS YOU KNOW WHAT YOU ARE DOING!
  110.  
  111. var maxPages = 20;
  112. var type = -1;
  113. var status_array = [];
  114. var shouldDing = false;
  115. var useTO = true;
  116.  
  117. var audiofile1 = document.createElement('audio');
  118. document.body.appendChild(audiofile1);
  119. audiofile1.src = 'data:audio/ogg;base64,T2dnUwACAAAAAAAAAAB8mpoRAAAAAFLKt9gBHgF2b3JiaXMAAAAAARErAAAAAAAAkGUAAAAAAACZAU9nZ1MAAAAAAAAAAAAAfJqaEQEAAACHYsq6Cy3///////////+1A3ZvcmJpcx0AAABYaXBoLk9yZyBsaWJWb3JiaXMgSSAyMDA1MDMwNAAAAAABBXZvcmJpcxJCQ1YBAAABAAxSFCElGVNKYwiVUlIpBR1jUFtHHWPUOUYhZBBTiEkZpXtPKpVYSsgRUlgpRR1TTFNJlVKWKUUdYxRTSCFT1jFloXMUS4ZJCSVsTa50FkvomWOWMUYdY85aSp1j1jFFHWNSUkmhcxg6ZiVkFDpGxehifDA6laJCKL7H3lLpLYWKW4q91xpT6y2EGEtpwQhhc+211dxKasUYY4wxxsXiUyiC0JBVAAABAABABAFCQ1YBAAoAAMJQDEVRgNCQVQBABgCAABRFcRTHcRxHkiTLAkJDVgEAQAAAAgAAKI7hKJIjSZJkWZZlWZameZaouaov+64u667t6roOhIasBADIAAAYhiGH3knMkFOQSSYpVcw5CKH1DjnlFGTSUsaYYoxRzpBTDDEFMYbQKYUQ1E45pQwiCENInWTOIEs96OBi5zgQGrIiAIgCAACMQYwhxpBzDEoGIXKOScggRM45KZ2UTEoorbSWSQktldYi55yUTkompbQWUsuklNZCKwUAAAQ4AAAEWAiFhqwIAKIAABCDkFJIKcSUYk4xh5RSjinHkFLMOcWYcowx6CBUzDHIHIRIKcUYc0455iBkDCrmHIQMMgEAAAEOAAABFkKhISsCgDgBAIMkaZqlaaJoaZooeqaoqqIoqqrleabpmaaqeqKpqqaquq6pqq5seZ5peqaoqp4pqqqpqq5rqqrriqpqy6ar2rbpqrbsyrJuu7Ks256qyrapurJuqq5tu7Js664s27rkearqmabreqbpuqrr2rLqurLtmabriqor26bryrLryratyrKua6bpuqKr2q6purLtyq5tu7Ks+6br6rbqyrquyrLu27au+7KtC7vourauyq6uq7Ks67It67Zs20LJ81TVM03X9UzTdVXXtW3VdW1bM03XNV1XlkXVdWXVlXVddWVb90zTdU1XlWXTVWVZlWXddmVXl0XXtW1Vln1ddWVfl23d92VZ133TdXVblWXbV2VZ92Vd94VZt33dU1VbN11X103X1X1b131htm3fF11X11XZ1oVVlnXf1n1lmHWdMLqurqu27OuqLOu+ruvGMOu6MKy6bfyurQvDq+vGseu+rty+j2rbvvDqtjG8um4cu7Abv+37xrGpqm2brqvrpivrumzrvm/runGMrqvrqiz7uurKvm/ruvDrvi8Mo+vquirLurDasq/Lui4Mu64bw2rbwu7aunDMsi4Mt+8rx68LQ9W2heHVdaOr28ZvC8PSN3a+AACAAQcAgAATykChISsCgDgBAAYhCBVjECrGIIQQUgohpFQxBiFjDkrGHJQQSkkhlNIqxiBkjknIHJMQSmiplNBKKKWlUEpLoZTWUmotptRaDKG0FEpprZTSWmopttRSbBVjEDLnpGSOSSiltFZKaSlzTErGoKQOQiqlpNJKSa1lzknJoKPSOUippNJSSam1UEproZTWSkqxpdJKba3FGkppLaTSWkmptdRSba21WiPGIGSMQcmck1JKSamU0lrmnJQOOiqZg5JKKamVklKsmJPSQSglg4xKSaW1kkoroZTWSkqxhVJaa63VmFJLNZSSWkmpxVBKa621GlMrNYVQUgultBZKaa21VmtqLbZQQmuhpBZLKjG1FmNtrcUYSmmtpBJbKanFFluNrbVYU0s1lpJibK3V2EotOdZaa0ot1tJSjK21mFtMucVYaw0ltBZKaa2U0lpKrcXWWq2hlNZKKrGVklpsrdXYWow1lNJiKSm1kEpsrbVYW2w1ppZibLHVWFKLMcZYc0u11ZRai621WEsrNcYYa2415VIAAMCAAwBAgAlloNCQlQBAFAAAYAxjjEFoFHLMOSmNUs45JyVzDkIIKWXOQQghpc45CKW01DkHoZSUQikppRRbKCWl1losAACgwAEAIMAGTYnFAQoNWQkARAEAIMYoxRiExiClGIPQGKMUYxAqpRhzDkKlFGPOQcgYc85BKRljzkEnJYQQQimlhBBCKKWUAgAAChwAAAJs0JRYHKDQkBUBQBQAAGAMYgwxhiB0UjopEYRMSielkRJaCylllkqKJcbMWomtxNhICa2F1jJrJcbSYkatxFhiKgAA7MABAOzAQig0ZCUAkAcAQBijFGPOOWcQYsw5CCE0CDHmHIQQKsaccw5CCBVjzjkHIYTOOecghBBC55xzEEIIoYMQQgillNJBCCGEUkrpIIQQQimldBBCCKGUUgoAACpwAAAIsFFkc4KRoEJDVgIAeQAAgDFKOSclpUYpxiCkFFujFGMQUmqtYgxCSq3FWDEGIaXWYuwgpNRajLV2EFJqLcZaQ0qtxVhrziGl1mKsNdfUWoy15tx7ai3GWnPOuQAA3AUHALADG0U2JxgJKjRkJQCQBwBAIKQUY4w5h5RijDHnnENKMcaYc84pxhhzzjnnFGOMOeecc4wx55xzzjnGmHPOOeecc84556CDkDnnnHPQQeicc845CCF0zjnnHIQQCgAAKnAAAAiwUWRzgpGgQkNWAgDhAACAMZRSSimllFJKqKOUUkoppZRSAiGllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimVUkoppZRSSimllFJKKaUAIN8KBwD/BxtnWEk6KxwNLjRkJQAQDgAAGMMYhIw5JyWlhjEIpXROSkklNYxBKKVzElJKKYPQWmqlpNJSShmElGILIZWUWgqltFZrKam1lFIoKcUaS0qppdYy5ySkklpLrbaYOQelpNZaaq3FEEJKsbXWUmuxdVJSSa211lptLaSUWmstxtZibCWlllprqcXWWkyptRZbSy3G1mJLrcXYYosxxhoLAOBucACASLBxhpWks8LR4EJDVgIAIQEABDJKOeecgxBCCCFSijHnoIMQQgghREox5pyDEEIIIYSMMecghBBCCKGUkDHmHIQQQgghhFI65yCEUEoJpZRSSucchBBCCKWUUkoJIYQQQiillFJKKSGEEEoppZRSSiklhBBCKKWUUkoppYQQQiillFJKKaWUEEIopZRSSimllBJCCKGUUkoppZRSQgillFJKKaWUUkooIYRSSimllFJKCSWUUkoppZRSSikhlFJKKaWUUkoppQAAgAMHAIAAI+gko8oibDThwgMQAAAAAgACTACBAYKCUQgChBEIAAAAAAAIAPgAAEgKgIiIaOYMDhASFBYYGhweICIkAAAAAAAAAAAAAAAABE9nZ1MABAgkAAAAAAAAfJqaEQIAAAB89IOyJjhEQUNNRE5TRENHS0xTRllHSEpISUdORk1GSEdISUNHP0ZHS1IhquPYHv5OAgC/7wFATp2pUBdX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';
  120. audiofile1.setAttribute("id", "ding_noise0");
  121. audiofile1.volume = 1;
  122.  
  123. var audiofile2 = document.createElement('audio');
  124. document.body.appendChild(audiofile2);
  125. audiofile2.src = 'data:audio/mp3;base64,SUQzBAAAAAABFVRYWFgAAAASAAADbWFqb3JfYnJhbmQAbXA0MgBUWFhYAAAAEQAAA21pbm9yX3ZlcnNpb24AMABUWFhYAAAAHAAAA2NvbXBhdGlibGVfYnJhbmRzAGlzb21tcDQyAFRERU4AAAAVAAADMjAxNC0wMi0xNiAxMzo0NDoxNgBUU1NFAAAADwAAA0xhdmY1My4zMi4xMDAA//uQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAWGluZwAAAAcAAAAlAAA+CQANDRQUFBoaGiEhKCgoLy8vNTU8PDxDQ0NKSkpQUFdXV15eXmVla2trcnJyeXl/f3+GhoaNjY2UlJqamqGhoaior6+vtbW1vLzDw8PKysrQ0NDX197e3uXl5evr8vLy+fn5//8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA//uQZAAP8AAAaQAAAAgAAA0gAAABAAABpAAAACAAADSAAAAETEFNRTMuOTkuM1VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVTEFNRTMuOTkuM1VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVV//uSZECP8AAAaQAAAAgAAA0gAAABAAABpAAAACAAADSAAAAEVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVYAGA2AAYsgQwh0DsmF2g2kgijWJF27brJ0vJilIk6SBUnSJ0mF98I7KLdQiTpMKMJk5R05ybh4XOSC0CZOowu2UcgjOcI1FtH5IC7ajCZhd6DJ7DPWTmkwj0ufIHI3oIzycs2C7cG1HI9UcK5I2kFz9QyGTmo5HqB6CKOmLns/qkf/7kmRAj/AAAGkAAAAIAAANIAAAAQAAAaQAAAAgAAA0gAAABJzKHgwmj0gVNwppif/rG+u8gVRyQMCPtbrK2Uqgtwqg1aky6xBr+g1mTJjmtJl1Bo05JkyYNGjRoyZMmTBo0aNGe6h6MzMTT0GUu38yZMmTBo0aNGTJkyYNGjSDJKH6xwIsVcQAIAiFtEAD+txLTkCjwprLYmCKLmMYch24IhEoiD+Pxneqcicvl7oMQjruM4lrkBcZnGYRjlhOAQgxxA03udiKR+LwGxwGhV/D+ClbZ5h/HnlERd9bbUPspFcyoe54UioH2S9QJEessavfwoDaciMbDQThCGV2yKUTNdVVjedbMdCvV76CnznUaHv4kM54zx5HLYqH9o7Wc6rNNzY1BEPwuCoV+p9Na0wvmR5EUD+icrqzyHePH1DjrhYzCZIUsOivniSxjTOtXudX86kVra/UiGNaHwsJyM/eZlUeG+TGmCDcYHoCABjEckEQufyIrRo6QCkn8uexCTAvdTAoKU6Iki4wPKcUM/RRd0l7m0MGgsJBuL6QGJo2F0b/+5Jk/40EvGw3CGlLckXgFAwEIwAdaab0TGHtyn22HFSUJbmNGUJIZD3NJBFG2mK26IBQgTFZi24Z/7QRR0gjUSMSHQ23DPDIe0ckCZPpGKxW0FAwqK3o53sEEPPNQMf/JoIfz2vNBFHPVGLRki4XDZYDCEVzVFbekBImjmmK26QIHI2yM2QBjEaNHOcEtttqTba1kQAkIji51RQ8vcAEgb5a9lgJIXZoWyK8W2g/DkGO4+k5all1/GhwBDsENpuAIjAckaQCBC+qcZjIA2c/STVfhs+rkKQ4olBBzZjICRgQWKqqzInoCBVqA41UNjiYCdDO5a+9AyNd650R10LMgsuogMVsLWGygtUW4AVr+hDCF0TDXLMOPW57EIrto8NIIysTGFtoLpju2/8WisOuErhnaVCbqWSVip28jMjoIhDF6elGVJunlcP34xTwK/bcITDduph/773///////5+G9frV0OeqQfnFkHKIrRdgGgNWEkFlAGzaOQow5znMsqIs285kzOf8zm52Yhg3Qiww4shcDsP04TgTEceD0DhMHAS//uSZIqA9udURmsZwvCOLEfwPYtuWUUbHAznC4j1FmMBhqkwDxZSdvtNi+rPzc68vffNgkDtAHALH8sQ4a+Ztr4qmTcvfGxtq2nPDKt074qzfefqJddN+Pl7/qX3XNvmPVPxFJnKp+990+Xwo8rdVTdHzf9azQj4QpUQCgxgGPi1RnssbMEYUZHWBcdKtJgqDJITaQiFKNbF0h0F4ACoMPMvXABh4jCy+6fSjrU6aGXiMcUDHjIAsInqduIXqM5EW6Aa00zKNRc55TKEnlETi0tSXHEAGyl2E0FH1UF4uK99FnBDBGuR1w2rJDr4RsZKzptHABKU/XkUzct59MZDhoOJGLhpwCNpzR2ptcd9QdLhn7wLJEjo8TtWSsYU7bgsAiqpUFwo2Pe7kbtS/crZwy9ocm3nuYiD6RSSvo48MMofgBrSK5S2IViPBXRskOLUyqi4CxDjOiaFn///jQKfxeFhgBgDAFjiIE8KQoFGDX////4uEKK5R7v//////+l9IqotIpRSoFJJSrkzVKAfBJCn0S46laOJRN2XM2WLvuyugf/7kmQQAASfSU9p+GL0MuM5SmCiOBCFJSmsPSvAyQwlPPOI4P65NtlS6XK3jjgIKKRIpxtxafh2QNvXz+OyaM0Utm4cjVyOxntjGgUg3ZNSBCngLjt1+1fuVrHp0ExXMWT1C+MjRqC6tdcHFMsJKy1Zo0sPzEtiVaKrlLpVNooVvwU+dnJ+Zoy+wvMVz/HrtFscv3O/1r/W38BFAAAEndSgARguGaSWsgFKLOIt8gj/06ihRJGUWScTBs6DX/+UD3/////+y/kMpqh0AAYgAAAAAQR8gI/IiKjTKIW0xhrL3WgY/jGU0dqTz2zEooaod2ix06ukOUTIWcZ43meoSaJI5npuo8pCGliVNVe98VmJgAhAmRrwRCGHprq4hcYHyImRComssueNJzOsJoyVN33ZXKn9E11osSilW+/HqwyeZnyOIrvyZjHPTV21f+ocAkJEqiEYC98N5q3EaLleuqHFxqAwA9fnt9vrgwoQg5klUUG+3/35X+v/////u60v68xVAAAEAAAAdBwNORROIFOE+lYy9zlBlFitydB9Emcnxzf/+5JkEAoEUUPIYw964DNjORwkpTgROUcWrODL0MQMpLShFOBihn8t6fQIT2qagxGJjQ05kw1qnBc0zIqlyhQ/wZYLtVQbIlxivW6MzPWHUKBG3Bi+a0bfj6xJWS76NmWkfM1MVg6+cwae+YL2mYtvamZ5q0/z/6QZvF8+VziNaOsYIgwG/2Z4KsEJhNHGAGBEN7C2erwCypOSWlSCICiX1GL9P6iBQ8IiLCRRokGB4OfV8t/V//U//u/373yPucq+hAkjDQiCFR9FDIACoaIzFvQ0dK9e65kqmLPM8DrzszHX5oquNDO/yfnrHeR93Vbr0UfuidVlcfeBAaDQBVBE9IO/vt/Wl+bqdSAS5Mhq6BR5pprOazl06i9KlT5pqP3MarLVzIPOz1ZGu9tENqWUYy63cOdCqVu/9s////9On2TeDg8etN1LB4mkiGBlNtACSBGlQnjOtdNWDNbTq3+uhisrFYIhEM+zdu+j//W//b/gAoYaMCzAAmzY6hUEgJWMgJJ0TOMZsAES9XasZTNAaFnOirWz9T0dZ/FbLwMBhyEU//uSZBKABA9Qx2MYQvI4YlktNEI4EY0rF4zhC8DhGCOgsRUwFSjod2eWZ38vxwpdRd7WlXHzzU6YOFnDEUzFBrX/vmiFIKA+cYJ6Rr+07RJL5EObOiYc5UJmJR2+rSV+5lr9u++47jqL6muUqZ4+OF7++fpSbC5KfM6/3Clz/1hMGBA5sqRii29YKg0PKt7GOFa/etW+HHtYMq/T0M7XA7lxn93nWdAyKkqUibSOhXcKJcK2tVdvU6VAAFDRKAASg2cRAoYDlEO6CVjwIhkZkRoqBixcqzL5bGXZkjdI9T0dWcpo3PdqR7eN+5n8wravu8xBQlrrBxpqHAa8h0o5b/42EUQhySD4nJNSRHX4mupFk5l0WKkdjj+VqWa3m5i+fGRZcVF/zA9ZSKqLodye/etY6pK2JTB1ITMm1VnFNDhIzW0S31Qk0AIfJQAuJCwvrl9Aaav/RvMZ8xAEGGAjbpf/b+0uvee70/utxXTM39f9aVta7+J5c7KPff1I1gEAzXkQDEjEfoDWX4R5WjRJOcqHAxgpAFDUk0DoWqboen7x+//7kmQSAAMXJ0lp7HpQO2PZnQjiSpJBiSOsPM3A4BUl9GCJMDk9XG769P9sDG4uMHB7o0DKLiKSjPunpr6p/Pv196RYMkDF67g9uOEZqoWbHQFtHP8styspS40l3JHLHU21VtNguMNANtxujgQk1tGqYJYQCQqP+cbf0fsbTcy/8O+w9nXklB7qInw8zHXWpzpFR3fb/v5S/pWlx+tKQmw05WQC2XbEiJLuBcLYA4rSxQirxIrAUomjN7oXlkMOd6h6iN04C/LuM70yHRJnGNRHCApzuLqXM7E4GYcoL86TJ8bdfHieHPJTddUoju86aPmOzOhie+tzmIE4IvKbJv3d04x71oKxs173tusyKbeNsmbbX3c5Q4kYj/+1tuPn/357/+/cZ9/7N27lF/LxDXQiZCUgoAsotvdxB96+8eZkmI/5lLgTRDJysdhn+8h1MSp0l9X/xlfDA4WSj/0N/fbk/ur7cUyc0moAAJABQ2XA6ZXI0UPCO4ooamgsQCk3Snm6z4AiOYR94muBuyvETgMCX6b2bysSwdBMUGfmaN/K2LT/+5JkHIhFY0rHyzl68C9luUEYRUwSJSUqDGcLyMQkafRRFX65LBTfLCxlu6QRxsp1LPXS22pl+YGlbO2JFXsa09iM/3YmZx339Xjq6tqPHqcM8Y6Fp9XofGG4XY+WZrnlZ0SWPXz8YBMIbnTBEeHIyKiHH9W0C+U8RjvaEVZjKjV6YuXxxV8b5h9P0mTiM0ctvbUqqxpxKbPrCVTy7z///6GAUmlpOokNA4t/sQjSdqen1eiMQmIBNk4GMF3tX0BVlpOp/GhGOpew+9tLDzKzf205F+UU0FjEg2C/FWrfp5gUoUIm9426d6S0ln6atSykhDhulp6SzH2QGWdPP082FuhHDLkjsdk2OEUk2+c3uSw9e/f6qsLl+fPxyfFTCxKK/JVKB0MSu3u8rZU2G9ff9uSzYHu3LN+abacqVLmHGekR5Ny3llK20prF7O9lKFwZa3n+U1ZySx6itlNEuDLKJAVy8n//9lmGYm3TX//ZOJM/5ke+3l/6n/5X6t9a//VP3AYk/r9nX9cpAwAg0hAAS22fMheJ4JLoyEc2AU72O/KU//uSZAyAND1KTmsYWvQyI3kcGEc4EYklOUxlq9DJl2MU9rUwngcuNWpdyJvIoGqDL7+P0iDqQy1RqaiCRrI1UVY5FftblT0FpYVKbGNS83ZNeQpFCWWADDBkHKfJMWv/WBPX9+mB0a1mww6CoTfFfbTl81nggeffyIo6z/L1CsVD4t/kEHt/+kIbp/9M+09/+qxYDAFwAAEDAXFiEjqBACol+b/9Q2PFYkikoE5ELDHEwLXAGJG8LluWEV3/xBlf//18sHbgkgQCAAmpVKX+UEhiSywSUpB4GYS/4ASrDSm3vWIDl9glTTYx//+siEnB3Gza+SvVV/mWNssBQNGOf3l8qARcmKNzYxJgyUFouubkp6nFgSjKzAcQD2W0zyBiA1Qnbt1nzF9ZgLgO9BJ8zLBM3rSRNBASO2ovEQJHrnFiMlN/NTwl//fqqQtVGUQ5SwIOoCLoqJrAdBtTuPBX//lAAWRwkiaGclguI5AUJJpYsBPymSwGUUOJ4Qf/////rL4l6rYHGEAwCABg2ARENkaSzOPLzEGqX7OWbw/DzSi+p//7kmQPAAR9Rk1rOpLgMiXYsGDtTBAFFT+sPauQu4zktPU04AA0Vzuz0smCqdHmFrL+e6KCNWGmr7vVYggNtW6tv64fsMuUzSYrLoFREbF1JmROjbc6hycHD6R4TeJ9dq0BuAQAd6y+VFpBqYKMcN+iN0cTl/OEWAjRPF0sJ0SJhaUkpE8iZCQD36Q+hEEVNlIcscUzVSQ250qgu95VYuCYJC9v/+gXDDmsBACo4YBzg5oa9hmAmI8i6DfB3n6i8j/////+SIKp2BIBAIhAAb0zP4gj03rR7QUI27rzMN6lHsCjavdwu7B6hYuH/+DJFHHw4er8FfD+YbBELhDMim2YA6RyFajWxWPAzPGvdbv1BbSXfyodpuktqgaRgzBPXOEqh5MCVHVaz+OEZZf9Yw2tsqHCpuorXcXb7v1jReoH0KczqYFhjxoaAA+wAGAuoxtQIJUZgFABCSJ9Hpt+l6RkXaLJdQnggoK+SRx9///yP6//3f//orKA0GCSCABOfoBYuhvFzFxbBOhaIBoKd9DTwD6QGnyIipN6G4yf/txvEnf/+5JkFYADoUHR6e9q5C6jKUo8TTgO4R9D7GILiNSM5LTWwOA//4MEl6XrBY9lQFKA0y1tIyElBZpN2RQ/Qb0iSLiHSJIc5cNOiOEewwDc4kZku+kMYkBxku+lSSPazE1UttRsBXxdIIFyH//0sABAAP7/x8Tq2vkF3AmfIIP85MGENPpppv/ruVAbwZhgyzu9+tv8Z+39tX9nv6mlgBQBQWAAADi65cMq6cVn1gQlKE6YBN46ekQtb38Ie3ZiahU7/6u8douY1fDmu7j6w0W/cszSAXwCQl+dWOsTlbqMD31mR/zpZKxq/LRUGNKraygREZc95TIeUk9EwGXGXKHWUCIEDNUOzKR6lIFk0bnFGZq2NpAAAAAAwA1tVTx/DfFuZtlQ/BOG86QIQ8lvlEc9Ropa/rRcdQygBoI+LXvR/////1/6f//XiJBaYGFkAAAKKGOQW4sykZsgGnOymAFmxnEkYFimnTDFskQPia3r5inKMyTP+uvwYvy95cAKo/konWiZBei4Z+p/1s/rLHq1FQnZLOt6xiDLKTegTzFBVZOC//uSZC0AQ19HUfsPauI3Iyk9Ng04DukfR+xhq4jZjGT1F6DgoJGiHKjEuHG6aKPzjfpGSDARkAAAAAxwOPqrVnQLwvMvSIAAujuaqWTARks+YDSMUoR+P/mCSgWYOwc5Z/d//4p/3//3+U+36ohgCWASbAEABCrKHheqxBXUpyJcRSqqw9H1HBUSeGPZR3PigaiFuWYZ1JcX3V5I9d5/sAs6x/peDlAAhA+FtaLHQkIv+t1v6JiSh/0kjVnqNnGFGglDes1Nh+Hmb+s0Dnn2pGQwwwRYUNSSKzBv1NqdRmb9FUup8EABBxlhlIpMpIsAnAKDZ9/KmFxEpX6rlVX/2OleWlhQymJNGgCgtNO6d9n9n/0f/+r////oUlnGCJAxmAAAAD7EVLokJX5lZbQne+6z+Nwh5g7Giam+sItdqkI0sM4F1SwVKSUrHaPH//SqWerOO840I7kXYf5vWXBHCKbrlRv+VH2+LI3bWs+O0LA2cwaYiqMs/1GQaRfNeuIYXn6llaD/R/WYp/c6l2AkYEAgkHyZ8d0pailA8lh1Horgqf/7kmRDgAONSNF7GGrwMaKpWj4lOI31I0fsYavAu4WodBKskggCfTLzf92A4cZ+YV+lTP//cytfBB3//od/U7qmcUIYCBqgAAIJQw+CF2pnNYhtBGV5jsFVH3mG6tGJu4fAVv/XivS9bzqSiKDAQgFnL9/6hyP8qkOFWpiSVB6IFn+Zc4NQU7/Mh5v7HEvycOcs7MYgyUza6x9GMPT1lYWogdcnEif7Tpp9X/SNNes/WwCD1AOOh/5I4JAy/4OEyw/gKJsnN5PW7X/16//P+Kf//Vpu6a2r5Lobq2Mc1aKWkSghQABlpe6tS0W3ZHNoMFO6Fjdxfq5E1yQBJHLkFSffxBIG9/Mr8afQMRw/nNWPRCit3LdC+4ECPZuTUfN/mTIsg39BRh+mOav3LhAD2oyJw8CEL9GsmSJjHjtaxueIuH7jBqdIsEwTA967sh+yvfMF7KXNnDgBAoYA1AAHuUE4cyozu/OAQCLiIA4POqN/UxBOU+Yb//9HvTbt///5AD/oKa2oZhpsAgAwQT9TF2kna6FkEtQWhLQWoyCicYzWeiP/+5JkYYATtklRYxmC9C2Guc0s4lyN2Q9Pp75rkMSYY8hntTLhHyCvInGbaspRRsz/Xv15Gsrq/0WQ2cAMxBj+tQxgskiqumZmrfopP6kW6zMnidJ1VJIuFk1RXmBcKKJqeqJsnyYKyataLr9fU9OdOOqOFQg/4pAjZLbjx+3qQDdgsB1s0b4MECFH1/oO+wBUkXW+qDiB3v6////////CuKZp5YlVkgMQQMAQAF8aXQudFJf7EY8ItKDN2HEuzKGAJbip7WEvl7vvvNpWUv//MHrfNd0vi0jjPsak+/j8dlQucBtgKGICpakibD4yXLBa4rUQCOKp9a7cdaP1kELRrVWwXmPTpPOlJRFqDnVhbUSmfJpLjpIx/q+tIrpfWWmKT//o/99lFABACAAkAEgkqAomb80hUlgJhA5CUIVTKFRJbwsA8Vin6DZ9tyTPJNZ/+VDX//+qQsQHq/WZRIAAeGbjTDHEWIpQmCJ6EkAYZftuyDjYU0yxhDg6q5HkgOXsMS1ty5nDzNYliT6/HAxi9HrSiTyZXZzb9C4AK3iAD3ed//uSZH4BBAtG0WsYmuQ2QqlNFec4kT0dPyxii4DMCSR0nDTgGPBswNMoDwyzgyojSecspuxotBnrGufpa0BBxED9aiGHAbGDZ2yyQAbSbJGZKjQBCJBseQOmrSGCAhEX+b+nOmXoqRKp8YAgAAB64C0f+aweCxj8G7q/u6gQz5X/uNZ28F/qTCmDkh4P19W69On6tv///2dWn+impUhQVQBIA3xfCDW1cMlArOFtsNjcpWo0ZXTwhUb2u9B1qF0sDi2pfGceayX2nzFXalVHc2/U9+7kKkQYQAEuIaVmdRRF2J9PLRzMXOSL7O7JJLRdQ+xST/GbIKVVIVizw1WSxl5DyJE9zpDAxAMcRM89Uah9ldTfQMCdPeowRI0hr/r/+naUQvuAJxjR3k+MlxAFDi6NgkaMTMJr9Fcg1+VHBIBOd53/yFR4dUCaiABWetSbq/7E2bEHU4wEQgQBtEy+46lS9fa5nzOhVEaFmbE1wugUIBc5TzM5eVSNkPWvWtrkFUZt+LqfAzEr+wnaV4RZAilHyR5sAYCJsnjM2QTKIb4SLv/7kmSCggQiRlFTGJrkNWOIoEdSShCRFz+sPouQyIhotLYc4tSSUt3rlIbyzz9AWw8tZqsxBIKKXLpq1axSpLX1jWAaPl8nmessCNSkxlzhv8sEDr84SwB26P/1dItASHDZEA/38d7lgAYOdKQ+Taf98rz0UNn/njggBe7J/kX///9v6u7Qun//V/8UbU76VTCAAABAEymog8psvB40pDN7AuoEBKaDM7TE1kB5CgZE4PgCL044Li0Hcv91uJEAY512nv3LDgBcTlVutEiwAkxeDIQbfAaSM7m/7W0yDFogBwyxsYu9GWqDAFlGp/v/TxDv/rcFBUALd3z/24SE2xuhfz5WKjsSI0v5rOYeERBx7LGeFd9BkMmGRwzJtL/M7BcAMHs3bx7990Xn5//UngwFZa//qUxQCMQAAgABwAAzBpeIED8YJkyTt0YmaLS/+bNpkyTf/yuxuIg9auTkgnf+4hwLgVL/rrBAFAAAB6ujUis86u0vmFmNosFjat0HO16bxxo5a5lJ80CyaL+//ylOtflqklFmUStCqj79+OoHgAL/+5JkiQAFRkZLW3vi4C8DiUpB6koSURc3TWarkLuN57SSlOJYApTEDUtFIa4CBaMA8ijYvhZpbtymj8uh3z3rUHnedLrEqEATBQDHn1JA38kG5SGiCxtImRVJnLg0wusfWZKsOo/6h/HA3VGPGiej//6iLHWthdxDM0TCq4AwESgJhtp7TYEcLISWTUCP7V+rHEwwXlqn/6ALb/vo/9Viv0epevbd/uWn/RWqdABJAAABXy3qPaYyqz+vieRoUGnvRllj+uAKiMx6Jkz4xKInFbLHSIxTncNfD4KJLd9RCigZ2k3SgA7zwYmiYCgWKSwZCiOcxnBUuYyWCczSE9/9EIMDb7jz/luX//6l70W+f/3UP08rf1YZpoka1itUty3quoYhOlPP5lOGCYGyqLTWOWLRgoq2uWN6nuOgoT3//6g6C3ti3/5XEIZb//do9//0DDEi8o8VcqfMkaRUjEgEgAAARQK75ahVuaEtuKROH+h7/qozKP2f/9AGr//09zlf///x1dXtp/q6mgABYAAANoAuAgVCNLxWV8jy5hpA1BYD//uSZHsAFY1Iy9NcyvQso3mdGOo4llkjLU1ui9CODiSUZp0qKQuEIB83YSaDI5mmopIFjQWOpbl/7mxoaVihcSanEoNfUmJcMsm6jAUqMJCxrxJhqBKOxD4wEGnG6K5XbUERJUN89UPr6ZaC1kqr50UqHbNp0UkdGMA2qMNvHc1ZgGdBQ6VXtQAwOcG8JODPEs5cMwNqCD30EEzRAMvBez0STC0kvj6SfEUDBZPc6V0f/+wRKKR8442s0cPBZRcIAHsHqkz/88fQSAIY3ZbLDlhxhfPRQHhWJfEwIjLf/HQhGWowAAIAAC8jK0lOpYywgyFTJUpAwbVEhWxGy8ZgQkGdCCXcf993AhbCgohi3UfqZ95ZEgy970SqB5RD4wCBoO5RCA2cFvyATGQ9qHJhBEh7EptyTBIJNXjARgB4qXLkfEJtLjLD72uRui///2ELTsZ9/F8AsAGi0j5HV8q7QQHqQBBqzj+ryhYQxsdxlS0VYzBdzPgIGlr8ULZAACA7VZ+v1ehhOQaE///guwgGSrOdqYxhX6ALvP3MQurfd2tQA//7kmRfhAXYSUnLnNLwKkIaLTwlOJh5JSbuc2vAsJzjgJa1cDM36/2Y9FaWw7BhYMC6P5raexhgDlZhz9rv9vr//2Zz/9no//0DkAAAAB+pSMjh6FqYpWGTpgkUUAgWCrZXiZyYFKBo0NpRYYS134wYrFRQaJHYz7y+PBeHs6lNu4ShJ/bVuMw86oCCJ0hhAoJiQRQFSCXu4YBNZlQFNwKAGj57MwYLqXyzCvXuX2o2f/90CAhp2dz/0+I0DMOkfYwzZzwooAQH7Z19duRiwVAlr+TUtBqicuWEw0iC0l5Ja1ICoBWOMGs6lEUbOYaHrM/+TbX3LFiGK75v3aDAlv6L/3Eo0/VM7/+7Renf6E0Q4NqIzLIjZb9E4JcWs2MULQADwcv0hypI+YjhKPqb//////////4fQGWa1XxMCQAAAFsFQDBYGcQw/jRDi8R5KrGilSBgEnyKi4CsovjUr8ekZN105bw5cHQdI65Q34Ml911oEyoZiNqKAEjOhLTAABpsZltxu5g6iBjXUIvd5HRkFIv/fMNPL///+sbD/w/aq7r/+5JkM4EFtUlL01vS9Cmhic0N4yaX1ScirfNr0LeGaDSJnJq0WWtXGZGFUPtTY56hhHhFvO3vLdOQzS5WVZkMdi4MBBQelDb525E5Y1KW839V/Ed2a6/7kfS8V9J+f92H2/ldv/8s0TJMJm03hVCkBR5k0UwLMQkRqBdegBRMNbncdhkV/sv6hRPFXfDLfVru+7//yX/7f/Qoopv62AAeIwAGUuJgB2HUd89jMStTQMMA4larDpDB02iNDlGoNVQMnGR6Nd//KoFa79Ndu5x8AAOO5VrbSIkYkFB20aDgLKwRPt5TFgMA64qzwGkbPY6FQkOMYc7+q2aHfLL//rp1JvvyWaggxIDLwt7j+VMMlRqwORAL9VMcy0ZiZY4ctgDjcFZTC0c88yUDatMJtqtJQs1VKWzGpzt6CliBCNTSfGGoqtkwUCWNnzUQiYhB5ajlR9r2o8RADKXvhjglB27doA/vG9SgahHH0kdA4ADsr7eOAFAvfSO+oHE/8XTubos//LZnQze7//+lI5BCAGA6ACAAKy53IXoWrl1o38WstzTG//uSZAwBBCdITfsyouIzw4ktPO1KEeUhNazqa4jHDiNI+E0oolLDXgzWTMSkKRIMASbp+IqmeRNDQxJUnn5sQIAlQW3CFSHZRG0AxHIsiXkqItZIIdTEd+PsrJdIpDbfnTICgQhCt2IgFALq1FwAgEDjZqqiVxggskLq2pmQlhs1aJiToqKPcjRCE1fWslCnjwMELAEAXUXD4a2eI2IeLiQxgCZU2+FggAc/1f/7mQKkSwcZQZY9x7qj3Pf6f+v//6P/R0KwRAASgAbXYyK06zWoycWyl4cMmrN5WiygsWs9x7xZZca1/cOZKRb/VNS4SmYh3//uLrktUEUhUyfK58fYC+wGJHUyZZRQFFLrdZNk79MyNvTIoVvOBlwNCJ430AyGGlDRSqTFMAzMAO0HTGjIE8LlAYGF7TR1MR5FjRJtIZcho7ecHQJwEfGumYF0niJPFAM8PhsfIWMlOAzyxzj8NP/WoL5Dtb6x8Hk/+oZ8GxoMDF8ussIlAZUKIYHnf7v//761vGMFAEABbSw7tK/WQhPWQBIrUaii272cwQqIIP/7kmQOABQnRs5TGZLkLcI6LzytOJNJGzGtUouQyBgigaVFMG56/8lspwWdf/5suv4c7r3Ds9/68wQgnOEQQrGaayUA0Y+pzEnVC6GRPoNrJBvoK+U7+Zh6zMrMg2Ud7ajoYYNXl4vF1TkNArAzRjrjXTQfUURsGSOpYpcnFIqnSGFvPf/mM2h9XsaGUKrISPCiHYnH4/s2KUnA+VkSMNZB6pv6r/xhQB9IT+tmtxL/q//2H/13Pu///1JQFAAEAEACXQYAeNCKkWEYmEokf0D0hGOomoGJBhcU3R0Az8FEqC+XAuKafF2iXKkSYAi3BYoPgYIswmiiOIDSizh1zQeiyGJxRDYirGDlgkPqJO/RF2yuoRMKBDroMgPsExxJptkAAMSAMDiFQNyaIMA4UBpAIjUxzE6IFJLyHiUxwsk2OWF1kVXQYXMZUnf7/fRnF9K97FpVBasbSrTDbqM5xRwYaj4FgUy/7kQsN+gnHf2WQwdYIcBMCWhTm6BNJl10f/////8cgcBYUwgA20NyZiAHAEQE7hxvXAXTKcImY4aLA4r/+5JkDYAEZkbLw1rK4DNGGc088EyRhRsxjOaLgMmYJLWDnTBG+YXhUiUELe+/thDT+8//xaxj+sJh6yWqu1/n0ZBI5aIXTpA+xrOlpFQ4/+f/nj///Ju/3//XXgvf+eo+Oitdw1rGadEI1pO67NvUEb2fuR6TMZBXY0C+eX/m/7m2v+vbpVQz3ec+VpuwLW7+FdVOKXjIpLoNENqMBnoo00MfN4/l3IEJF+d6GfWiK3/VTPLb+/trb3Uu2//+tgtsFH/+NovQt2tJCMMNQIBE21BosCtYdUgMOJtAAytM5ldLKTFSKA+XeXGxiFlKmOZ6/sqal3v/qJsj//tYsIGyFvi5xZJUHWHxg1RjiQNKljkCxJZouVfrKJz3J8dqWaqFrJhJamSIYEAkSVS1IDoBtgA0CL5asRQAEmHAkFJ7WK+JSKuuoXOVT/MBcguLoGIx5cIq1AAABDFwHw11xxqd3G4+QFc/ZIxT/bgubd+3b/QFotIX/0/p/X//9BCLfu3/v//19G3+qmmDQQSgAEUjWnIVpUUU1GRgfG3zckf32poY//uQZA0CBGJGTGM4kuAxBfk9POdMEDEnNUzhq9C5hia1FhyaBdWgX/w3Kg5DNscse6Xxfz7/5wXA36+zSlUge6HhkSTG6YA1MCPCTF0mz5gURzh31oKnGSQ1ufNeiscJq+6Ij0uoIGRmLWBVA6Y/dEbosRNHD6BQLoZIZDRPrpj7LRmmity6MkQZ358qGaNZwtFY2VYFw4gBUPkBxsB8HSkBhmkTsnhX5Y2Quk4tUQt1pdP/PQRxucKX///+3//0B4cGP0//+jsozgAq2Yty0hfiCiXih4PSTXRVT1mpe6Q0dYzi3Mt1ER32aRBEezo0w6Wvc1li2Gf5rsufQYUqKRRqdjk2F9kyaLnxA0GslU3W9RRq6j29SBIo60jQkzTMjdIG6JSeWtkxNU1InC6O8ZYO1TIZmco7Mkmj3Nt106PrP0j//+zv96Cqk3UqgLRbXU6IE6GJhSKC0jiPsbBAAkPlS2Br3LAn7Dxr/ZV/4Ne3//7/+r/11dhhCAACJZXZlfcpyxw4sZxX6YbKUQEcCuR9q6ckAY09M6iuld8YtPjf//uSZBSAA9RDzMsPmuA1Iyk6De04jCkbN6ew64DgDaS0Jazg1C6fVz9yejLhgMEwAnUArIgiTi0ieHSRE8pR01OImvrV2mSWjTYmi3ZI4Tgs4gxZQTcZUgpSNzA3KBRFkkNMDYxWo49DUkp+pSDoJuiXUQa43QGk4AwA2V6YRVE6a4mefg6Vafdc1+Dx/6hiAN42Q36NRuTBPESeZEbKf//3//5v+yvR//9ciSbqbwBZM/X2wkBC5DPCRGkyDaRJYRkIy4j0SrybE7fYjo+za3rM/jszNfXbQgAkSAdD5E9VuNWr06KOoXO7nPzHljjhqjIzsv0Oc5HoppvfodXqys6OYd6Fp76kU9P/1AAABxugXAZnsYSBCYJteSiUJjK/pMBIj/+88oX1/2xDyqAYJA7R2HZca2Xf6Lf/1v//M//9lP/QBABVKYAgxGUv3TViWQqqPBEQ5UjmXtWzMhUwGw08x6MUdlkjefV2j/4riz9hUbZAYz2cB1jDOIUk/mpCG99Hrml8UxrW863649t11/7f7tveIF4qVMQLugsFg0eFzv/7kmQvAANgLcljD3pgOAOpCiVJSg4w8yOnvauA6ozkILUs4LmT/TrUwn9RKfbiL3/qwCKWAWEAD52iNUIifIBS5KDA9LeQgpO+w+HTO//oC5XwkHRCDBIcaisPb5o9DH2af+t3/////4wIERxIwAIBIKQMAmKnV5dCcoWaYWoXpwqEuw2TGVL+BM26dPv5WyTT5kg3rGf9tiqA8XE7jsCZAS4KcOYFgSY+Bex5GiRkdRUmtTz9SPTZ9P6WtlJVLuqtSk6qlLQs9T+o1a4QuSy4VJYRkc+/UUOf/oBIaEDIrQVIFcDI0QZYyAmKv4UYy/Z6nTv/II1LTyTR5WDygPQ7kgaeHg///id/Faz+yj9z/qI//dM0VQAiAAKmV0HWoBYT2EsuAyD7BklvN1kWG4saoetjdm61r/5xJD99Ym/w2LhBLlJi+HeIEAXgurFyEmXjNFnWylGjpsklapq3SUhTRZkV2QMTc8s6ip3dNSJ1kD6dFObIDoDWjynERzTLbq9X9HxQMCFSWMgyOQd0mHEHtPKp8FQcH4+1Djl6UdCTCRD/+5JkRYADbjfIye+C4C+FqX0I4kwN/N8fLD4LgNcMJjQ0KOCnDCb99v//+n8E/X+f/s+v9/qADKBAWUixMVZKzVtJdhhRkktARoo0892YdyeWlGnFzFYUHi/xrNGy99f3hKCELGfMEd5cCfG2B+DYoujCImVUUDlFjxuZrepbImjHLJ36LoXY3ddG9fadUX0TqJ1JKbIWBiB3wVpOsDlj+mUuKWp1ho6XX2gfO6XgU6hhiB62poSiZjPwyEKUU2hvOKsn51SFy8a7+1TD2pn4R1+U1ehayZ3/q/rSABJAAlUf0iGwNiLFlfrtXwuEHlKVjazdLE4NrTtK0RC1n6lt8w/b78GRdjpC/UGhXwa4OYwhFwCsxjySsWurbxTTnb7z84p96zj2tbNoW7Xxi2949v4NI0XPvLlXvE0q7aIqgwPKuVj9HtV++sA0baZOHco9GsgfryA8fK/gWe///q893Oq51vFTYwK5CtuTH/xH8mTllYbchbStbvU3++xbLev0AgACFpjQlFE5ryb7vrqb9SwasOtvOtQkFPpziteIjO5R//uSZGMDg2owR8sPemA34yj1MQk4DNSnIIw+KYDlDGPklSzgtQYkTcObf8s7fMSYFkkuWBWRlRkA+wBlFBjyZEy9m2mNS76LLZ1Ld7LQOOpD2NAg0lAwfUNPLQeLlMQDMsBX7Xe9n3XfrarARVu2i5rNEo9deAUJviSQn//LYNAxALFP2L6rJYiBsEA7/qPJ+xHB91btKJH6kOLO0f9tLFN6KyAREARYY42kJCSaY8aaiISXEMgpsLvDM53wTpTrCk4ahTj/eIV6xWbOfmZW2wmBxYjJ4FMCwRQwHCgNGj6ICzzz5kTaaZsko30035k6WktSr9kWZKnZbLvdL1I+p7IqY2jjWs0OP7OnvV4k+8MajwL/A/fPopJANiiTX+qAtImdRWMLq3zjnv63PcYAdJP//fPq6c8zqZ5z8hnqO/yeA6WLdS+6MXsouqAAAERKBTJHwBGaUGbi8KpWMpfoTyAjAe8FA7DtYkQMJj+9ubybnP038YWywqYTs+VhdRCgSIAfjtB+LX282RROLUlmBagyTU0tF1LZFa3RWudPpJC9tv/7kmR/gANyPcbDL5rgOkXpOSTqTAxsuyGMMamA2ZPl9GOVKApmV2yx/jRj2V9ff9OabJ+0YUsql+pBJHJT3ouNgfliUxhQEPVa9R5HkIzXjGAIP0//q7pN9JRVKc7/Yu/b7/+n//rqAAFgAKUl0T1MMIEmscQQhhpeYwymEpJnikB9ny0Sn4n1aqTcY4s0HetKbclIDZJpeIIGCDaKdCCoDMAVkUCH8AXAiyGeTfRSMCfWm7KdmrqdWt0lNWtDQpK26HzVaAVXY8SPV8gD6nLVQoef9VywzvrSzJMslJStgSRwMATJiNvay5b9Y8weCg9Elv+FgGDPMI/LHt/+aYeQPKjYDk2EQ7Fix8p/+60UhIRAh7P//6hNUJkhoVgXfW3I0xZz0JyKSZAYjmJiAATg/oWBJHc7gigSj6sWLv+8wzOWzOIlOxhofCAq7enKZHaXdC7ohHxsDBhyVHwO54gOKUDwsqZd/DB0Wf+7/QRI7rf4BdtBLVO4GANLbmcGS1ffFRaTTN7//9iMlflEs2JdrDx+FEFxkKwYZdUInL//PO//+5JknQADnTZGQy+S4DwkSb0FB0qK1KU157CpgNyMpjRUmOD///66AQEgELQIyVVFgEYVwKoJaJUC2SYb+E4UDM0HhDVsk1dudr3r/E38u1LjSQHogq1BxxZgtBACUBkQuCRFaM1FCpmmTrTVu97pMt9SlUVo6RpamrrutO93XU9q1r/V9VT3Ron+lit/7fHdABiDckAERRZ6U1Vx2gaN3sHoKIS10Ye6We8r5+odJd811cTS1JgqGULO/Z+L/V+3Y38h/xdjVuk3U7EGKMAqaug07QE6Wd8dlGpcgKA0euG8Aj3M4VJduqjOTxta7DU2mdgRbKEjhQmUhXHiDuBXQ8nzIz2OVKSRXdfdTITZ2odd9a72WtqkK9dKtSu67KZ7ZxeaSWUJw/rbrz3WhCb/qAYKGUKAQRwYgu2zHKFJ+f9+/P+TYGMoqKiHoaXRES4yOtP/1632/+/X7d//jmK9qaP96NcAJiA6tnZSWiOIOQsKpMKhaaBRpJu0Fx+XT1acllJ8b3+3BXquzMsJPmwaBYIJ4CxRG2GBwAaBrCHiyFWa//uSZL0II0tHR0MvkuA7A0ktLMk4DIz7I4wxq4DQjOPgMSTgk7opLNzZBDRSsb1TlT66C9SltpKSMk27sfTZeZGGss24mtaQ2WIxSs/sN0Lcxei5SAAypDpCMDioC9+dPRgLqYYGKVFH5hZAqCKC+I2jMWDB6BOIr///UnxKyt///UjK//0diuoAQAMEOapaSChZEVLtmBEkKEeDmMQKrwMWikDuV4W+sUh+A7sopZFM15u3ySa+OPw839u0gqdUbQi9JYCikyYLEAjG8Letft3OYS+L3cpdG61iBpV9wgpHYijqCMod04IhglQTTs53kYEAKzxdVGdO/z5lOSUYyMYQYBvVk0fk8mn1unbOP9IAAiBQAphdJBpk65Acc3ZLyZJn//q7yQUJz6PcrG1w8eFRUUhX/L9fFXTn9Gt09R7Zz/Z167KujXrVAAJwKWtC/RCBe5eKZTEMBzes5VUsUrE1T6GnWhidVihbVA2RWXVbSMCTt2q671eWO1mgaJpBthYiAbwg4NWickBzjdjiRmblqxxMuJNmLZ2ZpJudOJIoV//7kmTdgANwNcdDDIrgNASJCAhKShDdbRStYE3A5Yzj8JCk4NFJq0WpKWg8ydRitBnqUcig8Fxli7S4Jkw4QNIRFGCstB3iwFTBIAB1osAClVjxKGSy2psKb6J1/8Nm2DsVLHTWbOecO2Sg1Xf97vTYtT6if7//s3MrUtbP/6aHbl6AJoZ1jBUchIGCAuEEFmUgNFn6gFxUTmxOLLX0gq9FY5i8ksqwLIY5ZoKCYl7hwG2dm85BSzGiueQCV2rrZi/i1DLyQSKTlafdkfO40jpzUrlD7zsR5PfApVh3TR66MVbVWZQhfD0EMEIgQEL1KmYlIHoQQ08pWDxORiHVTAQo/CNpSz8tf24c538sHtfz+slr+WxAg6tRqbB7mV7QxyAA3QL/Pix3EnvlTUoJ7f/t/5IIDMUrnKRRGCf/0+v/++xP27f6FQAkIBJ4TqhKhKHJKYIbFQAvcoMGoFroZBQWwL5i0YhlxY/ZnZfWqUuXZiDZ6IJayDB0odrRWkZ22KHVUhGcAUAuUFl4RkRiK4OYbLLjUybNiZLaJWRKKKVOigb/+5Jk6wgD9DlGQw+S4DcjKRwYSzgR9WMUrOhtwNQXJXQxCTBnElupNJdnNl0J5OZKs6S2eukipkVa1XWitnPJLc8ZpOupNaVNzcXc4dl2VKYOV4m9Z4mpgAAKpgYgzuQEVlzuRtBTUpn7ZFOp0hUBV0uCU9z8OsONM8veEQ8k/Lgd/9haGjo1y//v//zz6er/Tp1jAABsEJDSJW8syxVwljOsvlMI+SeALSDVKeTKEl9VCFIlUIUfqcjPoatm7zLeFmao4zZlUNO8wlIpRswDRxjL+0XaW7ZrXbNbPH8u467jrL/x1uyVRSkksqFFFG5SklvduoruklNbuUx3+7UvUVyuxrGrd81vf8/3f/f/9AAgJyFCr641CiZ6g15X87lax1kRW+ZYylYKebDCqFQCAhXws0eSv+p3//V//t7ej/+pTEFNRTMuOTkuM1VVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVV//uSZOyDBHZGRkM4kuA8gzkMFEw4D3SbHQw/CUjHjmRwMwkoVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVQBqs0jMnBWmgKng1FRU32lmZmb+N6/V+w+MygnAXTBSU3/Ffjw4KGwvhYijgbg038TRQL8U3//xv5cO4FZf/z8bdj8V3CpuILkwNl5P8KE+53G3eCoQUAIQxdpeAGCps5MdqtGpNehUFBQFS2qx1bDEzNSDATAKKlIBjsJGEQBFZnmMUrStzKW+pXKVDOZ+hnQ2jloarUNK2iGVjPlUVcHRF4NCI9e2HcNCJ4NQ0eEQlnbmh0FYaUxBTUUzLjk5LjNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVf/7kkScD/K0GL+QyBnCYEgHsjBlXAAAAaQAAAAgAAA0gAAABFVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVU=';
  126. audiofile2.setAttribute("id", "ding_noise1");
  127. audiofile2.volume = 1;
  128.  
  129. function newHits(dingNoise) {
  130. if (dingNoise || newHitDing)
  131. document.getElementById("ding_noise"+audio_index).play();
  132. }
  133.  
  134. //For editing the blocklist
  135. var div1 = document.createElement('div');
  136. var textarea1 = document.createElement('textarea');
  137.  
  138. div1.style.position = 'fixed';
  139. div1.style.width = '500px';
  140. div1.style.height = '235px';
  141. div1.style.left = '50%';
  142. div1.style.right = '50%';
  143. div1.style.margin = '-250px 0px 0px -250px';
  144. div1.style.top = '300px';
  145. div1.style.padding = '5px';
  146. div1.style.border = '2px';
  147. div1.style.backgroundColor = 'black';
  148. div1.style.color = 'white';
  149. div1.style.zIndex = '100';
  150. div1.setAttribute('id','block_div');
  151.  
  152. textarea1.style.padding = '2px';
  153. textarea1.style.width = '500px';
  154. textarea1.style.height = '180px';
  155. textarea1.title = 'Block list';
  156. textarea1.setAttribute('id','block_text');
  157.  
  158. div1.textContent = 'BLOCKLIST - Edit the blocklist with what you want to ignore/hide. Separate requester names and HIT titles with the ^ character. After clicking "Save", you\'ll need to scrape again to apply the changes.';
  159. div1.style.fontSize = '12px';
  160. div1.appendChild(textarea1);
  161.  
  162. var save_button1 = document.createElement('button');
  163. var cancel_button1 = document.createElement('button');
  164.  
  165. save_button1.textContent = 'Save';
  166. save_button1.setAttribute('id', 'save_blocklist');
  167. save_button1.style.height = '18px';
  168. save_button1.style.width = '100px';
  169. save_button1.style.fontSize = '10px';
  170. save_button1.style.paddingLeft = '3px';
  171. save_button1.style.paddingRight = '3px';
  172. save_button1.style.backgroundColor = 'white';
  173. save_button1.style.marginLeft = '5px';
  174.  
  175. cancel_button1.textContent = 'Cancel';
  176. cancel_button1.setAttribute('id', 'cancel_blocklist');
  177. cancel_button1.style.height = '18px';
  178. cancel_button1.style.width = '100px';
  179. cancel_button1.style.fontSize = '10px';
  180. cancel_button1.style.paddingLeft = '3px';
  181. cancel_button1.style.paddingRight = '3px';
  182. cancel_button1.style.backgroundColor = 'white';
  183. cancel_button1.style.marginLeft = '5px';
  184.  
  185. div1.appendChild(save_button1);
  186. div1.appendChild(cancel_button1);
  187. document.body.insertBefore(div1, document.body.firstChild);
  188. $("#block_div").hide();
  189.  
  190. // save and cancel for blocklist
  191. function save_blocklist() {
  192. //console.log("Save");
  193. var textarea = $("#block_text");
  194. var text = textarea.val();
  195. var block_list = text.split("^");
  196. var trimmed_list = [];
  197. for (var requester in block_list){
  198. if (block_list[requester].trim().length !== 0)
  199. trimmed_list.push(block_list[requester].toLowerCase().trim());
  200. }
  201. //console.log(trimmed_list);
  202. GM_setValue("scraper_ignore_list",trimmed_list.join('^'));
  203. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  204. //console.log("Save complete: ");
  205. //console.log(ignore_list);
  206. $("#block_div").hide();
  207. }
  208. save_button1.addEventListener("click", function(){ save_blocklist(); }, false);
  209. cancel_button1.addEventListener("click", function(){
  210. // reset textarea contents upon cancel
  211. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  212. var textarea = $("#block_text");
  213. var text = "";
  214. for (var i = 0; i < ignore_list.length; i++){
  215. text += ignore_list[i]+"^";
  216. }
  217. textarea.val(text.substring(0, text.length - 1));
  218. // close editor
  219. $("#block_div").hide();
  220. }, false);
  221.  
  222.  
  223. //For editing the include list
  224. var shouldInclude = false;
  225. var div2 = document.createElement('div');
  226. var textarea2 = document.createElement('textarea');
  227.  
  228. div2.style.position = 'fixed';
  229. div2.style.width = '500px';
  230. div2.style.height = '235px';
  231. div2.style.left = '50%';
  232. div2.style.right = '50%';
  233. div2.style.margin = '-250px 0px 0px -250px';
  234. div2.style.top = '300px';
  235. div2.style.padding = '5px';
  236. div2.style.border = '2px';
  237. div2.style.backgroundColor = 'black';
  238. div2.style.color = 'white';
  239. div2.style.zIndex = '100';
  240. div2.setAttribute('id','include_div');
  241.  
  242. textarea2.style.padding = '2px';
  243. textarea2.style.width = '500px';
  244. textarea2.style.height = '180px';
  245. textarea2.title = 'include list';
  246. textarea2.setAttribute('id','include_text');
  247.  
  248. div2.textContent = 'INCLUDELIST - Focus the results on your favorite requesters. Separate requester names and HIT titles with the ^ character. When \"Use includelist\" is selected, HIT Scraper only shows results matching the includelist.';
  249. div2.style.fontSize = '12px';
  250. div2.appendChild(textarea2);
  251.  
  252. var save_button2 = document.createElement('button');
  253. var cancel_button2 = document.createElement('button');
  254.  
  255. save_button2.textContent = 'Save';
  256. save_button2.setAttribute('id', 'save_includelist');
  257. save_button2.style.height = '18px';
  258. save_button2.style.width = '100px';
  259. save_button2.style.fontSize = '10px';
  260. save_button2.style.paddingLeft = '3px';
  261. save_button2.style.paddingRight = '3px';
  262. save_button2.style.backgroundColor = 'white';
  263. save_button2.style.marginLeft = '5px';
  264.  
  265. cancel_button2.textContent = 'Cancel';
  266. cancel_button2.setAttribute('id', 'cancel_includelist');
  267. cancel_button2.style.height = '18px';
  268. cancel_button2.style.width = '100px';
  269. cancel_button2.style.fontSize = '10px';
  270. cancel_button2.style.paddingLeft = '3px';
  271. cancel_button2.style.paddingRight = '3px';
  272. cancel_button2.style.backgroundColor = 'white';
  273. cancel_button2.style.marginLeft = '5px';
  274.  
  275. div2.appendChild(save_button2);
  276. div2.appendChild(cancel_button2);
  277. document.body.insertBefore(div2, document.body.firstChild);
  278. $("#include_div").hide();
  279.  
  280. // save and cancel for includelist
  281. function save_includelist() {
  282. //console.log("Save");
  283. var textarea = $("#include_text");
  284. var text = textarea.val();
  285. var includes = text.split("^");
  286. //console.log(includes);
  287. var trimmed_list = [];
  288. for (var requester in includes){
  289. if (includes[requester].trim().length !== 0)
  290. trimmed_list.push(includes[requester].toLowerCase().trim());
  291. }
  292. GM_setValue("scraper_include_list",trimmed_list.join('^'));
  293. include_list = GM_getValue("scraper_include_list").split('^');
  294. //console.log(include_list);
  295. $("#include_div").hide();
  296. }
  297. save_button2.addEventListener("click", function(){ save_includelist(); }, false);
  298. cancel_button2.addEventListener("click", function(){
  299. // reset textarea contents upon cancel
  300. include_list = GM_getValue("scraper_include_list").split('^');
  301. var textarea = $("#include_text");
  302. var text = "";
  303. for (var i = 0; i < include_list.length; i++){
  304. text += include_list[i]+"^";
  305. }
  306. textarea.val(text.substring(0, text.length - 1));
  307. // close editor
  308. $("#include_div").hide();
  309. }, false);
  310.  
  311.  
  312. var HITStorage = {};
  313. var indexedDB = window.indexedDB || window.webkitIndexedDB ||
  314. window.mozIndexedDB;
  315. window.IDBTransaction = window.IDBTransaction || window.webkitIDBTransaction || window.mozIDBTransaction;
  316. window.IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange || window.mozIDBKeyRange;
  317. HITStorage.IDBTransactionModes = { "READ_ONLY": "readonly", "READ_WRITE": "readwrite", "VERSION_CHANGE": "versionchange" };
  318. var IDBKeyRange = window.IDBKeyRange;
  319.  
  320. HITStorage.indexedDB = {};
  321. HITStorage.indexedDB = {};
  322. HITStorage.indexedDB.db = null;
  323.  
  324. HITStorage.indexedDB.onerror = function(e) {
  325. //console.log(e);
  326. };
  327.  
  328. var v=4; // why is this a global variable? Does this ever change value and is it ever used outside the checkTitle and checkRequester methods?
  329. HITStorage.indexedDB.checkTitle = function(title,button) {
  330. var request = indexedDB.open("HITDB", v);
  331. request.onsuccess = function(e) {
  332. HITStorage.indexedDB.db = e.target.result;
  333. var db = HITStorage.indexedDB.db;
  334. if (!db.objectStoreNames.contains("HIT"))
  335. {
  336. db.close();
  337. return;
  338. }
  339. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  340. var store = trans.objectStore("HIT");
  341.  
  342. var index = store.index("title");
  343. index.get(title).onsuccess = function(event)
  344. {
  345. if (event.target.result === undefined)
  346. {
  347. //console.log(title + ' not found');
  348. scraper_history[button].titledb = false;
  349. }
  350. else
  351. {
  352. //console.log(title + ' found');
  353. scraper_history[button].titledb = true;
  354. }
  355.  
  356. db.close();
  357. };
  358. };
  359. request.onerror = HITStorage.indexedDB.onerror;
  360. };
  361.  
  362. HITStorage.indexedDB.checkRequester = function(id,button) {
  363. var request = indexedDB.open("HITDB", v);
  364. request.onsuccess = function(e) {
  365. HITStorage.indexedDB.db = e.target.result;
  366. var db = HITStorage.indexedDB.db;
  367. if (!db.objectStoreNames.contains("HIT"))
  368. {
  369. db.close();
  370. return;
  371. }
  372. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  373. var store = trans.objectStore("HIT");
  374.  
  375. var index = store.index("requesterId");
  376. index.get(id).onsuccess = function(event)
  377. {
  378. if (event.target.result === undefined)
  379. {scraper_history[button].reqdb = false;
  380. //console.log(id + ' not found');
  381. }
  382. else
  383. {
  384. scraper_history[button].reqdb = true;
  385. //console.log(id + ' found');
  386. }
  387. db.close();
  388. };
  389. };
  390. request.onerror = HITStorage.indexedDB.onerror;
  391. };
  392.  
  393. var PAGES_TO_SCRAPE = 3;
  394. var MINIMUM_HITS = 100;
  395. var SEARCH_REFRESH=0;
  396. var MINIMUM_TO = -1;
  397. var URL_BASE = "/mturk/searchbar?searchWords=&selectedSearchType=hitgroups";
  398. var initial_url = URL_BASE;
  399. var TO_REQ_URL = "http://turkopticon.ucsd.edu/reports?id=";
  400. var found_key_list=[];
  401. var last_clear_time = new Date().getTime();
  402. var searched_once = false;
  403. var useBlocklist = true;
  404. var save_new_results_time = 120;
  405. var save_results_time = 3600;
  406. var default_type = 0;
  407. var block_no_to = false;
  408. var audio_index = 0;
  409. var cur_loc = window.location.href;
  410. var time_input = document.createElement("INPUT");
  411. time_input.value = 0;
  412. var page_input = document.createElement("INPUT");
  413. page_input.value = 3;
  414. var min_input = document.createElement("INPUT");
  415. var new_time_display_input = document.createElement("INPUT");
  416. new_time_display_input.value = 300;
  417. var reward_input = document.createElement("INPUT");
  418. var qual_input = document.createElement("INPUT");
  419. qual_input.type = "checkbox";
  420. var masters_input = document.createElement("INPUT");
  421. masters_input.type = "checkbox";
  422. var masters_hide = document.createElement("INPUT");
  423. masters_hide.type = "checkbox";
  424. var sort_input = document.createElement("SELECT");
  425. var sort_input1 = document.createElement("OPTION");
  426. sort_input1.text = "Latest";
  427. sort_input1.value = "late";
  428. var sort_input2 = document.createElement("OPTION");
  429. sort_input2.text = "Most Available";
  430. sort_input2.value = "most";
  431. var sort_input3 = document.createElement("OPTION");
  432. sort_input3.text = "Reward";
  433. sort_input3.value = "amount";
  434. var sort_input4 = document.createElement("OPTION");
  435. sort_input4.text = "Title";
  436. sort_input4.value = "alpha";
  437. sort_input.appendChild(sort_input1);
  438. sort_input.appendChild(sort_input2);
  439. sort_input.appendChild(sort_input3);
  440. sort_input.appendChild(sort_input4);
  441. var sort_input_invert = document.createElement("INPUT");
  442. sort_input_invert.type = "checkbox";
  443. var sort_to = document.createElement("INPUT");
  444. sort_to.type = "checkbox";
  445. var sort_to2 = document.createElement("INPUT");
  446. sort_to2.type = "checkbox";
  447. var sort_asc = document.createElement("INPUT");
  448. sort_asc.type = "radio";
  449. sort_asc.name = "sortOrder";
  450. var sort_dsc = document.createElement("INPUT");
  451. sort_dsc.type = "radio";
  452. sort_dsc.name = "sortOrder";
  453. var min_to = document.createElement("INPUT");
  454. var no_to_block = document.createElement("INPUT");
  455. no_to_block.type = "checkbox";
  456. var useTO_input = document.createElement("INPUT");
  457. useTO_input.type = "checkbox";
  458. var friesAreDone = document.createElement("INPUT");
  459. friesAreDone.type = "checkbox";
  460. var audio_option = document.createElement("SELECT");
  461. var audio1 = document.createElement("OPTION");
  462. var audio2 = document.createElement("OPTION");
  463. audio1.text = "Ding";
  464. audio1.value = 0;
  465. audio2.text = "Squee";
  466. audio2.value = 1;
  467. audio_option.appendChild(audio1);
  468. audio_option.appendChild(audio2);
  469. var correctForSkips = document.createElement("INPUT");
  470. correctForSkips.type = "checkbox";
  471. var useBlock = document.createElement("INPUT");
  472. useBlock.type = "checkbox";
  473. var matchOnly = document.createElement("INPUT");
  474. matchOnly.type = "checkbox";
  475. var highlightIncludes_input = document.createElement("INPUT");
  476. highlightIncludes_input.type = "checkbox";
  477. var opt_exportvb = document.createElement("INPUT");
  478. opt_exportvb.type = "checkbox";
  479. var opt_exportirc = document.createElement("INPUT");
  480. opt_exportirc.type = "checkbox";
  481. var opt_exportreddit = document.createElement("INPUT");
  482. opt_exportreddit.type = "checkbox";
  483. var sufTheme = "_thClassic";
  484. var show_checkboxes = document.createElement("INPUT");
  485. show_checkboxes.type = "checkbox";
  486. var search_input = document.createElement("INPUT");
  487.  
  488. var saveState = {};
  489. saveState.init = function() {
  490. if (localStorage["lastState_hit_scraper_with_export"])
  491. this.lastState = JSON.parse(localStorage["lastState_hit_scraper_with_export"]);
  492. else
  493. //this.lastState = this.lazyLoad();
  494. this.lastState = {};
  495.  
  496. time_input.value = this.lastState.refreshTime || 0;
  497. page_input.value = this.lastState.numPages || 3;
  498. min_input.value = this.lastState.minHits || "";
  499. new_time_display_input.value = this.lastState.newHitHighlight || 300;
  500. reward_input.value = this.lastState.reward || "";
  501. qual_input.checked = this.lastState.qualified || ((this.lastState.qualified === false) ? false : true);
  502. masters_input.checked = this.lastState.masters || false;
  503. masters_hide.checked = this.lastState.mShow || false;
  504. sort_input.selectedIndex = this.lastState.sort || 0;
  505. sort_input_invert.checked = this.lastState.invert || false;
  506. sort_to.checked = this.lastState.to || false;
  507. sort_to2.checked = this.lastState.to2 || false;
  508. sort_asc.checked = this.lastState.asc || false;
  509. sort_dsc.checked = this.lastState.dsc || ((this.lastState.dsc === false) ? false : true);
  510. min_to.value = this.lastState.minTO || "";
  511. no_to_block.checked = this.lastState.hideNTO || false;
  512. search_input.value = this.lastState.searchTerms || "";
  513. matchOnly.checked = this.lastState.useInclude || false;
  514. friesAreDone.checked = this.lastState.fries || false;
  515. audio_option.selectedIndex = this.lastState.whichfry || 0;
  516. useBlock.checked = this.lastState.blocklist || ((this.lastState.blocklist === false) ? false : true);
  517. correctForSkips.checked = this.lastState.skips || ((this.lastState.skips === false) ? false : true);
  518. useTO_input.checked = this.lastState.useTO || false;
  519. highlightIncludes_input.checked = this.lastState.highlightIncl || ((this.lastState.highlightIncl === false) ? false : true);
  520. opt_exportvb.checked = this.lastState.exvb || ((this.lastState.exvb === false) ? false : true);
  521. opt_exportirc.checked = this.lastState.exirc || ((this.lastState.exirc === false) ? false : true);
  522. opt_exportreddit.checked = this.lastState.exreddit || ((this.lastState.exreddit === false) ? false : true);
  523. sufTheme = this.lastState.theme || "_thClassic";
  524. show_checkboxes.checked = this.lastState.checkboxes || ((this.lastState.checkboxes === false) ? false : true);
  525. var k = ['customwisp','customsd','customsl','customclassic','customcustom'];
  526. for (k in this.lastState) {
  527. for (var j in this.lastState[k]) {
  528. if (this.lastState[k].hasOwnProperty(j)) {
  529. switch(k) {
  530. case "customwisp": uwcolors[j] = this.lastState[k][j]; break;
  531. case "customsd": usdcolors[j] = this.lastState[k][j]; break;
  532. case "customsl": uslcolors[j] = this.lastState[k][j]; break;
  533. case "customclassic": uclcolors[j] = this.lastState[k][j]; break;
  534. case "customcustom": ucucolors[j] = this.lastState[k][j]; break;
  535. }
  536. }
  537. }
  538. }
  539. if (Object.keys(uwcolors).length == 0)
  540. for (var key in wcolors) { if (wcolors.hasOwnProperty(key) && uwcolors[key] === undefined) uwcolors[key] = wcolors[key]; }
  541. if (Object.keys(uslcolors).length == 0)
  542. for (var key in slcolors) { if (slcolors.hasOwnProperty(key) && uslcolors[key] === undefined) uslcolors[key] = slcolors[key]; }
  543. if (Object.keys(usdcolors).length == 0)
  544. for (var key in sdcolors) { if (sdcolors.hasOwnProperty(key) && usdcolors[key] === undefined) usdcolors[key] = sdcolors[key]; }
  545. if (Object.keys(ucucolors).length == 0)
  546. for (var key in cucolors) { if (cucolors.hasOwnProperty(key) && ucucolors[key] === undefined) ucucolors[key] = cucolors[key]; }
  547. if (Object.keys(uclcolors).length == 0)
  548. for (var key in clcolors) { if (clcolors.hasOwnProperty(key) && uclcolors[key] === undefined) uclcolors[key] = clcolors[key]; }
  549. //console.dir(this.lastState);
  550. }
  551. saveState.setItem = function(key,value) {
  552. this.lastState[key] = value;
  553. };
  554. saveState.getItem = function(key) {
  555. return this.lastState[key];
  556. };
  557. saveState.removeItem = function(key) {
  558. delete this.lastState[key];
  559. };
  560. saveState.save = function() {
  561. localStorage["lastState_hit_scraper_with_export"] = JSON.stringify(this.lastState);
  562. }
  563. saveState.init();
  564.  
  565.  
  566. var LINK_BASE = "https://www.mturk.com";
  567. var STATUSDETAIL_DELAY = 250;
  568. var MPRE_DELAY = 3000;
  569.  
  570. var css = "<style type = 'text/css'>\n";
  571. css += ".body_thCustom {background-color: "+ucucolors.background+";}\n";
  572. css += ".controlpanel_thCustom {color: "+ucucolors.defaultText+"; margin: 0px auto 5px auto; padding: 8px 5px 1px; background: "+ucucolors.cpBackground+";}\n";
  573. css += ".cpInput_thCustom {color: "+ucucolors.inputText+"; border: 1px solid; background: "+ucucolors.background+"; text-align: center;}\n";
  574. css += ".cpButtons_thCustom {color: "+ucucolors.accent+"; border: 1px solid; background: transparent;}\n";
  575. css += ".cpSortdiv_thCustom {color: "+ucucolors.inputText+"; display: inline; font: 14px;}\n";
  576. css += ".cpSpans_thCustom {padding: 2px; cursor:default; color: "+ucucolors.defaultText+"; background: "+ucucolors.cpBackground+";} .cpSpans_thCustom:hover {background: "+ucucolors.hover+";}\n";
  577. css += ".cpSpansOn_thCustom {background: "+ucucolors.highlight+"; color: "+ucucolors.secondText+";}\n";
  578. css += ".taButtons_thCustom {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+ucucolors.export+";} .taButtonsOff_thCustom {display: none;}\n";
  579. css += ".link_thCustom:link {color: "+ucucolors.link+";} .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  580. css += ".toNone_thCustom .link_thCustom, .toNone_thCustom .tolink_thCustom {color: "+ucucolors.noTO+";}\n .toNone_thCustom .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  581. css += ".toHigh_thCustom .link_thCustom, .toHigh_thCustom .tolink_thCustom {color: "+ucucolors.highTO+";}\n .toHigh_thCustom .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  582. css += ".toGood_thCustom .link_thCustom, .toGood_thCustom .tolink_thCustom {color: "+ucucolors.goodTO+";}\n .toGood_thCustom .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  583. css += ".toAverage_thCustom .link_thCustom, .toAverage_thCustom .tolink_thCustom {color: "+ucucolors.averageTO+";}\n .toAverage_thCustom .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  584. css += ".toLow_thCustom .link_thCustom, .toLow_thCustom .tolink_thCustom {color: "+ucucolors.lowTO+";}\n .toLow_thCustom .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  585. css += ".toPoor_thCustom .link_thCustom, .toPoor_thCustom .tolink_thCustom {color: "+ucucolors.poorTO+";}\n .toPoor_thCustom .link_thCustom:visited {color: "+ucucolors.vlink+";}\n";
  586. css += ".nohitDB_thCustom {color: #000000; background: "+ucucolors.nohitDB+";}\n";
  587. css += ".tooweak_thCustom {color: #000000; background: "+ucucolors.unqualified+";}\n";
  588. css += ".yeshitDB_thCustom {color:#000000; background: "+ucucolors.hitDB+";}\n";
  589. css += ".needmaster_thCustom {color: #000000; background: "+ucucolors.reqmaster+";}\n";
  590. css += ".nomaster_thCustom {color: #000000; background: "+ucucolors.nomaster+";}\n";
  591. css += ".spacer_thCustom {color: "+ucucolors.accent+";}\n";
  592. css += ".mainlink_thCustom {color: "+ucucolors.defaultText+"; text-align: top;}\n";
  593. css += ".tabhead_thCustom {color: "+ucucolors.defaultText+";}\n";
  594. css += ".bodytable_thCustom {color: "+ucucolors.bodytable+";}\n";
  595. css += ".statusdiv_thCustom {font-size: 15px; color: "+ucucolors.secondText+";}\n";
  596. css += ".body_thRandom {background-color: "+rcolors.background+";}\n";
  597. css += ".controlpanel_thRandom {color: "+rcolors.defaultText+"; margin: 0px auto 5px auto; padding: 8px 5px 1px; background: "+rcolors.cpBackground+";}\n";
  598. css += ".cpInput_thRandom {color: "+rcolors.inputText+"; border: 1px solid; background: "+rcolors.background+"; text-align: center;}\n";
  599. css += ".cpButtons_thRandom {color: "+rcolors.accent+"; border: 1px solid; background: transparent;}\n";
  600. css += ".cpSortdiv_thRandom {color: "+rcolors.inputText+"; display: inline; font: 14px;}\n";
  601. css += ".cpSpans_thRandom {padding: 2px; cursor:default; color: "+rcolors.defaultText+"; background: "+rcolors.cpBackground+";} .cpSpans_thRandom:hover {background: "+rcolors.hover+";}\n";
  602. css += ".cpSpansOn_thRandom {background: "+rcolors.highlight+"; color: "+rcolors.secondText+";}\n";
  603. css += ".taButtons_thRandom {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+rcolors.export+";} .taButtonsOff_thRandom {display: none;}\n";
  604. css += ".link_thRandom:link {color: "+rcolors.link+";} .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  605. css += ".toNone_thRandom .link_thRandom, .toNone_thRandom .tolink_thRandom {color: "+rcolors.noTO+";}\n .toNone_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  606. css += ".toHigh_thRandom .link_thRandom, .toHigh_thRandom .tolink_thRandom {color: "+rcolors.highTO+";}\n .toHigh_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  607. css += ".toGood_thRandom .link_thRandom, .toGood_thRandom .tolink_thRandom {color: "+rcolors.goodTO+";}\n .toGood_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  608. css += ".toAverage_thRandom .link_thRandom, .toAverage_thRandom .tolink_thRandom {color: "+rcolors.averageTO+";}\n .toAverage_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  609. css += ".toLow_thRandom .link_thRandom, .toLow_thRandom .tolink_thRandom {color: "+rcolors.lowTO+";}\n .toLow_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  610. css += ".toPoor_thRandom .link_thRandom, .toPoor_thRandom .tolink_thRandom {color: "+rcolors.poorTO+";}\n .toPoor_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  611. css += ".nohitDB_thRandom {color: #000000; background: "+rcolors.nohitDB+";}\n";
  612. css += ".tooweak_thRandom {color: #000000; background: "+rcolors.unqualified+";}\n";
  613. css += ".yeshitDB_thRandom {color:#000000; background: "+rcolors.hitDB+";}\n";
  614. css += ".needmaster_thRandom {color: #000000; background: "+rcolors.reqmaster+";}\n";
  615. css += ".nomaster_thRandom {color: #000000; background: "+rcolors.nomaster+";}\n";
  616. css += ".spacer_thRandom {color: "+rcolors.accent+";}\n";
  617. css += ".mainlink_thRandom {color: "+rcolors.defaultText+"; text-align: top;}\n";
  618. css += ".tabhead_thRandom {color: "+rcolors.defaultText+";}\n";
  619. css += ".bodytable_thRandom {color: "+rcolors.bodytable+";}\n";
  620. css += ".statusdiv_thRandom {font-size: 15px; color: "+rcolors.secondText+";}\n";
  621. // -- Dark and Light themes adapted from the Solarized color scheme developed by Ethan Schoonover -- //
  622. css += ".body_thSLight {background-color: "+uslcolors.background+";}\n";
  623. css += ".controlpanel_thSLight {color: "+uslcolors.defaultText+"; margin: 0px auto 5px auto; padding: 8px 5px 1px; background: "+uslcolors.cpBackground+";}\n";
  624. css += ".cpInput_thSLight {color: "+uslcolors.inputText+"; border: 1px solid; background: "+uslcolors.background+"; text-align: center;}\n";
  625. css += ".cpButtons_thSLight {color: "+uslcolors.accent+"; border: 1px solid; background: transparent;}\n";
  626. css += ".cpSortdiv_thSLight {color: "+uslcolors.inputText+"; display: inline; font: 14px;}\n";
  627. css += ".cpSpans_thSLight {padding: 2px; cursor:default; color: "+uslcolors.defaultText+"; background: "+uslcolors.cpBackground+";} .cpSpans_thSLight:hover {background: "+uslcolors.hover+";}\n";
  628. css += ".cpSpansOn_thSLight {background: "+uslcolors.highlight+"; color: "+uslcolors.secondText+";}\n";
  629. css += ".taButtons_thSLight {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+uslcolors.export+";} .taButtonsOff_thSLight {display: none;}\n";
  630. css += ".link_thSLight:link {color: "+uslcolors.link+";} .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  631. css += ".toNone_thSLight .link_thSLight, .toNone_thSLight .tolink_thSLight {color: "+uslcolors.noTO+";}\n .toNone_thSLight .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  632. css += ".toHigh_thSLight .link_thSLight, .toHigh_thSLight .tolink_thSLight {color: "+uslcolors.highTO+";}\n .toHigh_thSLight .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  633. css += ".toGood_thSLight .link_thSLight, .toGood_thSLight .tolink_thSLight {color: "+uslcolors.goodTO+";}\n .toGood_thSLight .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  634. css += ".toAverage_thSLight .link_thSLight, .toAverage_thSLight .tolink_thSLight {color: "+uslcolors.averageTO+";}\n .toAverage_thSLight .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  635. css += ".toLow_thSLight .link_thSLight, .toLow_thSLight .tolink_thSLight {color: "+uslcolors.lowTO+";}\n .toLow_thSLight .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  636. css += ".toPoor_thSLight .link_thSLight, .toPoor_thSLight .tolink_thSLight {color: "+uslcolors.poorTO+";}\n .toPoor_thSLight .link_thSLight:visited {color: "+uslcolors.vlink+";}\n";
  637. css += ".nohitDB_thSLight {color: #000000; background: "+uslcolors.nohitDB+";}\n";
  638. css += ".tooweak_thSLight {color: #000000; background: "+uslcolors.unqualified+";}\n";
  639. css += ".yeshitDB_thSLight {color:#000000; background: "+uslcolors.hitDB+";}\n";
  640. css += ".needmaster_thSLight {color: #000000; background: "+uslcolors.reqmaster+";}\n";
  641. css += ".nomaster_thSLight {color: #000000; background: "+uslcolors.nomaster+";}\n";
  642. css += ".spacer_thSLight {color: "+uslcolors.accent+";}\n";
  643. css += ".mainlink_thSLight {color: "+uslcolors.defaultText+"; text-align: top;}\n";
  644. css += ".tabhead_thSLight {color: "+uslcolors.defaultText+";}\n";
  645. css += ".bodytable_thSLight {color: "+uslcolors.bodytable+";}\n";
  646. css += ".statusdiv_thSLight {font-size: 15px; color: "+uslcolors.secondText+";}\n";
  647. //css += ".tablerow_thSLight {color: "+colors.defaultText+";}\n";
  648. css += ".body_thSDark {background-color: "+usdcolors.background+";}\n";
  649. css += ".controlpanel_thSDark {color: "+usdcolors.defaultText+"; margin: 0px auto 5px auto; padding: 8px 5px 1px; background: "+usdcolors.cpBackground+";}\n";
  650. css += ".cpInput_thSDark {color: "+usdcolors.inputText+"; border: 1px solid; background: "+usdcolors.background+"; text-align: center;}\n";
  651. css += ".cpButtons_thSDark {color: "+usdcolors.accent+"; border: 1px solid; background: transparent;}\n";
  652. css += ".cpSortdiv_thSDark {color: "+usdcolors.inputText+"; display: inline; font: 14px;}\n";
  653. css += ".cpSpans_thSDark {padding: 2px; cursor:default; color: "+usdcolors.defaultText+"; background: "+usdcolors.cpBackground+";} .cpSpans_thSDark:hover {background: "+usdcolors.hover+";}\n";
  654. css += ".cpSpansOn_thSDark {background: "+usdcolors.highlight+"; color: "+usdcolors.secondText+";}\n";
  655. css += ".taButtons_thSDark {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+usdcolors.export+";} .taButtonsOff_thSDark {display: none;}\n";
  656. css += ".link_thSDark:link {color: "+usdcolors.link+";} .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  657. css += ".toNone_thSDark .link_thSDark, .toNone_thSDark .tolink_thSDark {color: "+usdcolors.noTO+";}\n .toNone_thSDark .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  658. css += ".toHigh_thSDark .link_thSDark, .toHigh_thSDark .tolink_thSDark {color: "+usdcolors.highTO+";}\n .toHigh_thSDark .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  659. css += ".toGood_thSDark .link_thSDark, .toGood_thSDark .tolink_thSDark {color: "+usdcolors.goodTO+";}\n .toGood_thSDark .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  660. css += ".toAverage_thSDark .link_thSDark, .toAverage_thSDark .tolink_thSDark {color: "+usdcolors.averageTO+";}\n .toAverage_thSDark .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  661. css += ".toLow_thSDark .link_thSDark, .toLow_thSDark .tolink_thSDark {color: "+usdcolors.lowTO+";}\n .toLow_thSDark .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  662. css += ".toPoor_thSDark .link_thSDark, .toPoor_thSDark .tolink_thSDark {color: "+usdcolors.poorTO+";}\n .toPoor_thSDark .link_thSDark:visited {color: "+usdcolors.vlink+";}\n";
  663. css += ".nohitDB_thSDark {color: #000000; background: "+usdcolors.nohitDB+";}\n";
  664. css += ".tooweak_thSDark {color: #000000; background: "+usdcolors.unqualified+";}\n";
  665. css += ".yeshitDB_thSDark {color:#000000; background: "+usdcolors.hitDB+";}\n";
  666. css += ".needmaster_thSDark {color: #000000; background: "+usdcolors.reqmaster+";}\n";
  667. css += ".nomaster_thSDark {color: #000000; background: "+usdcolors.nomaster+";}\n";
  668. css += ".spacer_thSDark {color: "+usdcolors.accent+";}\n";
  669. css += ".mainlink_thSDark {color: "+usdcolors.defaultText+"; text-align: top;}\n";
  670. css += ".tabhead_thSDark {color: "+usdcolors.defaultText+";}\n";
  671. css += ".bodytable_thSDark {color: "+usdcolors.bodytable+";}\n";
  672. css += ".statusdiv_thSDark {font-size: 15px; color: "+usdcolors.secondText+";}\n";
  673. css += ".body_thWisp {background-color: "+uwcolors.background+";}\n";
  674. css += ".controlpanel_thWisp {color: "+uwcolors.defaultText+"; margin: 0px auto 5px auto; padding: 8px 5px 1px; background: "+uwcolors.cpBackground+";}\n";
  675. css += ".cpInput_thWisp {color: "+uwcolors.inputText+"; border: 1px solid; background: "+uwcolors.background+"; text-align: center;}\n";
  676. css += ".cpButtons_thWisp {color: "+uwcolors.accent+"; border: 1px solid; background: transparent;}\n";
  677. css += ".cpSortdiv_thWisp {color: "+uwcolors.inputText+"; display: inline; font: 14px;}\n";
  678. css += ".cpSpans_thWisp {padding: 2px; cursor:default; color: "+uwcolors.defaultText+"; background: "+uwcolors.cpBackground+";} .cpSpans_thWisp:hover {background: "+uwcolors.hover+";}\n";
  679. css += ".cpSpansOn_thWisp {background: "+uwcolors.highlight+"; color: "+uwcolors.secondText+";}\n";
  680. css += ".taButtons_thWisp {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+uwcolors.export+";} .taButtonsOff_thWisp {display: none;}\n";
  681. css += ".link_thWisp:link {color: "+uwcolors.link+";} .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  682. css += ".toNone_thWisp .link_thWisp, .toNone_thWisp .tolink_thWisp {color: "+uwcolors.noTO+";}\n .toNone_thWisp .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  683. css += ".toHigh_thWisp .link_thWisp, .toHigh_thWisp .tolink_thWisp {color: "+uwcolors.highTO+";}\n .toHigh_thWisp .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  684. css += ".toGood_thWisp .link_thWisp, .toGood_thWisp .tolink_thWisp {color: "+uwcolors.goodTO+";}\n .toGood_thWisp .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  685. css += ".toAverage_thWisp .link_thWisp, .toAverage_thWisp .tolink_thWisp {color: "+uwcolors.averageTO+";}\n .toAverage_thWisp .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  686. css += ".toLow_thWisp .link_thWisp, .toLow_thWisp .tolink_thWisp {color: "+uwcolors.lowTO+";}\n .toLow_thWisp .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  687. css += ".toPoor_thWisp .link_thWisp, .toPoor_thWisp .tolink_thWisp {color: "+uwcolors.poorTO+";}\n .toPoor_thWisp .link_thWisp:visited {color: "+uwcolors.vlink+";}\n";
  688. css += ".nohitDB_thWisp {color: #000000; background: "+uwcolors.nohitDB+";}\n";
  689. css += ".tooweak_thWisp {color: #000000; background: "+uwcolors.unqualified+";}\n";
  690. css += ".yeshitDB_thWisp {color:#000000; background: "+uwcolors.hitDB+";}\n";
  691. css += ".needmaster_thWisp {color: #000000; background: "+uwcolors.reqmaster+";}\n";
  692. css += ".nomaster_thWisp {color: #000000; background: "+uwcolors.nomaster+";}\n";
  693. css += ".spacer_thWisp {color: "+uwcolors.accent+";}\n";
  694. css += ".mainlink_thWisp {color: "+uwcolors.defaultText+"; text-align: top;}\n";
  695. css += ".tabhead_thWisp {color: "+uwcolors.defaultText+";}\n";
  696. css += ".bodytable_thWisp {color: "+uwcolors.bodytable+";}\n";
  697. css += ".statusdiv_thWisp {font-size: 15px; color: "+uwcolors.secondText+";}\n";
  698. css += ".body_thClassic {background-color: "+uclcolors.background+";}\n";
  699. css += ".controlpanel_thClassic {color: "+uclcolors.defaultText+"; margin: 0px auto 5px auto; padding: 8px 5px 1px; background: "+uclcolors.cpBackground+";}\n";
  700. css += ".cpInput_thClassic {color: "+uclcolors.inputText+"; border: 1px solid; text-align: center;}\n";
  701. css += ".cpButtons_thClassic {border: 2px outset buttonface;}\n";
  702. css += ".cpSortdiv_thClassic {color: "+uclcolors.inputText+"; display: inline; font: 14px;}\n";
  703. css += ".cpSpans_thClassic {padding: 2px; cursor:default; color: "+uclcolors.defaultText+"; background: "+uclcolors.cpBackground+";} .cpSpans_thClassic:hover {background: "+uclcolors.hover+";}\n";
  704. css += ".cpSpansOn_thClassic {background: "+uclcolors.highlight+"; color: "+uclcolors.secondText+";}\n";
  705. css += ".taButtons_thClassic {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+uclcolors.export+";} .taButtonsOff_thClassic {display: none;}\n";
  706. css += ".link_thClassic:link {color: "+uclcolors.link+";} .link_thClassic:visited {color: "+uclcolors.vlink+";}\n";
  707. css += ".toNone_thClassic {background: "+uclcolors.noTO+";}\n";
  708. css += ".toHigh_thClassic {background: "+uclcolors.highTO+";}\n";
  709. css += ".toGood_thClassic {background: "+uclcolors.goodTO+";}\n";
  710. css += ".toAverage_thClassic {background: "+uclcolors.averageTO+";}\n";
  711. css += ".toLow_thClassic {background: "+uclcolors.lowTO+";}\n";
  712. css += ".toPoor_thClassic {background: "+uclcolors.poorTO+";}\n";
  713. css += ".nohitDB_thClassic {color: #000000; background: "+uclcolors.nohitDB+";}\n";
  714. css += ".tooweak_thClassic {color: #000000; background: "+uclcolors.unqualified+";}\n";
  715. css += ".yeshitDB_thClassic {color:#000000; background: "+uclcolors.hitDB+";}\n";
  716. css += ".needmaster_thClassic {color: #000000; background: "+uclcolors.reqmaster+";}\n";
  717. css += ".nomaster_thClassic {color: #000000; background: "+uclcolors.nomaster+";}\n";
  718. css += ".spacer_thClassic {color: "+uclcolors.accent+";}\n";
  719. css += ".mainlink_thClassic {color: "+uclcolors.defaultText+"; text-align: top;}\n";
  720. css += ".tabhead_thClassic {color: "+uclcolors.defaultText+";}\n";
  721. css += ".bodytable_thClassic {color: "+uclcolors.bodytable+";}\n";
  722. css += ".statusdiv_thClassic {font-size: 15px; color: "+uclcolors.secondText+";}\n";
  723. css += ".edspan_thCustom {padding: 0px; font-size: 12px; display: block; width: 370px; color: "+ucucolors.defaultText+";}\n";
  724. css += ".edspan_thRandom {padding: 0px; font-size: 12px; display: block; width: 370px; color: "+rcolors.defaultText+";}\n";
  725. css += ".edspan_thWisp {padding: 0px; font-size: 12px; display: block; width: 370px; color: "+uwcolors.defaultText+";}\n";
  726. css += ".edspan_thClassic {padding: 0px; font-size: 12px; display: block; width: 370px; color: "+uclcolors.defaultText+";}\n";
  727. css += ".edspan_thSDark {padding: 0px; font-size: 12px; display: block; width: 370px; color: "+usdcolors.defaultText+";}\n";
  728. css += ".edspan_thSLight {padding: 0px; font-size: 12px; display: block; width: 370px; color: "+uslcolors.defaultText+";}\n";
  729.  
  730. // ******************* below is universal and applies to all themes ********************************
  731. css += ".scraperBlockedRowOff td { border: 3pt solid #cc0000; }\n .scraperBlockedRow {display: none; }\n";
  732. css += ".scraperIncludelistedRow td { border: 3pt dashed #008800; }\n .scraperIncludelistedRowOff td { border: none; }\n";
  733. css += ".toBlockedRow {display:none;} .toBlockedRowOff td {border: 2px solid #00E5FF;}\n";
  734. css += "#thMenu {z-index: 2; position: absolute; top: 0.5%; right: 0.5%; text-align: center; display: inline; margin: 0; padding: 10px 5px; list-style: none;}\n";
  735. css += "#thMenu li {font: 15px; display: inline-block; position: relative; padding: 5px 5px; background: #E3E8E8; color: #000000; cursor: default}\n";
  736. css += "#thMenu li:hover {background: #616363; color: #ffffff;}\n";
  737. css += "#thMenu li ul {position: absolute; top: 24px; right: 0; width: 120px; padding: 0; display: none;}\n";
  738. css += "#thMenu li ul li {background: #616363; color: #ffffff; display: block; padding: 10px 10px;} #thMenu li ul li:hover {background: #8C8C8C;}\n";
  739. css += "#thMenu li:hover ul {display: block;}\n";
  740. css += ".checkboxesOff {display: none;}\n";
  741. css += ".settingsCloseSpan {font-weight: bold; font-size: 1.2em; margin-top: 1.33em; margin-bottom: 1.33em; color: white; background: black; cursor: default; position: relative;}\n";
  742. css += ".settingsPanelMainDiv {z-index: 105; position: fixed; background: white; box-shadow: -3px 3px 2px 2px #7B8C89; line-height: initial;}\n";
  743. css += ".settingsPanelMainDiv:focus {outline:0;}\n";
  744. css += ".settingsInnerDiv {left: 5px; top: 20px; padding: 2px; position: relative; border: 1px solid grey; float:left; line-height: initial;}\n";
  745. css += ".settingsSidebarMainDiv {width: 100px; min-width: 90px;}\n";
  746. css += ".settingsMainContainer {left: 10px; border: transparent; line-height: initial; overflow:auto;}\n";
  747. css += ".settingsSidebarSpans {float: left; margin-bottom: 5px; width: 100px; font-size: 1.05em; cursor: pointer;} .settingsSidebarSpans-selected {background: aquamarine;}\n";
  748. css += ".settingsDivInput {float: left; border: 1px solid black; display: inline; width: 25px; text-align: center; background: white; line-height: initial;}\n";
  749. css += ".settingsDivInput-large {display: block;}\n";
  750. css += ".settingsOptionContainers {margin-bottom: 15px; position: relative; background: #CCFFFA; margin-left: 5px; overflow: auto; padding-right: 10px; padding-bottom: 6px;}\n";
  751. css += ".settingsOptionSections {margin-left: 100px;}\n";
  752. css += ".settingsLabelSA {float: left; width: 51px;}\n";
  753. css += ".settingsLabelBlocklist {float: left; width: 95px; margin-left: 15px;}\n";
  754. css += ".ble {border: 1px solid black; border-collapse: collapse;} .blec {padding: 5px; text-align: left;}\n";
  755. css += "</style>";
  756. var header = document.getElementsByTagName("head");
  757. header[0].innerHTML = header[0].innerHTML + css;
  758.  
  759. var next_page = 1;
  760.  
  761. var API_PROXY_BASE = 'https://mturk-api.istrack.in/';
  762. var OFFICIAL_API_PROXY_BASE = 'https://turkopticon.ucsd.edu/';
  763. var API_MULTI_ATTRS_URL = API_PROXY_BASE + 'multi-attrs.php?ids=';
  764. var OFFICIAL_API_MULTI_ATTRS_URL = OFFICIAL_API_PROXY_BASE + 'api/multi-attrs.php?ids=';
  765. var REVIEWS_BASE = 'http://turkopticon.ucsd.edu/';
  766.  
  767.  
  768. document.body.className = "body"+sufTheme;
  769.  
  770. // --- theme buttons
  771. var btn01 = document.createElement("BUTTON"); // edit
  772. var btn02 = document.createElement("BUTTON"); // reset
  773. var thin = [ document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"),
  774. document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"),
  775. document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"),
  776. document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT"),
  777. document.createElement("INPUT"), document.createElement("INPUT"), document.createElement("INPUT") ];
  778. btn01.className = "cpButtons"+sufTheme;
  779. btn02.className = "cpButtons"+sufTheme;
  780. btn01.textContent = "Edit Current Theme";
  781. btn02.textContent = "Reset To Default Values";
  782. btn02.title = "Reset the values of the current theme to its default.";
  783. btn01.onclick = function() {
  784. if (btn01.textContent === "Edit Current Theme") {
  785. eddiv.style.display = 'initial';
  786. theme_editor();
  787. btn01.textContent = "Hide Editor";
  788. }
  789. else {
  790. eddiv.style.display = 'none';
  791. btn01.textContent = "Edit Current Theme";
  792. }
  793. };
  794. btn02.onclick = function() {
  795. var cc = {}, dc = {}, kc = "";
  796. if (sufTheme === "_thWisp") { cc = uwcolors; dc = wcolors; kc = "customwisp"; }
  797. else if (sufTheme === "_thSDark") { cc = usdcolors; dc = sdcolors; kc = "customsd"; }
  798. else if (sufTheme === "_thSLight") { cc = uslcolors; dc = slcolors; kc = "customsl"; }
  799. else if (sufTheme === "_thCustom") { cc = ucucolors; dc = cucolors; kc = "customcustom"; }
  800. else if (sufTheme === "_thClassic") { cc = uclcolors; dc = clcolors; kc = "customclassic"; }
  801. for (var key in dc) {
  802. if (dc.hasOwnProperty(key)) cc[key] = dc[key];
  803. }
  804. theme_editor();
  805. saveState.setItem(kc, cc);
  806. saveState.save();
  807. };
  808.  
  809. // --- control panel framing
  810. var control_panel = document.createElement("DIV");
  811. control_panel.className = "controlpanel"+sufTheme;
  812. var cpdiv = document.createElement("DIV");
  813. var eddiv = document.createElement("DIV");
  814. cpdiv.id = 'control_panel';
  815. var cptable = document.createElement("TABLE");
  816. cpdiv.appendChild(cptable);
  817. var testinput = document.createElement("INPUT");
  818. cptable.style.width = '100%';
  819. var cptr01 = document.createElement("TR");
  820. var cptd01 = document.createElement("TD"); //left
  821. cptd01.style.width = '900px';
  822. var cptd02 = document.createElement("TD"); //right
  823. cptable.appendChild(cptr01);
  824. cptr01.appendChild(cptd01); // left
  825. cptd01.appendChild(control_panel);
  826. cptr01.appendChild(cptd02); // right
  827. cptd02.appendChild(eddiv);
  828. eddiv.appendChild(btn02);
  829. var edspan = document.createElement("SPAN");
  830. edspan.className = "edspan"+sufTheme;
  831.  
  832. edspan.innerHTML = "<br>Mouseover the input boxes for a description. With the exception of the random theme, anything changed here will be saved under the current theme. <b>Note: Some values, along with resetting to default, will require a page refresh to take effect.</b><br><br>";
  833. eddiv.appendChild(edspan);
  834. for (var i=0; i<thin.length; i++) {
  835. eddiv.appendChild(thin[i]);
  836. eddiv.appendChild(document.createTextNode(" "));
  837. if ((i+1)%4 === 0) eddiv.appendChild(document.createElement("BR"));
  838. thin[i].size = '6';
  839. thin[i].className = "color {hash:true, pickerFaceColor:'transparent',pickerFace:0,pickerBorder:0,pickerInsetColor:'black'}";
  840. }
  841. eddiv.style.display = 'none';
  842. document.body.insertBefore(cpdiv, document.body.firstChild);
  843. $('body > :not(#control_panel)').hide(); //hide all nodes directly under the body
  844.  
  845. // --- settings panel
  846. var settings = {};
  847. settings.panel = { mainDiv: document.createElement("DIV"),
  848. containers: document.createElement("DIV"),
  849. sections: document.createElement("SECTION"),
  850. accessButton: document.createElement("BUTTON"),
  851. closeSpan: document.createElement("SPAN"),
  852. init: function() {
  853. this.mainDiv.className = "settings settingsPanelMainDiv";
  854. this.mainDiv.id = "settingsPanel";
  855. this.mainDiv.tabIndex = "0";
  856. this.mainDiv.style.display = "none";
  857. //this.mainDiv.addEventListener("blur", function() { settings.panel.mainDiv.style.display = "none"; });
  858. this.mainDiv.appendChild(this.closeSpan);
  859. this.mainDiv.appendChild(settings.sidebar.mainDiv);
  860. this.mainDiv.appendChild(settings.general.mainDiv);
  861. this.accessButton.className = "settings cpButtons"+sufTheme;
  862. this.accessButton.id = "settingsButton";
  863. this.accessButton.textContent = "Settings";
  864. this.accessButton.onclick = function() { settings.display(); };
  865. this.closeSpan.className = "settings settingsCloseSpan";
  866. this.closeSpan.innerHTML = "&#160;&#10008;&#160;";
  867. this.closeSpan.addEventListener("click", function() { settings.panel.mainDiv.style.display = "none"; });
  868. this.closeSpan.title = "Close";
  869. this.containers.className = "settings settingsOptionContainers";
  870. this.sections.className = "settings settingsOptionSections";
  871. }
  872. };
  873. settings.sidebar = { mainDiv: document.createElement("DIV"),
  874. generalSpan: document.createElement("SPAN"),
  875. blocklistSpan: document.createElement("SPAN"),
  876. init: function() {
  877. this.mainDiv.className = "settings settingsInnerDiv settingsSidebarMainDiv";
  878. this.mainDiv.appendChild(this.generalSpan);
  879. this.mainDiv.appendChild(this.blocklistSpan);
  880. this.generalSpan.className = "settings settingsSidebarSpans settingsSidebarGeneralSpan";
  881. this.generalSpan.textContent = "General";
  882. this.generalSpan.addEventListener("click", function() { settings.sidebar._click("generalSpan", "general") });
  883. this.blocklistSpan.className = "settings settingsSidebarSpans settingsSidebarBlocklistSpan";
  884. this.blocklistSpan.textContent = "Blocklist";
  885. this.blocklistSpan.addEventListener("click", function() { settings.sidebar._click("blocklistSpan", "blocklist") });
  886. },
  887. _click: function(v, div) {
  888. if (!settings.sidebar[v].classList.contains("settingsSidebarSpans-selected")) {
  889. settings.sidebar[v].classList.toggle("settingsSidebarSpans-selected");
  890. var last = settings.panel.mainDiv.lastChild;
  891. settings.sidebar[last.dataset.category.concat("Span")].classList.toggle("settingsSidebarSpans-selected");
  892. settings.panel.mainDiv.replaceChild(settings[div].mainDiv, last);
  893. }
  894. }
  895. };
  896. settings.general = { mainDiv: document.createElement("DIV"),
  897. colorTypeSimple: document.createElement("INPUT"),
  898. colorTypeAdjusted: document.createElement("INPUT"),
  899. sortTypeSimple: document.createElement("INPUT"),
  900. sortTypeAdjusted: document.createElement("INPUT"),
  901. colorTSLabel: document.createElement("LABEL"),
  902. colorTALabel: document.createElement("LABEL"),
  903. sortTSLabel: document.createElement("LABEL"),
  904. sortTALabel: document.createElement("LABEL"),
  905. commWeight: document.createElement("DIV"),
  906. payWeight: document.createElement("DIV"),
  907. fairWeight: document.createElement("DIV"),
  908. fastWeight: document.createElement("DIV"),
  909. commLabel: document.createElement("LABEL"),
  910. payLabel: document.createElement("LABEL"),
  911. fairLabel: document.createElement("LABEL"),
  912. fastLabel: document.createElement("LABEL"),
  913. init: function() {
  914. this.mainDiv.className = "settings settingsInnerDiv settingsMainContainer settingsGeneralMainDiv";
  915. this.mainDiv.dataset.category = "general";
  916. var colorTypeContainer = settings.panel.containers.cloneNode(false);
  917. var colorTypeSectionSimple = settings.panel.sections.cloneNode(false);
  918. var colorTypeSectionAdjusted = settings.panel.sections.cloneNode(false);
  919. var sortTypeContainer = settings.panel.containers.cloneNode(false);
  920. var sortTypeSectionSimple = settings.panel.sections.cloneNode(false);
  921. var sortTypeSectionAdjusted = settings.panel.sections.cloneNode(false);
  922. var weightContainer = settings.panel.containers.cloneNode(false);
  923. var weightSection = settings.panel.sections.cloneNode(false);
  924. this.mainDiv.appendChild(colorTypeContainer);
  925. this.mainDiv.appendChild(sortTypeContainer);
  926. this.mainDiv.appendChild(weightContainer);
  927. this.colorTypeSimple.className = "settings settingsRadio settingsGeneralColorTypeSimple";
  928. this.colorTypeSimple.type = "radio";
  929. this.colorTypeSimple.name = "colorType";
  930. this.colorTypeSimple.id = "colorTypeSimple";
  931. this.colorTypeSimple.addEventListener("click", function() { settings.options.commit("resultsColorType", "colorType", "sim"); });
  932. this.colorTSLabel.htmlFor = "colorTypeSimple";
  933. this.colorTSLabel.textContent = "Simple";
  934. this.colorTSLabel.className = "settings settingsLabelSA";
  935. this.colorTypeAdjusted.className = "settings settingsRadio settingsGeneralColorTypeAdjusted";
  936. this.colorTypeAdjusted.type = "radio";
  937. this.colorTypeAdjusted.name = "colorType";
  938. this.colorTypeAdjusted.id = "colorTypeAdjusted";
  939. this.colorTypeAdjusted.addEventListener("click", function() { settings.options.commit("resultsColorType", "colorType", "adj"); });
  940. this.colorTALabel.htmlFor = "colorTypeAdjusted";
  941. this.colorTALabel.textContent = "Adjusted";
  942. this.colorTALabel.className = "settings settingsLabelSA";
  943. var tempDiv = document.createElement("DIV")
  944. tempDiv.style.float = "left";
  945. tempDiv.style.marginLeft = "15px";
  946. tempDiv.innerHTML = "<span style='position: relative; left: -8px;'><b>Color Type</b></span><br>";
  947. tempDiv.appendChild(this.colorTSLabel);
  948. tempDiv.appendChild(this.colorTypeSimple);
  949. tempDiv.appendChild(document.createElement("P"));
  950. tempDiv.appendChild(this.colorTALabel);
  951. tempDiv.appendChild(this.colorTypeAdjusted);
  952. colorTypeContainer.appendChild(tempDiv);
  953. colorTypeContainer.appendChild(colorTypeSectionSimple);
  954. colorTypeContainer.appendChild(colorTypeSectionAdjusted);
  955. colorTypeSectionSimple.innerHTML = "<span style='position: relative; left: 10px;'><i>simple</i></span><br>HIT Scraper will use a simple weighted average to ";
  956. colorTypeSectionSimple.innerHTML += "determine the overall TO rating and colorize results off that value. Use this setting to equalize coloration between ";
  957. colorTypeSectionSimple.innerHTML += "HIT Scraper and Color Coded Search.";
  958. colorTypeSectionAdjusted.innerHTML = "<span style='position: relative; left: 10px;'><i>adjusted</i></span><br>HIT Scraper will calculate an adjusted average based on ";
  959. colorTypeSectionAdjusted.innerHTML += "confidence of the TO rating to colorize results. Confidence is proportional to the number of reviews.";
  960. this.sortTypeSimple.className = "settings settingsRadio settingsGeneralSortTypeSimple";
  961. this.sortTypeSimple.type = "radio";
  962. this.sortTypeSimple.name = "sortType";
  963. this.sortTypeSimple.id = "sortTypeSimple";
  964. this.sortTypeSimple.addEventListener("click", function() { settings.options.commit("resultsSortType", "sortType", "sim"); });
  965. this.sortTSLabel.htmlFor = "sortTypeSimple";
  966. this.sortTSLabel.textContent = "Simple";
  967. this.sortTSLabel.className = "settings settingsLabelSA";
  968. this.sortTypeAdjusted.className = "settings settingsRadio settingsGeneralSortTypeAdjusted";
  969. this.sortTypeAdjusted.type = "radio";
  970. this.sortTypeAdjusted.name = "sortType";
  971. this.sortTypeAdjusted.id = "sortTypeAdjusted";
  972. this.sortTypeAdjusted.addEventListener("click", function() { settings.options.commit("resultsSortType", "sortType", "adj"); });
  973. this.sortTALabel.htmlFor = "sortTypeAdjusted";
  974. this.sortTALabel.textContent = "Adjusted";
  975. this.sortTALabel.className = "settings settingsLabelSA";
  976. var tempDiv2 = tempDiv.cloneNode(false);
  977. tempDiv2.innerHTML = "<span style='position: relative; left: -8px;'><b>Sort Type</b></span><br>";
  978. tempDiv2.appendChild(this.sortTSLabel);
  979. tempDiv2.appendChild(this.sortTypeSimple);
  980. tempDiv2.appendChild(document.createElement("P"));
  981. tempDiv2.appendChild(this.sortTALabel);
  982. tempDiv2.appendChild(this.sortTypeAdjusted);
  983. sortTypeContainer.appendChild(tempDiv2);
  984. sortTypeContainer.appendChild(sortTypeSectionSimple);
  985. sortTypeContainer.appendChild(sortTypeSectionAdjusted);
  986. sortTypeSectionSimple.innerHTML = "<span style='position: relative; left: 10px;'><i>simple</i></span><br>";
  987. sortTypeSectionSimple.innerHTML += "HIT Scraper will sort results based simply on value regardless of the number of reviews.";
  988. sortTypeSectionAdjusted.innerHTML = "<span style='position: relative; left: 10px;'><i>adjusted</i></span><br>HIT Scraper will adjust ratings based on ";
  989. sortTypeSectionAdjusted.innerHTML += "reliability (ie. confidence) of the data. It factors in the number of reviews such that, for example, a requester ";
  990. sortTypeSectionAdjusted.innerHTML += "with 100 reviews rated at 4.6 will rightfully be ranked higher than a requester with 3 reviews rated at 5. ";
  991. sortTypeSectionAdjusted.innerHTML += "This gives a more accurate representation of the data.";
  992. this.commWeight.className = "settings settingsDivInput settingsGeneralCommWeight";
  993. this.commWeight.contentEditable = true;
  994. this.commWeight.title = "default value: 1";
  995. this.commWeight.addEventListener("keyup", function() {
  996. var num = Math.abs(Number(settings.general.commWeight.textContent));
  997. if (!isNaN(num))
  998. settings.options.commit("comm", "commWeight", num);
  999. });
  1000. this.commLabel.textContent = "comm: ";
  1001. this.commLabel.className = "settings settingsLabelSA";
  1002. this.payWeight.className = "settings settingsDivInput settingsGeneralPayWeight";
  1003. this.payWeight.contentEditable = true;
  1004. this.payWeight.title = "default value: 3";
  1005. this.payWeight.addEventListener("keyup", function() {
  1006. var num = Math.abs(Number(settings.general.payWeight.textContent));
  1007. if (!isNaN(num))
  1008. settings.options.commit("pay", "payWeight", num);
  1009. });
  1010. this.payLabel.textContent = "pay: ";
  1011. this.payLabel.className = "settings settingsLabelSA";
  1012. this.fairWeight.className = "settings settingsDivInput settingsGeneralFairWeight";
  1013. this.fairWeight.contentEditable = true;
  1014. this.fairWeight.title = "default value: 3";
  1015. this.fairWeight.addEventListener("keyup", function() {
  1016. var num = Math.abs(Number(settings.general.fairWeight.textContent));
  1017. if (!isNaN(num))
  1018. settings.options.commit("fair", "fairWeight", num);
  1019. });
  1020. this.fairLabel.textContent = "fair: ";
  1021. this.fairLabel.className = "settings settingsLabelSA";
  1022. this.fastWeight.className = "settings settingsDivInput settingsGeneralFastWeight";
  1023. this.fastWeight.contentEditable = true;
  1024. this.fastWeight.title = "default value: 1";
  1025. this.fastWeight.addEventListener("keyup", function() {
  1026. var num = Math.abs(Number(settings.general.fastWeight.textContent));
  1027. if (!isNaN(num))
  1028. settings.options.commit("fast", "fastWeight", num);
  1029. });
  1030. this.fastLabel.textContent = "fast: ";
  1031. this.fastLabel.className = "settings settingsLabelSA";
  1032. var tempDiv3 = tempDiv.cloneNode(false);
  1033. tempDiv3.innerHTML = "<span style='position: relative; left: -8px;'><b>TO Weighting</b></span><br>";
  1034. tempDiv3.appendChild(this.commLabel);
  1035. tempDiv3.appendChild(this.commWeight);
  1036. tempDiv3.appendChild(document.createElement("BR"));
  1037. tempDiv3.appendChild(this.payLabel);
  1038. tempDiv3.appendChild(this.payWeight);
  1039. tempDiv3.appendChild(document.createElement("BR"));
  1040. tempDiv3.appendChild(this.fairLabel);
  1041. tempDiv3.appendChild(this.fairWeight);
  1042. tempDiv3.appendChild(document.createElement("BR"));
  1043. tempDiv3.appendChild(this.fastLabel);
  1044. tempDiv3.appendChild(this.fastWeight);
  1045. weightContainer.appendChild(tempDiv3);
  1046. weightContainer.appendChild(weightSection);
  1047. weightSection.innerHTML = "<p>Specify weights for TO attributes to place greater importance on certain attributes over others.<br>";
  1048. weightSection.innerHTML += "The default values respect backwards compatibility; recommended settings are [1, 6, 3.5, 1].";
  1049. },
  1050. _init: function() {
  1051. var checked = settings.options.resultsColorType;
  1052. this.colorTypeSimple.checked = (checked === "adj") ? false : true;
  1053. this.colorTypeAdjusted.checked = (checked === "adj") ? true : false;
  1054. checked = settings.options.resultsSortType;
  1055. this.sortTypeSimple.checked = (checked === "adj") ? false : true;
  1056. this.sortTypeAdjusted.checked = (checked === "adj") ? true : false;
  1057. this.commWeight.textContent = settings.options.toWeighting.comm;
  1058. this.payWeight.textContent = settings.options.toWeighting.pay;
  1059. this.fairWeight.textContent = settings.options.toWeighting.fair;
  1060. this.fastWeight.textContent = settings.options.toWeighting.fast;
  1061. }
  1062. };
  1063. settings.blocklist = { mainDiv: document.createElement("DIV"),
  1064. theListBox: document.createElement("DIV"),
  1065. useWildcards: document.createElement("INPUT"),
  1066. wildcardLabel: document.createElement("LABEL"),
  1067. init: function() {
  1068. this.mainDiv.className = "settings settingsInnerDiv settingsMainContainer settingsBlocklistMainDiv";
  1069. this.mainDiv.dataset.category = "blocklist";
  1070. var wildcardContainer = settings.panel.containers.cloneNode(false);
  1071. var listEditorContainer = settings.panel.containers.cloneNode(false);
  1072. var wildcardSection = settings.panel.sections.cloneNode(false);
  1073. var tempDiv = document.createElement("DIV");
  1074. tempDiv.style.float = "left";
  1075. this.mainDiv.appendChild(wildcardContainer);
  1076. //this.mainDiv.appendChild(listEditorContainer);
  1077. this.useWildcards.className = "settings settingsCheckbox settingsBlocklistUseWildcards";
  1078. this.useWildcards.type = "checkbox";
  1079. this.useWildcards.id = "useWildcards";
  1080. this.useWildcards.addEventListener("click", function() { settings.options.commit("blocklistWildcards", "wildblocks", settings.blocklist.useWildcards.checked); });
  1081. this.wildcardLabel.className = "settings settingsLabelBlocklist";
  1082. this.wildcardLabel.htmlFor = "useWildcards";
  1083. this.wildcardLabel.textContent = "Allow Wildcards";
  1084. this.theListBox.className = "settings settingsDivInput settingsDivInput-large settingsBlocklistList";
  1085. this.theListBox.contentEditable = true;
  1086. wildcardContainer.innerHTML = "<span style='position: relative; left: 7px;'><b>Advanced Matching</b></span><p></p>";
  1087. wildcardContainer.appendChild(tempDiv);
  1088. wildcardContainer.appendChild(wildcardSection);
  1089. tempDiv.appendChild(this.wildcardLabel);
  1090. tempDiv.appendChild(this.useWildcards);
  1091. wildcardSection.style.marginLeft = "150px";
  1092. wildcardSection.innerHTML = "Allows for the use of asterisks <code>(*)</code> as wildcards in the blocklist for simple glob matching. Any blocklist entry without ";
  1093. wildcardSection.innerHTML += "an asterisk is treated the same as the default behavior; the entry must exactly match a HIT title or requester to trigger ";
  1094. wildcardSection.innerHTML += "a block. <p><em>Wildcards have the potential to block more HITs than intended if using a pattern that's too generic.</em></p>";
  1095. wildcardSection.innerHTML += "<p></p>Matching is not case sensitive regardless of the wildcard setting. Entries without an opening asterisk are expected ";
  1096. wildcardSection.innerHTML += "to match the beginning of a line, likewise, entries without a closing asterisk are expected to match the end of a line. ";
  1097. wildcardSection.innerHTML += "Example usage below. <p></p><table class='ble' style='left: -100px; position:relative; width:100%;'><tr><th class='blec ble'></th><th class='blec ble'>Matches</th><th class='blec ble'>\
  1098. Does not match</th><th class='blec ble'>Notes</th></tr><tr><td rowspan='2' class='blec ble'><code>foo*baz</code></td><td class='blec ble'>foo bar bat baz<td class='blec ble'>bar foo bat baz\
  1099. </td><td rowspan='2' class='blec ble'>no leading or closing asterisks; <code>foo</code> must be at the start of a line, and <code>baz</code> must be at the end of a line for a \
  1100. positive match</td></tr><tr><td class='blec ble'>foobarbatbaz</td><td class='blec ble'>foo bar bat</td></tr><tr><td class='blec ble'><code>*foo</code></td>\
  1101. <td class='ble blec'>bar baz foo</td><td class='blec ble'>foo baz</td><td class='ble blec'>matches and blocks any line ending in <code>foo</code>\
  1102. </td></tr><tr><td class='blec ble'><code>foo*</code></td><td class='ble blec'>foo bat bar</td><td class='ble blec'>bat foo baz</td><td class='ble blec'>\
  1103. matches and blocks any line beginning with <code>foo</code></td></tr><tr><td class='ble blec' rowspan='4'><code>*bar*</code></td><td class='blec ble'>foo bar bat baz</td>\
  1104. <td class='ble blec' rowspan='4'>foo bat baz</td><td class='ble blec' rowspan='4'>matches and blocks any line containing <code>bar</code></td></tr><tr><td class='ble blec'>bar bat baz</td>\
  1105. </tr><tr><td class='blec ble'>foo bar</td></tr><tr><td class='blec ble'>foobatbarbaz</td></tr><tr><td class='ble blec'><code>** foo</code></td><td class='ble blec'>** foo</td>\
  1106. <td class='ble blec'>** foo bar baz</td><td class='ble blec'>Multiple consecutive asterisks will be treated as a string rather than a wildcard. \
  1107. This makes it compatible with HITs using multiple asterisks in their titles, <i>eg.</i>, <code>*** contains peanuts ***</code>.</td></tr><tr><td class='ble blec'>\
  1108. <code>** *bar* ***</td><td class='blec ble'>** foo bar baz bat ***</td><td class='ble blec'>foo bar baz</td><td class='blec ble'>Consecutive asterisks used in conjunction with single asterisks.\
  1109. </td><tr><td class='blec ble'><code>*</code></td><td class='ble blec'></td><td class='ble blec'><i>everything</i></td><td class='ble blec'>A single asterisk would usually match anything and everything, \
  1110. but here, it matches nothing. This prevents accidentally blocking everything from the results table.</td></tr></table>";
  1111. listEditorContainer.innerHTML = "<span style='position: relative; left: 7px;'><b>Blocklist Editor</b></span><br>";
  1112. listEditorContainer.appendChild(this.theListBox)
  1113. },
  1114. _init: function() {
  1115. this.useWildcards.checked = settings.options.blocklistWildcards;
  1116. this.theListBox.textContent = "placeholder";
  1117. }
  1118. };
  1119.  
  1120. settings.options = { resultsColorType: null,
  1121. resultsSortType: null,
  1122. toWeighting: { comm: null, pay: null, fair: null, fast: null },
  1123. blocklistWildcards: null,
  1124. init: function() {
  1125. this.resultsColorType = saveState.getItem("colorType") || "sim";
  1126. this.resultsSortType = saveState.getItem("sortType") || "adj";
  1127. this.toWeighting = {comm: saveState.getItem("commWeight") || 1,
  1128. pay: saveState.getItem("payWeight") || 3,
  1129. fair: saveState.getItem("fairWeight") || 3,
  1130. fast: saveState.getItem("fastWeight") || 1
  1131. };
  1132. this.blocklistWildcards = saveState.getItem("wildblocks") || false;
  1133. },
  1134. commit: function(v, key, val) {
  1135. if (this[v] || this[v] === false)
  1136. this[v] = val;
  1137. else if (this.toWeighting[v] || this.toWeighting[v] == 0)
  1138. this.toWeighting[v] = val;
  1139. else {
  1140. return;
  1141. }
  1142. saveState.setItem(key, val);
  1143. saveState.save();
  1144. }
  1145. };
  1146. settings.display = function() {
  1147. var swidth = window.innerWidth*0.9;
  1148. var sheight = window.innerHeight*0.9;
  1149. var _display = document.querySelector(".settingsSidebarSpans-selected");
  1150. this.panel.mainDiv.style.height = String(sheight).concat("px");
  1151. this.panel.mainDiv.style.width = String(swidth).concat("px");
  1152. this.panel.mainDiv.style.top = String(window.innerHeight*0.05).concat("px");
  1153. this.panel.mainDiv.style.left = String(window.innerWidth*0.05).concat("px");
  1154. this.panel.closeSpan.style.left = String(swidth - 128).concat("px");
  1155. this.sidebar.mainDiv.style.height = String(sheight-45).concat("px");
  1156. this.sidebar.mainDiv.style.maxHeight = this.sidebar.mainDiv.style.height;
  1157. if (!_display)
  1158. this.sidebar.generalSpan.className += " settingsSidebarSpans-selected";
  1159. this.general.mainDiv.style.height = String(sheight-59).concat("px");
  1160. this.general.mainDiv.style.width = String(swidth-126).concat("px");
  1161. this.blocklist.mainDiv.style.height = String(sheight-59).concat("px");
  1162. this.blocklist.mainDiv.style.width = String(swidth-126).concat("px");
  1163. var containers = document.querySelectorAll(".settingsOptionContainers");
  1164. for (var i=0; i<containers.length; i++) {
  1165. containers[i].style.width = String(swidth-126-16).concat("px");
  1166. };
  1167. this.general._init();
  1168. this.blocklist._init();
  1169. this.panel.mainDiv.style.display = "block";
  1170. this.panel.mainDiv.focus();
  1171. };
  1172. settings.init = function() {
  1173. document.body.insertBefore(this.panel.mainDiv, document.body.firstChild);
  1174. this.options.init();
  1175. this.panel.init();
  1176. this.sidebar.init();
  1177. this.general.init();
  1178. this.blocklist.init();
  1179. };
  1180. settings.init();
  1181.  
  1182.  
  1183. var big_red_button = document.createElement("BUTTON");
  1184. var reset_blocks = document.createElement("BUTTON");
  1185. var include_button = document.createElement("BUTTON");
  1186. var progress_report = document.createTextNode("Stopped");
  1187. var status_report = document.createTextNode("None");
  1188. var text_area = document.createElement("TABLE");
  1189. text_area.id = "body_table";
  1190. //text_area.className = "bodytable"+sufTheme;
  1191. var showButton = document.createElement("BUTTON");
  1192. showButton.className += " cpButtons"+sufTheme;
  1193. showButton.onclick = function() { show_TO_rows(); };
  1194. showButton.textContent = "Show ignored hits";
  1195. showButton.id = "show_TO_stuff_button";
  1196.  
  1197. function show_TO_rows() {
  1198. var mr = document.getElementsByClassName("toBlockedRow");
  1199. var mt = mr || "[]";
  1200. if (showButton.textContent == "Show ignored hits") {
  1201. for (var i=mt.length-1; i>-1; --i) { mt[i].className = mt[i].className.replace(/toBlockedRow/, "toBlockedRowOff"); }
  1202. showButton.textContent = "Hide ignored hits";
  1203. }
  1204. else {
  1205. mt = document.getElementsByClassName("toBlockedRowOff");
  1206. for (var i=mt.length-1; i>-1; i--) { mt[i].className = mt[i].className.replace(/toBlockedRowOff/, "toBlockedRow"); }
  1207. showButton.textContent = "Show ignored hits";
  1208. }
  1209. }
  1210.  
  1211. show_interface();
  1212.  
  1213. // --- theme randomization
  1214. function yiqBrightness(hc,yiq) { // hc type "#000000", "000000", "[00, 00, 00]"
  1215. var r = 0, b = 0, g = 0;
  1216. if (yiq===true) {
  1217. if (typeof hc === 'object') { r = hc[0]; g = hc[1]; b = hc[2]; }
  1218. else if (hc.substr(0,1) == "#") {
  1219. r = parseInt(hc.substr(1,2), 16);
  1220. g = parseInt(hc.substr(3,2), 16);
  1221. b = parseInt(hc.substr(5,2), 16);
  1222. }
  1223. else {
  1224. r = parseInt(hc.substr(0,2), 16);
  1225. g = parseInt(hc.substr(2,2), 16);
  1226. b = parseInt(hc.substr(4,2), 16);
  1227. }
  1228. return r*0.299 + g*0.587 + b*0.114; // y value
  1229. }
  1230. else {
  1231. return hc[0] + hc[1]*0.956 + hc[2]*0.621; // get r value from full yiq
  1232. }
  1233. }
  1234. function randomizeScheme() {
  1235. // TODO: make randomization faster
  1236.  
  1237. if (!("background" in rcolors))
  1238. for (var l in cucolors) { if (cucolors.hasOwnProperty(l)) rcolors[l] = cucolors[l]; }
  1239. var r = 0, g = 0, b = 0;
  1240. var k = Object.keys(rcolors);
  1241. for (k in rcolors) {
  1242. rcolors[k] = getNewColor();
  1243. }
  1244. console.groupCollapsed("Random theme init routine");
  1245. var bg1 = yiqBrightness(rcolors.cpBackground, true), bg0 = yiqBrightness(rcolors.background,true);
  1246. var colorslice = {};
  1247. var chex = "";
  1248. k = ["defaultText", "inputText", "secondText", "export", "accent"];
  1249. for (var i in k) { colorslice[k[i]] = rcolors[k[i]]; }
  1250. for (k in colorslice) {
  1251. var iterations = 0;
  1252. if (colorslice.hasOwnProperty(k)) {
  1253. var c1 = yiqBrightness(rcolors[k],true);
  1254. while (Math.abs(bg1-c1)<69 || Math.abs(bg0-c1)<69) { // increase readability
  1255. if (iterations > 61000) {
  1256. console.log("Skipping: "+k+". Final value: " + rcolors[k]);
  1257. break;
  1258. }
  1259. chex = getNewColor();
  1260. c1 = yiqBrightness(chex,true);
  1261. rcolors[k] = chex;
  1262. iterations++;
  1263. }
  1264. console.log(k+": "+rcolors[k]+" after "+iterations + " cycles");
  1265. }
  1266. }
  1267. console.groupEnd();
  1268. }
  1269. function getNewColor() {
  1270. var ri = Math.floor( Math.random()*255 ).toString(16);
  1271. var gi = Math.floor( Math.random()*255 ).toString(16);
  1272. var bi = Math.floor( Math.random()*255 ).toString(16);
  1273. return "#".concat( ri.length<2 ? ("0".concat(ri)) : ri ).concat( gi.length<2 ? ("0".concat(gi)) : gi ).concat( bi.length<2 ? ("0".concat(bi)) : bi);
  1274. }
  1275.  
  1276. // ---
  1277. var global_run = false;
  1278. var statusdetail_loop_finished = false;
  1279. var date_header = "";
  1280. var scraper_history = {};
  1281. var wait_loop;
  1282. var dings = 0;
  1283.  
  1284. function set_progress_report(text, force)
  1285. {
  1286. if (global_run == true || force == true)
  1287. {
  1288. progress_report.textContent = text;
  1289. var status_text = status_array.join("; ");
  1290. status_report.textContent = status_text;
  1291. }
  1292. }
  1293.  
  1294. function get_progress_report()
  1295. {
  1296. return progress_report.textContent;
  1297. }
  1298.  
  1299. function wait_until_stopped()
  1300. {
  1301. if (global_run == true)
  1302. {
  1303. if (statusdetail_loop_finished == true)
  1304. {
  1305. big_red_button.textContent = "Start";
  1306. set_progress_report("Finished", false);
  1307. }
  1308. else
  1309. {
  1310. setTimeout(function(){wait_until_stopped();}, 500);
  1311. }
  1312. }
  1313. }
  1314.  
  1315. function display_wait_time(wait_time)
  1316. {
  1317. if (global_run == true)
  1318. {
  1319. var current_progress = get_progress_report();
  1320. if (current_progress.indexOf("Searching again in")!==-1)
  1321. {
  1322. set_progress_report(current_progress.replace(/Searching again in \d+ seconds/ , "Searching again in " + wait_time + " seconds"),false);
  1323. }
  1324. else
  1325. set_progress_report(current_progress + " Searching again in " + wait_time + " seconds.", false);
  1326. if (wait_time>1)
  1327. setTimeout(function(){display_wait_time(wait_time-1);}, 1000);
  1328. }
  1329. }
  1330.  
  1331. function dispArr(ar)
  1332. {
  1333. var disp = "";
  1334. for (var z = 0; z < ar.length; z++)
  1335. {
  1336. disp += "id " + z + " is " + ar[z] + " ";
  1337. }
  1338. //console.log(disp);
  1339. }
  1340.  
  1341. function scrape($src)
  1342. {
  1343. var $requester = $src.find('a[href^="/mturk/searchbar?selectedSearchType=hitgroups&requester"]');
  1344. if ($requester.length == 0)
  1345. $requester = $src.find('span.requesterIdentity');
  1346. var $title = $src.find('a[class="capsulelink"]');
  1347. var $reward = $src.find('span[class="reward"]');
  1348. var $preview = $src.find('a[href^="/mturk/preview?"]');
  1349. var $qualified = $src.find('a[href^="/mturk/notqualified?"],a[id^="private_hit"]');
  1350. var $times = $src.find('a[id^="duration_to_complete"]');
  1351. var $descriptions = $src.find('a[id^="description"]');
  1352. var not_qualified_group_IDs=[];
  1353. var $quals = $src.find('a[id^="qualificationsRequired"]');
  1354. var $mixed;
  1355. $qualified.each(function(){
  1356. var groupy = $(this).attr('href');
  1357. if (groupy){
  1358. groupy = groupy.replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1");
  1359. groupy = groupy.replace(/\/mturk\/notqualified\?hitId=([A-Z0-9]+)(&.+)?/,"$1");
  1360. groupy = groupy.replace(/\&hitGroupId=([A-Z0-9]+)(&.+)?/,"");
  1361. groupy = groupy.replace(/\&hitId=([A-Z0-9]+)(&.+)?/,"");
  1362. }
  1363. not_qualified_group_IDs.push(groupy);
  1364. });
  1365. //console.log(not_qualified_group_IDs);
  1366. if (document.getElementById('lnkWorkerSignin'))
  1367. $mixed = $src.find('a[href^="/mturk/preview?"],a[id^="private_hit"]');
  1368. else
  1369. $mixed = $src.find('a[href^="/mturk/preview?"],a[href^="/mturk/notqualified?"]');
  1370. //console.log($mixed);
  1371. var listy =[];
  1372. $mixed.each(function(){
  1373. var groupy = $(this).attr('href');
  1374. if (groupy)
  1375. {
  1376. groupy = groupy.replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1");
  1377. groupy = groupy.replace(/\/mturk\/notqualified\?hitId=([A-Z0-9]+)(&.+)?/,"$1");
  1378. groupy = groupy.replace(/\&hitGroupId=([A-Z0-9]+)(&.+)?/,"");
  1379. groupy = groupy.replace(/\&hitId=([A-Z0-9]+)(&.+)?/,"");
  1380. groupy = groupy.replace("/mturk/preview?groupId=","");
  1381. }
  1382. else
  1383. groupy = "";
  1384. //console.log($(this));
  1385. //console.log(groupy);
  1386. if (listy.indexOf(groupy) == -1)
  1387. listy.push(groupy);
  1388. else if (groupy == "")
  1389. listy.push(groupy);
  1390. });
  1391. //console.log(listy);
  1392. //listy = listy.filter(function(elem, pos) {
  1393. // return listy.indexOf(elem) == pos;
  1394. //});
  1395. //console.log(listy);
  1396. for (var j = 0; j < $requester.length; j++)
  1397. {
  1398. var $hits = $requester.eq(j).parent().parent().parent().parent().parent().parent().find('td[class="capsule_field_text"]');
  1399. var requester_name = $requester.eq(j).text().trim();
  1400. var requester_link = $requester.eq(j).attr('href');
  1401. if (!requester_link)
  1402. {
  1403. requester_link = "https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords="+ requester_name.replace(/ /g,"+");
  1404. requester_name += " (search)";
  1405. }
  1406. var group_ID=(listy[j] ? listy[j] : "");
  1407. var masters = false;
  1408. var title = $title.eq(j).text().trim();
  1409. var preview_link = "/mturk/preview?groupId=" + group_ID;
  1410. //console.log(listy[j]);
  1411. //console.log(title+" "+group_ID +" "+ listy[j]);
  1412. if (!group_ID || group_ID.length == 0){
  1413. preview_link = requester_link;
  1414. title += " (Preview link unavailable)";
  1415. }
  1416. var reward = $reward.eq(j).text().trim();
  1417. var hits = $hits.eq(4).text().trim();
  1418. if (hits.length == 0)
  1419. hits = "Scraper logged out";
  1420. var time = $times.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  1421. var description = $descriptions.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  1422. //console.log(description);
  1423. var requester_id = requester_link.replace('/mturk/searchbar?selectedSearchType=hitgroups&requesterId=','');
  1424. var accept_link;
  1425. accept_link = preview_link.replace('preview','previewandaccept');
  1426.  
  1427. /*HIT SCRAPER ADDITION*/
  1428. var qElements = $quals.eq(j).parent().parent().parent().find('tr');
  1429. //console.log(qElements);
  1430.  
  1431. var qualifications = [];
  1432. for (var i = 1; i < qElements.length; i++) {
  1433. qualifications.push((qElements[i].childNodes[1].textContent.trim().replace(/\s+/g, ' ').indexOf("Masters") != -1 ? "[color=red][b]"+qElements[i].childNodes[1].textContent.trim().replace(/\s+/g, ' ')+"[/b][/color]" : qElements[i].childNodes[1].textContent.trim().replace(/\s+/g, ' ')));
  1434. if (qElements[i].childNodes[1].textContent.trim().replace(/\s+/g, ' ').indexOf("Masters") != -1)
  1435. masters=true;
  1436. }
  1437. var qualList = (qualifications.join('; ') ? qualifications.join('; ') : "None");
  1438.  
  1439. var key = requester_name+title+reward+group_ID;
  1440. if (found_key_list.indexOf(key) == -1)
  1441. found_key_list.push(key);
  1442. else
  1443. {
  1444. console.log("DUPE: "+key);
  1445. continue;
  1446. }
  1447. if (scraper_history[key] === undefined)
  1448. {
  1449. scraper_history[key] = {requester:"", title:"", description:"", reward:"", hits:"", req_link:"", quals:"", prev_link:"", rid:"", acc_link:"", new_result:"", dinged:"", qualified:"", found_this_time:"", initial_time:"", reqdb:"",titledb:"",time:"",masters:false};
  1450. scraper_history[key].req_link = requester_link;
  1451. scraper_history[key].prev_link = preview_link;
  1452. scraper_history[key].requester = requester_name;
  1453. scraper_history[key].title = title;
  1454. scraper_history[key].reward = reward;
  1455. scraper_history[key].hits = hits;
  1456. scraper_history[key].rid = requester_id;
  1457. scraper_history[key].acc_link = accept_link;
  1458. scraper_history[key].time = time;
  1459. scraper_history[key].quals = qualList;
  1460. scraper_history[key].description = description.replace(/'/g, "&apos;").replace(/"/g, "&quot;");
  1461. scraper_history[key].masters = masters;
  1462. scraper_history[key].requester_strip = requester_name.replace(" (search)","");
  1463. HITStorage.indexedDB.checkRequester(requester_id,key);
  1464. HITStorage.indexedDB.checkTitle(title,key);
  1465. if (searched_once)
  1466. {
  1467. scraper_history[key].initial_time = new Date().getTime();//-1000*(save_new_results_time - SEARCH_REFRESH);
  1468. scraper_history[key].new_result = 0;
  1469. scraper_history[key].dinged = 0;
  1470. }
  1471. else
  1472. {
  1473. scraper_history[key].initial_time = new Date().getTime()-1000*save_new_results_time;
  1474. scraper_history[key].new_result = 1000*save_new_results_time;
  1475. scraper_history[key].dinged = 1;
  1476. }
  1477. if (not_qualified_group_IDs.indexOf(group_ID) !== -1)
  1478. scraper_history[key].qualified = false;
  1479. else
  1480. scraper_history[key].qualified = true;
  1481.  
  1482. scraper_history[key].found_this_time = true;
  1483. }
  1484. else
  1485. {
  1486. scraper_history[key].new_result = new Date().getTime() - scraper_history[key].initial_time;
  1487. scraper_history[key].found_this_time = true;
  1488. scraper_history[key].hits = hits;
  1489. }
  1490. }
  1491. }
  1492.  
  1493. function statusdetail_loop(next_URL)
  1494. {
  1495. if (global_run == true)
  1496. {
  1497. if (next_URL.length != 0)
  1498. {
  1499. $.get(next_URL, function(data)
  1500. {
  1501. var $src = $(data);
  1502. var maxpagerate = $src.find('td[class="error_title"]:contains("You have exceeded the maximum allowed page request rate for this website.")');
  1503. if (maxpagerate.length == 0)
  1504. {
  1505. if (next_page > PAGES_TO_SCRAPE)
  1506. {
  1507. if(status_array.indexOf("Correcting for skips") == -1 && type != 2){
  1508. status_array.push("Correcting for skips");
  1509. }
  1510. }
  1511. set_progress_report("Processing page " + next_page, false);
  1512. scrape($src);
  1513.  
  1514. var $next_URL = $src.find('a[href^="/mturk/viewsearchbar"]:contains("Next")');
  1515. next_URL = ($next_URL.length != 0) ? $next_URL.attr("href") : "";
  1516. next_page++;
  1517. if (document.getElementById('lnkWorkerSignin'))
  1518. maxPages--;
  1519. if (maxPages == 0)
  1520. {
  1521. maxPages = 20;
  1522. next_URL = "";
  1523. next_page = -1;
  1524. }
  1525.  
  1526. if (default_type == 1)
  1527. {
  1528. var hmin = MINIMUM_HITS+1;
  1529. for (var j = 0; j < found_key_list.length; j++)
  1530. {
  1531. //console.log(scraper_history[found_key_list[j]]);
  1532. if (scraper_history[found_key_list[j]].hits < hmin)
  1533. {
  1534. next_URL = "";
  1535. next_page = -1;
  1536. break;
  1537. }
  1538. }
  1539. }
  1540. else if (next_page > PAGES_TO_SCRAPE && correct_for_skips)
  1541. {
  1542. var skipped_hits = 0;
  1543. var added_pages = 0;
  1544. for (var k = 0; k < found_key_list.length; k++)
  1545. {
  1546. var obj = scraper_history[found_key_list[k]];
  1547. if (!ignore_check(obj.requester.replace(" (search)",""),obj.title))
  1548. skipped_hits++;
  1549. }
  1550. added_pages = Math.floor(skipped_hits/10);
  1551. if (skipped_hits%10 >6)
  1552. added_pages++;
  1553. if (next_page > PAGES_TO_SCRAPE + added_pages)
  1554. {
  1555. next_URL = "";
  1556. next_page = -1;
  1557. }
  1558.  
  1559. }
  1560. else if (next_page > PAGES_TO_SCRAPE)
  1561. {
  1562. next_URL = "";
  1563. next_page = -1;
  1564. }
  1565.  
  1566. setTimeout(function(){statusdetail_loop(next_URL);}, STATUSDETAIL_DELAY);
  1567. }
  1568. else
  1569. {
  1570. //console.log("MPRE");
  1571. setTimeout(function(){statusdetail_loop(next_URL);}, MPRE_DELAY);
  1572. }
  1573. }).fail(function() {
  1574. if (qual_input)
  1575. alert("Error when searching. Are you logged out?");
  1576. global_run = false;
  1577. big_red_button.textContent = "Start";
  1578. });
  1579. }
  1580. else
  1581. {
  1582. maxPages = 20;
  1583. searched_once = true;
  1584. var found_hits = found_key_list.length;
  1585. var shown_hits = 0;
  1586. var new_hits = 0;
  1587. var blocklist_hits = 0, belowThreshold_hits = 0;
  1588. dings = 0;
  1589. var ridString = "";
  1590. var rids = [];
  1591. var rnames = [];
  1592. var lastRow = text_area.rows.length - 1;
  1593. for (var i = lastRow; i>0; i--)
  1594. text_area.deleteRow(i);
  1595. for (var j = 0; j < found_key_list.length; j++)
  1596. {
  1597. //(function(url,rids,j) {
  1598. var obj = scraper_history[found_key_list[j]];
  1599. //console.dir(obj);
  1600. var ignored = ignore_check(obj.requester.replace(" (search)",""),obj.title);
  1601. var includelisted = include_check(obj.requester.replace(" (search)", ""),obj.title);
  1602. if (!(obj.masters == true && masters_hide.checked) && (ignored && obj.found_this_time) || (!ignored && obj.found_this_time)){
  1603. ++shown_hits;
  1604. //hit export will update col_heads[1]
  1605. var col_heads = [];
  1606. if(obj.req_link == obj.rid)
  1607. {
  1608. if (useTO)
  1609. col_heads = ["<a class=\"link"+sufTheme+"\" href='"+ obj.req_link +"' target='_blank'>" + obj.requester + "</a>","<a class=\"link"+sufTheme+"\" href='"+ obj.prev_link +"' target='_blank' title='"+ obj.description +"'>" + obj.title + "</a>",obj.reward,obj.hits,"TO down",(obj.title.indexOf("(Preview link unavailable)") > -1) ? "Scraper logged out" : "<a href='"+LINK_BASE+obj.acc_link+"' target='_blank'>Accept</a>","M"];
  1610. else
  1611. col_heads = ["<a class=\"link"+sufTheme+"\" href='"+ obj.req_link +"' target='_blank'>" + obj.requester + "</a>","<a class=\"link"+sufTheme+"\" href='"+ obj.prev_link +"' target='_blank' title='"+ obj.description +"'>" + obj.title + "</a>",obj.reward,obj.hits,"TO disabled",(obj.title.indexOf("(Preview link unavailable)") > -1) ? "Scraper logged out" : "<a href='"+LINK_BASE+obj.acc_link+"' target='_blank'>Accept</a>","M"];
  1612. }
  1613. else
  1614. {
  1615. if (useTO)
  1616. col_heads = ["<a class=\"link"+sufTheme+"\" href='"+ LINK_BASE+obj.req_link +"' target='_blank'>" + obj.requester + "</a>","<a class=\"link"+sufTheme+"\" href='"+ LINK_BASE+obj.prev_link +"' target='_blank' title='"+ obj.description +"'>" + obj.title + "</a>",obj.reward,obj.hits,"TO down","<a class=\"link"+sufTheme+"\" href='"+ LINK_BASE+obj.acc_link +"' target='_blank'>Accept</a>","M"];
  1617. else
  1618. col_heads = ["<a class=\"link"+sufTheme+"\" href='"+ LINK_BASE+obj.req_link +"' target='_blank'>" + obj.requester + "</a>","<a class=\"link"+sufTheme+"\" href='"+ LINK_BASE+obj.prev_link +"' target='_blank' title='"+ obj.description +"'>" + obj.title + "</a>",obj.reward,obj.hits,"TO disabled","<a class=\"link"+sufTheme+"\" href='"+ LINK_BASE+obj.acc_link +"' target='_blank'>Accept</a>","M"];
  1619. }
  1620. var row = text_area.insertRow(text_area.rows.length);
  1621. if (!ignored && obj.found_this_time){
  1622. if (useBlock.checked) row.setAttribute("class","scraperBlockedRow");
  1623. else row.className = "scraperBlockedRowOff";
  1624. blocklist_hits++;
  1625. obj.dinged = 1;
  1626. //console.log(blocklist_hits);
  1627. }
  1628. else if (!includelisted && obj.found_this_time){
  1629. if (highlightIncludes_input.checked) row.setAttribute("class","scraperIncludelistedRow");
  1630. else row.setAttribute("class", "scraperIncludelistedRowOff");
  1631. }
  1632. //url += obj.rid + ',';
  1633. //url2 += obj.rid + ',';
  1634. if (obj.rid == obj.req_link)
  1635. rnames.push(obj.requester_strip);
  1636. else{
  1637. ridString += obj.rid + ',';
  1638. }
  1639. rids.push(obj.rid);
  1640. if (check_hitDB)
  1641. {
  1642. col_heads.push("R");
  1643. col_heads.push("T");
  1644. }
  1645. if (!obj.qualified)
  1646. {
  1647. col_heads.push("Not Qualified");
  1648. }
  1649. for (i=0; i<col_heads.length; i++)
  1650. {
  1651. var this_cell = row.insertCell(i);
  1652. row.cells[i].style.fontSize = default_text_size+"px";
  1653. this_cell.innerHTML = col_heads[i];
  1654. if (i>1)
  1655. this_cell.style.textAlign = 'center';
  1656. if (i==6)
  1657. {
  1658. var listOfQuals = obj.quals.replace(/;/g,'\n');
  1659. listOfQuals = listOfQuals.replace(/\[(.*?)\]/g,'');
  1660. this_cell.title=listOfQuals;
  1661. if (obj.masters)
  1662. {
  1663. this_cell.className = "needmaster"+sufTheme;
  1664. this_cell.innerHTML="Y";
  1665. }
  1666. else
  1667. {
  1668. this_cell.className = "nomaster"+sufTheme;
  1669. this_cell.innerHTML="N";
  1670. }
  1671. }
  1672. if (check_hitDB)
  1673. {
  1674. if (i==7)
  1675. {
  1676. if (obj.reqdb){
  1677. this_cell.className = "yeshitDB"+sufTheme;
  1678. this_cell.title = "This requester name was found in your HitDB.";
  1679. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,0);}})(obj));
  1680. }
  1681. else {
  1682. this_cell.className = "nohitDB"+sufTheme;
  1683. this_cell.title = "This requester name was not found in your HitDB, or you don't have a HitDB.";
  1684. }
  1685. }
  1686. else if (i==8)
  1687. {
  1688. if (obj.titledb){
  1689. this_cell.className = "yeshitDB"+sufTheme;
  1690. this_cell.title = "This HIT title was found in your HitDB.";
  1691. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,1);}})(obj));
  1692. }
  1693. else {
  1694. this_cell.className = "nohitDB"+sufTheme;
  1695. this_cell.title = "This HIT title was not found in your HitDB, or you don't have a HitDB.";
  1696. }
  1697. }
  1698. else if (i==9)
  1699. this_cell.className = "tooweak"+sufTheme;
  1700. }
  1701. else if (i==7)
  1702. this_cell.className = "tooweak"+sufTheme;
  1703. }
  1704. if (Object.keys(scraper_history).length > 0)
  1705. {
  1706. if (obj.dinged == 0)
  1707. {
  1708. if (shouldInclude && !includelisted)
  1709. {
  1710. dings++;
  1711. obj.dinged = 2; //#hookdinged
  1712. }
  1713. else if (!shouldInclude)
  1714. {
  1715. dings++;
  1716. obj.dinged = 2;
  1717. }
  1718. }
  1719. if (obj.new_result < 1000*save_new_results_time)
  1720. {
  1721. if (ignored)
  1722. new_hits++;
  1723. for (var h = 0; h < col_heads.length; h++)
  1724. {
  1725. row.cells[h].style.fontSize = default_text_size + 1+"px";
  1726. row.cells[h].style.fontWeight = "bold";
  1727. }
  1728. }
  1729. }
  1730.  
  1731. var button1 = document.createElement('button'); // FORUM VBCODE EXPORT BUTTON
  1732. button1.textContent = 'vB';
  1733. button1.title = 'Export this HIT description as vBulletin formatted text';
  1734. button1.className = "vbButton";
  1735. button1.className += opt_exportvb.checked ? (" taButtons"+sufTheme) : (" taButtonsOff"+sufTheme);
  1736. button1.style.width = '30px';
  1737.  
  1738. var button2 = document.createElement('button'); //BUTTON TO BLOCK REQUESTER
  1739. button2.textContent = 'R';
  1740. button2.title = 'Add requester to block list';
  1741. button2.style.width = '15px';
  1742. button2.className += " taButtons"+sufTheme;
  1743.  
  1744. var button3 = document.createElement('button'); //BUTTON TO BLOCK TITLE
  1745. button3.textContent = 'T';
  1746. button3.title = 'Add title to block list';
  1747. button3.style.width = '15px';
  1748. button3.className += " taButtons"+sufTheme;
  1749.  
  1750. var button4 = document.createElement('button'); // IRC EXPORT BUTTON
  1751. button4.textContent = 'IRC';
  1752. button4.className = "ircButton";
  1753. button4.className += opt_exportirc.checked ? (" taButtons"+sufTheme) : (" taButtonsOff"+sufTheme);
  1754. button4.style.width = '30px';
  1755. button4.title = 'Click to save HIT information to your clipboard. Please wait while shortened URLs are retrieved.';
  1756.  
  1757. var button5 = document.createElement('button'); // REDDIT EXPORT BUTTON
  1758. button5.textContent = 'HWTF';
  1759. button5.style.width = '33px';
  1760. button5.className = "redditButton";
  1761. button5.className += opt_exportreddit.checked ? (" taButtons"+sufTheme) : (" taButtonsOff"+sufTheme);
  1762. button5.title = 'Export this HIT to r/HITsWorthTurkingFor title standards.'
  1763.  
  1764. button1.addEventListener("click", (function (obj,j) { return function() {export_sel_deleg(obj,j,"vb");}})(obj,j));
  1765. row.cells[1].appendChild(document.createTextNode(" "));
  1766. if (obj.rid != obj.req_link)
  1767. row.cells[1].appendChild(button1);
  1768. button4.addEventListener("click", (function (obj,j) { return function() {export_sel_deleg(obj,j,"irc");}})(obj,j));
  1769. row.cells[1].appendChild(document.createTextNode(" "));
  1770. row.cells[1].appendChild(button4);
  1771. button5.addEventListener("click", (function (obj,j) { return function() {export_sel_deleg(obj,j,"reddit");}})(obj,j));
  1772. row.cells[1].appendChild(document.createTextNode(" "));
  1773. row.cells[1].appendChild(button5);
  1774.  
  1775. button2.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,0);}})(obj,j));
  1776. row.cells[0].appendChild(document.createTextNode(" "));
  1777. row.cells[0].appendChild(button2);
  1778. button3.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,1);}})(obj,j));
  1779. row.cells[0].appendChild(button3);
  1780. }
  1781. //});
  1782.  
  1783. }
  1784. //url = url.substring(0,url.length - 1);
  1785. //console.log(url);
  1786. var success_flag = false;
  1787. var rdata = "";
  1788. if (useTO){
  1789. if (ridString != "")
  1790. rdata = getTOMulti(ridString);
  1791. else
  1792. rdata = "Scraper logged out";
  1793. }
  1794. else
  1795. rdata = "TO Down";
  1796. //var rdata = "TO Down";
  1797. //console.log(rdata);
  1798. if (rdata != "TO Down")
  1799. {
  1800. var globalMeans = { "pay":0, "paycount":0, "quality":0, "divisor":0, "reviews":0, "reviewcount":0 };
  1801.  
  1802. for (var r = 0; r < rids.length; r++) { // calculate global values across entire table
  1803. if (rdata[rids[r]]) {
  1804. if (rdata[rids[r]].attrs.pay > 0) {
  1805. globalMeans.quality += rdata[rids[r]].attrs.pay * settings.options.toWeighting.pay;
  1806. globalMeans.pay += rdata[rids[r]].attrs.pay * 1.0;
  1807. globalMeans.paycount++;
  1808. globalMeans.divisor += settings.options.toWeighting.pay;
  1809. }
  1810. if (rdata[rids[r]].attrs.comm.trim() > 0) {
  1811. globalMeans.quality += rdata[rids[r]].attrs.comm.trim()*settings.options.toWeighting.comm;
  1812. globalMeans.divisor += settings.options.toWeighting.comm;
  1813. }
  1814. if (rdata[rids[r]].attrs.fast.trim() > 0) {
  1815. globalMeans.quality += rdata[rids[r]].attrs.fast.trim()*settings.options.toWeighting.fast;
  1816. globalMeans.divisor += settings.options.toWeighting.fast;
  1817. }
  1818. if (rdata[rids[r]].attrs.fair.trim() > 0) {
  1819. globalMeans.quality += rdata[rids[r]].attrs.fair.trim()*settings.options.toWeighting.fair;
  1820. globalMeans.divisor += settings.options.toWeighting.fair;
  1821. }
  1822. if (rdata[rids[r]].reviews > 0) {
  1823. globalMeans.reviews += rdata[rids[r]].reviews*1.0;
  1824. globalMeans.reviewcount++;
  1825. }
  1826. }
  1827. }
  1828. globalMeans.quality = globalMeans.divisor > 0 ? globalMeans.quality/globalMeans.divisor : 0;
  1829. globalMeans.pay = globalMeans.paycount > 0 ? globalMeans.pay/globalMeans.paycount : 0;
  1830. globalMeans.reviews = globalMeans.reviewcount > 0 ? globalMeans.reviews/globalMeans.reviewcount : 0;
  1831.  
  1832. for (var r = 0; r < rids.length; r++)
  1833. {
  1834. if (rdata[rids[r]])
  1835. {
  1836. var pay = rdata[rids[r]].attrs.pay*1.0;
  1837. var reviews = rdata[rids[r]].reviews*1.0;
  1838. var quality = 0;
  1839. var sum = 0;
  1840. var divisor = 0;
  1841. var count = 0;
  1842. var wpayrank = 0;
  1843. var wqualityrank = 0;
  1844. //console.log(rdata[rids[r]]);
  1845. var comm = rdata[rids[r]].attrs.comm.trim()*1.0;
  1846. var fair = rdata[rids[r]].attrs.fair.trim()*1.0;
  1847. var fast = rdata[rids[r]].attrs.fast.trim()*1.0;
  1848. var tos = rdata[rids[r]].tos_flags;
  1849. if (comm > 0)
  1850. {
  1851. sum += settings.options.toWeighting.comm*comm;
  1852. divisor += settings.options.toWeighting.comm;
  1853. }
  1854. if (pay > 0)
  1855. {
  1856. sum += settings.options.toWeighting.pay*pay;
  1857. divisor += settings.options.toWeighting.pay;
  1858. }
  1859. if (fair > 0)
  1860. {
  1861. sum += settings.options.toWeighting.fair*fair;
  1862. divisor += settings.options.toWeighting.fair;
  1863. }
  1864. if (fast > 0)
  1865. {
  1866. sum += settings.options.toWeighting.fast*fast;
  1867. divisor += settings.options.toWeighting.fast;
  1868. }
  1869. if (divisor > 0)
  1870. {
  1871. quality = sum/divisor;
  1872. }
  1873. var apay = (reviews * pay + 15) / (reviews + 5);
  1874. //wpayrank = ((globalMeans.pay - (pay/globalMeans.paycount)) + reviews * pay) / globalMeans.reviewcount; // simplification
  1875. //wpayrank = (globalMeans.reviews * (globalMeans.pay - (pay/globalMeans.paycount)) + (reviews * pay)) / globalMeans.reviewcount; // unit derivation
  1876. wpayrank = apay - 1.645 * Math.sqrt((Math.pow(1.061 * apay,2) - Math.pow(apay,2)) / (reviews + 5)); // pnormal alpha = 0.10
  1877. //wpayrank = (10*globalMeans.pay + reviews*pay)/(globalMeans.reviewcount+10); // minimalist, no zs/conf - large fluctuations
  1878. var aqual = (quality * reviews + 15)/(reviews + 5);
  1879. //wqualityrank = (15*globalMeans.quality + reviews*quality)/(reviews+15);
  1880. wqualityrank = aqual - 1.645 * Math.sqrt((Math.pow(1.0693 * aqual,2) - Math.pow(aqual,2)) / (reviews + 5)); // pnormal, alpha = 0.10
  1881. var titleText = "";
  1882. titleText += "comm: "+comm+"\nfair: "+fair+"\nfast: "+fast+"\npay: "+pay+"\nReviews: "+reviews+"\nTOS violations: "+tos;
  1883. titleText += "\n\nAdjusted pay rating: " + wpayrank.toPrecision(4) + "\nGlobal pay mean: " + globalMeans.pay.toPrecision(4);
  1884. titleText += "\nWeighted quality: " + quality.toPrecision(4) + "\nAdjusted quality rating: " + wqualityrank.toPrecision(4) +
  1885. "\nGlobal quality mean: " + globalMeans.quality.toPrecision(4);
  1886. text_area.rows[r+1].cells[4].innerHTML = "<a class=\"tolink"+sufTheme+"\" href='"+ TO_REQ_URL+rids[r] +"' target='_blank'>" + pay.toFixed(2) + "</a>";
  1887. text_area.rows[r+1].cells[4].title += titleText;
  1888. text_area.rows[r+1].cells[4].dataset.simPay = pay;
  1889. text_area.rows[r+1].cells[4].dataset.adjPay = wpayrank.toPrecision(4);
  1890. text_area.rows[r+1].cells[4].dataset.simQual = quality.toPrecision(4);
  1891. text_area.rows[r+1].cells[4].dataset.adjQual = wqualityrank.toPrecision(4);
  1892. if (pay < MINIMUM_TO && text_area.rows[r+1].className.match(/(scraperBlocked|toBlocked)/) == null) {
  1893. text_area.rows[r+1].className += (showButton.textContent == "Hide ignored hits") ? " toBlockedRowOff" : " toBlockedRow";
  1894. belowThreshold_hits++;
  1895. for (var m=0; m<found_key_list.length; m++) {
  1896. if (dings > 0 && scraper_history[found_key_list[m]].rid == rids[r] && scraper_history[found_key_list[m]].dinged == 2) {
  1897. //console.log("[threshold] "+scraper_history[found_key_list[m]].requester+"; initial dings: "+dings);
  1898. dings--;
  1899. scraper_history[found_key_list[m]].dinged = 1;
  1900. //console.log("dings left: "+dings);
  1901. }
  1902. }
  1903. }
  1904. if (reviews > 4)
  1905. {
  1906. var copt = saveState.getItem("colorType") || "sim";
  1907. if (copt == "sim") {
  1908. if (quality > 4)
  1909. text_area.rows[r+1].className += " toHigh"+sufTheme;
  1910. else if (quality > 3)
  1911. text_area.rows[r+1].className += " toGood"+sufTheme;
  1912. else if (quality > 2)
  1913. text_area.rows[r+1].className += " toAverage"+sufTheme;
  1914. else if (quality > 0)
  1915. text_area.rows[r+1].className += " toPoor"+sufTheme;
  1916. }
  1917. else {
  1918. if (wqualityrank > 4.05)
  1919. text_area.rows[r+1].className += " toHigh"+sufTheme;
  1920. else if (wqualityrank > 3.06)
  1921. text_area.rows[r+1].className += " toGood"+sufTheme;
  1922. else if (wqualityrank > 2.4)
  1923. text_area.rows[r+1].className += " toAverage"+sufTheme;
  1924. else if (wqualityrank > 1.7)
  1925. text_area.rows[r+1].className += " toLow"+sufTheme;
  1926. else if (wqualityrank > 0)
  1927. text_area.rows[r+1].className += " toPoor"+sufTheme;
  1928. }
  1929. }
  1930. else if (text_area.rows[r+1].className.match(/toNone/) == null) text_area.rows[r+1].className += " toNone"+sufTheme;
  1931. }
  1932. else if(rdata == "Scraper logged out")
  1933. {
  1934. text_area.rows[r+1].cells[4].innerHTML = "<a class=\"link"+sufTheme+"\" href='https://turkopticon.ucsd.edu/main/php_search?field=name&query="+rnames[r].replace(/ /g,'+') +"' target='_blank'>Search TO</a>";
  1935. text_area.rows[r+1].cells[4].setAttribute("title", "No Data");
  1936. text_area.rows[r+1].className += " toNone"+sufTheme;
  1937. if (block_no_to && text_area.rows[r+1].className.match(/(scraperBlocked|toBlocked)/) == null) {
  1938. text_area.rows[r+1].className += (showButton.textContent == "Hide ignored hits") ? " toBlockedRowOff" : " toBlockedRow";
  1939. belowThreshold_hits++;
  1940. for (var m=0; m<found_key_list.length; m++) {
  1941. if (dings > 0 && scraper_history[found_key_list[m]].rid == rids[r] && scraper_history[found_key_list[m]].dinged == 2) {
  1942. //console.log("[nodata]"+scraper_history[found_key_list[m]].requester+"; initial dings: "+dings);
  1943. dings--;
  1944. scraper_history[found_key_list[m]].dinged = 1;
  1945. //console.log("dings left: "+dings);
  1946. }
  1947. }
  1948. }
  1949. }
  1950. else
  1951. {
  1952. text_area.rows[r+1].cells[4].innerHTML = "<a class=\"link"+sufTheme+"\" href='"+ TO_REQ_URL+rids[r] +"' target='_blank'>No Data</a>";
  1953. text_area.rows[r+1].cells[4].setAttribute("title", "No Data");
  1954. text_area.rows[r+1].className += " toNone"+sufTheme;
  1955. if (block_no_to && text_area.rows[r+1].className.match(/(scraperBlocked|toBlocked)/) == null) {
  1956. text_area.rows[r+1].className += (showButton.textContent == "Hide ignored hits") ? " toBlockedRowOff" : " toBlockedRow";
  1957. belowThreshold_hits++;
  1958. for (var m=0; m<found_key_list.length; m++) {
  1959. if (dings > 0 && scraper_history[found_key_list[m]].rid == rids[r] && scraper_history[found_key_list[m]].dinged == 2) {
  1960. //console.log("[nodata]"+scraper_history[found_key_list[m]].requester+"; initial dings: "+dings);
  1961. dings--;
  1962. scraper_history[found_key_list[m]].dinged = 1;
  1963. //console.log("dings left: "+dings);
  1964. }
  1965. }
  1966. }
  1967. }
  1968. }
  1969. if (sort_TO) table_sort("pay");
  1970. if (sort_TO2) table_sort("qual");
  1971.  
  1972. success_flag = true;
  1973. }
  1974. var pStr = "Scrape complete. " + shown_hits + " HITs found (" + new_hits + " new results). ";
  1975. if (!useBlock.checked) {
  1976. if (blocklist_hits > 0) pStr += blocklist_hits + " HITs shown from blocklist.";
  1977. if (belowThreshold_hits > 0) pStr += " " + belowThreshold_hits + " HITs below TO threshold.";
  1978. set_progress_report(pStr, false);
  1979. }
  1980. else {
  1981. if (blocklist_hits > 0 || belowThreshold_hits > 0) pStr += (belowThreshold_hits+blocklist_hits) + " HITs ignored: ";
  1982. if (blocklist_hits > 0) pStr += blocklist_hits + " from blocklist ";
  1983. if (belowThreshold_hits > 0) pStr += belowThreshold_hits + " below TO threshold.";
  1984. set_progress_report(pStr, false);
  1985. }
  1986. if (!success_flag)
  1987. for (var s = 0; s < rids.length; s++)
  1988. if (text_area.rows[s+1].className.match(/toNone/) === null) text_area.rows[s+1].className += " toNone"+sufTheme;
  1989.  
  1990. statusdetail_loop_finished = true;
  1991. if (dings > 0){
  1992. newHits(shouldDing);
  1993. }
  1994. if (SEARCH_REFRESH>0)
  1995. {
  1996. wait_loop = setTimeout(function(){if (global_run) start_it();}, 1000*SEARCH_REFRESH);
  1997. display_wait_time(SEARCH_REFRESH);
  1998. }
  1999. else
  2000. {
  2001. global_run = false;
  2002. big_red_button.textContent = "Start";
  2003. }
  2004. }
  2005. }
  2006. }
  2007.  
  2008. function table_sort(stype) {
  2009. //--- Get the table we want to sort.
  2010. var jTableToSort = $("#body_table");
  2011.  
  2012. //--- Get the rows to sort, but skip the first row, since it contains column titles.
  2013. var jRowsToSort = jTableToSort.find ("tr:gt(0)");
  2014.  
  2015. //--- Sort the rows in place.
  2016. jRowsToSort.sort(function(a,b) {
  2017. var opt = saveState.getItem("sortType") || "adj";
  2018. var aVal = Number( $(a).find("td:eq(4)").attr("data-"+opt+"-"+stype) );
  2019. var bVal = Number( $(b).find("td:eq(4)").attr("data-"+opt+"-"+stype) );
  2020. //--- no TO goes to the bottom
  2021. if (isNaN(aVal)) aVal = 0;
  2022. if (isNaN(bVal)) bVal = 0;
  2023. //--- sort
  2024. if (sort_asc.checked) return aVal - bVal;
  2025. else return bVal - aVal; // descending order
  2026. }).appendTo(jTableToSort);
  2027. }
  2028.  
  2029. //--- check block list for requester name and HIT title
  2030. function ignore_check(r,t){
  2031. var tempList = ignore_list.map(function(item) { return item.toLowerCase().replace(/\s+/g," "); });
  2032. var foundR = -1;
  2033. var foundT = -1;
  2034. if (!settings.options.blocklistWildcards) { // wildcard matching disabled
  2035. foundR = tempList.indexOf(r.toLowerCase().replace(/\s+/g," "));
  2036. foundT = tempList.indexOf(t.toLowerCase().replace(/\s+/g," "));
  2037. }
  2038. else {
  2039. var blockWilds = [], blockExact = [];
  2040. blockExact = tempList.filter(function(item) { // separate wildcard glob patterns from regular exact patterns
  2041. if (item.search(".*?[*].*")) return true; else if (item.length > 1) {blockWilds.push(item); return false;}
  2042. });
  2043. // run default matching first
  2044. foundR = blockExact.indexOf(r.toLowerCase().replace(/\s+/g," "));
  2045. foundT = blockExact.indexOf(t.toLowerCase().replace(/\s+/g," "));
  2046. // if no match, try globs
  2047. if (foundR == -1 && foundT == -1) {
  2048. for (var i=0; i<blockWilds.length; i++) {
  2049. blockWilds[i] = blockWilds[i].replace(/([+${}[\](\)^|?.])/g, "\\$1");
  2050. blockWilds[i] = "^".concat(blockWilds[i].replace(/([^*]|^)[*](?!\*)/g, "$1.*").replace(/\*{2,}/g, function(s) { return s.replace(/\*/g, "\\*") })).concat("$");
  2051. foundR = r.toLowerCase().replace(/\s+/g," ").search(blockWilds[i]);
  2052. foundT = t.toLowerCase().replace(/\s+/g," ").search(blockWilds[i]);
  2053. if (foundR != -1 || foundT != -1)
  2054. break;
  2055. }
  2056. }
  2057. }
  2058. if (shouldInclude){ // if in Use Includelist mode, treat all HITs not matching the includelist as if they were on the blocklist
  2059. //console.log(include_list);
  2060. var temp = include_list.map(function(item) { return item.toLowerCase().replace(/\s+/g," "); }).indexOf(r.toLowerCase().replace(/\s+/g," "));
  2061. console.log(temp);
  2062. if (temp != -1)
  2063. foundR = -1;
  2064. else
  2065. foundR = 0;
  2066. }
  2067. var found = foundR == -1 && foundT == -1;
  2068. //console.log("r: "+r+" t: "+t+" f: "+found);
  2069. return found; // returns false (making !(include_check(x,y)) true) if HIT should be blocked, returns true if it shouldn't be blocked
  2070. }
  2071.  
  2072. // check include list for requester name and HIT title
  2073. function include_check(r,t){
  2074. var tempList = include_list.map(function(item) { return item.toLowerCase().replace(/\s+/g," "); });
  2075. var foundR = -1;
  2076. var foundT = -1;
  2077. foundR = tempList.indexOf(r.toLowerCase().replace(/\s+/g," "));
  2078. foundT = tempList.indexOf(t.toLowerCase().replace(/\s+/g," "));
  2079. var found = foundR == -1 && foundT == -1;
  2080. return found; // returns false (making !(include_check(x,y)) true) if HIT should be highlighted, returns true if it shouldn't be highlighted
  2081. }
  2082.  
  2083. function start_running()
  2084. {
  2085. if (big_red_button.textContent == "Start")
  2086. {
  2087. status_array=[];
  2088. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  2089. if (GM_getValue("scraper_include_list"))
  2090. include_list = GM_getValue("scraper_include_list").split('^');
  2091. global_run = true;
  2092. initial_url = URL_BASE;
  2093.  
  2094. if (search_input.value.length>0)
  2095. {
  2096. initial_url = initial_url.replace("searchWords=", "searchWords=" + search_input.value);
  2097. }
  2098.  
  2099. if (time_input.value.replace(/[^0-9]+/g,"") != "")
  2100. {
  2101. SEARCH_REFRESH = Number(time_input.value);
  2102. }
  2103.  
  2104. if (page_input.value.replace(/[^0-9]+/g,"") != "")
  2105. {
  2106. PAGES_TO_SCRAPE = Number(page_input.value);
  2107. if (PAGES_TO_SCRAPE > 20 && document.getElementById('lnkWorkerSignin')){
  2108. status_array.push("Search limited to 20 pages when logged out");
  2109. PAGES_TO_SCRAPE = 20;
  2110. }
  2111. }
  2112.  
  2113. if (new_time_display_input.value.replace(/[^0-9]+/g,"") != "")
  2114. {
  2115. save_new_results_time = Number(new_time_display_input.value);
  2116. }
  2117.  
  2118. if (reward_input.value.replace(/[^0-9]+/g,"") != "")
  2119. {
  2120. initial_url += "&minReward=" + reward_input.value;
  2121. }
  2122. else
  2123. {
  2124. initial_url += "&minReward=0.00";
  2125. }
  2126.  
  2127. if (qual_input.checked)
  2128. {
  2129. if (document.getElementById('lnkWorkerSignin')){
  2130. status_array.push("Logged out, ignoring qualified");
  2131. initial_url += "&qualifiedFor=off";
  2132. }
  2133. else
  2134. initial_url += "&qualifiedFor=on";
  2135. }
  2136. else
  2137. {
  2138. initial_url += "&qualifiedFor=off";
  2139. }
  2140.  
  2141. if (masters_input.checked)
  2142. {
  2143. initial_url += "&requiresMasterQual=on";
  2144. }
  2145.  
  2146. if (masters_hide.checked)
  2147. {
  2148. status_array.push("Masters hits hidden");
  2149. }
  2150.  
  2151. switch (sort_input[sort_input.selectedIndex].value)
  2152. {
  2153. case "late":
  2154. initial_url+= "&sortType=LastUpdatedTime%3A";
  2155. type=1;
  2156. default_type = 0;
  2157. break;
  2158. case "most":
  2159. initial_url+= "&sortType=NumHITs%3A";
  2160. default_type = 1;
  2161. type=2;
  2162. status_array.push("Sorting by NumHITs ignores Correct For Skips in favor of minimum batch size");
  2163. break;
  2164. case "amount":
  2165. initial_url+= "&sortType=Reward%3A";
  2166. type=3;
  2167. default_type = 0;
  2168. break;
  2169. case "alpha":
  2170. type=4;
  2171. initial_url += "&sortType=Title%3A";
  2172. break;
  2173. default:
  2174. alert("I don't know how you did it, but you broke it. Good job.");
  2175. }
  2176.  
  2177. if (min_input.value.replace(/[^0-9]+/g,"") != "")
  2178. {
  2179. if (type != 2)
  2180. status_array.push("Minimum Batch Size requires sorting by Most Available");
  2181. MINIMUM_HITS = Number(min_input.value);
  2182. }
  2183.  
  2184. if (sort_input_invert.checked)
  2185. {
  2186. if (sort_input[sort_input.selectedIndex].value == "alpha")
  2187. initial_url += "1";
  2188. else
  2189. initial_url += "0";
  2190. }
  2191. else
  2192. {
  2193. if (sort_input[sort_input.selectedIndex].value == "alpha")
  2194. initial_url += "0";
  2195. else
  2196. initial_url += "1";
  2197. }
  2198.  
  2199. if (sort_to.checked)
  2200. {
  2201. if (document.getElementById('lnkWorkerSignin')){
  2202. logged_array.push("Logged out, ignoring TO");
  2203. sort_TO = false;
  2204. sort_TO2 = false;
  2205. }
  2206. else
  2207. {
  2208. sort_TO = true;
  2209. sort_TO2 = false;
  2210. status_array.push("Sorting by TO pay is still in testing");
  2211. }
  2212. }
  2213. else if (sort_to2.checked)
  2214. {
  2215. if (document.getElementById('lnkWorkerSignin')){
  2216. status_array.push("Logged out, ignoring TO");
  2217. sort_TO = false;
  2218. sort_TO2 = false;
  2219. }
  2220. else
  2221. {
  2222. sort_TO2 = true;
  2223. sort_TO= false;
  2224. status_array.push("Sorting by TO quality is still in testing");
  2225. }
  2226. }
  2227. else
  2228. {
  2229. sort_TO = false;
  2230. sort_TO2 = false;
  2231. }
  2232.  
  2233. if (useTO_input.checked)
  2234. {
  2235. useTO = false;
  2236. }
  2237. else
  2238. {
  2239. useTO = true;
  2240. }
  2241.  
  2242. if (min_to.value.replace(/[^0-9]+/g,"") != "")
  2243. {
  2244. MINIMUM_TO = Number(min_to.value);
  2245. $("#show_TO_stuff_button").show();
  2246. }
  2247. else
  2248. {
  2249. MINIMUM_TO = -1;
  2250. }
  2251.  
  2252. if (friesAreDone.checked)
  2253. {
  2254. shouldDing = true;
  2255. }
  2256. else
  2257. {
  2258. shouldDing = false;
  2259. }
  2260.  
  2261. if (no_to_block.checked)
  2262. {
  2263. block_no_to = true;
  2264. $("#show_TO_stuff_button").show();
  2265. }
  2266. else
  2267. {
  2268. block_no_to = false;
  2269. }
  2270.  
  2271. if (!block_no_to && MINIMUM_TO == -1)
  2272. $("#show_TO_stuff_button").hide();
  2273.  
  2274. if (correctForSkips.checked)
  2275. {
  2276. if (matchOnly.checked)
  2277. {
  2278. status_array.push("Use Includelist (match only) checked, so ignoring Correct For Skips to prevent issues");
  2279. correct_for_skips = false;
  2280. }
  2281. else{
  2282. correct_for_skips = true;
  2283. }
  2284. }
  2285. else
  2286. {
  2287. correct_for_skips = false;
  2288. }
  2289.  
  2290. if (useBlock.checked)
  2291. {
  2292. useBlocklist = true;
  2293. }
  2294. else
  2295. {
  2296. useBlocklist = false;
  2297. }
  2298.  
  2299. if (matchOnly.checked)
  2300. {
  2301. if (include_list.length == 0){
  2302. status_array.push("No items in includelist, so ignoring Use Includelist checkbox");
  2303. shouldInclude = false;
  2304. }
  2305. else
  2306. {
  2307. shouldInclude = true;
  2308. if (!useBlock.checked)
  2309. {
  2310. status_array.push("Use Includelist checked, so forcing Use Blocklist for intended output behavior");
  2311. useBlocklist = true;
  2312. }
  2313. }
  2314. }
  2315. else
  2316. {
  2317. shouldInclude = false;
  2318. }
  2319.  
  2320. initial_url += "&pageNumber=1&searchSpec=HITGroupSearch";
  2321. audio_index = audio_option[audio_option.selectedIndex].value;
  2322. start_it();
  2323. }
  2324. else
  2325. {
  2326. global_run = false;
  2327. clearTimeout(wait_loop);
  2328. big_red_button.textContent = "Start";
  2329. set_progress_report("Stopped", true);
  2330. }
  2331. }
  2332.  
  2333. function start_it()
  2334. {
  2335. statusdetail_loop_finished = false;
  2336. big_red_button.textContent = "Stop";
  2337. found_key_list = [];
  2338. var ctime = new Date().getTime();
  2339. if (ctime - last_clear_time > save_results_time*666)
  2340. {
  2341. var last_history = scraper_history;
  2342. scraper_history = {};
  2343. for (var key in last_history)
  2344. {
  2345. if (last_history[key].new_result<save_results_time*1000)
  2346. {
  2347. scraper_history[key] = last_history[key];
  2348. if (last_history[key].found_this_time)
  2349. {
  2350. last_history[key].found_this_time = false;
  2351. if (last_history[key].new_result>save_new_results_time*1000)
  2352. last_history[key].initial_time = ctime-1000*save_new_results_time;
  2353. }
  2354. }
  2355.  
  2356. }
  2357. last_clear_time = ctime;
  2358. }
  2359. next_page = 1;
  2360. statusdetail_loop(initial_url);
  2361. }
  2362.  
  2363. function themeSwitchAux(th, force) {
  2364. var carr = $("[class*='"+sufTheme+"']");
  2365. for (var i=0; i<carr.length; i++) {
  2366. carr[i].className = carr[i].className.replace(/_[a-zA-Z]+/g, th);
  2367. }
  2368. sufTheme = th;
  2369. saveState.setItem("theme",sufTheme);
  2370. }
  2371. function UIAux(span, checkbox, statename) {
  2372. if (checkbox.checked) {
  2373. checkbox.checked = false;
  2374. span.className = span.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2375. } else {
  2376. checkbox.checked = true;
  2377. span.className += " cpSpansOn"+sufTheme;
  2378. }
  2379. saveState.setItem(statename, checkbox.checked);
  2380. saveState.save();
  2381. }
  2382. function colorSet(arr, loc, value, key) {
  2383. var cc = {}, kc = "";
  2384. if (sufTheme === "_thWisp") { cc = uwcolors; kc = "customwisp"; }
  2385. else if (sufTheme === "_thSDark") { cc = usdcolors; kc = "customsd"; }
  2386. else if (sufTheme === "_thSLight") { cc = uslcolors; kc = "customsl"; }
  2387. else if (sufTheme === "_thCustom") { cc = ucucolors; kc = "customcustom"; }
  2388. else if (sufTheme === "_thClassic") { cc = uclcolors; kc = "customclassic"; }
  2389. cc[key] = '#'+value.toString();
  2390. for (var j=0; j<arr.length; j++) {
  2391. if (loc == "bg") arr[j].style.background = cc[key];
  2392. else if (loc == "fg") arr[j].style.color = cc[key];
  2393. }
  2394. saveState.setItem(kc, cc);
  2395. saveState.save();
  2396. }
  2397. function theme_editor() { //testinput.className = "color {hash:true, pickerFaceColor:'transparent',pickerFace:0,pickerBorder:0,pickerInsetColor:'black'}";
  2398. var col = {};
  2399. if (sufTheme === "_thWisp") col = uwcolors;
  2400. else if (sufTheme === "_thSDark") col = usdcolors;
  2401. else if (sufTheme === "_thSLight") col = uslcolors;
  2402. else if (sufTheme === "_thCustom") col = ucucolors;
  2403. else if (sufTheme === "_thClassic") col = uclcolors;
  2404. else if (sufTheme === "_thRandom") col = rcolors;
  2405. for (var i=0; i<thin.length; i++) {
  2406. var rebind = [];
  2407. switch (i) {
  2408. case 0: new jscolor.color(thin[i], {}).fromString(col.highlight);
  2409. thin[i].title = "HIGHLIGHT: distinguishes between active and inactive states in the control panel";
  2410. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2411. function() { colorSet(document.querySelectorAll(".cpSpansOn"+sufTheme), 'bg', this, 'highlight'); }});
  2412. col.highlight = thin[i].value;
  2413. break;
  2414. case 1: new jscolor.color(thin[i], {}).fromString(col.background);
  2415. thin[i].title = "BACKGROUND: background color";
  2416. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2417. function() { colorSet(document.querySelectorAll(".body"+sufTheme+", .cpInput"+sufTheme), 'bg', this, 'background'); }});
  2418. break;
  2419. case 2: new jscolor.color(thin[i], {}).fromString(col.cpBackground);
  2420. thin[i].title = "CP-BACKGROUND: background color of the control panel";
  2421. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2422. function() { colorSet(document.querySelectorAll(".cpSpans"+sufTheme+", .controlpanel"+sufTheme), 'bg', this, 'cpBackground'); }});
  2423. break;
  2424. case 3: new jscolor.color(thin[i], {}).fromString(col.accent);
  2425. thin[i].title = "ACCENT: color of spacer text and control panel buttons";
  2426. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2427. function() { colorSet(document.querySelectorAll(".spacer"+sufTheme+", .cpButtons"+sufTheme), 'fg', this, 'accent'); }});
  2428. break;
  2429. case 4: new jscolor.color(thin[i], {}).fromString(col.bodytable);
  2430. thin[i].title = "BODYTABLE: default text color of results table";
  2431. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2432. function() { colorSet(document.querySelectorAll(".bodytable"+sufTheme), 'fg', this, 'bodytable'); }});
  2433. break;
  2434. case 5: new jscolor.color(thin[i], {}).fromString(col.highTO);
  2435. thin[i].title = "HIGH-TO: color for results with high TO";
  2436. if (sufTheme === "_thClassic")
  2437. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2438. function() { colorSet(document.querySelectorAll(".toHigh"+sufTheme), 'bg', this, 'highTO'); }});
  2439. else
  2440. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2441. function() { colorSet(document.querySelectorAll(".toHigh"+sufTheme+" .link"+sufTheme+", .toHigh"+sufTheme+" .tolink"+sufTheme), 'fg', this, 'highTO'); }});
  2442. break;
  2443. case 6: new jscolor.color(thin[i], {}).fromString(col.goodTO);
  2444. thin[i].title = "GOOD-TO: color for results with good TO";
  2445. if (sufTheme === "_thClassic")
  2446. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2447. function() { colorSet(document.querySelectorAll(".toGood"+sufTheme), 'bg', this, 'goodTO'); }});
  2448. else
  2449. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2450. function() { colorSet(document.querySelectorAll(".toGood"+sufTheme+" .link"+sufTheme+", .toGood"+sufTheme+" .tolink"+sufTheme), 'fg', this, 'goodTO'); }});
  2451. break;
  2452. case 7: new jscolor.color(thin[i], {}).fromString(col.averageTO);
  2453. thin[i].title = "AVERAGE-TO: color for results with mediocre TO";
  2454. if (sufTheme === "_thClassic")
  2455. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2456. function() { colorSet(document.querySelectorAll(".toAverage"+sufTheme), 'bg', this, 'toAverage'); }});
  2457. else
  2458. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2459. function() { colorSet(document.querySelectorAll(".toAverage"+sufTheme+" .link"+sufTheme+", .toAverage"+sufTheme+" .tolink"+sufTheme), 'fg', this, 'toAverage'); }});
  2460. break;
  2461. case 8: new jscolor.color(thin[i], {}).fromString(col.lowTO);
  2462. thin[i].title = "LOW-TO: color for results with low TO";
  2463. if (sufTheme === "_thClassic")
  2464. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2465. function() { colorSet(document.querySelectorAll(".toLow"+sufTheme), 'bg', this, 'lowTO'); }});
  2466. else
  2467. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2468. function() { colorSet(document.querySelectorAll(".toLow"+sufTheme+" .link"+sufTheme+", .toLow"+sufTheme+" .tolink"+sufTheme), 'fg', this, 'lowTO'); }});
  2469. break;
  2470. case 9: new jscolor.color(thin[i], {}).fromString(col.poorTO);
  2471. thin[i].title = "POOR-TO: color for results with terrible TO";
  2472. if (sufTheme === "_thClassic")
  2473. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2474. function() { colorSet(document.querySelectorAll(".toPoor"+sufTheme), 'bg', this, 'poorTO'); }});
  2475. else
  2476. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2477. function() { colorSet(document.querySelectorAll(".toPoor"+sufTheme+" .link"+sufTheme+", .toPoor"+sufTheme+" .tolink"+sufTheme), 'fg', this, 'poorTO'); }});
  2478. break;
  2479. case 10: new jscolor.color(thin[i], {}).fromString(col.noTO);
  2480. thin[i].title = "NO-TO: color for results with no TO";
  2481. if (sufTheme === "_thClassic")
  2482. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2483. function() { colorSet(document.querySelectorAll(".toNone"+sufTheme), 'bg', this, 'noTO'); }});
  2484. else
  2485. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2486. function() { colorSet(document.querySelectorAll(".toNone"+sufTheme+" .link"+sufTheme+", .toNone"+sufTheme+" .tolink"+sufTheme), 'fg', this, 'noTO'); }});
  2487. break;
  2488. case 11: new jscolor.color(thin[i], {}).fromString(col.hitDB);
  2489. thin[i].title = "HITDB: color for results found in your hitDB";
  2490. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2491. function() { colorSet(document.querySelectorAll(".yeshitDB"+sufTheme), 'bg', this, 'hitDB') }});
  2492. break;
  2493. case 12: new jscolor.color(thin[i], {}).fromString(col.nohitDB);
  2494. thin[i].title = "NOHITDB: color for results not found in your hitDB";
  2495. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2496. function() { colorSet(document.querySelectorAll(".nohitDB"+sufTheme), 'bg', this, 'nohitDB'); }});
  2497. break;
  2498. case 13: new jscolor.color(thin[i], {}).fromString(col.unqualified);
  2499. thin[i].title = "UNQUALIFIED: color for results for which you are unqualified";
  2500. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2501. function() { colorSet(document.querySelectorAll(".tooweak"+sufTheme), 'bg', this, 'unqualified'); }});
  2502. break;
  2503. case 14: new jscolor.color(thin[i], {}).fromString(col.reqmaster);
  2504. thin[i].title = "REQMASTER: color for results which require masters";
  2505. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2506. function() { colorSet(document.querySelectorAll(".needmaster"+sufTheme), 'bg', this, 'reqmaster'); }});
  2507. break;
  2508. case 15: new jscolor.color(thin[i], {}).fromString(col.nomaster);
  2509. thin[i].title = "NOMASTER: color for results which do not require masters";
  2510. mat = document.querySelectorAll(".nomaster"+sufTheme);
  2511. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2512. function() { colorSet(document.querySelectorAll(".nomaster"+sufTheme), 'bg', this, 'nomaster'); }});
  2513. break;
  2514. case 16: new jscolor.color(thin[i], {}).fromString(col.defaultText);
  2515. thin[i].title = "DEFAULT-TEXT: default text color";
  2516. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2517. function() { colorSet(document.querySelectorAll(".controlpanel"+sufTheme+", .cpspans"+sufTheme+", .mainlink"+sufTheme+", .tabhead"+sufTheme), 'fg', this, 'defaultText'); }});
  2518. break;
  2519. case 17: new jscolor.color(thin[i], {}).fromString(col.inputText);
  2520. thin[i].title = "INPUT-TEXT: color for the input boxes";
  2521. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2522. function() { colorSet(document.querySelectorAll(".cpInput"+sufTheme+", .cpSortdiv"+sufTheme), 'fg', this, 'inputText'); }});
  2523. break;
  2524. case 18: new jscolor.color(thin[i], {}).fromString(col.secondText);
  2525. thin[i].title = "SECOND-TEXT: color of secondary text (status bar, highlighted control panel buttons)";
  2526. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2527. function() { colorSet(document.querySelectorAll(".cpSpansOn"+sufTheme+", .statusdiv"+sufTheme), 'fg', this, 'secondText'); }});
  2528. break;
  2529. case 19: new jscolor.color(thin[i], {}).fromString(col.link);
  2530. thin[i].title = "LINK: default link color";
  2531. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2532. function() { colorSet(document.querySelectorAll(".link"+sufTheme+":link"), 'fg', this, 'link'); }});
  2533. break;
  2534. case 20: new jscolor.color(thin[i], {}).fromString(col.vlink);
  2535. thin[i].title = "VLINK: default visited link color";
  2536. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2537. function() { colorSet(document.querySelectorAll(".link"+sufTheme+":visited"), 'fg', this, 'vlink'); }});
  2538. break;
  2539. case 21: new jscolor.color(thin[i], {}).fromString(col.export);
  2540. thin[i].title = "EXPORT: color of buttons within the results table";
  2541. mat = document.querySelectorAll(".toButtons"+sufTheme);
  2542. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2543. function() { colorSet(document.querySelectorAll(".toButtons"+sufTheme), 'fg', this, 'export'); }});
  2544. break;
  2545. case 22: new jscolor.color(thin[i], {}).fromString(col.hover);
  2546. thin[i].title = "HOVER: color of control panel options on mouseover";
  2547. rebind[i] = new jscolor.color(thin[i], {"hash":true,"pickerBorder":0,"pickerFace":2,"onImmediateChange":
  2548. function() { colorSet(document.querySelectorAll(".cpSpans"+sufTheme+":hover"), 'bg', this, 'hover'); }});
  2549. break;
  2550. }
  2551. }
  2552. }
  2553.  
  2554. function show_interface()
  2555. {
  2556. //--- dynamic theme switching
  2557. //theme_editor();
  2558. var themeMenu = document.createElement("UL");
  2559. var thC2 = document.createElement("UL");
  2560. var thT1 = document.createElement("LI");
  2561. var thT2 = document.createElement("LI");
  2562. var thT3 = document.createElement("LI");
  2563. var thT4 = document.createElement("LI");
  2564. var thT5 = document.createElement("LI");
  2565. var thT6 = document.createElement("LI");
  2566. var thT7 = document.createElement("LI");
  2567. var thE0 = document.createElement("LI");
  2568. themeMenu.id = "thMenu";
  2569. thE0.innerHTML = " &#9670; ";
  2570. thE0.style.width = '20px';
  2571. thT1.textContent = "Solarium:Dark Theme";
  2572. thT2.textContent = "Solarium:Light Theme";
  2573. thT3.textContent = "Whisper Theme";
  2574. thT6.textContent = "Classic Theme";
  2575. thT4.textContent = "Deluge Theme";
  2576. thT7.textContent = !show_checkboxes.checked ? "Show checkboxes!" : "Remove Checkboxes!";
  2577. thT5.textContent = "I'm Feelin' Lucky";
  2578. thT7.addEventListener("click", function() {
  2579. if (show_checkboxes.checked){
  2580. var mt = document.getElementsByClassName("checkboxes")
  2581. for (var i=mt.length-1; i>-1; i--) mt[i].className = mt[i].className+"Off";
  2582. show_checkboxes.checked = false;
  2583. thT7.textContent = "Show Checkboxes!";
  2584. saveState.setItem("checkboxes", show_checkboxes.checked);
  2585. }
  2586. else {
  2587. var mt = document.getElementsByClassName("checkboxesOff")
  2588. for (var i=mt.length-1; i>-1; i--) mt[i].className = mt[i].className.replace(/Off/, "");
  2589. show_checkboxes.checked = true;
  2590. thT7.textContent = "Hide Checkboxes!";
  2591. saveState.setItem("checkboxes", show_checkboxes.checked);
  2592. }
  2593. saveState.setItem("checkboxes", show_checkboxes.checked);
  2594. saveState.save();
  2595. });
  2596. thT1.addEventListener("click", function() { themeSwitchAux("_thSDark"); saveState.save(); }, false);
  2597. thT2.addEventListener("click", function() { themeSwitchAux("_thSLight"); saveState.save();}, false);
  2598. thT3.addEventListener("click", function() { themeSwitchAux("_thWisp"); saveState.save();}, false);
  2599. thT4.addEventListener("click", function() { themeSwitchAux("_thCustom"); saveState.save();}, false);
  2600. thT5.addEventListener("click", function() { themeSwitchAux("_thRandom"); saveState.save();}, false);
  2601. thT6.addEventListener("click", function() { themeSwitchAux("_thClassic"); saveState.save();}, false);
  2602. thC2.appendChild(thT7);
  2603. thC2.appendChild(thT6);
  2604. thC2.appendChild(thT2);
  2605. thC2.appendChild(thT3);
  2606. thC2.appendChild(thT1);
  2607. thC2.appendChild(thT4);
  2608. thC2.appendChild(thT5);
  2609. thE0.appendChild(thC2);
  2610. themeMenu.appendChild(thE0);
  2611. control_panel.parentNode.insertBefore(themeMenu, control_panel);
  2612. //---
  2613.  
  2614. var spacer = document.createElement("SPAN");
  2615. spacer.className = "spacer"+sufTheme;
  2616. spacer.appendChild(document.createTextNode(SPACER_TEXT));
  2617.  
  2618. control_panel.style.fontSize = "14px";
  2619. //control_panel.removeChild(big_red_button);
  2620. control_panel.appendChild(document.createTextNode("Auto-refresh delay: "));
  2621. time_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  2622. time_input.title = "Enter search refresh delay in seconds.\n" + "Enter 0 for no auto-refresh.\n" + "Default is 0 (no auto-refresh).";
  2623. time_input.size = 3;
  2624. time_input.className = "cpInput"+sufTheme;
  2625. time_input.addEventListener("keyup", function() {
  2626. time_input.value = time_input.value.replace(/.*?(\d+)?.*/, '$1');
  2627. if (time_input.value >= 0) {
  2628. saveState.setItem("refreshTime", time_input.value);
  2629. saveState.save();
  2630. }
  2631. } );
  2632. control_panel.appendChild(time_input);
  2633.  
  2634. control_panel.appendChild(document.createTextNode(" "));
  2635.  
  2636. control_panel.appendChild(spacer);
  2637. control_panel.appendChild(document.createTextNode("Pages to scrape: "));
  2638. page_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  2639. page_input.title = "Enter number of pages to scrape. Default is 3.\n" + "Has no effect in a batch search (Most Available sort).";
  2640. page_input.size = 3;
  2641. page_input.className = "cpInput"+sufTheme;
  2642. page_input.addEventListener("keyup", function() {
  2643. page_input.value = page_input.value.replace(/.*?(\d+)?.*/, '$1');
  2644. if (page_input.value > 0) {
  2645. saveState.setItem("numPages", page_input.value);
  2646. saveState.save();
  2647. }
  2648. } );
  2649. control_panel.appendChild(page_input);
  2650.  
  2651. var skipspan = document.createElement("SPAN");
  2652. skipspan.appendChild(document.createTextNode("Correct for skips"));
  2653. skipspan.className += " cpSpans"+sufTheme;
  2654. skipspan.title = "Searches additional pages to get a more consistent number of results. Helpful if you're blocking a lot of items.";
  2655. if (correctForSkips.checked) skipspan.className += " cpSpansOn"+sufTheme;
  2656. skipspan.addEventListener("click", function() {
  2657. UIAux(skipspan, correctForSkips, "skips");
  2658. }, false);
  2659. if (show_checkboxes.checked) correctForSkips.className = "checkboxes";
  2660. else correctForSkips.className = "checkboxesOff";
  2661. skipspan.appendChild(correctForSkips);
  2662. correctForSkips.onclick = function () { UIAux(skipspan, correctForSkips, "skips"); };
  2663. control_panel.appendChild(spacer.cloneNode(true));
  2664. control_panel.appendChild(skipspan);
  2665.  
  2666. control_panel.appendChild(document.createTextNode(" "));
  2667.  
  2668. control_panel.appendChild(spacer.cloneNode(true));
  2669. control_panel.appendChild(document.createTextNode("Minimum batch size: "));
  2670. min_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  2671. min_input.title = "Enter minimum HITs for batch search (must Sort by Most Available).\n" + "Default is 100.";
  2672. min_input.size = 3;
  2673. min_input.className = "cpInput"+sufTheme;
  2674. min_input.addEventListener("keyup", function() {
  2675. min_input.value = min_input.value.replace(/.*?(\d+)?.*/, '$1');
  2676. if (min_input.value >= 0) {
  2677. saveState.setItem("minHits", min_input.value);
  2678. saveState.save();
  2679. }
  2680. } );
  2681. control_panel.appendChild(min_input);
  2682. control_panel.appendChild(document.createTextNode(" "));
  2683.  
  2684. control_panel.appendChild(document.createElement("P"));
  2685. control_panel.appendChild(document.createTextNode("Minimum reward: "));
  2686. reward_input.size = 3;
  2687. reward_input.title = "Enter the minimum desired pay per HIT, such as 0.10";
  2688. reward_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  2689. reward_input.className = "cpInput"+sufTheme;
  2690. reward_input.addEventListener("keyup", function() {
  2691. reward_input.value = reward_input.value.replace(/.*?(\d*\.?\d*)?.*/, '$1');
  2692. if (reward_input.value >= 0) {
  2693. saveState.setItem("reward", reward_input.value);
  2694. saveState.save();
  2695. }
  2696. } );
  2697. control_panel.appendChild(reward_input);
  2698.  
  2699. var qualspan = document.createElement("SPAN");
  2700. qualspan.appendChild(document.createTextNode("Qualified"));
  2701. qualspan.className += " cpSpans"+sufTheme;
  2702. qualspan.title = "Only show HITs you're currently qualified for (must be logged in).";
  2703. if (qual_input.checked) qualspan.className += " cpSpansOn"+sufTheme;
  2704. qualspan.addEventListener("click", function() {
  2705. UIAux(qualspan, qual_input, "qualified");
  2706. }, false);
  2707. if (show_checkboxes.checked) qual_input.className = "checkboxes";
  2708. else qual_input.className = "checkboxesOff";
  2709. qualspan.appendChild(qual_input);
  2710. qual_input.onclick = function () { UIAux(qualspan, qual_input, "qualified"); };
  2711. control_panel.appendChild(spacer.cloneNode(true));
  2712. control_panel.appendChild(qualspan);
  2713.  
  2714. var forcemastersspan = document.createElement("SPAN");
  2715. forcemastersspan.appendChild(document.createTextNode("Masters Only"));
  2716. forcemastersspan.className += " cpSpans"+sufTheme;
  2717. forcemastersspan.title = "Only show HITs that require Masters qualifications.";
  2718. if (masters_input.checked) forcemastersspan.className += " cpSpansOn"+sufTheme;
  2719. forcemastersspan.addEventListener("click", function() {
  2720. UIAux(forcemastersspan, masters_input, "masters");
  2721. }, false);
  2722. if (show_checkboxes.checked) masters_input.className = "checkboxes";
  2723. else masters_input.className = "checkboxesOff";
  2724. forcemastersspan.appendChild(masters_input);
  2725. masters_input.onclick = function () { UIAux(forcemastersspan, masters_input, "masters"); };
  2726. control_panel.appendChild(spacer.cloneNode(true));
  2727. control_panel.appendChild(forcemastersspan);
  2728.  
  2729. var mhidespan = document.createElement("SPAN");
  2730. mhidespan.appendChild(document.createTextNode("Hide Masters"));
  2731. mhidespan.className += " cpSpans"+sufTheme;
  2732. mhidespan.title = "Remove masters hits from the results if selected, otherwise display both masters and non-masters HITS.\nThe \"qualified\" setting superceedes this option.";
  2733. if (masters_hide.checked) mhidespan.className += " cpSpansOn"+sufTheme;
  2734. mhidespan.addEventListener("click", function() {
  2735. UIAux(mhidespan, masters_hide, "mShow");
  2736. }, false);
  2737. if (show_checkboxes.checked) masters_hide.className = "checkboxes";
  2738. else masters_hide.className = "checkboxesOff";
  2739. mhidespan.appendChild(masters_hide);
  2740. masters_hide.onclick = function () { UIAux(mhidespan, masters_hide, "mShow"); };
  2741. control_panel.appendChild(spacer.cloneNode(true));
  2742. control_panel.appendChild(mhidespan);
  2743.  
  2744. control_panel.appendChild(spacer.cloneNode(true));
  2745. control_panel.appendChild(document.createTextNode("Sort type: "));
  2746. control_panel.appendChild(sort_input);
  2747. sort_input.className += " cpInput"+sufTheme;
  2748. sort_input.addEventListener('change', function() {
  2749. var search = this.selectedIndex;
  2750. console.log(search);
  2751. saveState.setItem("sort", search);
  2752. saveState.save();
  2753. });
  2754. sort_input.title = "Get search results by...\n Latest = HIT Creation Date (newest first),\n Most Available = HITs Available (most first),\n Reward = Reward Amount (most first),\n Title = Title (A-Z)";
  2755.  
  2756. var sinvertspan = document.createElement("SPAN");
  2757. sinvertspan.appendChild(document.createTextNode("Invert"));
  2758. sinvertspan.className += " cpSpans"+sufTheme;
  2759. sinvertspan.title = "Reverse the order of the Sort Type choice, so...\n Latest = HIT Creation Date (oldest first),\n Most Available = HITs Available (least first),\n Reward = Reward Amount (least first),\n Title = Title (Z-A)";
  2760. if (sort_input_invert.checked) sinvertspan.className += " cpSpansOn"+sufTheme;
  2761. sinvertspan.addEventListener("click", function() {
  2762. UIAux(sinvertspan, sort_input_invert, "invert");
  2763. }, false);
  2764. if (show_checkboxes.checked) sort_input_invert.className = "checkboxes";
  2765. else sort_input_invert.className = "checkboxesOff";
  2766. sinvertspan.appendChild(sort_input_invert);
  2767. sort_input_invert.onclick = function () { UIAux(sinvertspan, sort_input_invert, "invert"); };
  2768. control_panel.appendChild(spacer.cloneNode(true));
  2769. control_panel.appendChild(sinvertspan);
  2770.  
  2771. control_panel.appendChild(document.createElement("P"));
  2772. control_panel.appendChild(document.createTextNode("New HIT highlighting: "));
  2773. new_time_display_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  2774. new_time_display_input.title = "Enter time (in seconds) to keep new HITs highlighted.\n" + "Default is 300 (5 minutes).";
  2775. new_time_display_input.size = 6;
  2776. new_time_display_input.className = "cpInput"+sufTheme;
  2777. new_time_display_input.addEventListener("keyup", function() {
  2778. new_time_display_input.value = new_time_display_input.value.replace(/.*?(\d*)?.*/, '$1');
  2779. if (new_time_display_input.value >= 0) {
  2780. saveState.setItem("newHitHighlight", new_time_display_input.value);
  2781. saveState.save();
  2782. }
  2783. } );
  2784. control_panel.appendChild(new_time_display_input);
  2785.  
  2786. var soundspan = document.createElement("SPAN");
  2787. soundspan.appendChild(document.createTextNode("Sound on new HIT "));
  2788. soundspan.className += " cpSpans"+sufTheme;
  2789. soundspan.title = "Play a sound when new results are found.";
  2790. if (friesAreDone.checked) soundspan.className += " cpSpansOn"+sufTheme;
  2791. audio_option.className += " cpInput"+sufTheme;
  2792. soundspan.addEventListener("click", function() {
  2793. if (friesAreDone.checked) {
  2794. friesAreDone.checked = false;
  2795. soundspan.className = soundspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2796. control_panel.removeChild(audio_option);
  2797. } else {
  2798. friesAreDone.checked = true;
  2799. soundspan.className += " cpSpansOn"+sufTheme;
  2800. control_panel.insertBefore(audio_option, soundspan.nextSibling);
  2801. }
  2802. saveState.setItem("fries", friesAreDone.checked);
  2803. saveState.save();
  2804. }, false);
  2805. if (show_checkboxes.checked) friesAreDone.className = "checkboxes";
  2806. else friesAreDone.className = "checkboxesOff";
  2807. soundspan.appendChild(friesAreDone);
  2808. friesAreDone.onclick = function () { UIAux(soundspan, friesAreDone, "fries"); };
  2809. control_panel.appendChild(spacer.cloneNode(true));
  2810. control_panel.appendChild(soundspan);
  2811. audio_option.title = "Select which sound will be played.";
  2812. audio_option.addEventListener('change', function() {
  2813. var sound = this.selectedIndex;
  2814. saveState.setItem("whichfry", sound);
  2815. saveState.save();
  2816. });
  2817. if (friesAreDone.checked) control_panel.insertBefore(audio_option, soundspan.nextSibling);
  2818.  
  2819. var ascspan = document.createElement("SPAN");
  2820. ascspan.className += " cpSpans"+sufTheme;
  2821. ascspan.style.fontSize = '15px';
  2822. if (sort_asc.checked) ascspan.className += " cpSpansOn"+sufTheme;
  2823. ascspan.title = "Sort results in ascending (low to high) order.";
  2824. var dscspan = document.createElement("SPAN");
  2825. dscspan.className += " cpSpans"+sufTheme;
  2826. dscspan.style.fontSize = '15px';
  2827. if (sort_dsc.checked) dscspan.className += " cpSpansOn"+sufTheme;
  2828. dscspan.title = "Sort results in descending (high to low) order.";
  2829. ascspan.addEventListener("click", function() {
  2830. if (!sort_asc.checked) {
  2831. sort_asc.checked = true;
  2832. sort_dsc.checked = false;
  2833. saveState.setItem("asc", sort_asc.checked);
  2834. saveState.setItem("dsc", sort_dsc.checked);
  2835. saveState.save();
  2836. ascspan.className += " cpSpansOn"+sufTheme;
  2837. dscspan.className = dscspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2838. }
  2839. }, false);
  2840. sort_asc.addEventListener("click", function() {
  2841. saveState.setItem("asc", sort_asc.checked);
  2842. saveState.setItem("dsc", sort_dsc.checked);
  2843. saveState.save();
  2844. ascspan.className += " cpSpansOn"+sufTheme;
  2845. dscspan.className = dscspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2846. }, false);
  2847. dscspan.addEventListener("click", function() {
  2848. if (!sort_dsc.checked) {
  2849. sort_dsc.checked = true;
  2850. sort_asc.checked = false;
  2851. saveState.setItem("asc", sort_asc.checked);
  2852. saveState.setItem("dsc", sort_dsc.checked);
  2853. saveState.save();
  2854. dscspan.className += " cpSpansOn"+sufTheme;
  2855. ascspan.className = ascspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2856. }
  2857. }, false);
  2858. sort_dsc.addEventListener("click", function() {
  2859. saveState.setItem("asc", sort_asc.checked);
  2860. saveState.setItem("dsc", sort_dsc.checked);
  2861. saveState.save();
  2862. dscspan.className += " cpSpansOn"+sufTheme;
  2863. ascspan.className = ascspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2864. }, false);
  2865. if (show_checkboxes.checked) {
  2866. sort_asc.className = "checkboxes";
  2867. sort_dsc.className = "checkboxes";
  2868. }
  2869. else {
  2870. sort_asc.className = "checkboxesOff";
  2871. sort_dsc.className = "checkboxesOff";
  2872. }
  2873. var sortdiv = document.createElement("DIV");
  2874. sortdiv.className = "cpSortdiv"+sufTheme;
  2875. sortdiv.appendChild(document.createTextNode(" ("));
  2876. sortdiv.appendChild(ascspan);
  2877. sortdiv.appendChild(dscspan);
  2878. sortdiv.appendChild(document.createTextNode(")"));
  2879. ascspan.innerHTML = " &#9650; ";
  2880. ascspan.appendChild(sort_asc);
  2881. //ascspan.appendChild(sort_asc);
  2882. dscspan.innerHTML = "&nbsp;&#9660; ";
  2883. dscspan.appendChild(sort_dsc);
  2884. //dscspan.appendChild(sort_dsc);
  2885.  
  2886. var topayspan = document.createElement("SPAN");
  2887. var toqualspan = document.createElement("SPAN");
  2888. topayspan.appendChild(document.createTextNode("Sort by TO pay"));
  2889. topayspan.className += " cpSpans"+sufTheme;
  2890. topayspan.title = "After getting search results based on the other selected options,\n" + "re-sort the results based on their average Turkopticon pay ratings. (Bayesian adjusted)";
  2891. if (sort_to.checked) {
  2892. topayspan.className += " cpSpansOn"+sufTheme;
  2893. } else topayspan.className = topayspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2894. topayspan.addEventListener("click", function() {
  2895. if (sort_to.checked) {
  2896. sort_to.checked = false;
  2897. control_panel.removeChild(sortdiv);
  2898. topayspan.className = topayspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2899. } else {
  2900. sort_to.checked = true;
  2901. sort_to2.checked = false;
  2902. control_panel.insertBefore(sortdiv, topayspan.nextSibling);
  2903. toqualspan.className = toqualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2904. topayspan.className += " cpSpansOn"+sufTheme;
  2905. }
  2906. saveState.setItem("to", sort_to.checked);
  2907. saveState.setItem("to2", sort_to2.checked);
  2908. saveState.save();
  2909. }, false);
  2910. if (show_checkboxes.checked) sort_to.className = "checkboxes";
  2911. else sort_to.className = "checkboxesOff";
  2912. topayspan.appendChild(sort_to);
  2913. sort_to.onclick = function () { UIAux(topayspan, sort_to, "to"); };
  2914. control_panel.appendChild(spacer.cloneNode(true));
  2915. control_panel.appendChild(topayspan);
  2916. if (sort_to.checked) control_panel.insertBefore(sortdiv, topayspan.nextSibling);
  2917.  
  2918. toqualspan.style.cursor = 'default';
  2919. toqualspan.appendChild(document.createTextNode("Sort by TO overall"));
  2920. toqualspan.className += " cpSpans"+sufTheme;
  2921. toqualspan.title = "After getting search results based on the other selected options,\n" + "re-sort the results by their overall Turkopticon rating. (Bayesian adjusted)";
  2922. if (sort_to2.checked) {
  2923. toqualspan.className += " cpSpansOn"+sufTheme;
  2924. } else toqualspan.className = toqualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2925. toqualspan.addEventListener("click", function() {
  2926. if (sort_to2.checked) {
  2927. sort_to2.checked = false;
  2928. control_panel.removeChild(sortdiv);
  2929. toqualspan.className = toqualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2930. } else {
  2931. sort_to2.checked = true;
  2932. sort_to.checked = false;
  2933. control_panel.insertBefore(sortdiv, toqualspan.nextSibling);
  2934. topayspan.className = topayspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2935. toqualspan.className += " cpSpansOn"+sufTheme;
  2936. }
  2937. saveState.setItem("to", sort_to.checked);
  2938. saveState.setItem("to2", sort_to2.checked);
  2939. saveState.save();
  2940. }, false);
  2941. if (show_checkboxes.checked) sort_to2.className = "checkboxes";
  2942. else sort_to2.className = "checkboxesOff";
  2943. toqualspan.appendChild(sort_to2);
  2944. sort_to2.onclick = function () { UIAux(toqualspan, sort_to2, "to2"); };
  2945. control_panel.appendChild(spacer.cloneNode(true));
  2946. control_panel.appendChild(toqualspan);
  2947. if (sort_to2.checked) control_panel.insertBefore(sortdiv, toqualspan.nextSibling);
  2948.  
  2949. control_panel.appendChild(document.createElement("P"));
  2950. control_panel.appendChild(document.createTextNode("Min pay TO: "));
  2951. control_panel.appendChild(min_to);
  2952. min_to.size = 3;
  2953. min_to.className = "cpInput"+sufTheme;
  2954. min_to.addEventListener("keyup", function() {
  2955. min_to.value = min_to.value.replace(/.*?(\d*\.?\d*)?.*/, '$1');
  2956. if (min_to.value >= 0) {
  2957. saveState.setItem("minTO", min_to.value);
  2958. saveState.save();
  2959. }
  2960. } );
  2961. min_to.title = "After getting search results based on the other selected options,\n" + "hide any results below this average Turkopticon pay rating.\n" + "Minimum is 1, maximum is 5, decimals up to 2 places, such as 3.25";
  2962.  
  2963. var hidenotospan = document.createElement("SPAN");
  2964. hidenotospan.appendChild(document.createTextNode("Hide no TO"));
  2965. hidenotospan.className += " cpSpans"+sufTheme;
  2966. hidenotospan.title = "After getting search results based on the other selected options,\n" + "hide any results that have no Turkopticon pay ratings yet.";
  2967. if (no_to_block.checked) hidenotospan.className += " cpSpansOn"+sufTheme;
  2968. hidenotospan.addEventListener("click", function() {
  2969. UIAux(hidenotospan, no_to_block, "hideNTO");
  2970. }, false);
  2971. if (show_checkboxes.checked) no_to_block.className = "checkboxes";
  2972. else no_to_block.className = "checkboxesOff";
  2973. hidenotospan.appendChild(no_to_block);
  2974. no_to_block.onclick = function () { UIAux(hidenotospan, no_to_block, "hideNTO"); };
  2975. control_panel.appendChild(spacer.cloneNode(true));
  2976. control_panel.appendChild(hidenotospan);
  2977.  
  2978. var disabletospan = document.createElement("SPAN");
  2979. disabletospan.appendChild(document.createTextNode("Disable TO"));
  2980. disabletospan.className += " cpSpans"+sufTheme;
  2981. disabletospan.title = "Disable attempts to download ratings data from Turkopticon for the results table.";
  2982. if (useTO_input.checked) disabletospan.className += " cpSpansOn"+sufTheme;
  2983. disabletospan.addEventListener("click", function() {
  2984. UIAux(disabletospan, useTO_input, "useTO");
  2985. }, false);
  2986. if (show_checkboxes.checked) useTO_input.className = "checkboxes";
  2987. else useTO_input.className = "checkboxesOff";
  2988. disabletospan.appendChild(useTO_input);
  2989. useTO_input.onclick = function () { UIAux(disabletospan, useTO_input, "useTO"); };
  2990. control_panel.appendChild(spacer.cloneNode(true));
  2991. control_panel.appendChild(disabletospan);
  2992.  
  2993. control_panel.appendChild(spacer.cloneNode(true));
  2994. control_panel.appendChild(document.createTextNode("Display export buttons: "));
  2995. var vbspan = document.createElement("SPAN");
  2996. vbspan.className += " cpSpans"+sufTheme;
  2997. vbspan.title = 'Show a button to export the specified HIT with vBulletin formatted text.';
  2998. if (opt_exportvb.checked) vbspan.className += " cpSpansOn"+sufTheme;
  2999. vbspan.addEventListener("click", function(){
  3000. var carr = document.getElementsByClassName('vbButton');
  3001. if (!opt_exportvb.checked) {
  3002. opt_exportvb.checked=true; vbspan.className += " cpSpansOn"+sufTheme;
  3003. for (var i=0; i<carr.length; i++)
  3004. carr[i].className = carr[i].className.replace(/taButtonsOff/, "taButtons");
  3005. } else {
  3006. for (var i=0; i<carr.length; i++)
  3007. carr[i].className = carr[i].className.replace(/taButtons/, "taButtonsOff");
  3008. opt_exportvb.checked=false; vbspan.className = vbspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  3009. }
  3010. saveState.setItem("exvb", opt_exportvb.checked);
  3011. saveState.save();
  3012. }, false);
  3013. vbspan.appendChild(document.createTextNode(" VB "));
  3014. if (show_checkboxes.checked) opt_exportvb.className = "checkboxes";
  3015. else opt_exportvb.className = "checkboxesOff";
  3016. vbspan.appendChild(opt_exportvb);
  3017. opt_exportvb.onclick = function () { UIAux(vbspan, opt_exportvb, "exvb"); };
  3018. control_panel.appendChild(vbspan);
  3019.  
  3020. var ircspan = document.createElement("SPAN");
  3021. ircspan.className += " cpSpans"+sufTheme;
  3022. ircspan.title = 'Show a button to export the specified HIT streamlined for IRC.';
  3023. if (opt_exportirc.checked) ircspan.className += " cpSpansOn"+sufTheme;
  3024. ircspan.addEventListener("click", function(){
  3025. var carr = document.getElementsByClassName('ircButton');
  3026. if (!opt_exportirc.checked) {
  3027. for (var i=0; i<carr.length; i++)
  3028. carr[i].className = carr[i].className.replace(/taButtonsOff/, "taButtons");
  3029. opt_exportirc.checked=true; ircspan.className += " cpSpansOn"+sufTheme;
  3030. } else {
  3031. for (var i=0; i<carr.length; i++)
  3032. carr[i].className = carr[i].className.replace(/taButtons/, "taButtonsOff");
  3033. opt_exportirc.checked=false; ircspan.className = ircspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  3034. }
  3035. saveState.setItem("exirc", opt_exportirc.checked);
  3036. saveState.save();
  3037. }, false);
  3038. ircspan.appendChild(document.createTextNode(" IRC "));
  3039. if (show_checkboxes.checked) opt_exportirc.className = "checkboxes";
  3040. else opt_exportirc.className = "checkboxesOff";
  3041. ircspan.appendChild(opt_exportirc);
  3042. opt_exportirc.onclick = function () { UIAux(ircspan, opt_exportirc, "exirc"); };
  3043. control_panel.appendChild(ircspan);
  3044.  
  3045. var redditspan = document.createElement("SPAN");
  3046. redditspan.className += " cpSpans"+sufTheme;
  3047. redditspan.title = 'Show a button to export the specified HIT formatted to r/HITsWorthTurkingFor standards .';
  3048. if (opt_exportreddit.checked) redditspan.className += " cpSpansOn"+sufTheme;
  3049. redditspan.addEventListener("click", function(){
  3050. var carr = document.getElementsByClassName('redditButton');
  3051. if (!opt_exportreddit.checked) {
  3052. for (var i=0; i<carr.length; i++)
  3053. carr[i].className = carr[i].className.replace(/taButtonsOff/, "taButtons");
  3054. opt_exportreddit.checked=true; redditspan.className += " cpSpansOn"+sufTheme;
  3055. } else {
  3056. for (var i=0; i<carr.length; i++)
  3057. carr[i].className = carr[i].className.replace(/taButtons/, "taButtonsOff");
  3058. opt_exportreddit.checked=false; redditspan.className = redditspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  3059. }
  3060. saveState.setItem("exreddit", opt_exportreddit.checked);
  3061. saveState.save();
  3062. }, false);
  3063. redditspan.appendChild(document.createTextNode(" HWTF "));
  3064. if (show_checkboxes.checked) opt_exportreddit.className = "checkboxes";
  3065. else opt_exportreddit.className = "checkboxesOff";
  3066. redditspan.appendChild(opt_exportreddit);
  3067. opt_exportreddit.onclick = function () { UIAux(redditspan, opt_exportreddit, "exreddit"); };
  3068. control_panel.appendChild(redditspan);
  3069.  
  3070. control_panel.appendChild(document.createElement("P"));
  3071. control_panel.appendChild(document.createTextNode("Search Terms: "));
  3072. control_panel.appendChild(search_input);
  3073. search_input.size = 20;
  3074. search_input.title = "Enter keywords to search for.\n" + "Default is blank (no search terms).";
  3075. search_input.placeholder="Enter search terms here";
  3076. search_input.style.textAlign = "left";
  3077. search_input.className = "cpInput"+sufTheme;
  3078. search_input.addEventListener('keyup', function() {
  3079. var value = this.value;
  3080. saveState.setItem("searchTerms", value);
  3081. saveState.save();
  3082. });
  3083. var blockspan = document.createElement("SPAN");
  3084. blockspan.appendChild(document.createTextNode("Hide blocklisted"));
  3085. blockspan.className += " cpSpans"+sufTheme;
  3086. blockspan.title = "When enabled, hide HITs that match your blocklist.\n" + "When disabled, HITs that match your blocklist will be displayed with a red border.";
  3087. if (useBlock.checked) blockspan.className += " cpSpansOn"+sufTheme;
  3088. blockspan.addEventListener("click", function() {
  3089. if (useBlock.checked) {
  3090. var mt = document.getElementsByClassName("scraperBlockedRow");
  3091. if (mt.length > 0) for (var i=mt.length-1; i>-1; i--)
  3092. mt[i].className = mt[i].className.replace(/scraperBlockedRow/, "scraperBlockedRowOff");
  3093. useBlock.checked = false;
  3094. blockspan.className = blockspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  3095. }
  3096. else {
  3097. var mt = document.getElementsByClassName("scraperBlockedRowOff");
  3098. if (mt.length > 0) for (var i=mt.length-1; i>-1; i--)
  3099. mt[i].className = mt[i].className.replace(/scraperBlockedRowOff/, "scraperBlockedRow");
  3100. useBlock.checked = true;
  3101. blockspan.className += " cpSpansOn"+sufTheme;
  3102. }
  3103. saveState.setItem("blocklist", useBlock.checked);
  3104. saveState.save();
  3105. }, false);
  3106. if (show_checkboxes.checked) useBlock.className = "checkboxes";
  3107. else useBlock.className = "checkboxesOff";
  3108. blockspan.appendChild(useBlock);
  3109. useBlock.onclick = function () { UIAux(blockspan, useBlock, "blocklist"); };
  3110. control_panel.appendChild(spacer.cloneNode(true));
  3111. control_panel.appendChild(blockspan);
  3112.  
  3113. var includelistspan = document.createElement("SPAN");
  3114. includelistspan.appendChild(document.createTextNode("Restrict to includelist"));
  3115. includelistspan.className += " cpSpans"+sufTheme;
  3116. includelistspan.title = "Show only HITs that match your includelist.\n" + "Be sure to edit your includelist first or no results will be displayed.";
  3117. if (matchOnly.checked) includelistspan.className += " cpSpansOn"+sufTheme;
  3118. includelistspan.addEventListener("click", function() {
  3119. if (matchOnly.checked) {
  3120. matchOnly.checked = false;
  3121. includelistspan.className = includelistspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  3122. }
  3123. else {
  3124. matchOnly.checked = true;
  3125. includelistspan.className += " cpSpansOn"+sufTheme;
  3126. }
  3127. saveState.setItem("useInclude", matchOnly.checked)
  3128. saveState.save();
  3129. }, false);
  3130. if (show_checkboxes.checked) matchOnly.className = "checkboxes";
  3131. else matchOnly.className = "checkboxesOff";
  3132. includelistspan.appendChild(matchOnly);
  3133. matchOnly.onclick = function () { UIAux(includelistspan, matchOnly, "useInclude"); };
  3134. control_panel.appendChild(spacer.cloneNode(true));
  3135. control_panel.appendChild(includelistspan);
  3136.  
  3137. var highincspan = document.createElement("SPAN");
  3138. highincspan.appendChild(document.createTextNode("Highlight Includelisted"));
  3139. highincspan.className += " cpSpans"+sufTheme;
  3140. highincspan.title = "Outline HITs that match your includelist with a dashed green border.";
  3141. if (highlightIncludes_input.checked) highincspan.className += " cpSpansOn"+sufTheme;
  3142. highincspan.addEventListener("click", function() {
  3143. if (highlightIncludes_input.checked) {
  3144. var mt = document.getElementsByClassName("scraperIncludelistedRow");
  3145. if (mt.length > 0)
  3146. for (var i=mt.length-1; i>-1; i--) { mt[i].className = mt[i].className.replace(/scraperIncludelistedRow/, "scraperIncludelistedRowOff"); }
  3147. highlightIncludes_input.checked = false;
  3148. highincspan.className = highincspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  3149. }
  3150. else {
  3151. var mt = document.getElementsByClassName("scraperIncludelistedRowOff");
  3152. if ( mt.length > 0)
  3153. for (var i=mt.length-1; i>-1; i--) { mt[i].className = mt[i].className.replace(/scraperIncludelistedRowOff/, "scraperIncludelistedRow"); }
  3154. highlightIncludes_input.checked = true;
  3155. highincspan.className += " cpSpansOn"+sufTheme;
  3156. }
  3157. saveState.setItem("highlightIncl", highlightIncludes_input.checked)
  3158. saveState.save();
  3159. }, false);
  3160. if (show_checkboxes.checked) highlightIncludes_input.className = "checkboxes";
  3161. else highlightIncludes_input.className = "checkboxesOff";
  3162. highincspan.appendChild(highlightIncludes_input);
  3163. highlightIncludes_input.onclick = function () { UIAux(highincspan, highlightIncludes_input, "highlightIncl"); };
  3164. control_panel.appendChild(spacer.cloneNode(true));
  3165. control_panel.appendChild(highincspan);
  3166. control_panel.appendChild(document.createElement("BR"));
  3167. control_panel.appendChild(document.createTextNode(" "));
  3168. //------
  3169. big_red_button.textContent = "Start";
  3170. big_red_button.className += " cpButtons"+sufTheme;
  3171. big_red_button.onclick = function(){start_running();};
  3172.  
  3173. // open blocklist editor
  3174. reset_blocks.textContent = "Edit Blocklist";
  3175. reset_blocks.className += " cpButtons"+sufTheme;
  3176. reset_blocks.setAttribute("id","blocklist_reset_button");
  3177. reset_blocks.onclick = function(){
  3178. //console.log("in");
  3179. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  3180. var textarea = $("#block_text");
  3181. var text = "";
  3182. for (var i = 0; i < ignore_list.length; i++){
  3183. text += ignore_list[i]+"^";
  3184. }
  3185. textarea.val(text.substring(0, text.length - 1));
  3186. $("#block_div").show();
  3187. };
  3188.  
  3189. // open includelist editor
  3190. include_button.textContent = "Edit Includelist";
  3191. include_button.className += " cpButtons"+sufTheme;
  3192. include_button.id = "includes_reset_button";
  3193. include_button.onclick = function() {
  3194. include_list = GM_getValue("scraper_include_list").split('^');
  3195. var textarea = $("#include_text");
  3196. var text = "";
  3197. for (var i = 0; i < include_list.length; i++){
  3198. text += include_list[i]+"^";
  3199. }
  3200. textarea.val(text.substring(0, text.length - 1));
  3201. $("#include_div").show();
  3202. };
  3203.  
  3204. control_panel.appendChild(document.createElement("P"));
  3205. text_area.style.fontWeight = "normal";
  3206. text_area.createCaption().innerHTML = '<a class="mainlink'+sufTheme+'" target="_blank" href="https://greasyfork.org/en/scripts/2002-hit-scraper-with-export" title="Read the documentation for HIT Scraper With Export on its Greasyfork page.">HIT Scraper</a> Results';
  3207. var col_heads = ['Requester','Title','Reward','# Avail','TO pay','Accept HIT','M?'];
  3208. var row = text_area.createTHead().insertRow(0);
  3209. text_area.caption.style.fontWeight = 800;
  3210. text_area.caption.className = "tabhead"+sufTheme;
  3211. text_area.caption.style.lineHeight = "1.25em";
  3212. if (default_text_size > 10)
  3213. text_area.cellPadding=Math.min(Math.max(1,Math.floor((default_text_size-10)/2)),5);
  3214. //console.log(text_area.cellPadding);
  3215. //text_area.cellPadding=2;
  3216. text_area.caption.style.fontSize = "24px";
  3217. text_area.rows[0].style.fontWeight = 800;
  3218. for (var i=0; i<col_heads.length; i++)
  3219. {
  3220. var this_cell = row.insertCell(i);
  3221. this_cell.innerHTML = col_heads[i];
  3222. this_cell.style.fontSize = "14px";
  3223. this_cell.className = "tabhead"+sufTheme;
  3224. if (i > 1)
  3225. this_cell.style.textAlign = 'center';
  3226. }
  3227. text_area.style.width = '100%';
  3228. var table_div = document.createElement('div');
  3229. table_div.className = "bodytable"+sufTheme;
  3230. text_area.rows[0].className = "tabHead"+sufTheme;
  3231. table_div.style.fontSize = "14px";
  3232. document.body.insertBefore(table_div,control_panel.nextSibling);
  3233. var header_hide_button = document.createElement('button');
  3234. table_div.appendChild(big_red_button); // start/stop
  3235. table_div.appendChild(document.createTextNode(" "));
  3236. table_div.appendChild(header_hide_button); // hide control panel
  3237. table_div.appendChild(document.createTextNode(" "));
  3238. table_div.appendChild(reset_blocks); // blocklist editor
  3239. table_div.appendChild(document.createTextNode(" "));
  3240. table_div.appendChild(include_button); // includelist editor
  3241. table_div.appendChild(document.createTextNode(" "));
  3242. table_div.appendChild(showButton); // toggle ignored
  3243. table_div.appendChild(document.createTextNode(" "));
  3244. table_div.appendChild(btn01); // theme editor
  3245. table_div.appendChild(document.createTextNode(" "));
  3246. table_div.appendChild(settings.panel.accessButton); // settings
  3247. $("#show_TO_stuff_button").hide();
  3248.  
  3249. table_div.appendChild(document.createElement("P"));
  3250. var statusdiv = document.createElement("DIV");
  3251. statusdiv.className = "statusdiv"+sufTheme;
  3252. statusdiv.appendChild(progress_report);
  3253. statusdiv.appendChild(document.createElement("P"));
  3254. statusdiv.appendChild(document.createTextNode("Status messages: "));
  3255. statusdiv.appendChild(status_report);
  3256. table_div.appendChild(statusdiv);
  3257. if (document.getElementById('lnkWorkerSignin'))
  3258. {
  3259. var logged_out_warning = document.createElement("P");
  3260. logged_out_warning.style.color = "#ff3300";
  3261. logged_out_warning.innerHTML = "<b>ATTENTION: SCRAPER IS RUNNING WHILE LOGGED OUT. SOME DATA WILL BE MISSING. GO TO REQUESTER'S LINK IN LOGGED-IN WINDOW FOR FORUM EXPORT.</b> <a class=\"link"+sufTheme+"\" href='https://www.mturk.com/mturk/beginsignin'>Click here to sign in</a>";
  3262. table_div.appendChild(logged_out_warning);
  3263. table_div.appendChild(document.createElement("P"));
  3264. }
  3265. table_div.appendChild(text_area);
  3266. header_hide_button.textContent = "Hide Panel";
  3267. header_hide_button.className += " cpButtons"+sufTheme;
  3268. header_hide_button.onclick = function(){
  3269. if (header_hide_button.textContent == "Hide Panel"){
  3270. $('#control_panel').hide(300);
  3271. btn01.style.display = 'none';
  3272. header_hide_button.textContent = "Show Panel";
  3273. }
  3274. else{
  3275. $('#control_panel').show(300);
  3276. btn01.style.display = 'initial';
  3277. header_hide_button.textContent = "Hide Panel";
  3278. }
  3279. };
  3280. }
  3281.  
  3282. /********HIT EXPORT ADDITIONS*****/
  3283.  
  3284. var EDIT = false;
  3285. var HIT;
  3286.  
  3287. var TO_BASE = "http://turkopticon.ucsd.edu/";
  3288. var API_BASE = "https://mturk-api.istrack.in/";
  3289. var API_URL = API_BASE + "multi-attrs.php?ids=";
  3290.  
  3291. var DEFAULT_TEMPLATE = '[table][tr][td][b]Title:[/b] [url={prev_link}][COLOR=blue]{title}[/COLOR][/url]\n';
  3292. DEFAULT_TEMPLATE += '[b]Requester:[/b] [url=https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId={rid}][COLOR=blue]{requester}[/COLOR][/url]';
  3293. DEFAULT_TEMPLATE += ' [{rid}] ([url='+TO_BASE+'{rid}][COLOR=blue]TO[/COLOR][/url])';
  3294. DEFAULT_TEMPLATE += '\n[b]TO Ratings:[/b]{to_stuff}';
  3295. DEFAULT_TEMPLATE += '\n[b]Description:[/b] {description}';
  3296. DEFAULT_TEMPLATE += '\n[b]Time:[/b] {time}';
  3297. DEFAULT_TEMPLATE += '\n[b]Hits Available:[/b] {hits}';
  3298. DEFAULT_TEMPLATE += '\n[b]Reward:[/b] [COLOR=green][b]{reward}[/b][/COLOR]';
  3299. DEFAULT_TEMPLATE += '\n[b]Qualifications:[/b] {quals}[/td][/tr][/table]';
  3300.  
  3301. var TEMPLATE;
  3302. var EASYLINK;
  3303.  
  3304. if (typeof GM_getValue === 'undefined')
  3305. TEMPLATE = null;
  3306. else {
  3307. TEMPLATE = GM_getValue('HITScraper Template');
  3308. EASYLINK = GM_getValue('HITScraper Easylink');
  3309. }
  3310. if (TEMPLATE == null) {
  3311. TEMPLATE = DEFAULT_TEMPLATE;
  3312. }
  3313.  
  3314. function buildXhrUrl(rai) {
  3315. var url = API_URL;
  3316. var ri = rai;
  3317. url += rai;
  3318. return url;
  3319. }
  3320.  
  3321. function makeXhrQuery(url) {
  3322. var xhr = new XMLHttpRequest();
  3323. try{
  3324. xhr.open('GET', url, false);
  3325. xhr.send(null);
  3326. return $.parseJSON(xhr.response);
  3327. }
  3328. catch(err){
  3329. return "TO DOWN";
  3330. }
  3331. }
  3332.  
  3333. function getNamesForEmptyResponses(rai, resp) {
  3334. for (var rid in rai) {
  3335. if (rai.hasOwnProperty(rid) && resp[rid] == "") {
  3336. resp[rid] = $.parseJSON('{"name": "' + rai[rid][0].innerHTML + '"}');
  3337. }
  3338. }
  3339. return resp;
  3340. }
  3341.  
  3342. function getKeys(obj) {
  3343. var keys = [];
  3344. for (var key in obj) {
  3345. keys.push(key);
  3346. }
  3347. return keys;
  3348. }
  3349.  
  3350. function export_sel_deleg(item,index,extype) {
  3351. //console.log(item);
  3352. if (extype == "vb") export_func(item);
  3353. else display(item, extype);
  3354. }
  3355.  
  3356. function block_deleg(item,index) {
  3357. //console.log(item);
  3358. block(item,index);
  3359. }
  3360.  
  3361. function block(hit,index){
  3362. var blockType = ["requester_strip","title"];
  3363. var blockThis = hit[blockType[index]].replace(/\s+/g," ").toLowerCase().trim();
  3364. var r = confirm("Do you really want to block hits matching requester/title \""+blockThis+"\"?");
  3365. if(r)
  3366. {
  3367. ignore_list.push(blockThis);
  3368. GM_setValue("scraper_ignore_list",ignore_list.join('^'));
  3369. }
  3370. }
  3371.  
  3372. function search_deleg(item,index) {
  3373. //console.log(item);
  3374. var searches = ["rid","title"];
  3375. search(item,searches[index]);
  3376. }
  3377.  
  3378. function hit_sort_func()
  3379. {
  3380. return function(a,b) {
  3381. if (a.date == b.date) {
  3382. if (a.requesterName < b.requesterName)
  3383. return -1;
  3384. if (a.requesterName > b.requesterName)
  3385. return 1;
  3386. if (a.title < b.title)
  3387. return -1;
  3388. if (a.title > b.title)
  3389. return 1;
  3390. if (a.status < b.status)
  3391. return -1;
  3392. if (a.status > b.status)
  3393. return 1;
  3394. }
  3395. if (a.date > b.date)
  3396. return 1;
  3397. if (a.date < b.date)
  3398. return -1;
  3399. };
  3400. }
  3401.  
  3402. function escapeRegExp(str) {
  3403. return str.replace(/[-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
  3404. }
  3405.  
  3406. function search(item,search_type){
  3407. //return true;/*
  3408.  
  3409. var request = indexedDB.open("HITDB", v);
  3410. request.onsuccess = function(e) {
  3411. HITStorage.indexedDB.db = e.target.result;
  3412. var db = HITStorage.indexedDB.db;
  3413. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  3414. var store = trans.objectStore("HIT");
  3415.  
  3416. var req;
  3417. var results = [];
  3418. var index;
  3419. var range;
  3420. req = store.openCursor();
  3421. req.onsuccess = function(event) {
  3422. var cursor = event.target.result;
  3423. if (cursor) {
  3424. hit = cursor.value;
  3425. var keys = ['title', 'requesterId'];
  3426. var re = new RegExp(escapeRegExp(item[search_type]),"ig");
  3427. for (var k in keys)
  3428. {
  3429. if (hit[keys[k]] != null && re.test(hit[keys[k]].trim())){
  3430. results.push(cursor.value);
  3431. }
  3432. }
  3433.  
  3434. cursor.continue();
  3435. }
  3436. else {
  3437. //console.log(results);
  3438. results.sort(hit_sort_func());
  3439. show_results(results);
  3440. }
  3441. db.close();
  3442. };
  3443. request.onerror = HITStorage.indexedDB.onerror;/**/
  3444. };
  3445. }
  3446.  
  3447. function format_hit_line (hit, odd, status_color, new_day)
  3448. {
  3449. var line = '<tr style="background-color:';
  3450. if (odd)
  3451. line += '#f1f3eb;';
  3452. else
  3453. line += 'white;';
  3454. line += ' valign=top;';
  3455. if (new_day)
  3456. line += ' border: 0px dotted #000000; border-width: 2px 0px 0px 0px">';
  3457. else
  3458. line += '">';
  3459.  
  3460. line += '<td>' + hit.date + '</td>';
  3461. if (hit.requesterLink != null)
  3462. line += '<td style="width:165px"><a href="' + hit.requesterLink + '" title="Contact this Requester">' + hit.requesterName + '</a></td>';
  3463. else
  3464. line += '<td style="width:165px">' + hit.requesterName + '</td>';
  3465. line += '<td style="width:213px">' + hit.title + '</td>';
  3466. line += '<td style="width:45px">$' + hit.reward.toFixed(2) + '</td>';
  3467. line += '<td style="color:' + status_color + '; width:55px">' + hit.status + '</td>';
  3468. line += '<td><div style="width:225px; overflow:hidden">' + hit.feedback + '</div></td>';
  3469. line += '</tr>\n';
  3470. return line;
  3471. }
  3472.  
  3473. function status_color (status)
  3474. {
  3475. var color = "green";
  3476.  
  3477. if (status.match("Pending Approval"))
  3478. color = "orange";
  3479. else if (status.match("Rejected"))
  3480. color = "red";
  3481.  
  3482. return color;
  3483. }
  3484.  
  3485. function show_results (results){
  3486. var resultsWindow = window.open();
  3487. resultsWindow.document.write("<html><head><title>Status Detail Search Results</title></head><body>\n");
  3488. resultsWindow.document.write("<h1>HITs matching your search:</h1>\n");
  3489. resultsWindow.document.write('<table style="border: 1px solid black;border-collapse:collapse;width:90%;margin-left:auto;margin-right:auto;">\n');
  3490. resultsWindow.document.write('<tr style="background-color:lightgrey"><th>Date</th><th>Requester</th><th>HIT Title</th><th>Reward</th><th>Status</th><th>Feedback</th></tr>\n');
  3491.  
  3492. var odd = true;
  3493. var sum = 0;
  3494. var sum_rejected = 0;
  3495. var sum_approved = 0;
  3496. var sum_pending = 0;
  3497.  
  3498.  
  3499. var new_day = false;
  3500.  
  3501. for (var i=0; i<results.length; i++) {
  3502. odd = !odd;
  3503. sum += results[i].reward;
  3504. if (results[i].status == 'Rejected')
  3505. sum_rejected += results[i].reward;
  3506. else if (results[i].status == 'Pending Approval')
  3507. sum_pending += results[i].reward;
  3508. else
  3509. sum_approved += results[i].reward;
  3510.  
  3511. if (i>0 && (results[i-1].date != results[i].date))
  3512. new_day = true;
  3513. else
  3514. new_day = false;
  3515. resultsWindow.document.write(format_hit_line(results[i], odd, status_color(results[i].status), new_day ));
  3516. }
  3517.  
  3518. resultsWindow.document.write('<tr style="background-color:lightgrey"><th></th><th></th><th></th><th>$' + sum.toFixed(2) + '</th><th></th><th></th></tr>\n');
  3519. resultsWindow.document.write("</table>");
  3520. resultsWindow.document.write("<p>Found " + results.length + " matching HITs. $" + sum_approved.toFixed(2) + " approved, " +
  3521. "$" + sum_rejected.toFixed(2) + " rejected and $" + sum_pending.toFixed(2) + " pending.</p>");
  3522. resultsWindow.document.write("</body></html>");
  3523. resultsWindow.document.close();
  3524. }
  3525.  
  3526.  
  3527. // ---- vB
  3528.  
  3529. function export_func(item) {
  3530. HIT = item;
  3531. gedit_button.textContent = 'Edit Template';
  3532. apply_template(item);
  3533. gdiv.style.display = 'block';
  3534. gtextarea.select();
  3535. }
  3536.  
  3537. function apply_template(hit_data) {
  3538. var txt = TEMPLATE;
  3539.  
  3540. var vars = ['title', 'requester', 'rid', 'description', 'reward', 'quals', 'prev_link', 'time', 'hits', 'to_stuff', 'to_text'];
  3541.  
  3542. var resp = null;
  3543. if (txt.indexOf('{to_text}') >= 0 || txt.indexOf('{to_stuff}') >= 0){
  3544. var url = buildXhrUrl(hit_data["rid"]);
  3545. resp = getTOMulti(hit_data["rid"]);
  3546. //console.log(resp);
  3547. }
  3548. var toText = "";
  3549. var toStuff = "";
  3550. var toData = "";
  3551. var numResp = (resp == null || resp == "TO DOWN" ? "n/a" : resp[hit_data["rid"]].reviews);
  3552. if (resp == "TO DOWN"){
  3553. toStuff = " [URL=\""+TO_BASE+hit_data['rid']+"\"]TO down.[/URL]";
  3554. toText = toStuff;
  3555. }
  3556. else if (resp == null || resp[hit_data["rid"]].attrs == null && resp != "TO DOWN") {
  3557. toStuff = " No TO ";
  3558. toText = " No TO ";
  3559. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  3560. toStuff += "(Submit a new TO rating for this requester)[/URL]";
  3561. }
  3562. else {
  3563. for (var key in resp[hit_data["rid"]].attrs) {
  3564. //toText += "\n[*]"+key+": "+resp[hit_data["requesterId"]].attrs[key]+"\n";
  3565. var i = 0;
  3566. var color = "green";
  3567. var name = key;
  3568. var num = Math.floor(resp[hit_data["rid"]].attrs[key]);
  3569. switch (key){
  3570. case "comm":
  3571. name = "Communicativity";
  3572. break;
  3573. case "pay":
  3574. name = "Generosity";
  3575. break;
  3576. case "fast":
  3577. name = "Promptness";
  3578. break;
  3579. case "fair":
  3580. name = "Fairness";
  3581. break;
  3582. default:
  3583. name = key;
  3584. break;
  3585. }
  3586. switch (num){
  3587. case 0:
  3588. color = "red";
  3589. break;
  3590. case 1:
  3591. color = "red";
  3592. break;
  3593. case 2:
  3594. color = "orange";
  3595. break;
  3596. case 3:
  3597. color = "yellow";
  3598. break;
  3599. default:
  3600. break;
  3601. }
  3602. toText += (num > 0 ? "\n[color="+color+"]" : "\n");
  3603. for (i; i < num; i++){
  3604. toText += "[b]"+symbol+"[/b]";
  3605. }
  3606. toText += (num > 0 ? "[/color]" : "");
  3607. if (i < 5){
  3608. toText += "[color=white]";
  3609. for (i; i < 5; i++)
  3610. toText += "[b]"+symbol+"[/b]";
  3611. toText += "[/color]";
  3612. }
  3613. toText += " "+Number(resp[hit_data["rid"]].attrs[key]).toFixed(2)+" "+name;
  3614. toData += Number(resp[hit_data["rid"]].attrs[key]).toFixed(2) + ",";
  3615. }
  3616. //toText += "[/list]";
  3617. toText += (txt.indexOf('{to_stuff}') >= 0 ? "" : "\nNumber of Reviews: "+numResp+"\n[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"](Submit a new TO rating for this requester)[/URL]");
  3618. toStuff = '\n[img]http://data.istrack.in/to/' + toData.slice(0,-1) + '.png[/img]';
  3619. toStuff += (txt.indexOf('{to_stuff}') >= 0 ? (txt.indexOf('{to_text}') >= 0 ? "" : toText) : "");
  3620. toStuff += "\nNumber of Reviews: "+numResp;
  3621. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  3622. toStuff += "\n(Submit a new TO rating for this requester)[/URL]";
  3623. }
  3624.  
  3625. for (var u = 0; u < vars.length; u++) {
  3626. var t = new RegExp('\{' + vars[u] + '\}', 'g');
  3627. if (vars[u] == "to_stuff") {
  3628. txt = txt.replace(t, toStuff);
  3629. }
  3630. else if (vars[u] == "to_text") {
  3631. txt = txt.replace(t, toText);
  3632. }
  3633. else if (vars[u] == "prev_link") {
  3634. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[u]]);
  3635. }
  3636. else if (vars[u] == "acc_link") {
  3637. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[u]]);
  3638. }
  3639. else
  3640. txt = txt.replace(t, hit_data[vars[u]]);
  3641. }
  3642. gtextarea.value = txt;
  3643. }
  3644.  
  3645. function export_irc(item) {
  3646. display(item);
  3647. }
  3648.  
  3649. function apply_template(hit_data) {
  3650. var txt = TEMPLATE;
  3651.  
  3652. var vars = ['title', 'requester', 'rid', 'description', 'reward', 'quals', 'prev_link', 'time', 'hits', 'to_stuff', 'to_text'];
  3653.  
  3654. var resp = null;
  3655. if (txt.indexOf('{to_text}') >= 0 || txt.indexOf('{to_stuff}') >= 0){
  3656. var url = buildXhrUrl(hit_data["rid"]);
  3657. resp = getTOMulti(hit_data["rid"]);
  3658. //console.log(resp);
  3659. }
  3660. var toText = "";
  3661. var toStuff = "";
  3662. var toData = "";
  3663. var numResp = (resp == null || resp == "TO DOWN" ? "n/a" : resp[hit_data["rid"]].reviews);
  3664. if (resp == "TO DOWN"){
  3665. toStuff = " [URL=\""+TO_BASE+hit_data['rid']+"\"]TO down.[/URL]";
  3666. toText = toStuff;
  3667. }
  3668. else if (resp == null || resp[hit_data["rid"]].attrs == null && resp != "TO DOWN") {
  3669. toStuff = " No TO ";
  3670. toText = " No TO ";
  3671. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  3672. toStuff += "(Submit a new TO rating for this requester)[/URL]";
  3673. }
  3674. else {
  3675. for (var key in resp[hit_data["rid"]].attrs) {
  3676. //toText += "\n[*]"+key+": "+resp[hit_data["requesterId"]].attrs[key]+"\n";
  3677. var i = 0;
  3678. var color = "green";
  3679. var name = key;
  3680. var num = Math.floor(resp[hit_data["rid"]].attrs[key]);
  3681. switch (key){
  3682. case "comm":
  3683. name = "Communicativity";
  3684. break;
  3685. case "pay":
  3686. name = "Generosity";
  3687. break;
  3688. case "fast":
  3689. name = "Promptness";
  3690. break;
  3691. case "fair":
  3692. name = "Fairness";
  3693. break;
  3694. default:
  3695. name = key;
  3696. break;
  3697. }
  3698. switch (num){
  3699. case 0:
  3700. color = "red";
  3701. break;
  3702. case 1:
  3703. color = "red";
  3704. break;
  3705. case 2:
  3706. color = "orange";
  3707. break;
  3708. case 3:
  3709. color = "yellow";
  3710. break;
  3711. default:
  3712. break;
  3713. }
  3714. toText += (num > 0 ? "\n[color="+color+"]" : "\n");
  3715. for (i; i < num; i++){
  3716. toText += "[b]"+symbol+"[/b]";
  3717. }
  3718. toText += (num > 0 ? "[/color]" : "");
  3719. if (i < 5){
  3720. toText += "[color=white]";
  3721. for (i; i < 5; i++)
  3722. toText += "[b]"+symbol+"[/b]";
  3723. toText += "[/color]";
  3724. }
  3725. toText += " "+Number(resp[hit_data["rid"]].attrs[key]).toFixed(2)+" "+name;
  3726. toData += Number(resp[hit_data["rid"]].attrs[key]).toFixed(2) + ",";
  3727. }
  3728. //toText += "[/list]";
  3729. toText += (txt.indexOf('{to_stuff}') >= 0 ? "" : "\nNumber of Reviews: "+numResp+"\n[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"](Submit a new TO rating for this requester)[/URL]");
  3730. toStuff = '\n[img]http://data.istrack.in/to/' + toData.slice(0,-1) + '.png[/img]';
  3731. toStuff += (txt.indexOf('{to_stuff}') >= 0 ? (txt.indexOf('{to_text}') >= 0 ? "" : toText) : "");
  3732. toStuff += "\nNumber of Reviews: "+numResp;
  3733. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  3734. toStuff += "\n(Submit a new TO rating for this requester)[/URL]";
  3735. }
  3736.  
  3737. for (var u = 0; u < vars.length; u++) {
  3738. var t = new RegExp('\{' + vars[u] + '\}', 'g');
  3739. if (vars[u] == "to_stuff") {
  3740. txt = txt.replace(t, toStuff);
  3741. }
  3742. else if (vars[u] == "to_text") {
  3743. txt = txt.replace(t, toText);
  3744. }
  3745. else if (vars[u] == "prev_link") {
  3746. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[u]]);
  3747. }
  3748. else if (vars[u] == "acc_link") {
  3749. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[u]]);
  3750. }
  3751. else
  3752. txt = txt.replace(t, hit_data[vars[u]]);
  3753. }
  3754. gtextarea.value = txt;
  3755. }
  3756.  
  3757. function hide_func(div) {
  3758. if (EDIT == false)
  3759. div.style.display = 'none';
  3760. }
  3761.  
  3762. function edit_func() {
  3763. if (EDIT == true) {
  3764. EDIT = false;
  3765. TEMPLATE = gtextarea.value;
  3766. gedit_button.textContent = 'Edit Template';
  3767. apply_template(HIT);
  3768. }
  3769. else {
  3770. //console.log("Editing");
  3771. EDIT = true;
  3772. gedit_button.textContent = 'Show Changes';
  3773. gsave_button.disabled = false;
  3774. gtextarea.value = TEMPLATE;
  3775. }
  3776. }
  3777.  
  3778. function default_func() {
  3779. GM_deleteValue('HITScraper Template');
  3780. TEMPLATE = DEFAULT_TEMPLATE;
  3781. EDIT = false;
  3782. gedit_button.textContent = 'Edit Template';
  3783. apply_template(HIT);
  3784. }
  3785.  
  3786. function save_func() {
  3787. if (EDIT)
  3788. TEMPLATE = gtextarea.value;
  3789. GM_setValue('HITScraper Template', TEMPLATE);
  3790. }
  3791.  
  3792. var gdiv = document.createElement('div');
  3793. var gtextarea = document.createElement('textarea');
  3794. var gdiv2 = document.createElement('label');
  3795.  
  3796. gdiv.style.position = 'fixed';
  3797. gdiv.style.width = '500px';
  3798. gdiv.style.height = '235px';
  3799. gdiv.style.left = '50%';
  3800. gdiv.style.right = '50%';
  3801. gdiv.style.margin = '-250px 0px 0px -250px';
  3802. gdiv.style.top = '300px';
  3803. gdiv.style.padding = '5px';
  3804. gdiv.style.border = '2px';
  3805. gdiv.style.backgroundColor = 'black';
  3806. gdiv.style.color = 'white';
  3807. gdiv.style.zIndex = '100';
  3808.  
  3809. gtextarea.style.padding = '2px';
  3810. gtextarea.style.width = '500px';
  3811. gtextarea.style.height = '200px';
  3812. gtextarea.title = '{title}\n{requester}\n{rid}\n{description}\n{reward}\n{quals}\n{prev_link}\n{time}\n{hit}\n{to_stuff}\n{to_text}';
  3813.  
  3814. gdiv.textContent = 'Press Ctrl+C to copy to clipboard. Click textarea to close';
  3815. gdiv.style.fontSize = '12px';
  3816. gdiv.appendChild(gtextarea);
  3817.  
  3818. var gedit_button = document.createElement('button');
  3819. var gsave_button = document.createElement('button');
  3820. var gdefault_button = document.createElement('button');
  3821. var geasy_button = document.createElement('button');
  3822.  
  3823. gedit_button.textContent = 'Edit Template';
  3824. gedit_button.setAttribute('id', 'edit_button');
  3825. gedit_button.style.height = '18px';
  3826. gedit_button.style.width = '100px';
  3827. gedit_button.style.fontSize = '10px';
  3828. gedit_button.style.paddingLeft = '3px';
  3829. gedit_button.style.paddingRight = '3px';
  3830. gedit_button.style.backgroundColor = 'white';
  3831.  
  3832. gsave_button.textContent = 'Save Template';
  3833. gsave_button.setAttribute('id', 'save_button');
  3834. gsave_button.style.height = '18px';
  3835. gsave_button.style.width = '100px';
  3836. gsave_button.style.fontSize = '10px';
  3837. gsave_button.style.paddingLeft = '3px';
  3838. gsave_button.style.paddingRight = '3px';
  3839. gsave_button.style.backgroundColor = 'white';
  3840. gsave_button.style.marginLeft = '5px';
  3841.  
  3842. geasy_button.textContent = 'Change Adfly Url';
  3843. geasy_button.setAttribute('id', 'easy_button');
  3844. geasy_button.style.height = '18px';
  3845. geasy_button.style.width = '100px';
  3846. geasy_button.style.fontSize = '10px';
  3847. geasy_button.style.paddingLeft = '3px';
  3848.  
  3849. gdefault_button.textContent = ' D ';
  3850. gdefault_button.setAttribute('id', 'default_button');
  3851. gdefault_button.style.height = '18px';
  3852. gdefault_button.style.width = '20px';
  3853. gdefault_button.style.fontSize = '10px';
  3854. gdefault_button.style.paddingLeft = '3px';
  3855. gdefault_button.style.paddingRight = '3px';
  3856. gdefault_button.style.backgroundColor = 'white';
  3857. gdefault_button.style.marginLeft = '5px';
  3858. gdefault_button.title = 'Return default template';
  3859.  
  3860. gdiv.appendChild(gedit_button);
  3861. gdiv.appendChild(gsave_button);
  3862. gdiv.appendChild(gdefault_button);
  3863. gdiv.appendChild(geasy_button);
  3864. gsave_button.disabled = true;
  3865.  
  3866. gdiv.style.display = 'none';
  3867. gtextarea.addEventListener("click", function() {hide_func(gdiv);}, false);
  3868. gedit_button.addEventListener("click", function() {edit_func();}, false);
  3869. gsave_button.addEventListener("click", function() {save_func();}, false);
  3870. gdefault_button.addEventListener("click", function() {default_func();}, false);
  3871. document.body.insertBefore(gdiv, document.body.firstChild);
  3872.  
  3873.  
  3874. //Functions below were added for the irc export with the help of clickhappier and Cristo
  3875.  
  3876. var accountStatus = "loggedOut";
  3877. if ( !document.getElementById("lnkWorkerSignin") ) // if sign-in link not present
  3878. {
  3879. accountStatus = "loggedIn";
  3880. }
  3881.  
  3882.  
  3883. function getUrlVariable(url, variable)
  3884. {
  3885. var query = url.split('?');
  3886. var vars = query[1].split("&");
  3887. for ( var i=0; i<vars.length; i++ )
  3888. {
  3889. var pair = vars[i].split("=");
  3890. if ( pair[0] == variable )
  3891. { return pair[1]; }
  3892. }
  3893. return(false);
  3894. }
  3895.  
  3896. function getTO(f){
  3897. var toComp = [];
  3898. var toUrl2 = 'https://mturk-api.istrack.in/multi-attrs.php?ids='+f;
  3899. var toUrl = 'https://turkopticon.ucsd.edu/api/multi-attrs.php?ids='+f;
  3900. var requestTO = new XMLHttpRequest();
  3901. try{ // first try Miku's TO mirror server (istrack.in)
  3902. requestTO.onreadystatechange = function () {
  3903. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  3904. if (requestTO.responseText.split(':').length > 2) {
  3905. var toInfo = requestTO.responseText.split('{')[3].split('}')[0].split(',');
  3906. for (var t = 0; t < 4; t++) {
  3907. var arrTo = toInfo[t].split(':');
  3908. toComp.push(arrTo[1].substring(1,4));
  3909. }
  3910. }
  3911. else { toComp = ['-','-','-','-']; }
  3912. }
  3913. };
  3914. requestTO.open('GET', toUrl, false);
  3915. requestTO.send(null);
  3916. return toComp;
  3917. }
  3918. catch(err){ // if mirror unavailable, try main TO server
  3919. try{
  3920. requestTO.onreadystatechange = function () {
  3921. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  3922. if (requestTO.responseText.split(':').length > 2) {
  3923. var toInfo = requestTO.responseText.split('{')[3].split('}')[0].split(',');
  3924. for (var t = 0; t < 4; t++) {
  3925. var arrTo = toInfo[t].split(':');
  3926. toComp.push(arrTo[1].substring(1,4));
  3927. }
  3928. }
  3929. else { toComp = ['-','-','-','-']; }
  3930. }
  3931. };
  3932. requestTO.open('GET', toUrl2, false);
  3933. requestTO.send(null);
  3934. return toComp;
  3935. }
  3936. catch(err){ // if both unavailable, return 'na's
  3937. toComp = ['na','na','na','na'];
  3938. return toComp;
  3939. }
  3940. }
  3941. }
  3942.  
  3943. function getTOMulti(f){
  3944. var toComp = {};
  3945. var toUrl2 = 'https://mturk-api.istrack.in/multi-attrs.php?ids='+f;
  3946. var toUrl = 'https://turkopticon.ucsd.edu/api/multi-attrs.php?ids='+f;
  3947. var rids = f.split(',');
  3948. var requestTO = new XMLHttpRequest();
  3949. try{ // first try Miku's TO mirror server (istrack.in)
  3950. requestTO.onreadystatechange = function () {
  3951. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  3952. if (requestTO.responseText.split(':').length > 2)
  3953. toComp = $.parseJSON(requestTO.responseText);
  3954. else
  3955. toComp = null;
  3956. }
  3957. };
  3958. requestTO.open('GET', toUrl, false);
  3959. requestTO.send(null);
  3960. return toComp;
  3961. }
  3962. catch(err){ // if mirror unavailable, try main TO server
  3963. try{
  3964. requestTO.onreadystatechange = function () {
  3965. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  3966. if (requestTO.responseText.split(':').length > 2)
  3967. toComp = $.parseJSON(requestTO.responseText);
  3968. else
  3969. toComp = null;
  3970. }
  3971. };
  3972. requestTO.open('GET', toUrl2, false);
  3973. requestTO.send(null);
  3974. return toComp;
  3975. }
  3976. catch(err){ // if both unavailable, return 'na's
  3977. toComp = "TO DOWN";
  3978. return toComp;
  3979. }
  3980. }
  3981. }
  3982.  
  3983. function sleep(ms){ // from http://www.digimantra.com/tutorials/sleep-or-wait-function-in-javascript/
  3984. var dt = new Date();
  3985. dt.setTime(dt.getTime() + ms);
  3986. while (new Date().getTime() < dt.getTime());
  3987. }
  3988. function ns4tbulkshorten(urlArr){
  3989. console.log("ns4t bulk shorten");
  3990. var urlT = "https://ns4t.net/yourls-api.php?action=bulkshortener&title=MTurk&signature=39f6cf4959";
  3991. for (var i = 0; i < urlArr.length; i++)
  3992. {
  3993. urlT += "&urls[]="+urlArr[i].replace(/&/g, "%26");
  3994. }
  3995. console.log(urlT);
  3996. var requestNs4t = new XMLHttpRequest();
  3997. try{
  3998. requestNs4t.onreadystatechange = function () {
  3999. if (requestNs4t.readyState == 4) {
  4000. if (requestNs4t.status == 200) {
  4001. shortUrl = requestNs4t.responseText;
  4002. console.log("ns4t.net response: " + requestNs4t.status + " " + requestNs4t.statusText + " " + requestNs4t.responseText);
  4003. }
  4004. else {
  4005. console.log('ns4t.net unsuccessful: ' + requestNs4t.status + " " + requestNs4t.statusText);
  4006. }
  4007. }
  4008. };
  4009. requestNs4t.open('GET', urlT, false);
  4010. requestNs4t.send(null);
  4011. var urls = shortUrl.split(";");
  4012. return {preview:urls[0], panda:urls[1], req:urls[2], to:urls[3]}
  4013. }
  4014. catch(err){
  4015. return shortUrl;
  4016. }
  4017. }
  4018.  
  4019. function ns4tShorten(url){ // mturk-only URL shortener on Tjololo's server ns4t.net
  4020. console.log("ns4tShorten function");
  4021. var shortUrl;
  4022. var urlT = "https://ns4t.net/yourls-api.php" + "?action=shorturl&url=" + encodeURIComponent(url) + "&format=simple&title=MTurk&signature=39f6cf4959";
  4023. var requestNs4t = new XMLHttpRequest();
  4024. try{
  4025. requestNs4t.onreadystatechange = function () {
  4026. if (requestNs4t.readyState == 4) {
  4027. if (requestNs4t.status == 200) {
  4028. shortUrl = requestNs4t.responseText;
  4029. console.log("ns4t.net response: " + requestNs4t.status + " " + requestNs4t.statusText + " " + requestNs4t.responseText);
  4030. }
  4031. else {
  4032. console.log('ns4t.net unsuccessful: ' + requestNs4t.status + " " + requestNs4t.statusText);
  4033. }
  4034. }
  4035. };
  4036. requestNs4t.open('GET', urlT, false);
  4037. requestNs4t.send(null);
  4038. return shortUrl;
  4039. }
  4040. catch(err){
  4041. return shortUrl;
  4042. }
  4043. }
  4044. function tnyimShorten(url){ // Tny.im URL Shortener - http://tny.im/aboutapi.php - this is only possible this way because their server has the "Access-Control-Allow-Origin = *" headers enabled (the above TO mirror server does too)
  4045. console.log("tnyimShorten function");
  4046. var shortUrl;
  4047. var urlT = "https://tny.im/yourls-api.php" + "?action=shorturl&url=" + encodeURIComponent(url) + "&format=simple&title=MTurk";
  4048. var requestTnyim = new XMLHttpRequest();
  4049. try{
  4050. requestTnyim.onreadystatechange = function () {
  4051. if (requestTnyim.readyState == 4) {
  4052. if (requestTnyim.status == 200) {
  4053. shortUrl = requestTnyim.responseText;
  4054. console.log("tny.im response: " + requestTnyim.status + " " + requestTnyim.statusText + " " + requestTnyim.responseText);
  4055. }
  4056. else {
  4057. console.log('tny.im unsuccessful: ' + requestTnyim.status + " " + requestTnyim.statusText);
  4058. }
  4059. }
  4060. };
  4061. requestTnyim.open('GET', urlT, false);
  4062. requestTnyim.send(null);
  4063. return shortUrl;
  4064. }
  4065. catch(err){
  4066. return shortUrl;
  4067. }
  4068. }
  4069. function googlShorten(url){ // Goo.gl URL Shortener
  4070. console.log("googlShorten function");
  4071. var shortUrl;
  4072. var urlG = "https://www.googleapis.com/urlshortener/v1/url";
  4073. var requestGoogl = new XMLHttpRequest();
  4074. try{
  4075. requestGoogl.open("POST", urlG, false);
  4076. requestGoogl.setRequestHeader("Content-Type", "application/json");
  4077. requestGoogl.onreadystatechange = function() {
  4078. if (requestGoogl.readyState == 4) {
  4079. if (requestGoogl.status == 200) {
  4080. shortUrl = JSON.parse(requestGoogl.response).id;
  4081. console.log("goo.gl response: " + requestGoogl.status + " " + requestGoogl.statusText + " " + JSON.parse(requestGoogl.response).id );
  4082. }
  4083. else {
  4084. console.log('goo.gl unsuccessful: ' + requestGoogl.status + " " + requestGoogl.statusText);
  4085. }
  4086. }
  4087. };
  4088. var data = new Object();
  4089. data.longUrl = url;
  4090. requestGoogl.send(JSON.stringify(data));
  4091. return shortUrl;
  4092. }
  4093. catch(err){
  4094. return shortUrl;
  4095. }
  4096. }
  4097. function shortenUrl(url){
  4098. sleep(500); // milliseconds delay - wait some milliseconds (currently half a second) between shortens to reduce chance of hitting usage limits
  4099. var shortUrl;
  4100. shortUrl = ns4tShorten(url);
  4101. if ( shortUrl === undefined ) { // if you reached the ns4t.net URL shortener's temporary usage limits or the server is otherwise unavailable
  4102. shortUrl = tnyimShorten(url);
  4103. if ( shortUrl === undefined ) { // if you reached the tny.im URL shortener's temporary limits or the server is otherwise unavailable
  4104. shortUrl = googlShorten(url);
  4105. if ( shortUrl === undefined ) { // if you reached the Google URL shortener's temporary limits too or the server is otherwise unavailable
  4106. shortUrl = "(x)";
  4107. }
  4108. }
  4109. }
  4110. return shortUrl;
  4111. }
  4112.  
  4113. // output display box
  4114. // this is messy
  4115. // TODO: make it cleaner
  4116. var exportdiv = document.createElement('div');
  4117. var exporttextarea1 = document.createElement('textarea'); var exporttextarea2 = document.createElement('textarea');
  4118. var exporttextarea3 = document.createElement('textarea'); var exporttextarea4 = document.createElement('textarea');
  4119. var exporttextarea5 = document.createElement('textarea');
  4120. var exportclosebutton = document.createElement('button');var exporttitlebutton = document.createElement('button');var exporturlbutton = document.createElement('button');
  4121. var exportreqbutton = document.createElement('button');var exportpandabutton = document.createElement('button');var exporttobutton = document.createElement('button');
  4122. exportclosebutton.style.backgroundColor = 'black';
  4123. exportclosebutton.style.color = 'white';
  4124. //exportclosebutton.style.border = 'none';
  4125. exportclosebutton.style.width = '505px';
  4126. exportclosebutton.style.align = 'center';
  4127. exportclosebutton.textContent='Close';
  4128. exporttitlebutton.style.backgroundColor = 'black';
  4129. exporttitlebutton.style.color = 'white';
  4130. //exporttitlebutton.style.border = 'none';
  4131. exporttitlebutton.style.width = '101px';
  4132. exporttitlebutton.style.align = 'center';
  4133. exporttitlebutton.textContent='Title';
  4134. exporturlbutton.style.backgroundColor = 'black';
  4135. exporturlbutton.style.color = 'white';
  4136. //exporturlbutton.style.border = 'none';
  4137. exporturlbutton.style.width = '101px';
  4138. exporturlbutton.style.align = 'center';
  4139. exporturlbutton.textContent='URL';
  4140. exportreqbutton.style.backgroundColor = 'black';
  4141. exportreqbutton.style.color = 'white';
  4142. //exportreqbutton.style.border = 'none';
  4143. exportreqbutton.style.width = '101px';
  4144. exportreqbutton.style.align = 'center';
  4145. exportreqbutton.textContent='Req';
  4146. exportpandabutton.style.backgroundColor = 'black';
  4147. exportpandabutton.style.color = 'white';
  4148. //exportpandabutton.style.border = 'none';
  4149. exportpandabutton.style.width = '101px';
  4150. exportpandabutton.style.align = 'center';
  4151. exportpandabutton.textContent='PandA';
  4152. exporttobutton.style.backgroundColor = 'black';
  4153. exporttobutton.style.color = 'white';
  4154. //exporttobutton.style.border = 'none';
  4155. exporttobutton.style.width = '101px';
  4156. exporttobutton.style.align = 'center';
  4157. exporttobutton.textContent='TO';
  4158. exportdiv.style.position = 'fixed';
  4159. exportdiv.style.width = '505px';
  4160. exportdiv.style.height = '155px';
  4161. exportdiv.style.left = '50%';
  4162. exportdiv.style.right = '50%';
  4163. exportdiv.style.margin = '-250px 0px 0px -250px';
  4164. exportdiv.style.top = '300px';
  4165. exportdiv.style.padding = 'none'; // def 5px
  4166. exportdiv.style.border = 'none'; //def 2px
  4167. exportdiv.style.backgroundColor = 'black';
  4168. exportdiv.style.color = 'white';
  4169. exportdiv.style.zIndex = '100';
  4170. exportdiv.style.display = 'none';
  4171. document.body.insertBefore(exportdiv, document.body.firstChild);
  4172. exporttextarea1.style.padding = 'none';
  4173. exporttextarea1.style.width = '500px';
  4174. exporttextarea2.style.padding = 'none';
  4175. exporttextarea2.style.width = '500px';
  4176. exporttextarea2.style.height = '30px';
  4177. exporttextarea2.style.overflow = 'hidden';
  4178. exporttextarea2.setAttribute('id','hwtf_url');
  4179. exporttextarea3.style.padding = 'none';
  4180. exporttextarea3.style.width = '500px';
  4181. exporttextarea3.style.height = '30px';
  4182. exporttextarea3.style.overflow = 'hidden';
  4183. exporttextarea3.setAttribute('id','hwtf_req');
  4184. exporttextarea4.style.padding = 'none';
  4185. exporttextarea4.style.width = '500px';
  4186. exporttextarea4.style.height = '30px';
  4187. exporttextarea4.style.overflow = 'hidden';
  4188. exporttextarea4.setAttribute('id','hwtf_panda');
  4189. exporttextarea5.style.padding = '2px';
  4190. exporttextarea5.style.width = '500px';
  4191. exporttextarea5.style.height = '20px';
  4192. exporttextarea5.style.overflow = 'hidden';
  4193. exporttextarea5.setAttribute('id','hwtf_to');
  4194. exportdiv.style.fontSize = '12px';
  4195. //exportdiv.style.display = 'block';
  4196.  
  4197. function display(hit, extype){
  4198.  
  4199. var thisReqName = hit["requester_strip"];
  4200. var thisReqId = "unavailable";
  4201. if ( accountStatus == "loggedIn" )
  4202. {
  4203. thisReqId = hit["rid"];
  4204. }
  4205.  
  4206. var thisTitle = hit["title"].replace(" (Requester link substituted)","");
  4207.  
  4208. var thisReward = hit["reward"];
  4209.  
  4210. var thisTimeLimit = hit["time"];
  4211.  
  4212. var thisHitsAvail = "??";
  4213. if ( accountStatus == "loggedIn" )
  4214. {
  4215. thisHitsAvail = hit["hits"];
  4216. }
  4217.  
  4218. var qualList = hit["quals"];
  4219. var qualColl = qualList.split(';');
  4220. var masterQual = '';
  4221. var locationStat = 'ICA';
  4222. var locationIndex = null;
  4223. for ( var m = 0; m < qualColl.length; m++ ) {
  4224. if ( qualColl[m].indexOf('Masters') > -1 ) {
  4225. masterQual = (extype == "irc") ? 'MASTERS • ' : ' [MASTERS]';
  4226. }
  4227. else if (qualColl[m].indexOf('is US') > -1) {
  4228. locationStat = ' US';
  4229. locationIndex = m;
  4230. }
  4231. else if (qualColl[m].match(/approval rate/)) {
  4232. qualColl[m] = qualColl[m].replace(/[A-Za-z ]+\(%\) is (?:not less than (\d+)|greater than (\d+))/, '>$1$2%');
  4233. }
  4234. else if (qualColl[m].match(/approved HITs/)) {
  4235. qualColl[m] = qualColl[m].replace(/Total approved HITs is (?:not less than (\d+)|greater than (\d+))/, '>$1$2');
  4236. qualColl[m] = qualColl[m].replace(/Total approved HITs is (?:not greater than (\d+)|less than (\d+))/, '<$1$2');
  4237. }
  4238. }
  4239.  
  4240. if (extype == "reddit" && locationIndex != null) qualColl.splice(locationIndex,1);
  4241. if (qualColl[0] == null) qualColl = "None";
  4242. var urlsToShorten = [];
  4243. var prevUrl = (thisReqId == "unavailable") ? "Unavailable due to logged out scraper" : LINK_BASE+hit["prev_link"];
  4244. var accUrl = (thisReqId == "unavailable") ? "Unavailable due to logged out scraper" : LINK_BASE+hit["acc_link"];
  4245. if (thisReqId != "unavailable")
  4246. {
  4247. urlsToShorten.push(prevUrl);
  4248. urlsToShorten.push(accUrl);
  4249. }
  4250. var thisPreviewUrl = prevUrl;
  4251. var thisPandaUrl = accUrl;
  4252. var thisReqUrl = "(url n/a)";
  4253. if ( thisReqId != "unavailable")
  4254. {
  4255. if (extype == "irc"){
  4256. urlsToShorten.push('https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=' + thisReqId);
  4257. //thisReqUrl = shortenUrl('https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=' + thisReqId);
  4258. }
  4259. else
  4260. thisReqUrl = 'https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=' + thisReqId;
  4261. }
  4262. else if ( thisReqId == "unavailable") // handle 2015-07-20 loss of logged-out requester ids
  4263. {
  4264. if (extype == "irc"){
  4265. urlsToShorten.push('https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=' + thisReqName.replace(/ /g, "+"));
  4266. //thisReqUrl = shortenUrl('https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=' + thisReqName.replace(/ /g, "+") ) + " (search)";
  4267. }
  4268. else
  4269. thisReqUrl = 'https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=' + thisReqName.replace(/ /g, "+") + " (Requester search URL due to logged out scraper)";
  4270. }
  4271. var hitLinkUnav = '';
  4272. //if ( capGId == 'unavailable' ) { capUrl = capReqUrl; pandaUrl = ""; hitLinkUnav = " (preview link unavailable)"; } // handle logged-out export requests for HITs with no preview/notqualified links ** This is handled at line 647
  4273. if (hit["prev_link"].indexOf("requesterId") > -1)
  4274. {
  4275. hitLinkUnav = " (preview link unavailable)";
  4276. thisPandaUrl = "";
  4277. }
  4278. var thisTOUrl = "(url n/a)";
  4279. var thisTOStats = "??";
  4280. if ( thisReqId != "unavailable" )
  4281. {
  4282. urlsToShorten.push(TO_BASE+thisReqId);
  4283. thisTOUrl = TO_BASE+thisReqId;
  4284. thisTOStats = getTO(thisReqId);
  4285. }
  4286. else if ( thisReqId == "unavailable") // handle 2015-07-20 loss of logged-out requester ids
  4287. {
  4288. urlsToShorten.push('https://turkopticon.ucsd.edu/main/php_search?query=' + thisReqName.replace(/ /g, "+"));
  4289. thisTOUrl = 'https://turkopticon.ucsd.edu/main/php_search?query=' + thisReqName.replace(/ /g, "+") +" (TO search URL due to logged out scraper)";
  4290. }
  4291. var shortUrlUnav = '';
  4292. if ( (thisPreviewUrl == "(x)") && (thisHitGroup != "unavailable") )
  4293. {
  4294. shortUrlUnav = " \r\n^ https://www.mturk.com/mturk/preview?groupId=" + thisHitGroup;
  4295. }
  4296. if (extype == "irc")
  4297. {
  4298. if (urlsToShorten.length == 4)
  4299. {
  4300. var shortUrls = ns4tbulkshorten(urlsToShorten);
  4301. console.log(shortUrls);
  4302. thisPreviewUrl = shortUrls["preview"];
  4303. thisPandaUrl = shortUrls["panda"];
  4304. thisReqUrl = shortUrls["req"];
  4305. thisTOUrl = shortUrls["to"];
  4306. if ( thisReqId == "unavailable" ){
  4307. thisTOUrl += " (search)";
  4308. thisReqUrl += " (search)";
  4309. }
  4310. }
  4311. else
  4312. {
  4313. if ( thisReqId != "unavailable" ) // handle logged-out export requests for HITs with no preview/notqualified links
  4314. {
  4315. thisPreviewUrl = shortenUrl('https://www.mturk.com/mturk/preview?groupId=' + thisHitGroup);
  4316. thisPandaUrl = shortenUrl('https://www.mturk.com/mturk/previewandaccept?groupId=' + thisHitGroup);
  4317. thisReqUrl = shortenUrl('https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId=' + thisReqId);
  4318. thisTOUrl = shortenUrl('http://turkopticon.ucsd.edu/' + thisReqId);
  4319. }
  4320. else if ( thisReqId == "unavailable" ) // handle 2015-07-20 loss of logged-out requester ids
  4321. {
  4322. thisReqUrl = shortenUrl('https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&searchWords=' + thisReqName.replace(/ /g, "+") ) + " (search)";
  4323. thisTOUrl = shortenUrl('https://turkopticon.ucsd.edu/main/php_search?query=' + thisReqName.replace(/ /g, "+") ) + " (search)";
  4324. }
  4325. }
  4326. }
  4327. var exportOutput = "";
  4328. var loggedOutApology = " (Info missing since logged out.)";
  4329. var temp = [];
  4330. for (var i = 0; i < qualColl.length; i++)
  4331. {
  4332. var search = /b\](.*?)\[/.exec(qualColl[i]);
  4333. if (search)
  4334. temp[i] = search[1].trim();
  4335. else
  4336. temp[i] = qualColl[i].trim();
  4337. }
  4338. if (temp[0] != "None")
  4339. var qualString = temp.join(", ");
  4340. else
  4341. qualString = temp;
  4342. var exTitle = locationStat + ' - ' + thisTitle.replace(" (Preview link unavailable)","") + ' - ' + thisReqName + ' - ' + thisReward + '/' + 'COMTIME - ' + '(' + qualString + ')' + masterQual;
  4343. var exUrl = 'URL: ' + thisPreviewUrl +'\n\n';
  4344. var exReq = 'Req: ' + thisReqUrl +'\n\n';
  4345. var exTO = 'TO: ' + thisTOUrl +'\n\n';
  4346. var exPanda = 'PandA: ' + thisPandaUrl +'\n\n';
  4347. if ( accountStatus == "loggedIn" )
  4348. {
  4349. if (extype == "irc") {
  4350. exportOutput = masterQual + 'Requester: ' + thisReqName + ' ' + thisReqUrl + ' • ' + 'HIT: ' + thisTitle + ' ' + thisPreviewUrl + ' • ' + 'Pay: ' + thisReward + ' • ' + 'Avail: ' + thisHitsAvail + ' • ' + 'Limit: ' + thisTimeLimit + ' • ' + 'TO: ' + 'Pay='+thisTOStats[1] + ' Fair='+thisTOStats[2] + ' Comm='+thisTOStats[0] + ' ' + thisTOUrl + ' • ' + 'PandA: ' + thisPandaUrl + shortUrlUnav ;
  4351. exporttextarea1.value = exportOutput;
  4352. exporttextarea1.style.height = '130px';
  4353. exporttextarea1.setAttribute('id', 'ircexport_text');
  4354. exportdiv.textContent = 'IRC Export: Press Ctrl+C to (re-)copy to clipboard. Click textarea to close.';
  4355. exportdiv.appendChild(exporttextarea1);
  4356. exporttextarea1.addEventListener("click", function(){ exportdiv.style.display = 'none'; }, false);
  4357. exportdiv.style.display = 'block';
  4358. exporttextarea1.select();
  4359. if (GM_setClipboard) { GM_setClipboard(exportOutput); }
  4360. }
  4361. else if (extype == "reddit") {
  4362. exporttextarea1.value = exTitle;
  4363. exporttextarea2.value = exUrl;
  4364. exporttextarea3.value = exReq;
  4365. exporttextarea4.value = exPanda;
  4366. exporttextarea5.value = exTO;
  4367. exporttextarea1.style.height = '50px';
  4368. exporttextarea1.setAttribute('id','hwtf_title');
  4369. exportdiv.textContent = 'r/HitsWorthTurkingFor Export: Use the buttons for single-click copying.';
  4370. exportdiv.appendChild(exporttextarea1);
  4371. exportdiv.appendChild(exporttextarea2);
  4372. exportdiv.appendChild(exporttextarea3);
  4373. exportdiv.appendChild(exporttextarea4);
  4374. exportdiv.appendChild(exporttextarea5);
  4375. exportdiv.appendChild(exporttitlebutton);
  4376. exportdiv.appendChild(exporturlbutton);
  4377. exportdiv.appendChild(exportreqbutton);
  4378. exportdiv.appendChild(exportpandabutton);
  4379. exportdiv.appendChild(exporttobutton);
  4380. exportdiv.appendChild(exportclosebutton);
  4381. document.body.insertBefore(exportdiv, document.body.firstChild);
  4382. exportclosebutton.addEventListener("click", function(){ exportdiv.style.display = 'none'; }, false);
  4383. exporttitlebutton.addEventListener("click", function(){ exporttextarea1.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea1.value); } }, false);
  4384. exporturlbutton.addEventListener("click", function(){ exporttextarea2.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea2.value); } }, false);
  4385. exportreqbutton.addEventListener("click", function(){ exporttextarea3.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea3.value); } }, false);
  4386. exportpandabutton.addEventListener("click", function(){ exporttextarea4.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea4.value); } }, false);
  4387. exporttobutton.addEventListener("click", function(){ exporttextarea5.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea5.value); } }, false);
  4388. exportdiv.style.display = 'block';
  4389. //exporttextarea1.focus();
  4390. var tindex = exTitle.search(/COMTIME/);
  4391. exporttextarea1.setSelectionRange(tindex, tindex+7); // pre-select completion time
  4392. }
  4393. }
  4394. else if ( accountStatus == "loggedOut" )
  4395. {
  4396. if (extype == "irc") {
  4397. exportOutput = masterQual + 'Requester: ' + thisReqName + ' ' + thisReqUrl + ' • ' + 'HIT: ' + thisTitle + ' ' + thisPreviewUrl + ' • ' + 'Pay: ' + thisReward + ' • ' + 'Avail: ' + thisHitsAvail + ' • ' + 'Limit: ' + thisTimeLimit + ' • ' + 'TO: ?? ' + thisTOUrl + ' • ' + 'PandA: ' + thisPandaUrl + loggedOutApology + shortUrlUnav ;
  4398. exporttextarea1.value = exportOutput;
  4399. exporttextarea1.style.height = '130px';
  4400. exporttextarea1.setAttribute('id', 'ircexport_text');
  4401. exportdiv.textContent = 'IRC Export: Press Ctrl+C to (re-)copy to clipboard. Click textarea to close.';
  4402. exportdiv.appendChild(exporttextarea1);
  4403. exporttextarea1.addEventListener("click", function(){ exportdiv.style.display = 'none'; }, false);
  4404. exportdiv.style.display = 'block';
  4405. exporttextarea1.select();
  4406. if (GM_setClipboard) { GM_setClipboard(exportOutput); }
  4407. }
  4408. else if (extype == "reddit") {
  4409. exporttextarea1.value = exTitle;
  4410. exporttextarea2.value = exUrl;
  4411. exporttextarea3.value = exReq;
  4412. exporttextarea4.value = exPanda;
  4413. exporttextarea5.value = exTO;
  4414. exporttextarea1.style.height = '50px';
  4415. exporttextarea1.setAttribute('id','hwtf_title');
  4416. exportdiv.textContent = 'r/HitsWorthTurkingFor Export: Use the buttons for single-click copying.';
  4417. exportdiv.appendChild(exporttextarea1);
  4418. exportdiv.appendChild(exporttextarea2);
  4419. exportdiv.appendChild(exporttextarea3);
  4420. exportdiv.appendChild(exporttextarea4);
  4421. exportdiv.appendChild(exporttextarea5);
  4422. exportdiv.appendChild(exporttitlebutton);
  4423. exportdiv.appendChild(exporturlbutton);
  4424. exportdiv.appendChild(exportreqbutton);
  4425. exportdiv.appendChild(exportpandabutton);
  4426. exportdiv.appendChild(exporttobutton);
  4427. exportdiv.appendChild(exportclosebutton);
  4428. document.body.insertBefore(exportdiv, document.body.firstChild);
  4429. exportclosebutton.addEventListener("click", function(){ exportdiv.style.display = 'none'; }, false);
  4430. exporttitlebutton.addEventListener("click", function(){ exporttextarea1.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea1.value); } }, false);
  4431. exporturlbutton.addEventListener("click", function(){ exporttextarea2.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea2.value); } }, false);
  4432. exportreqbutton.addEventListener("click", function(){ exporttextarea3.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea3.value); } }, false);
  4433. exportpandabutton.addEventListener("click", function(){ exporttextarea4.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea4.value); } }, false);
  4434. exporttobutton.addEventListener("click", function(){ exporttextarea5.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea5.value); } }, false);
  4435. exportdiv.style.display = 'block';
  4436. //exporttextarea1.focus();
  4437. var tindex = exTitle.search(/COMTIME/);
  4438. exporttextarea1.setSelectionRange(tindex, tindex+7); // pre-select completion time
  4439. }
  4440. }
  4441. }
  4442.  
  4443. // TODO: cleanup, shift OO, readability, modularity