HIT Scraper WITH AWESOME

Snag HITs.

目前为 2015-06-25 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name HIT Scraper WITH AWESOME
  3. // @author Kerek, TJ, clickhappier, feiht
  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?match=true#hit_scraper*
  8. // @match https://www.mturk.com/mturk/findhits?match=true?hit_scraper*
  9. // @version 3.0.1
  10. // @grant GM_xmlhttpRequest
  11. // @grant GM_setClipboard
  12. // @require http://code.jquery.com/jquery-latest.min.js
  13. // ==/UserScript==
  14.  
  15. //Fixes it so as not to rely on GM's storage schemes
  16. this.GM_getValue=function (key,def) {
  17. return localStorage[key] || def;
  18. };
  19. this.GM_setValue=function (key,value) {
  20. return localStorage[key]=value;
  21. };
  22. this.GM_deleteValue=function (key) {
  23. return localStorage.removeItem(key);
  24. };
  25.  
  26. var default_list = ["oscar smith", "Diamond Tip Research LLC", "jonathon weber", "jerry torres", "Crowdsource", "we-pay-you-fast", "turk experiment", "jon brelig"];
  27. var ignore_list = default_list;
  28. if (GM_getValue("scraper_ignore_list"))
  29. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  30. else
  31. GM_setValue("scraper_ignore_list", default_list.join('^'));
  32.  
  33. var include_list = [];
  34. if (!GM_getValue("scraper_include_list"))
  35. GM_setValue("scraper_include_list","nothing includelisted yet");
  36. if (GM_getValue("scraper_include_list"))
  37. include_list = GM_getValue("scraper_include_list").split('^');
  38.  
  39. //This is to update the hit export symbol
  40. var symbol = "☭";
  41.  
  42. //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.
  43. var correct_for_skips = true;
  44.  
  45. //This is to test sorting by TO
  46. var sort_TO = false; // by pay
  47. var sort_TO2 = false; // by overall user-weighted ratings
  48.  
  49. // this fixes mismatched names due to Turkopticon using an older value after the requester changed it on mturk.
  50. var fix_names = true;
  51.  
  52. //weight the four TO ratings for the coloring / sorting / ranking
  53. var COMM_WEIGHT = 1;
  54. var PAY_WEIGHT = 6;
  55. var FAIR_WEIGHT = 3.5;
  56. var FAST_WEIGHT = 1;
  57.  
  58. //Used for theming, change the colors to change how scraper looks
  59. var HIGHLIGHT_COLOR = '#1F3847'; // highlighting control panel selections
  60. var HOVER_COLOR = '#426075'; // control panel options' mouseover color
  61. var BACKGROUND_COLOR = '#434e56'; //Background of page
  62. var ACCENT_COLOR = '#fbde2d'; // used for spacers and main control buttons
  63. var CP_BORDER_COLOR = '#131313'; // lower bound border for the control panel
  64. var CP_BG_COLOR = '#384147'; // background color of the control panel
  65. var EXPORT_COLOR = '#ADC6EE'; // color for export buttons
  66. var HIGH_TO_COLOR = '#6FFA3C';
  67. var GOOD_TO_COLOR = '#D9FC35';
  68. var AVERAGE_TO_COLOR = '#fbde2d';
  69. var LOW_TO_COLOR = '#FAB050';
  70. var POOR_TO_COLOR = '#FA6F50';
  71. var NO_TO_COLOR = '#697048';
  72. var HITDB_COLOR = '#d8fa3c'; // found in hitdb
  73. var NOHITDB_COLOR = '#DA95A8'; // not found in hitdb
  74. var UNQUALIFIED_COLOR = '#ADC6EE'; // No HITDB, "Not Qualified"
  75. var REQMASTER_CELLCOLOR = '#BFADEE'; // requires masters
  76. var DEFMASTER_CELLCOLOR = '#ADEEDF'; // doesn't require masters
  77. var DEFAULT_TEXT_COLOR = '#f8f8f8';
  78. var SECONDARY_TEXT_COLOR= '#ADC6EE';
  79. var INPUT_TEXT_COLOR = '#D8FA3C'; // for the input boxes
  80. var LINK_COLOR = '#BFADEE';
  81. var VISITED_LINK_COLOR = '#DCEEAD';
  82. var SPACER_TEXT = " | "; // spacing for the control panel
  83.  
  84. var colors = { "highlight" :HIGHLIGHT_COLOR, "background" :BACKGROUND_COLOR, "accent" :ACCENT_COLOR, "cpBorder" :CP_BORDER_COLOR,
  85. "cpBackground" :CP_BG_COLOR, "highTO" :HIGH_TO_COLOR, "goodTO" :GOOD_TO_COLOR, "averageTO" :AVERAGE_TO_COLOR,
  86. "lowTO" :LOW_TO_COLOR, "poorTO" :POOR_TO_COLOR, "hitDB" :HITDB_COLOR, "nohitDB" :NOHITDB_COLOR,
  87. "unqualified" :UNQUALIFIED_COLOR, "reqmaster" :REQMASTER_CELLCOLOR, "nomaster" :DEFMASTER_CELLCOLOR,"defaultText" :DEFAULT_TEXT_COLOR,
  88. "inputText" :INPUT_TEXT_COLOR, "secondText" :SECONDARY_TEXT_COLOR,"link" :LINK_COLOR, "vlink" :VISITED_LINK_COLOR,
  89. "noTO" :NO_TO_COLOR, "export" :EXPORT_COLOR, "hover" :HOVER_COLOR
  90. };
  91. var rcolors = {};
  92.  
  93. // Random themes are still in testing stages! It can, on occasion, cause minor lag on initial page load.
  94. // NOTE: The issue is on the inital page initialization; it does not affect searching/scraping.
  95. // Comment out the next line to disable randomization.
  96. randomizeScheme();
  97.  
  98. //display your hitdb records if applicable
  99. var check_hitDB = true;
  100.  
  101. //default text size
  102. var default_text_size=11;
  103.  
  104. //set to "true" to override checkbox setting and ding on new hits
  105. var newHitDing = false;
  106.  
  107. //DO NOT EDIT ANYTHING BELOW THIS LINE UNLESS YOU KNOW WHAT YOU ARE DOING!
  108.  
  109.  
  110. var maxPages = 20;
  111. var type = -1;
  112. var status_array = [];
  113. var shouldDing = false;
  114. var useTO = true;
  115.  
  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,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VVVVVVVVVVVVVQBqs0jMnBWmgKng1FRU32lmZmb+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. //console.log(dingNoise);
  131. if (dingNoise || newHitDing)
  132. document.getElementById("ding_noise"+audio_index).play();
  133. }
  134.  
  135.  
  136. //$("head").append('<style>tr.scraperBlockedRow td { border: 3pt solid #cc0000; }</style>');
  137. //$("head").append('<style>tr.scraperIncludelistedRow td { border: 3pt dashed #008800; }</style>');
  138.  
  139. //For editing the blocklist
  140. var div1 = document.createElement('div');
  141. var textarea1 = document.createElement('textarea');
  142.  
  143. div1.style.position = 'fixed';
  144. div1.style.width = '500px';
  145. div1.style.height = '235px';
  146. div1.style.left = '50%';
  147. div1.style.right = '50%';
  148. div1.style.margin = '-250px 0px 0px -250px';
  149. div1.style.top = '300px';
  150. div1.style.padding = '5px';
  151. div1.style.border = '2px';
  152. div1.style.backgroundColor = 'black';
  153. div1.style.color = 'white';
  154. div1.style.zIndex = '100';
  155. div1.setAttribute('id','block_div');
  156.  
  157. textarea1.style.padding = '2px';
  158. textarea1.style.width = '500px';
  159. textarea1.style.height = '180px';
  160. textarea1.title = 'Block list';
  161. textarea1.setAttribute('id','block_text');
  162.  
  163. 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.';
  164. div1.style.fontSize = '12px';
  165. div1.appendChild(textarea1);
  166.  
  167. var save_button1 = document.createElement('button');
  168. var cancel_button1 = document.createElement('button');
  169.  
  170. save_button1.textContent = 'Save';
  171. save_button1.setAttribute('id', 'save_blocklist');
  172. save_button1.style.height = '18px';
  173. save_button1.style.width = '100px';
  174. save_button1.style.fontSize = '10px';
  175. save_button1.style.paddingLeft = '3px';
  176. save_button1.style.paddingRight = '3px';
  177. save_button1.style.backgroundColor = 'white';
  178. save_button1.style.marginLeft = '5px';
  179.  
  180. cancel_button1.textContent = 'Cancel';
  181. cancel_button1.setAttribute('id', 'cancel_blocklist');
  182. cancel_button1.style.height = '18px';
  183. cancel_button1.style.width = '100px';
  184. cancel_button1.style.fontSize = '10px';
  185. cancel_button1.style.paddingLeft = '3px';
  186. cancel_button1.style.paddingRight = '3px';
  187. cancel_button1.style.backgroundColor = 'white';
  188. cancel_button1.style.marginLeft = '5px';
  189.  
  190. div1.appendChild(save_button1);
  191. div1.appendChild(cancel_button1);
  192. document.body.insertBefore(div1, document.body.firstChild);
  193. $("#block_div").hide();
  194.  
  195. // save and cancel for blocklist
  196. function save_blocklist() {
  197. //console.log("Save");
  198. var textarea = $("#block_text");
  199. var text = textarea.val();
  200. var block_list = text.split("^");
  201. var trimmed_list = [];
  202. for (var requester in block_list){
  203. if (block_list[requester].trim().length !== 0)
  204. trimmed_list.push(block_list[requester].toLowerCase().trim());
  205. }
  206. //console.log(trimmed_list);
  207. GM_setValue("scraper_ignore_list",trimmed_list.join('^'));
  208. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  209. //console.log("Save complete: ");
  210. //console.log(ignore_list);
  211. $("#block_div").hide();
  212. }
  213. save_button1.addEventListener("click", function(){ save_blocklist(); }, false);
  214. cancel_button1.addEventListener("click", function(){
  215. // reset textarea contents upon cancel
  216. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  217. var textarea = $("#block_text");
  218. var text = "";
  219. for (var i = 0; i < ignore_list.length; i++){
  220. text += ignore_list[i]+"^";
  221. }
  222. textarea.val(text.substring(0, text.length - 1));
  223. // close editor
  224. $("#block_div").hide();
  225. }, false);
  226.  
  227.  
  228. //For editing the include list
  229. var shouldInclude = false;
  230. var div2 = document.createElement('div');
  231. var textarea2 = document.createElement('textarea');
  232.  
  233. div2.style.position = 'fixed';
  234. div2.style.width = '500px';
  235. div2.style.height = '235px';
  236. div2.style.left = '50%';
  237. div2.style.right = '50%';
  238. div2.style.margin = '-250px 0px 0px -250px';
  239. div2.style.top = '300px';
  240. div2.style.padding = '5px';
  241. div2.style.border = '2px';
  242. div2.style.backgroundColor = 'black';
  243. div2.style.color = 'white';
  244. div2.style.zIndex = '100';
  245. div2.setAttribute('id','include_div');
  246.  
  247. textarea2.style.padding = '2px';
  248. textarea2.style.width = '500px';
  249. textarea2.style.height = '180px';
  250. textarea2.title = 'include list';
  251. textarea2.setAttribute('id','include_text');
  252.  
  253. 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.';
  254. div2.style.fontSize = '12px';
  255. div2.appendChild(textarea2);
  256.  
  257. var save_button2 = document.createElement('button');
  258. var cancel_button2 = document.createElement('button');
  259.  
  260. save_button2.textContent = 'Save';
  261. save_button2.setAttribute('id', 'save_includelist');
  262. save_button2.style.height = '18px';
  263. save_button2.style.width = '100px';
  264. save_button2.style.fontSize = '10px';
  265. save_button2.style.paddingLeft = '3px';
  266. save_button2.style.paddingRight = '3px';
  267. save_button2.style.backgroundColor = 'white';
  268. save_button2.style.marginLeft = '5px';
  269.  
  270. cancel_button2.textContent = 'Cancel';
  271. cancel_button2.setAttribute('id', 'cancel_includelist');
  272. cancel_button2.style.height = '18px';
  273. cancel_button2.style.width = '100px';
  274. cancel_button2.style.fontSize = '10px';
  275. cancel_button2.style.paddingLeft = '3px';
  276. cancel_button2.style.paddingRight = '3px';
  277. cancel_button2.style.backgroundColor = 'white';
  278. cancel_button2.style.marginLeft = '5px';
  279.  
  280. div2.appendChild(save_button2);
  281. div2.appendChild(cancel_button2);
  282. document.body.insertBefore(div2, document.body.firstChild);
  283. $("#include_div").hide();
  284.  
  285. // save and cancel for includelist
  286. function save_includelist() {
  287. //console.log("Save");
  288. var textarea = $("#include_text");
  289. var text = textarea.val();
  290. var includes = text.split("^");
  291. //console.log(includes);
  292. var trimmed_list = [];
  293. for (var requester in includes){
  294. if (includes[requester].trim().length !== 0)
  295. trimmed_list.push(includes[requester].toLowerCase().trim());
  296. }
  297. GM_setValue("scraper_include_list",trimmed_list.join('^'));
  298. include_list = GM_getValue("scraper_include_list").split('^');
  299. //console.log(include_list);
  300. $("#include_div").hide();
  301. }
  302. save_button2.addEventListener("click", function(){ save_includelist(); }, false);
  303. cancel_button2.addEventListener("click", function(){
  304. // reset textarea contents upon cancel
  305. include_list = GM_getValue("scraper_include_list").split('^');
  306. var textarea = $("#include_text");
  307. var text = "";
  308. for (var i = 0; i < include_list.length; i++){
  309. text += include_list[i]+"^";
  310. }
  311. textarea.val(text.substring(0, text.length - 1));
  312. // close editor
  313. $("#include_div").hide();
  314. }, false);
  315.  
  316.  
  317. var HITStorage = {};
  318. var indexedDB = window.indexedDB || window.webkitIndexedDB ||
  319. window.mozIndexedDB;
  320. window.IDBTransaction = window.IDBTransaction || window.webkitIDBTransaction || window.mozIDBTransaction;
  321. window.IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange || window.mozIDBKeyRange;
  322. HITStorage.IDBTransactionModes = { "READ_ONLY": "readonly", "READ_WRITE": "readwrite", "VERSION_CHANGE": "versionchange" };
  323. var IDBKeyRange = window.IDBKeyRange;
  324.  
  325. HITStorage.indexedDB = {};
  326. HITStorage.indexedDB = {};
  327. HITStorage.indexedDB.db = null;
  328.  
  329. HITStorage.indexedDB.onerror = function(e) {
  330. //console.log(e);
  331. };
  332.  
  333. var v=4;
  334.  
  335. HITStorage.indexedDB.checkTitle = function(title,button) {
  336. var request = indexedDB.open("HITDB", v);
  337. request.onsuccess = function(e) {
  338. HITStorage.indexedDB.db = e.target.result;
  339. var db = HITStorage.indexedDB.db;
  340. if (!db.objectStoreNames.contains("HIT"))
  341. {
  342. db.close();
  343. return;
  344. }
  345. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  346. var store = trans.objectStore("HIT");
  347.  
  348. var index = store.index("title");
  349. index.get(title).onsuccess = function(event)
  350. {
  351. if (event.target.result === undefined)
  352. {
  353. //console.log(title + ' not found');
  354. scraper_history[button].titledb = false;
  355. }
  356. else
  357. {
  358. //console.log(title + ' found');
  359. scraper_history[button].titledb = true;
  360. }
  361.  
  362. db.close();
  363. };
  364. };
  365. request.onerror = HITStorage.indexedDB.onerror;
  366. };
  367.  
  368. HITStorage.indexedDB.checkRequester = function(id,button) {
  369. var request = indexedDB.open("HITDB", v);
  370. request.onsuccess = function(e) {
  371. HITStorage.indexedDB.db = e.target.result;
  372. var db = HITStorage.indexedDB.db;
  373. if (!db.objectStoreNames.contains("HIT"))
  374. {
  375. db.close();
  376. return;
  377. }
  378. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  379. var store = trans.objectStore("HIT");
  380.  
  381. var index = store.index("requesterId");
  382. index.get(id).onsuccess = function(event)
  383. {
  384. if (event.target.result === undefined)
  385. {scraper_history[button].reqdb = false;
  386. //console.log(id + ' not found');
  387. }
  388. else
  389. {
  390. scraper_history[button].reqdb = true;
  391. //console.log(id + ' found');
  392. }
  393. db.close();
  394. };
  395. };
  396. request.onerror = HITStorage.indexedDB.onerror;
  397. };
  398.  
  399. var PAGES_TO_SCRAPE = 3;
  400. var MINIMUM_HITS = 100;
  401. var SEARCH_REFRESH=0;
  402. var MINIMUM_TO = -1;
  403. var URL_BASE = "/mturk/searchbar?searchWords=&selectedSearchType=hitgroups";
  404. var initial_url = URL_BASE;
  405. var TO_REQ_URL = "http://turkopticon.ucsd.edu/reports?id=";
  406. var found_key_list=[];
  407. var last_clear_time = new Date().getTime();
  408. var searched_once = false;
  409. var useBlocklist = true;
  410. var save_new_results_time = 120;
  411. var save_results_time = 3600;
  412. var default_type = 0;
  413. var block_no_to = false;
  414. var audio_index = 0;
  415. var cur_loc = window.location.href;
  416. var time_input = document.createElement("INPUT");
  417. time_input.value = 0;
  418. var page_input = document.createElement("INPUT");
  419. page_input.value = 3;
  420. var min_input = document.createElement("INPUT");
  421. var new_time_display_input = document.createElement("INPUT");
  422. new_time_display_input.value = 300;
  423. var reward_input = document.createElement("INPUT");
  424. var qual_input = document.createElement("INPUT");
  425. qual_input.type = "checkbox";
  426. var masters_input = document.createElement("INPUT");
  427. masters_input.type = "checkbox";
  428. var masters_hide = document.createElement("INPUT");
  429. masters_hide.type = "checkbox";
  430. var sort_input = document.createElement("SELECT");
  431. var sort_input1 = document.createElement("OPTION");
  432. sort_input1.text = "Latest";
  433. sort_input1.value = "late";
  434. var sort_input2 = document.createElement("OPTION");
  435. sort_input2.text = "Most Available";
  436. sort_input2.value = "most";
  437. var sort_input3 = document.createElement("OPTION");
  438. sort_input3.text = "Reward";
  439. sort_input3.value = "amount";
  440. var sort_input4 = document.createElement("OPTION");
  441. sort_input4.text = "Title";
  442. sort_input4.value = "alpha";
  443. sort_input.appendChild(sort_input1);
  444. sort_input.appendChild(sort_input2);
  445. sort_input.appendChild(sort_input3);
  446. sort_input.appendChild(sort_input4);
  447. var sort_input_invert = document.createElement("INPUT");
  448. sort_input_invert.type = "checkbox";
  449. var sort_to = document.createElement("INPUT");
  450. sort_to.type = "checkbox";
  451. var sort_to2 = document.createElement("INPUT");
  452. sort_to2.type = "checkbox";
  453. var sort_asc = document.createElement("INPUT");
  454. sort_asc.type = "radio";
  455. sort_asc.name = "sortOrder";
  456. var sort_dsc = document.createElement("INPUT");
  457. sort_dsc.type = "radio";
  458. sort_dsc.name = "sortOrder";
  459. sort_dsc.checked = true;
  460. var min_to = document.createElement("INPUT");
  461. var no_to_block = document.createElement("INPUT");
  462. no_to_block.type = "checkbox";
  463. var useTO_input = document.createElement("INPUT");
  464. useTO_input.type = "checkbox";
  465. var friesAreDone = document.createElement("INPUT");
  466. friesAreDone.type = "checkbox";
  467. var audio_option = document.createElement("SELECT");
  468. var audio1 = document.createElement("OPTION");
  469. var audio2 = document.createElement("OPTION");
  470. audio1.text = "Ding";
  471. audio1.value = 0;
  472. audio2.text = "Squee";
  473. audio2.value = 1;
  474. audio_option.appendChild(audio1);
  475. audio_option.appendChild(audio2);
  476. var correctForSkips = document.createElement("INPUT");
  477. correctForSkips.type = "checkbox";
  478. correctForSkips.checked = true;
  479. var useBlock = document.createElement("INPUT");
  480. useBlock.type = "checkbox";
  481. useBlock.checked = true;
  482. var matchOnly = document.createElement("INPUT");
  483. matchOnly.type = "checkbox";
  484. var highlightIncludes_input = document.createElement("INPUT");
  485. highlightIncludes_input.type = "checkbox";
  486. var opt_exportvb = document.createElement("INPUT");
  487. opt_exportvb.type = "checkbox";
  488. var opt_exportirc = document.createElement("INPUT");
  489. opt_exportirc.type = "checkbox";
  490. var opt_exportreddit = document.createElement("INPUT");
  491. opt_exportreddit.type = "checkbox";
  492. var sufTheme = "_thCustom";
  493.  
  494. var search_input = document.createElement("INPUT");
  495.  
  496. //In order to recover from the last state, let's see if it exists
  497. if (localStorage["lastState_hit_scraper_with_export"])
  498. {
  499. var lastStateObj = JSON.parse(localStorage["lastState_hit_scraper_with_export"]);
  500. time_input.value = lastStateObj["refreshTime"];
  501. page_input.value = lastStateObj["numPages"];
  502. min_input.value = lastStateObj["minHits"];
  503. new_time_display_input.value = lastStateObj["newHitHighlight"];
  504. reward_input.value = lastStateObj["reward"];
  505. qual_input.checked = lastStateObj["qualified"];
  506. masters_input.checked = lastStateObj["masters"];
  507. masters_hide.checked = lastStateObj["mShow"];
  508. sort_input.selectedIndex = lastStateObj["sort"];
  509. sort_input_invert.checked = lastStateObj["invert"];
  510. sort_to.checked = lastStateObj["to"];
  511. sort_to2.checked = lastStateObj["to2"];
  512. sort_asc.checked = lastStateObj["asc"];
  513. sort_dsc.checked = lastStateObj["dsc"];
  514. min_to.value = lastStateObj["minTO"];
  515. no_to_block.checked = lastStateObj["hideNTO"];
  516. search_input.value = lastStateObj["searchTerms"];
  517. matchOnly.checked = lastStateObj["useInclude"];
  518. friesAreDone.checked = lastStateObj["fries"];
  519. audio_option.selectedIndex = lastStateObj["whichfry"];
  520. useBlock.checked = lastStateObj["blocklist"];
  521. correctForSkips.checked = lastStateObj["skips"];
  522. useTO_input.checked = lastStateObj["useTO"];
  523. highlightIncludes_input.checked = lastStateObj["highlightIncl"];
  524. opt_exportvb.checked = lastStateObj["exvb"];
  525. opt_exportirc.checked = lastStateObj["exirc"];
  526. opt_exportreddit.checked = lastStateObj["exreddit"];
  527. sufTheme = (lastStateObj["theme"] != null) ? lastStateObj["theme"] : "_thCustom";
  528. }
  529. else
  530. {
  531. time_input.value = 0;
  532. page_input.value = 3;
  533. min_input.value = "";
  534. new_time_display_input.value = 300;
  535. reward_input.value = "";
  536. if (!document.getElementById('lnkWorkerSignin'))
  537. qual_input.checked = true;
  538. masters_input.checked = false;
  539. masters_hide.checked = true;
  540. sort_input.selectedIndex = 0;
  541. sort_input_invert.checked = false;
  542. sort_to.checked = false;
  543. sort_to2.checked = false;
  544. //sort_asc.checked = false;
  545. sort_dsc.checked = true;
  546. min_to.value = "";
  547. no_to_block.checked = false;
  548. search_input.value = "";
  549. matchOnly.checked = false;
  550. friesAreDone.checked = false;
  551. audio_option.selectedIndex = 0;
  552. useBlock.checked = true;
  553. correctForSkips.checked = true;
  554. highlightIncludes_input.checked = false;
  555. opt_exportvb.checked = true;
  556. opt_exportirc.checked = true;
  557. opt_exportreddit.checked = true;
  558. }
  559.  
  560. var LINK_BASE = "https://www.mturk.com";
  561. var STATUSDETAIL_DELAY = 250;
  562. var MPRE_DELAY = 3000;
  563.  
  564. var css = "<style type = 'text/css'>\n";
  565. css += ".body_thCustom {background-color: "+colors.background+";}\n";
  566. css += ".controlpanel_thCustom {color: "+colors.defaultText+"; margin: 0px auto 5px auto; border-bottom: 1px solid "+colors.cpBorder+"; background: "+colors.cpBackground+";}\n";
  567. css += ".cpInput_thCustom {color: "+colors.inputText+"; border: 1px solid; background: "+colors.background+"; text-align: center;}\n";
  568. css += ".cpButtons_thCustom {color: "+colors.accent+"; border: 1px solid; background: transparent;}\n";
  569. css += ".cpSortdiv_thCustom {color: "+colors.inputText+"; display: inline; font: 14px;}\n";
  570. css += ".cpSpans_thCustom {padding: 2px; cursor:default; color: "+colors.defaultText+"; background: "+colors.cpBackground+";} .cpSpans_thCustom:hover {background: "+colors.hover+";}\n";
  571. css += ".cpSpansOn_thCustom {background: "+colors.highlight+"; color: "+colors.secondText+";}\n";
  572. css += ".taButtons_thCustom {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+colors.export+";} .taButtonsOff_thCustom {display: none;}\n";
  573. css += ".link_thCustom:link {color: "+colors.link+";} .link_thCustom:visited {color: "+colors.vlink+";}\n";
  574. css += ".toNone_thCustom .link_thCustom, .toNone_thCustom .tolink_thCustom {color: "+colors.defaultText+";}\n .toNone_thCustom .link_thCustom:visited {color: "+colors.vlink+";}\n";
  575. css += ".toHigh_thCustom .link_thCustom, .toHigh_thCustom .tolink_thCustom {color: "+colors.highTO+";}\n .toHigh_thCustom .link_thCustom:visited {color: "+colors.vlink+";}\n";
  576. css += ".toGood_thCustom .link_thCustom, .toGood_thCustom .tolink_thCustom {color: "+colors.goodTO+";}\n .toGood_thCustom .link_thCustom:visited {color: "+colors.vlink+";}\n";
  577. css += ".toAverage_thCustom .link_thCustom, .toAverage_thCustom .tolink_thCustom {color: "+colors.averageTO+";}\n .toAverage_thCustom .link_thCustom:visited {color: "+colors.vlink+";}\n";
  578. css += ".toLow_thCustom .link_thCustom, .toLow_thCustom .tolink_thCustom {color: "+colors.lowTO+";}\n .toLow_thCustom .link_thCustom:visited {color: "+colors.vlink+";}\n";
  579. css += ".toPoor_thCustom .link_thCustom, .toPoor_thCustom .tolink_thCustom {color: "+colors.poorTO+";}\n .toPoor_thCustom .link_thCustom:visited {color: "+colors.vlink+";}\n";
  580. css += ".nohitDB_thCustom {color: #000000; background: "+colors.nohitDB+";}\n";
  581. css += ".tooweak_thCustom {color: #000000; background: "+colors.unqualified+";}\n";
  582. css += ".yeshitDB_thCustom {color:#000000; background: "+colors.hitDB+";}\n";
  583. css += ".needmaster_thCustom {color: #000000; background: "+colors.reqmaster+";}\n";
  584. css += ".nomaster_thCustom {color: #000000; background: "+colors.nomaster+";}\n";
  585. css += ".spacer_thCustom {color: "+colors.accent+";}\n";
  586. css += ".mainlink_thCustom {color: "+colors.defaultText+"; text-align: top;}\n";
  587. css += ".tabhead_thCustom {color: "+colors.defaultText+";}\n";
  588. css += ".bodytable_thCustom {color: "+colors.defaultText+";}\n";
  589. css += ".statusdiv_thCustom {font-size: 15px; color: "+colors.secondText+";}\n";
  590. css += ".body_thRandom {background-color: "+rcolors.background+";}\n";
  591. css += ".controlpanel_thRandom {color: "+rcolors.defaultText+"; margin: 0px auto 5px auto; border-bottom: 1px solid "+rcolors.cpBorder+"; margin-bottom: 5px; background: "+rcolors.cpBackground+";}\n";
  592. css += ".cpInput_thRandom {color: "+rcolors.inputText+"; border: 1px solid; background: "+rcolors.background+"; text-align: center;}\n";
  593. css += ".cpButtons_thRandom {color: "+rcolors.accent+"; border: 1px solid; background: transparent;}\n";
  594. css += ".cpSortdiv_thRandom {color: "+rcolors.inputText+"; display: inline; font: 14px;}\n";
  595. css += ".cpSpans_thRandom {padding: 2px; cursor:default; color: "+rcolors.defaultText+"; background: "+rcolors.cpBackground+";} .cpSpans_thRandom:hover {background: "+rcolors.hover+";}\n";
  596. css += ".cpSpansOn_thRandom {background: "+rcolors.highlight+"; color: "+rcolors.secondText+";}\n";
  597. css += ".taButtons_thRandom {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: "+rcolors.export+";} .taButtonsOff_thRandom {display: none;}\n";
  598. css += ".link_thRandom:link {color: "+rcolors.link+";} .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  599. css += ".toNone_thRandom .link_thRandom, .toNone_thRandom .tolink_thRandom {color: "+rcolors.defaultText+";}\n .toNone_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  600. css += ".toHigh_thRandom .link_thRandom, .toHigh_thRandom .tolink_thRandom {color: "+rcolors.highTO+";}\n .toHigh_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  601. css += ".toGood_thRandom .link_thRandom, .toGood_thRandom .tolink_thRandom {color: "+rcolors.goodTO+";}\n .toGood_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  602. css += ".toAverage_thRandom .link_thRandom, .toAverage_thRandom .tolink_thRandom {color: "+rcolors.averageTO+";}\n .toAverage_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  603. css += ".toLow_thRandom .link_thRandom, .toLow_thRandom .tolink_thRandom {color: "+rcolors.lowTO+";}\n .toLow_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  604. css += ".toPoor_thRandom .link_thRandom, .toPoor_thRandom .tolink_thRandom {color: "+rcolors.poorTO+";}\n .toPoor_thRandom .link_thRandom:visited {color: "+rcolors.vlink+";}\n";
  605. css += ".nohitDB_thRandom {color: #000000; background: "+rcolors.nohitDB+";}\n";
  606. css += ".tooweak_thRandom {color: #000000; background: "+rcolors.unqualified+";}\n";
  607. css += ".yeshitDB_thRandom {color:#000000; background: "+rcolors.hitDB+";}\n";
  608. css += ".needmaster_thRandom {color: #000000; background: "+rcolors.reqmaster+";}\n";
  609. css += ".nomaster_thRandom {color: #000000; background: "+rcolors.nomaster+";}\n";
  610. css += ".spacer_thRandom {color: "+rcolors.accent+";}\n";
  611. css += ".mainlink_thRandom {color: "+rcolors.defaultText+"; text-align: top;}\n";
  612. css += ".tabhead_thRandom {color: "+rcolors.defaultText+";}\n";
  613. css += ".bodytable_thRandom {color: "+rcolors.defaultText+";}\n";
  614. css += ".statusdiv_thRandom {font-size: 15px; color: "+rcolors.secondText+";}\n";
  615. // -- Dark and Light themes adapted from the Solarized color scheme developed by Ethan Schoonover -- //
  616. css += ".body_thSLight {background-color: #fdf6e3;}\n";
  617. css += ".controlpanel_thSLight {color: #657b83; margin: 0px auto 5px auto; border-bottom: 1px solid #000000; margin-bottom: 5px; background: #eee8d5;}\n";
  618. css += ".cpInput_thSLight {color: #6FA3A3; border: 1px solid; background: #fdf6e3; text-align: center;}\n";
  619. css += ".cpButtons_thSLight {color: #b58900; border: 1px solid; background: transparent;}\n";
  620. css += ".cpSortdiv_thSLight {color: #98D6D6; display: inline; font: 14px;}\n";
  621. css += ".cpSpans_thSLight {padding: 2px; cursor:default; color: #657b83; background: #eee8d5;} .cpSpans_thSLight:hover {background: #C7D2D6;}\n";
  622. css += ".cpSpansOn_thSLight {background: #657b83; color: #A6BABA;}\n";
  623. css += ".taButtons_thSLight {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent;} .taButtonsOff_thSLight {display: none;}\n";
  624. css += ".link_thSLight:link {color: #000000;} .link_thSLight:visited {color: #6c71c4;}\n";
  625. css += ".toNone_thSLight .link_thSLight, .toNone_thSLight .tolink_thSLight {color: #657b83;}\n .toNone_thSLight .link_thSLight:visited {color: #6c71c4;}\n";
  626. css += ".toHigh_thSLight .link_thSLight, .toHigh_thSLight .tolink_thSLight {color: #859900;}\n .toHigh_thSLight .link_thSLight:visited {color: #6c71c4;}\n";
  627. css += ".toGood_thSLight .link_thSLight, .toGood_thSLight .tolink_thSLight {color: #A2BA00;}\n .toGood_thSLight .link_thSLight:visited {color: #6c71c4;}\n";
  628. css += ".toAverage_thSLight .link_thSLight, .toAverage_thSLight .tolink_thSLight {color: #b58900;}\n .toAverage_thSLight .link_thSLight:visited {color: #6c71c4;}\n";
  629. css += ".toLow_thSLight .link_thSLight, .toLow_thSLight .tolink_thSLight {color: #cb4b16;}\n .toLow_thSLight .link_thSLight:visited {color: #6c71c4;}\n";
  630. css += ".toPoor_thSLight .link_thSLight, .toPoor_thSLight .tolink_thSLight {color: #dc322f;}\n .toPoor_thSLight .link_thSLight:visited {color: #6c71c4;}\n";
  631. css += ".nohitDB_thSLight {color: #000000; background: #36D0D3;}\n";
  632. css += ".tooweak_thSLight {color: #000000; background: #9F9F9F;}\n";
  633. css += ".yeshitDB_thSLight {color:#000000; background: #82D336;}\n";
  634. css += ".needmaster_thSLight {color: #000000; background: #B58900;}\n";
  635. css += ".nomaster_thSLight {color: #000000; background: #6C71C4;}\n";
  636. css += ".spacer_thSLight {color: #b58900;}\n";
  637. css += ".mainlink_thSLight {color: #657b83; text-align: top;}\n";
  638. css += ".tabhead_thSLight {color: #657b83;}\n";
  639. css += ".bodytable_thSLight {color: #657b83;}\n";
  640. css += ".statusdiv_thSLight {font-size: 15px; color: #93a1a1;}\n";
  641. //css += ".tablerow_thSLight {color: "+colors.defaultText+";}\n";
  642. css += ".body_thSDark {background-color: #002b36;}\n";
  643. css += ".controlpanel_thSDark {color: #839496; margin: 0px auto 5px auto; border-bottom: 1px solid #000000; margin-bottom: 5px; background: #073642;}\n";
  644. css += ".cpInput_thSDark {color: #eee8d5; border: 1px solid; background: #002b36; text-align: center;}\n";
  645. css += ".cpButtons_thSDark {color: #b58900; border: 1px solid; background: transparent;}\n";
  646. css += ".cpSortdiv_thSDark {color: #eee8d5; display: inline; font: 14px;}\n";
  647. css += ".cpSpans_thSDark {padding: 2px; cursor:default; color: #839496; background: #073642;} .cpSpans_thSDark:hover {background: #122A30;}\n";
  648. css += ".cpSpansOn_thSDark {background: #657b83; color: #A6BABA;}\n";
  649. css += ".taButtons_thSDark {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: #CCC6B4} .taButtonsOff_thSDark {display: none;}\n";
  650. css += ".link_thSDark:link {color: #000000;} .link_thSDark:visited {color: #6c71c4;}\n";
  651. css += ".toNone_thSDark .link_thSDark, .toNone_thSDark .tolink_thSDark {color: #839496;}\n .toNone_thSDark .link_thSDark:visited {color: #6c71c4;}\n";
  652. css += ".toHigh_thSDark .link_thSDark, .toHigh_thSDark .tolink_thSDark {color: #859900;}\n .toHigh_thSDark .link_thSDark:visited {color: #6c71c4;}\n";
  653. css += ".toGood_thSDark .link_thSDark, .toGood_thSDark .tolink_thSDark {color: #A2BA00;}\n .toGood_thSDark .link_thSDark:visited {color: #6c71c4;}\n";
  654. css += ".toAverage_thSDark .link_thSDark, .toAverage_thSDark .tolink_thSDark {color: #b58900;}\n .toAverage_thSDark .link_thSDark:visited {color: #6c71c4;}\n";
  655. css += ".toLow_thSDark .link_thSDark, .toLow_thSDark .tolink_thSDark {color: #cb4b16;}\n .toLow_thSDark .link_thSDark:visited {color: #6c71c4;}\n";
  656. css += ".toPoor_thSDark .link_thSDark, .toPoor_thSDark .tolink_thSDark {color: #dc322f;}\n .toPoor_thSDark .link_thSDark:visited {color: #6c71c4;}\n";
  657. css += ".nohitDB_thSDark {color: #000000; background: #D33682;}\n";
  658. css += ".tooweak_thSDark {color: #000000; background: #9F9F9F;}\n";
  659. css += ".yeshitDB_thSDark {color:#000000; background: #82D336;}\n";
  660. css += ".needmaster_thSDark {color: #000000; background: #B58900;}\n";
  661. css += ".nomaster_thSDark {color: #000000; background: #839496;}\n";
  662. css += ".spacer_thSDark {color: #b58900;}\n";
  663. css += ".mainlink_thSDark {color: #839496; text-align: top;}\n";
  664. css += ".tabhead_thSDark {color: #839496;}\n";
  665. css += ".bodytable_thSDark {color: #839496;}\n";
  666. css += ".statusdiv_thSDark {font-size: 15px; color: #93a1a1;}\n";
  667. css += ".body_thWisp {background-color: #232A2F;}\n";
  668. css += ".controlpanel_thWisp {color: #AFCCDE; margin: 0px auto 5px auto; border-bottom: 1px solid #000000; margin-bottom: 5px; background: #394752;}\n";
  669. css += ".cpInput_thWisp {color: #98D6D6; border: 1px solid; background: #232A2F; text-align: center;}\n";
  670. css += ".cpButtons_thWisp {color: #00ffff; border: 1px solid; background: transparent;}\n";
  671. css += ".cpSortdiv_thWisp {color: #98D6D6; display: inline; font: 14px;}\n";
  672. css += ".cpSpans_thWisp {padding: 2px; cursor:default; color: #AFCCDE; background: #394752;} .cpSpans_thWisp:hover {background: #1E303B;}\n";
  673. css += ".cpSpansOn_thWisp {background: #1F3847; color: #808080;}\n";
  674. css += ".taButtons_thWisp {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: #86939C;} .taButtonsOff_thWisp {display: none;}\n";
  675. css += ".link_thWisp:link {color: #7B8F8F;} .link_thWisp:visited {color: #40F0F0;}\n";
  676. css += ".toNone_thWisp .link_thWisp, .toNone_thWisp .tolink_thWisp {color: #AFCCDE;}\n .toNone_thWisp .link_thWisp:visited {color: #40F0F0;}\n";
  677. css += ".toHigh_thWisp .link_thWisp, .toHigh_thWisp .tolink_thWisp {color: #009DFF;}\n .toHigh_thWisp .link_thWisp:visited {color: #40F0F0;}\n";
  678. css += ".toGood_thWisp .link_thWisp, .toGood_thWisp .tolink_thWisp {color: #40B6FF;}\n .toGood_thWisp .link_thWisp:visited {color: #40F0F0;}\n";
  679. css += ".toAverage_thWisp .link_thWisp, .toAverage_thWisp .tolink_thWisp {color: #7ACCFF;}\n .toAverage_thWisp .link_thWisp:visited {color: #40F0F0;}\n";
  680. css += ".toLow_thWisp .link_thWisp, .toLow_thWisp .tolink_thWisp {color: #B5E3FF;}\n .toLow_thWisp .link_thWisp:visited {color: #40F0F0;}\n";
  681. css += ".toPoor_thWisp .link_thWisp, .toPoor_thWisp .tolink_thWisp {color: #DEF1FC;}\n .toPoor_thWisp .link_thWisp:visited {color: #40F0F0;}\n";
  682. css += ".nohitDB_thWisp {color: #000000; background: #DA95A8;}\n";
  683. css += ".tooweak_thWisp {color: #000000; background: #808080;}\n";
  684. css += ".yeshitDB_thWisp {color:#000000; background: #CADA95;}\n";
  685. css += ".needmaster_thWisp {color: #000000; background: #C1E1F6;}\n";
  686. css += ".nomaster_thWisp {color: #000000; background: #D6C1F6;}\n";
  687. css += ".spacer_thWisp {color: #00ffff;}\n";
  688. css += ".mainlink_thWisp {color: #AFCCDE; text-align: top;}\n";
  689. css += ".tabhead_thWisp {color: #AFCCDE;}\n";
  690. css += ".bodytable_thWisp {color: #AFCCDE;}\n";
  691. css += ".statusdiv_thWisp {font-size: 15px; color: #808080;}\n";
  692. css += ".body_thClassic {background-color: #131313;}\n";
  693. css += ".controlpanel_thClassic {color: #94704D; margin: 0px auto 5px auto; border-bottom: 1px solid #000000; margin-bottom: 5px; background: #131313;}\n";
  694. css += ".cpInput_thClassic {border: 1px solid; text-align: left;}\n";
  695. css += ".cpButtons_thClassic {border: 2px outset buttonface;}\n";
  696. css += ".cpSortdiv_thClassic {color: #94704D; display: inline; font: 14px;}\n";
  697. css += ".cpSpans_thClassic {padding: 2px; cursor:default; color: #94704D; background: #131313;} .cpSpans_thClassic:hover {background: #21211F;}\n";
  698. css += ".cpSpansOn_thClassic {background: #30302F; color: #997553;}\n";
  699. css += ".taButtons_thClassic {height: 14px; font-size: 8px; border: 1px solid; padding: 0px; background: transparent; color: #000000;} .taButtonsOff_thClassic {display: none;}\n";
  700. //css += ".link_thClassic:link {color: #7B8F8F;} .link_thClassic:visited {color: #40F0F0;}\n";
  701. css += ".toNone_thClassic {background: lightGrey;}\n";
  702. css += ".toHigh_thClassic {background: #66CC66;}\n";
  703. css += ".toGood_thClassic {background: #ADFF2F;}\n";
  704. css += ".toAverage_thClassic {background: #FFD700;}\n";
  705. css += ".toLow_thClassic {background: #FF9900;}\n";
  706. css += ".toPoor_thClassic {background: #FF3030;}\n";
  707. css += ".nohitDB_thClassic {color: #000000; background: #FF3030;}\n";
  708. css += ".tooweak_thClassic {color: #000000; background: #9F9F9F;}\n";
  709. css += ".yeshitDB_thClassic {color:#000000; background: #66CC66;}\n";
  710. css += ".needmaster_thClassic {color: #000000; background: #551A8B;}\n";
  711. css += ".nomaster_thClassic {color: #000000; background: #0066CC;}\n";
  712. css += ".spacer_thClassic {color: #94704D;}\n";
  713. css += ".mainlink_thClassic {color: #94704D; text-align: top;}\n";
  714. css += ".bodytable_thClassic {color: #000000;}\n";
  715. css += ".tabhead_thClassic {color: #94704D;}\n";
  716. css += ".statusdiv_thClassic {font-size: 15px; color: #94704D;}\n";
  717.  
  718. // ******************* below is universal and applies to all themes ********************************
  719. css += ".scraperBlockedRowOff td { border: 3pt solid #cc0000; }\n .scraperBlockedRow {display: none; }\n";
  720. css += ".scraperIncludelistedRow td { border: 3pt dashed #008800; }\n .scraperIncludelistedRowOff td { border: none; }\n";
  721. css += ".toBlockedRow {display:none;} .toBlockedRowOff td {border: 2px solid #00E5FF;}";
  722. css += "#thMenu {z-index: 2; position: fixed; top: 0.5%; right: 0.5%; text-align: center; display: inline; margin: 0; padding: 10px 5px; list-style: none;}\n";
  723. css += "#thMenu li {font: 15px; display: inline-block; position: relative; padding: 5px 5px; background: #E3E8E8; color: #000000; cursor: default}\n";
  724. css += "#thMenu li:hover {background: #616363; color: #ffffff;}\n";
  725. css += "#thMenu li ul {position: absolute; top: 24px; right: 0; width: 120px; padding: 0; display: none;}\n";
  726. css += "#thMenu li ul li {background: #616363; color: #ffffff; display: block; padding: 10px 10px;} #thMenu li ul li:hover {background: #8C8C8C;}\n";
  727. css += "#thMenu li:hover ul {display: block;}\n";
  728. css += "</style>";
  729. var header = document.getElementsByTagName("head");
  730. header[0].innerHTML = header[0].innerHTML + css;
  731.  
  732. var next_page = 1;
  733.  
  734. var API_PROXY_BASE = 'https://mturk-api.istrack.in/';
  735. var OFFICIAL_API_PROXY_BASE = 'https://turkopticon.ucsd.edu/';
  736. var API_MULTI_ATTRS_URL = API_PROXY_BASE + 'multi-attrs.php?ids=';
  737. var OFFICIAL_API_MULTI_ATTRS_URL = OFFICIAL_API_PROXY_BASE + 'api/multi-attrs.php?ids=';
  738. var REVIEWS_BASE = 'http://turkopticon.ucsd.edu/';
  739. var control_panel = document.createElement("DIV");
  740. control_panel.id = "control_panel";
  741. control_panel.className = "controlpanel"+sufTheme;
  742. document.body.className = "body"+sufTheme;
  743. document.body.insertBefore(control_panel, document.body.firstChild);
  744. $('body > :not(#control_panel)').hide(); //hide all nodes directly under the body
  745. var big_red_button = document.createElement("BUTTON");
  746. var reset_blocks = document.createElement("BUTTON");
  747. var include_button = document.createElement("BUTTON");
  748. var progress_report = document.createTextNode("Stopped");
  749. var status_report = document.createTextNode("None");
  750. var text_area = document.createElement("TABLE");
  751. text_area.id = "body_table";
  752. //text_area.className = "bodytable"+sufTheme;
  753. var showButton = document.createElement("BUTTON");
  754. showButton.className += " cpButtons"+sufTheme;
  755. showButton.onclick = function() { show_TO_rows(); };
  756. showButton.textContent = "Show ignored hits";
  757. showButton.id = "show_TO_stuff_button";
  758.  
  759. //function show_TO_rows() { $("#body_table tr:hidden").each(function () { $(this).show(); console.log($(this)); }); }
  760.  
  761. function show_TO_rows() {
  762. var mr = document.getElementsByClassName("toBlockedRow");
  763. var mt = mr || "[]";
  764. if (showButton.textContent == "Show ignored hits") {
  765. for (var i=mt.length-1; i>-1; --i) { mt[i].className = mt[i].className.replace(/(?:^| +)toBlockedRow/, "toBlockedRowOff"); }
  766. showButton.textContent = "Hide ignored hits";
  767. }
  768. else {
  769. mt = document.getElementsByClassName("toBlockedRowOff");
  770. for (var i=mt.length-1; i>-1; i--) { mt[i].className = mt[i].className.replace(/(?:^| +)toBlockedRowOff/, "toBlockedRow"); }
  771. showButton.textContent = "Show ignored hits";
  772. }
  773. }
  774.  
  775. show_interface();
  776.  
  777. function yiqBrightness(hc,yiq) { // hc type "#000000", "000000", "[00, 00, 00]"
  778. var r = 0, b = 0, g = 0;
  779. if (yiq===true) {
  780. if (typeof hc === 'object') { console.log("b3");r = hc[0]; g = hc[1]; b = hc[2]; }
  781. else if (hc.substr(0,1) == "#") {
  782. r = parseInt(hc.substr(1,2), 16);
  783. g = parseInt(hc.substr(3,2), 16);
  784. b = parseInt(hc.substr(5,2), 16);
  785. }
  786. else {
  787. r = parseInt(hc.substr(0,2), 16);
  788. g = parseInt(hc.substr(2,2), 16);
  789. b = parseInt(hc.substr(4,2), 16);
  790. }
  791. return r*0.299 + g*0.587 + b*0.114; // y value
  792. }
  793. else {
  794. return hc[0] + hc[1]*0.956 + hc[2]*0.621; // get r value from full yiq
  795. }
  796. }
  797.  
  798. function randomizeScheme() {
  799. // basic rudimentary placeholder
  800. // TODO: make randomization faster
  801.  
  802. if (!("background" in rcolors))
  803. for (var l in colors) { if (colors.hasOwnProperty(l)) rcolors[l] = colors[l]; }
  804. var r = 0, g = 0, b = 0;
  805. var k = Object.keys(rcolors);
  806. for (k in rcolors) {
  807. rcolors[k] = getNewColor();
  808. }
  809. var bg1 = yiqBrightness(rcolors.cpBackground, true), bg0 = yiqBrightness(rcolors.background,true);
  810. var colorslice = {};
  811. var chex = "";
  812. k = ["defaultText", "inputText", "secondText", "export", "accent"];
  813. for (var i in k) { colorslice[k[i]] = rcolors[k[i]]; }
  814. for (k in colorslice) {
  815. var iterations = 0;
  816. //console.log(k+": "+colorslice[k]);
  817. if (colorslice.hasOwnProperty(k)) {
  818. var c1 = yiqBrightness(rcolors[k],true);
  819. while (Math.abs(bg1-c1)<69 || Math.abs(bg0-c1)<69) { // increase readability
  820. chex = getNewColor();
  821. c1 = yiqBrightness(chex,true);
  822. rcolors[k] = chex;
  823. iterations++;
  824. }
  825. console.log(k+": "+rcolors[k]+" after "+iterations + " cycles");
  826. }
  827. }
  828. }
  829. function getNewColor() {
  830. var ri = Math.floor( Math.random()*255 ).toString(16);
  831. var gi = Math.floor( Math.random()*255 ).toString(16);
  832. var bi = Math.floor( Math.random()*255 ).toString(16);
  833. 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);
  834. }
  835.  
  836. var global_run = false;
  837. var statusdetail_loop_finished = false;
  838. var date_header = "";
  839. var scraper_history = {};
  840. var wait_loop;
  841. var dings = 0;
  842.  
  843. function set_progress_report(text, force)
  844. {
  845. if (global_run == true || force == true)
  846. {
  847. progress_report.textContent = text;
  848. var status_text = status_array.join("; ");
  849. status_report.textContent = status_text;
  850. }
  851. }
  852.  
  853. function get_progress_report()
  854. {
  855. return progress_report.textContent;
  856. }
  857.  
  858. function wait_until_stopped()
  859. {
  860. if (global_run == true)
  861. {
  862. if (statusdetail_loop_finished == true)
  863. {
  864. big_red_button.textContent = "Start";
  865. set_progress_report("Finished", false);
  866. }
  867. else
  868. {
  869. setTimeout(function(){wait_until_stopped();}, 500);
  870. }
  871. }
  872. }
  873.  
  874. function display_wait_time(wait_time)
  875. {
  876. if (global_run == true)
  877. {
  878. var current_progress = get_progress_report();
  879. if (current_progress.indexOf("Searching again in")!==-1)
  880. {
  881. set_progress_report(current_progress.replace(/Searching again in \d+ seconds/ , "Searching again in " + wait_time + " seconds"),false);
  882. }
  883. else
  884. set_progress_report(current_progress + " Searching again in " + wait_time + " seconds.", false);
  885. if (wait_time>1)
  886. setTimeout(function(){display_wait_time(wait_time-1);}, 1000);
  887. }
  888. }
  889.  
  890. function dispArr(ar)
  891. {
  892. var disp = "";
  893. for (var z = 0; z < ar.length; z++)
  894. {
  895. disp += "id " + z + " is " + ar[z] + " ";
  896. }
  897. console.log(disp);
  898. }
  899.  
  900. function scrape($src)
  901. {
  902. var $requester = $src.find('a[href^="/mturk/searchbar?selectedSearchType=hitgroups&requester"]');
  903. var $title = $src.find('a[class="capsulelink"]');
  904. var $reward = $src.find('span[class="reward"]');
  905. var $preview = $src.find('a[href^="/mturk/preview?"]');
  906. var $qualified = $src.find('a[href^="/mturk/notqualified?"],a[id^="private_hit"]');
  907. var $times = $src.find('a[id^="duration_to_complete"]');
  908. var $descriptions = $src.find('a[id^="description"]');
  909. var not_qualified_group_IDs=[];
  910. var $quals = $src.find('a[id^="qualificationsRequired"]');
  911. var $mixed;
  912. $qualified.each(function(){
  913. var groupy = $(this).attr('href');
  914. if (groupy){
  915. groupy = groupy.replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1");
  916. groupy = groupy.replace(/\/mturk\/notqualified\?hitId=([A-Z0-9]+)(&.+)?/,"$1");
  917. groupy = groupy.replace(/\&hitGroupId=([A-Z0-9]+)(&.+)?/,"");
  918. groupy = groupy.replace(/\&hitId=([A-Z0-9]+)(&.+)?/,"");
  919. }
  920. not_qualified_group_IDs.push(groupy);
  921. });
  922. //console.log(not_qualified_group_IDs);
  923. if (document.getElementById('lnkWorkerSignin'))
  924. $mixed = $src.find('a[href^="/mturk/preview?"],a[id^="private_hit"]');
  925. else
  926. $mixed = $src.find('a[href^="/mturk/preview?"],a[href^="/mturk/notqualified?"]');
  927. //console.log($mixed);
  928. var listy =[];
  929. $mixed.each(function(){
  930. var groupy = $(this).attr('href');
  931. if (groupy)
  932. {
  933. groupy = groupy.replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1");
  934. groupy = groupy.replace(/\/mturk\/notqualified\?hitId=([A-Z0-9]+)(&.+)?/,"$1");
  935. groupy = groupy.replace(/\&hitGroupId=([A-Z0-9]+)(&.+)?/,"");
  936. groupy = groupy.replace(/\&hitId=([A-Z0-9]+)(&.+)?/,"");
  937. groupy = groupy.replace("/mturk/preview?groupId=","");
  938. }
  939. else
  940. groupy = "";
  941. //console.log($(this));
  942. //console.log(groupy);
  943. if (listy.indexOf(groupy) == -1)
  944. listy.push(groupy);
  945. else if (groupy == "")
  946. listy.push(groupy);
  947. });
  948. //console.log(listy);
  949. //listy = listy.filter(function(elem, pos) {
  950. // return listy.indexOf(elem) == pos;
  951. //});
  952. //console.log(listy);
  953. for (var j = 0; j < $requester.length; j++)
  954. {
  955. var $hits = $requester.eq(j).parent().parent().parent().parent().parent().parent().find('td[class="capsule_field_text"]');
  956. var requester_name = $requester.eq(j).text().trim();
  957. var requester_link = $requester.eq(j).attr('href');
  958. var group_ID=(listy[j] ? listy[j] : "");
  959. var masters = false;
  960. var title = $title.eq(j).text().trim();
  961. var preview_link = "/mturk/preview?groupId=" + group_ID;
  962. //console.log(listy[j]);
  963. //console.log(title+" "+group_ID +" "+ listy[j]);
  964. if (!group_ID || group_ID.length == 0){
  965. preview_link = requester_link;
  966. title += " (Requester link substituted)";
  967. }
  968. var reward = $reward.eq(j).text().trim();
  969. var hits = $hits.eq(4).text().trim();
  970. var time = $times.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  971. var description = $descriptions.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  972. //console.log(description);
  973. var requester_id = requester_link.replace('/mturk/searchbar?selectedSearchType=hitgroups&requesterId=','');
  974. var accept_link;
  975. accept_link = preview_link.replace('preview','previewandaccept');
  976.  
  977. /*HIT SCRAPER ADDITION*/
  978. var qElements = $quals.eq(j).parent().parent().parent().find('tr');
  979. //console.log(qElements);
  980.  
  981. var qualifications = [];
  982. for (var i = 1; i < qElements.length; i++) {
  983. 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, ' ')));
  984. if (qElements[i].childNodes[1].textContent.trim().replace(/\s+/g, ' ').indexOf("Masters") != -1)
  985. masters=true;
  986. }
  987. var qualList = (qualifications.join(', ') ? qualifications.join(', ') : "None");
  988.  
  989. var key = requester_name+title+reward+group_ID;
  990. if (found_key_list.indexOf(key) == -1)
  991. found_key_list.push(key);
  992. else
  993. {
  994. console.log("DUPE: "+key);
  995. continue;
  996. }
  997. if (scraper_history[key] === undefined)
  998. {
  999. 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};
  1000. scraper_history[key].req_link = requester_link;
  1001. scraper_history[key].prev_link = preview_link;
  1002. scraper_history[key].requester = requester_name;
  1003. scraper_history[key].title = title;
  1004. scraper_history[key].reward = reward;
  1005. scraper_history[key].hits = hits;
  1006. scraper_history[key].rid = requester_id;
  1007. scraper_history[key].acc_link = accept_link;
  1008. scraper_history[key].time = time;
  1009. scraper_history[key].quals = qualList;
  1010. scraper_history[key].description = description.replace(/'/g, "&apos;").replace(/"/g, "&quot;");
  1011. scraper_history[key].masters = masters;
  1012. HITStorage.indexedDB.checkRequester(requester_id,key);
  1013. HITStorage.indexedDB.checkTitle(title,key);
  1014. if (searched_once)
  1015. {
  1016. scraper_history[key].initial_time = new Date().getTime();//-1000*(save_new_results_time - SEARCH_REFRESH);
  1017. scraper_history[key].new_result = 0;
  1018. scraper_history[key].dinged = 0;
  1019. }
  1020. else
  1021. {
  1022. scraper_history[key].initial_time = new Date().getTime()-1000*save_new_results_time;
  1023. scraper_history[key].new_result = 1000*save_new_results_time;
  1024. scraper_history[key].dinged = 1;
  1025. }
  1026. if (not_qualified_group_IDs.indexOf(group_ID) !== -1)
  1027. scraper_history[key].qualified = false;
  1028. else
  1029. scraper_history[key].qualified = true;
  1030.  
  1031. scraper_history[key].found_this_time = true;
  1032. }
  1033. else
  1034. {
  1035. scraper_history[key].new_result = new Date().getTime() - scraper_history[key].initial_time;
  1036. scraper_history[key].found_this_time = true;
  1037. scraper_history[key].hits = hits;
  1038. }
  1039. }
  1040. }
  1041.  
  1042. function statusdetail_loop(next_URL)
  1043. {
  1044. if (global_run == true)
  1045. {
  1046. if (next_URL.length != 0)
  1047. {
  1048. $.get(next_URL, function(data)
  1049. {
  1050. var $src = $(data);
  1051. var maxpagerate = $src.find('td[class="error_title"]:contains("You have exceeded the maximum allowed page request rate for this website.")');
  1052. if (maxpagerate.length == 0)
  1053. {
  1054. if (next_page > PAGES_TO_SCRAPE)
  1055. {
  1056. if(status_array.indexOf("Correcting for skips") == -1 && type != 2){
  1057. status_array.push("Correcting for skips");
  1058. }
  1059. }
  1060. set_progress_report("Processing page " + next_page, false);
  1061. scrape($src);
  1062.  
  1063. var $next_URL = $src.find('a[href^="/mturk/viewsearchbar"]:contains("Next")');
  1064. next_URL = ($next_URL.length != 0) ? $next_URL.attr("href") : "";
  1065. next_page++;
  1066. if (document.getElementById('lnkWorkerSignin'))
  1067. maxPages--;
  1068. if (maxPages == 0)
  1069. {
  1070. maxPages = 20;
  1071. next_URL = "";
  1072. next_page = -1;
  1073. }
  1074.  
  1075. if (default_type == 1)
  1076. {
  1077. var hmin = MINIMUM_HITS+1;
  1078. for (var j = 0; j < found_key_list.length; j++)
  1079. {
  1080. //console.log(scraper_history[found_key_list[j]]);
  1081. if (scraper_history[found_key_list[j]].hits < hmin)
  1082. {
  1083. next_URL = "";
  1084. next_page = -1;
  1085. break;
  1086. }
  1087. }
  1088. }
  1089. else if (next_page > PAGES_TO_SCRAPE && correct_for_skips)
  1090. {
  1091. var skipped_hits = 0;
  1092. var added_pages = 0;
  1093. for (var k = 0; k < found_key_list.length; k++)
  1094. {
  1095. var obj = scraper_history[found_key_list[k]];
  1096. if (!ignore_check(obj.requester,obj.title))
  1097. skipped_hits++;
  1098. }
  1099. added_pages = Math.floor(skipped_hits/10);
  1100. if (skipped_hits%10 >6)
  1101. added_pages++;
  1102. if (next_page > PAGES_TO_SCRAPE + added_pages)
  1103. {
  1104. next_URL = "";
  1105. next_page = -1;
  1106. }
  1107.  
  1108. }
  1109. else if (next_page > PAGES_TO_SCRAPE)
  1110. {
  1111. next_URL = "";
  1112. next_page = -1;
  1113. }
  1114.  
  1115. setTimeout(function(){statusdetail_loop(next_URL);}, STATUSDETAIL_DELAY);
  1116. }
  1117. else
  1118. {
  1119. //console.log("MPRE");
  1120. setTimeout(function(){statusdetail_loop(next_URL);}, MPRE_DELAY);
  1121. }
  1122. }).fail(function() {
  1123. if (qual_input)
  1124. alert("Error when searching. Are you logged out?");
  1125. global_run = false;
  1126. big_red_button.textContent = "Start";
  1127. });
  1128. }
  1129. else
  1130. {
  1131. maxPages = 20;
  1132. searched_once = true;
  1133. var found_hits = found_key_list.length;
  1134. var shown_hits = 0;
  1135. var new_hits = 0;
  1136. var blocklist_hits = 0, belowThreshold_hits = 0;
  1137. dings = 0;
  1138. var ridString = "";
  1139. var rids = [];
  1140. var lastRow = text_area.rows.length - 1;
  1141. for (var i = lastRow; i>0; i--)
  1142. text_area.deleteRow(i);
  1143. for (var j = 0; j < found_key_list.length; j++)
  1144. {
  1145. //(function(url,rids,j) {
  1146. var obj = scraper_history[found_key_list[j]];
  1147. var ignored = ignore_check(obj.requester,obj.title);
  1148. var includelisted = include_check(obj.requester,obj.title);
  1149. //if (!(obj.masters == true && masters_hide.checked) && (ignored && obj.found_this_time) || (!ignored && obj.found_this_time && !useBlocklist)){
  1150. if (!(obj.masters == true && masters_hide.checked) && (ignored && obj.found_this_time) || (!ignored && obj.found_this_time)){
  1151. ++shown_hits;
  1152. //console.log(obj);
  1153. //hit export will update col_heads[1]
  1154. var col_heads = [];
  1155. if (useTO)
  1156. 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"];
  1157. else
  1158. 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"];
  1159. var row = text_area.insertRow(text_area.rows.length);
  1160. //row.className = "tablerow"+sufTheme;
  1161. if (!ignored && obj.found_this_time){
  1162. if (useBlock.checked) row.setAttribute("class","scraperBlockedRow");
  1163. else row.className = "scraperBlockedRowOff";
  1164. blocklist_hits++;
  1165. }
  1166. else if (!includelisted && obj.found_this_time){
  1167. if (highlightIncludes_input.checked) row.setAttribute("class","scraperIncludelistedRow");
  1168. else row.setAttribute("class", "scraperIncludelistedRowOff");
  1169. }
  1170. //url += obj.rid + ',';
  1171. //url2 += obj.rid + ',';
  1172. ridString += obj.rid + ',';
  1173. rids.push(obj.rid);
  1174. if (check_hitDB)
  1175. {
  1176. col_heads.push("R");
  1177. col_heads.push("T");
  1178. }
  1179. if (!obj.qualified)
  1180. {
  1181. col_heads.push("Not Qualified");
  1182. }
  1183. for (i=0; i<col_heads.length; i++)
  1184. {
  1185. var this_cell = row.insertCell(i);
  1186. row.cells[i].style.fontSize = default_text_size+"px";
  1187. this_cell.innerHTML = col_heads[i];
  1188. if (i>1)
  1189. this_cell.style.textAlign = 'center';
  1190. if (i==4 && fix_names) {
  1191. this_cell.title=obj.requester;
  1192. }
  1193. if (i==6)
  1194. {
  1195. var listOfQuals = obj.quals.replace(/,/g,'\n');
  1196. listOfQuals = listOfQuals.replace(/\[(.*?)\]/g,'');
  1197. this_cell.title=listOfQuals;
  1198. if (obj.masters)
  1199. {
  1200. this_cell.className = "needmaster"+sufTheme;
  1201. this_cell.innerHTML="Y";
  1202. }
  1203. else
  1204. {
  1205. this_cell.className = "nomaster"+sufTheme;
  1206. this_cell.innerHTML="N";
  1207. }
  1208. }
  1209. if (check_hitDB)
  1210. {
  1211. if (i==7)
  1212. {
  1213. if (obj.reqdb){
  1214. this_cell.className = "yeshitDB"+sufTheme;
  1215. this_cell.title = "This requester name was found in your HitDB.";
  1216. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,0);}})(obj));
  1217. }
  1218. else {
  1219. this_cell.className = "nohitDB"+sufTheme;
  1220. this_cell.title = "This requester name was not found in your HitDB, or you don't have a HitDB.";
  1221. }
  1222. }
  1223. else if (i==8)
  1224. {
  1225. if (obj.titledb){
  1226. this_cell.className = "yeshitDB"+sufTheme;
  1227. this_cell.title = "This HIT title was found in your HitDB.";
  1228. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,1);}})(obj));
  1229. }
  1230. else {
  1231. this_cell.className = "nohitDB"+sufTheme;
  1232. this_cell.title = "This HIT title was not found in your HitDB, or you don't have a HitDB.";
  1233. }
  1234. }
  1235. else if (i==9)
  1236. this_cell.className = "tooweak"+sufTheme;
  1237. }
  1238. else if (i==7)
  1239. this_cell.className = "tooweak"+sufTheme;
  1240. }
  1241. if (Object.keys(scraper_history).length > 0)
  1242. {
  1243. if (obj.dinged == 0)
  1244. {
  1245. dings++;
  1246. obj.dinged = 2; //#hookdinged
  1247. }
  1248. if (obj.new_result < 1000*save_new_results_time)
  1249. {
  1250. new_hits++;
  1251. for (var h = 0; h < col_heads.length; h++)
  1252. {
  1253. row.cells[h].style.fontSize = default_text_size + 1+"px";
  1254. row.cells[h].style.fontWeight = "bold";
  1255. }
  1256. }
  1257. }
  1258.  
  1259. var button1 = document.createElement('button'); // FORUM VBCODE EXPORT BUTTON
  1260. button1.textContent = 'vB';
  1261. button1.title = 'Export this HIT description as vBulletin formatted text';
  1262. button1.className = "vbButton";
  1263. button1.className += opt_exportvb.checked ? (" taButtons"+sufTheme) : (" taButtonsOff"+sufTheme);
  1264. button1.style.width = '30px';
  1265.  
  1266. var button2 = document.createElement('button'); //BUTTON TO BLOCK REQUESTER
  1267. button2.textContent = 'R';
  1268. button2.title = 'Add requester to block list';
  1269. button2.style.width = '15px';
  1270. button2.className += " taButtons"+sufTheme;
  1271.  
  1272. var button3 = document.createElement('button'); //BUTTON TO BLOCK TITLE
  1273. button3.textContent = 'T';
  1274. button3.title = 'Add title to block list';
  1275. button3.style.width = '15px';
  1276. button3.className += " taButtons"+sufTheme;
  1277.  
  1278. var button4 = document.createElement('button'); // IRC EXPORT BUTTON
  1279. button4.textContent = 'IRC';
  1280. button4.className = "ircButton";
  1281. button4.className += opt_exportirc.checked ? (" taButtons"+sufTheme) : (" taButtonsOff"+sufTheme);
  1282. button4.style.width = '30px';
  1283. button4.title = 'Click to save HIT information to your clipboard.';
  1284.  
  1285. var button5 = document.createElement('button'); // REDDIT EXPORT BUTTON
  1286. button5.textContent = 'HWTF';
  1287. button5.style.width = '33px';
  1288. button5.className = "redditButton";
  1289. button5.className += opt_exportreddit.checked ? (" taButtons"+sufTheme) : (" taButtonsOff"+sufTheme);
  1290. button5.title = 'Export this HIT to r/HITsWorthTurkingFor title standards.'
  1291.  
  1292.  
  1293. button1.addEventListener("click", (function (obj,j) { return function() {export_sel_deleg(obj,j,"vb");}})(obj,j));
  1294. row.cells[1].appendChild(document.createTextNode(" "));
  1295. row.cells[1].appendChild(button1);
  1296. button4.addEventListener("click", (function (obj,j) { return function() {export_sel_deleg(obj,j,"irc");}})(obj,j));
  1297. row.cells[1].appendChild(document.createTextNode(" "));
  1298. row.cells[1].appendChild(button4);
  1299. button5.addEventListener("click", (function (obj,j) { return function() {export_sel_deleg(obj,j,"reddit");}})(obj,j));
  1300. row.cells[1].appendChild(document.createTextNode(" "));
  1301. row.cells[1].appendChild(button5);
  1302.  
  1303. button2.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,0);}})(obj,j));
  1304. row.cells[0].appendChild(document.createTextNode(" "));
  1305. row.cells[0].appendChild(button2);
  1306. button3.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,1);}})(obj,j));
  1307. row.cells[0].appendChild(button3);
  1308. }
  1309. //});
  1310.  
  1311. }
  1312. //url = url.substring(0,url.length - 1);
  1313. //console.log(url);
  1314. var success_flag = false;
  1315. var rdata = "";
  1316. if (useTO)
  1317. rdata = getTOMulti(ridString);
  1318. else
  1319. rdata = "TO Down";
  1320. //var rdata = "TO Down";
  1321. //console.log(rdata);
  1322. if (rdata != "TO Down")
  1323. {
  1324. var globalMeans = { "pay":0, "paycount":0, "quality":0, "divisor":0, "reviews":0, "reviewcount":0 };
  1325.  
  1326. for (var r = 0; r < rids.length; r++) { // calculate global values across entire table
  1327. if (rdata[rids[r]]) {
  1328. if (rdata[rids[r]].attrs.pay > 0) {
  1329. globalMeans.quality += rdata[rids[r]].attrs.pay * PAY_WEIGHT;
  1330. globalMeans.pay += rdata[rids[r]].attrs.pay * 1.0;
  1331. globalMeans.paycount++;
  1332. globalMeans.divisor += PAY_WEIGHT;
  1333. }
  1334. if (rdata[rids[r]].attrs.comm.trim() > 0) {
  1335. globalMeans.quality += rdata[rids[r]].attrs.comm.trim()*COMM_WEIGHT;
  1336. globalMeans.divisor += COMM_WEIGHT;
  1337. }
  1338. if (rdata[rids[r]].attrs.fast.trim() > 0) {
  1339. globalMeans.quality += rdata[rids[r]].attrs.fast.trim()*FAST_WEIGHT;
  1340. globalMeans.divisor += FAST_WEIGHT;
  1341. }
  1342. if (rdata[rids[r]].attrs.fair.trim() > 0) {
  1343. globalMeans.quality += rdata[rids[r]].attrs.fair.trim()*FAIR_WEIGHT;
  1344. globalMeans.divisor += FAIR_WEIGHT;
  1345. }
  1346. if (rdata[rids[r]].reviews > 0) {
  1347. globalMeans.reviews += rdata[rids[r]].reviews*1.0;
  1348. globalMeans.reviewcount++;
  1349. }
  1350. }
  1351. }
  1352. globalMeans.quality = globalMeans.divisor > 0 ? globalMeans.quality/globalMeans.divisor : 0;
  1353. globalMeans.pay = globalMeans.paycount > 0 ? globalMeans.pay/globalMeans.paycount : 0;
  1354. globalMeans.reviews = globalMeans.reviewcount > 0 ? globalMeans.reviews/globalMeans.reviewcount : 0;
  1355. for (r = 0; r < rids.length; r++) // find
  1356. {
  1357. //if (text_area.rows[r+1].className.match(/toNone/) == null) text_area.rows[r+1].className += " toNone"+sufTheme;
  1358. if (rdata[rids[r]])
  1359. {
  1360. var pay = rdata[rids[r]].attrs.pay*1.0;
  1361. var reviews = rdata[rids[r]].reviews*1.0;
  1362. var quality = 0;
  1363. var sum = 0;
  1364. var divisor = 0;
  1365. var count = 0;
  1366. var wpayrank = 0;
  1367. var wqualityrank = 0;
  1368. //console.log(rdata[rids[r]]);
  1369. var comm = rdata[rids[r]].attrs.comm.trim()*1.0;
  1370. var fair = rdata[rids[r]].attrs.fair.trim()*1.0;
  1371. var fast = rdata[rids[r]].attrs.fast.trim()*1.0;
  1372. var tos = rdata[rids[r]].tos_flags;
  1373. if (comm > 0)
  1374. {
  1375. sum += COMM_WEIGHT*comm;
  1376. divisor += COMM_WEIGHT;
  1377. }
  1378. if (pay > 0)
  1379. {
  1380. sum += PAY_WEIGHT*pay;
  1381. divisor += PAY_WEIGHT;
  1382. }
  1383. if (fair > 0)
  1384. {
  1385. sum += FAIR_WEIGHT*fair;
  1386. divisor += FAIR_WEIGHT;
  1387. }
  1388. if (fast > 0)
  1389. {
  1390. sum += FAST_WEIGHT*fast;
  1391. divisor += FAST_WEIGHT;
  1392. }
  1393. if (divisor > 0)
  1394. {
  1395. quality = sum/divisor;
  1396. }
  1397. var apay = (reviews * pay + 15) / (reviews + 5);
  1398. //wpayrank = ((globalMeans.pay - (pay/globalMeans.paycount)) + reviews * pay) / globalMeans.reviewcount; // simplification
  1399. //wpayrank = (globalMeans.reviews * (globalMeans.pay - (pay/globalMeans.paycount)) + (reviews * pay)) / globalMeans.reviewcount; // unit derivation
  1400. wpayrank = apay - 1.645 * Math.sqrt((Math.pow(1.061 * apay,2) - Math.pow(apay,2)) / (reviews + 5)); // pnormal alpha = 0.10
  1401. //wpayrank = (10*globalMeans.pay + reviews*pay)/(globalMeans.reviewcount+10); // minimalist, no zs/conf - large fluctuations
  1402. var aqual = (quality * reviews + 15)/(reviews + 5);
  1403. //wqualityrank = (15*globalMeans.quality + reviews*quality)/(reviews+15);
  1404. wqualityrank = aqual - 1.645 * Math.sqrt((Math.pow(1.0693 * aqual,2) - Math.pow(aqual,2)) / (reviews + 5)); // pnormal, alpha = 0.10
  1405. var titleText = "\n\n"
  1406. titleText += "comm: "+comm+"\nfair: "+fair+"\nfast: "+fast+"\npay: "+pay+"\nReviews: "+reviews+"\nTOS violations: "+tos;
  1407. titleText += "\n\nAdjusted pay rating: " + wpayrank.toPrecision(4) + "\nGlobal pay mean: " + globalMeans.pay.toPrecision(4);
  1408. titleText += "\nWeighted quality: " + quality.toPrecision(4) + "\nAdjusted quality rating: " + wqualityrank.toPrecision(4) +
  1409. "\nGlobal quality mean: " + globalMeans.quality.toPrecision(4);
  1410. text_area.rows[r+1].cells[4].innerHTML = "<a class=\"tolink"+sufTheme+"\" href='"+ TO_REQ_URL+rids[r] +"' target='_blank'>" + pay + "</a>";
  1411. text_area.rows[r+1].cells[4].title += titleText;
  1412. if (pay < MINIMUM_TO && text_area.rows[r+1].className.match(/Blocked/) == null) {
  1413. text_area.rows[r+1].className += " toBlockedRow";
  1414. belowThreshold_hits++;
  1415. for (var m in found_key_list) {
  1416. if (dings > 0 && m.includes(r) && scraper_history[found_key_list[m]].dinged == 2) {
  1417. //console.log("[threshold] "+scraper_history[found_key_list[m]].requester+"; initial dings: "+dings);
  1418. dings--;
  1419. scraper_history[found_key_list[m]].dinged = 1;
  1420. //console.log("dings left: "+dings);
  1421. }
  1422. }
  1423. }
  1424.  
  1425. if (reviews > 4)
  1426. {
  1427. if (wqualityrank > 4.1)
  1428. text_area.rows[r+1].className += " toHigh"+sufTheme;
  1429. else if (wqualityrank > 3.1)
  1430. text_area.rows[r+1].className += " toGood"+sufTheme;
  1431. else if (wqualityrank > 2.4)
  1432. text_area.rows[r+1].className += " toAverage"+sufTheme;
  1433. else if (wqualityrank > 1.7)
  1434. text_area.rows[r+1].className += " toLow"+sufTheme;
  1435. else if (wqualityrank > 0)
  1436. text_area.rows[r+1].className += " toPoor"+sufTheme;
  1437. }
  1438. else if (text_area.rows[r+1].className.match(/toNone/) == null) text_area.rows[r+1].className += " toNone"+sufTheme;
  1439. }
  1440. else
  1441. {
  1442. text_area.rows[r+1].cells[4].innerHTML = "<a class=\"link"+sufTheme+"\" href='"+ TO_REQ_URL+rids[r] +"' target='_blank'>No Data</a>";
  1443. text_area.rows[r+1].cells[4].setAttribute("title", "No Data");
  1444. text_area.rows[r+1].className += " toNone"+sufTheme;
  1445. if (block_no_to && text_area.rows[r+1].className.match(/Blocked/) == null) {
  1446. text_area.rows[r+1].className += " toBlockedRow";
  1447. //if (text_area.rows[r+1].className.match(/ignored/) == null) text_area.rows[r+1].className += " ignoredRow";
  1448. belowThreshold_hits++;
  1449. for (var m in found_key_list) {
  1450. if (dings > 0 && m.includes(r) && scraper_history[found_key_list[m]].dinged == 2) {
  1451. //console.log("[nodata]"+scraper_history[found_key_list[m]].requester+"; initial dings: "+dings);
  1452. dings--;
  1453. scraper_history[found_key_list[m]].dinged = 1;
  1454. //console.log("dings left: "+dings);
  1455. }
  1456. }
  1457. }
  1458. }
  1459. }
  1460. if (sort_TO) table_sort("pay");
  1461. if (sort_TO2) table_sort("quality");
  1462.  
  1463. success_flag = true;
  1464. }
  1465. var pStr = "Scrape complete. " + shown_hits + " HITs found (" + new_hits + " new results). ";
  1466. if (!useBlock.checked) {
  1467. if (blocklist_hits > 0) pStr += blocklist_hits + " HITs shown from blocklist.";
  1468. if (belowThreshold_hits > 0) pStr += " " + belowThreshold_hits + " HITs below TO threshold.";
  1469. set_progress_report(pStr, false);
  1470. }
  1471. else {
  1472. if (belowThreshold_hits > 0) pStr += (belowThreshold_hits+blocklist_hits) + " HITs ignored: " + blocklist_hits + " from blocklist, " + belowThreshold_hits + " below TO threshold.";
  1473. set_progress_report(pStr, false);
  1474. }
  1475. if (!success_flag)
  1476. for (var s = 0; s < rids.length; s++)
  1477. if (text_area.rows[r+1].className.match(/toNone/) === null) text_area.rows[r+1].className += " toNone"+sufTheme;
  1478.  
  1479. statusdetail_loop_finished = true;
  1480. if (dings > 0){
  1481. newHits(shouldDing);
  1482. }
  1483. if (SEARCH_REFRESH>0)
  1484. {
  1485. wait_loop = setTimeout(function(){if (global_run) start_it();}, 1000*SEARCH_REFRESH);
  1486. display_wait_time(SEARCH_REFRESH);
  1487. }
  1488. else
  1489. {
  1490. global_run = false;
  1491. big_red_button.textContent = "Start";
  1492. }
  1493. }
  1494. }
  1495. }
  1496.  
  1497. function table_sort(stype) {
  1498. //--- Get the table we want to sort.
  1499. var jTableToSort = $("#body_table");
  1500.  
  1501. //--- Get the rows to sort, but skip the first row, since it contains column titles.
  1502. var jRowsToSort = jTableToSort.find ("tr:gt(0)");
  1503.  
  1504. //--- Sort the rows in place.
  1505. //if (stype == "rank") jRowsToSort.sort(SortByTORank).appendTo(jTableToSort);
  1506. //else jRowsToSort.sort(SortByTOPay).appendTo(jTableToSort);
  1507. jRowsToSort.sort(function(a,b) {
  1508. var ValA_Text = $(a).find("td:eq(4)").attr("title");
  1509. var ValB_Text = $(b).find("td:eq(4)").attr("title");
  1510. var aVal = "";
  1511. var bVal = "";
  1512. if (stype == "pay") { // extract pay data
  1513. aVal = ValA_Text.match(/pay rating: ([0-9.]+)/);
  1514. bVal = ValB_Text.match(/pay rating: ([0-9.]+)/);
  1515. }
  1516. else if (stype == "quality") { // extract rank data
  1517. aVal = ValA_Text.match(/quality rating: ([0-9.]+)/);
  1518. bVal = ValB_Text.match(/quality rating: ([0-9.]+)/);
  1519. }
  1520. // no TO goes to the bottom
  1521. aVal = aVal==null ? [0,0] : aVal;
  1522. bVal = bVal==null ? [0,0] : bVal;
  1523. //sort
  1524. if (sort_asc.checked) return aVal[1] - bVal[1];
  1525. else return bVal[1] - aVal[1]; // descending order
  1526. }).appendTo(jTableToSort);
  1527. }
  1528.  
  1529. // check block list for requester name and HIT title
  1530. function ignore_check(r,t){
  1531. var tempList = ignore_list.map(function(item) { return item.toLowerCase().replace(/\s+/g," "); });
  1532. var foundR = -1;
  1533. var foundT = -1;
  1534. foundR = tempList.indexOf(r.toLowerCase().replace(/\s+/g," "));
  1535. foundT = tempList.indexOf(t.toLowerCase().replace(/\s+/g," "));
  1536.  
  1537. if (shouldInclude){ // if in Use Includelist mode, treat all HITs not matching the includelist as if they were on the blocklist
  1538. //console.log(include_list);
  1539. var temp = include_list.map(function(item) { return item.toLowerCase().replace(/\s+/g," "); }).indexOf(r.toLowerCase().replace(/\s+/g," "));
  1540. //console.log(temp);
  1541. if (temp != -1)
  1542. foundR = -1;
  1543. else
  1544. foundR = 0;
  1545. }
  1546.  
  1547. var found = foundR == -1 && foundT == -1;
  1548. //console.log("r: "+r+" t: "+t+" f: "+found);
  1549. return found; // returns false (making !(include_check(x,y)) true) if HIT should be blocked, returns true if it shouldn't be blocked
  1550. //return -1 == ignore_list.map(function(item) { return item.toLowerCase(); }).indexOf(r.toLowerCase());
  1551. }
  1552.  
  1553. // check include list for requester name and HIT title
  1554. function include_check(r,t){
  1555. var tempList = include_list.map(function(item) { return item.toLowerCase().replace(/\s+/g," "); });
  1556. var foundR = -1;
  1557. var foundT = -1;
  1558. foundR = tempList.indexOf(r.toLowerCase().replace(/\s+/g," "));
  1559. foundT = tempList.indexOf(t.toLowerCase().replace(/\s+/g," "));
  1560. var found = foundR == -1 && foundT == -1;
  1561. return found; // returns false (making !(include_check(x,y)) true) if HIT should be highlighted, returns true if it shouldn't be highlighted
  1562. }
  1563.  
  1564. function start_running()
  1565. {
  1566. if (big_red_button.textContent == "Start")
  1567. {
  1568. status_array=[];
  1569. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  1570. if (GM_getValue("scraper_include_list"))
  1571. include_list = GM_getValue("scraper_include_list").split('^');
  1572. global_run = true;
  1573. initial_url = URL_BASE;
  1574.  
  1575. if (search_input.value.length>0)
  1576. {
  1577. initial_url = initial_url.replace("searchWords=", "searchWords=" + search_input.value);
  1578. }
  1579.  
  1580. if (time_input.value.replace(/[^0-9]+/g,"") != "")
  1581. {
  1582. SEARCH_REFRESH = Number(time_input.value);
  1583. }
  1584.  
  1585. if (page_input.value.replace(/[^0-9]+/g,"") != "")
  1586. {
  1587. PAGES_TO_SCRAPE = Number(page_input.value);
  1588. if (PAGES_TO_SCRAPE > 20 && document.getElementById('lnkWorkerSignin')){
  1589. status_array.push("Search limited to 20 pages when logged out");
  1590. PAGES_TO_SCRAPE = 20;
  1591. }
  1592. }
  1593.  
  1594. if (new_time_display_input.value.replace(/[^0-9]+/g,"") != "")
  1595. {
  1596. save_new_results_time = Number(new_time_display_input.value);
  1597. }
  1598.  
  1599. if (reward_input.value.replace(/[^0-9]+/g,"") != "")
  1600. {
  1601. initial_url += "&minReward=" + reward_input.value;
  1602. }
  1603. else
  1604. {
  1605. initial_url += "&minReward=0.00";
  1606. }
  1607.  
  1608. if (qual_input.checked)
  1609. {
  1610. if (document.getElementById('lnkWorkerSignin')){
  1611. status_array.push("Logged out, ignoring qualified");
  1612. initial_url += "&qualifiedFor=off";
  1613. }
  1614. else
  1615. initial_url += "&qualifiedFor=on";
  1616. }
  1617. else
  1618. {
  1619. initial_url += "&qualifiedFor=off";
  1620. }
  1621.  
  1622. if (masters_input.checked)
  1623. {
  1624. initial_url += "&requiresMasterQual=on";
  1625. }
  1626.  
  1627. if (masters_hide.checked)
  1628. {
  1629. status_array.push("Masters hits hidden");
  1630. }
  1631.  
  1632. switch (sort_input[sort_input.selectedIndex].value)
  1633. {
  1634. case "late":
  1635. initial_url+= "&sortType=LastUpdatedTime%3A";
  1636. type=1;
  1637. default_type = 0;
  1638. break;
  1639. case "most":
  1640. initial_url+= "&sortType=NumHITs%3A";
  1641. default_type = 1;
  1642. type=2;
  1643. status_array.push("Sorting by NumHITs ignores Correct For Skips in favor of minimum batch size");
  1644. break;
  1645. case "amount":
  1646. initial_url+= "&sortType=Reward%3A";
  1647. type=3;
  1648. default_type = 0;
  1649. break;
  1650. case "alpha":
  1651. type=4;
  1652. initial_url += "&sortType=Title%3A";
  1653. break;
  1654. default:
  1655. alert("I don't know how you did it, but you broke it. Good job.");
  1656. }
  1657.  
  1658. if (min_input.value.replace(/[^0-9]+/g,"") !== "")
  1659. {
  1660. if (type != 2)
  1661. status_array.push("Minimum Batch Size requires sorting by Most Available");
  1662. MINIMUM_HITS = Number(min_input.value);
  1663. }
  1664.  
  1665. if (sort_input_invert.checked)
  1666. {
  1667. if (sort_input[sort_input.selectedIndex].value == "alpha")
  1668. initial_url += "1";
  1669. else
  1670. initial_url += "0";
  1671. }
  1672. else
  1673. {
  1674. if (sort_input[sort_input.selectedIndex].value == "alpha")
  1675. initial_url += "0";
  1676. else
  1677. initial_url += "1";
  1678. }
  1679.  
  1680. if (sort_to.checked)
  1681. {
  1682. sort_TO = true;
  1683. sort_TO2 = false;
  1684. status_array.push("Sorting by TO pay is still in testing");
  1685. }
  1686. else if (sort_to2.checked)
  1687. {
  1688. sort_TO2 = true;
  1689. sort_TO= false;
  1690. status_array.push("Sorting by TO quality is still in testing");
  1691. }
  1692. else
  1693. {
  1694. sort_TO = false;
  1695. sort_TO2 = false;
  1696. }
  1697.  
  1698. if (useTO_input.checked)
  1699. {
  1700. useTO = false;
  1701. }
  1702. else
  1703. {
  1704. useTO = true;
  1705. }
  1706.  
  1707. if (min_to.value.replace(/[^0-9]+/g,"") != "")
  1708. {
  1709. MINIMUM_TO = Number(min_to.value);
  1710. $("#show_TO_stuff_button").show();
  1711. }
  1712. else
  1713. {
  1714. MINIMUM_TO = -1;
  1715. }
  1716.  
  1717. if (friesAreDone.checked)
  1718. {
  1719. shouldDing = true;
  1720. }
  1721. else
  1722. {
  1723. shouldDing = false;
  1724. }
  1725.  
  1726. if (no_to_block.checked)
  1727. {
  1728. block_no_to = true;
  1729. $("#show_TO_stuff_button").show();
  1730. }
  1731. else
  1732. {
  1733. block_no_to = false;
  1734. }
  1735.  
  1736. if (!block_no_to && MINIMUM_TO == -1)
  1737. $("#show_TO_stuff_button").hide();
  1738.  
  1739. if (correctForSkips.checked)
  1740. {
  1741. if (matchOnly.checked)
  1742. {
  1743. status_array.push("Use Includelist (match only) checked, so ignoring Correct For Skips to prevent issues");
  1744. correct_for_skips = false;
  1745. }
  1746. else{
  1747. correct_for_skips = true;
  1748. }
  1749. }
  1750. else
  1751. {
  1752. correct_for_skips = false;
  1753. }
  1754.  
  1755. if (useBlock.checked)
  1756. {
  1757. useBlocklist = true;
  1758. }
  1759. else
  1760. {
  1761. useBlocklist = false;
  1762. }
  1763.  
  1764. if (matchOnly.checked)
  1765. {
  1766. if (include_list.length == 0){
  1767. status_array.push("No items in includelist, so ignoring Use Includelist checkbox");
  1768. shouldInclude = false;
  1769. }
  1770. else
  1771. {
  1772. shouldInclude = true;
  1773. if (!useBlock.checked)
  1774. {
  1775. status_array.push("Use Includelist checked, so forcing Use Blocklist for intended output behavior");
  1776. useBlocklist = true;
  1777. }
  1778. }
  1779. }
  1780. else
  1781. {
  1782. shouldInclude = false;
  1783. }
  1784.  
  1785. initial_url += "&pageNumber=1&searchSpec=HITGroupSearch";
  1786. audio_index = audio_option[audio_option.selectedIndex].value;
  1787. //To store the last state of the items
  1788. var lastState = {"refreshTime":time_input.value,"numPages":page_input.value,"minHits":min_input.value,"newHitHighlight":new_time_display_input.value,
  1789. "reward":reward_input.value,"qualified":qual_input.checked,"masters":masters_input.checked,"mShow":masters_hide.checked,
  1790. "sort":sort_input.selectedIndex,"invert":sort_input_invert.checked,"to":sort_to.checked,"to2":sort_to2.checked,
  1791. "minTO":min_to.value,"hideNTO":no_to_block.checked,"searchTerms":search_input.value,"useInclude":matchOnly.checked,
  1792. "highlightIncl":highlightIncludes_input.checked,"fries":friesAreDone.checked,"whichfry":audio_option.selectedIndex,
  1793. "blocklist":useBlock.checked,"skips":correctForSkips.checked,"useTO":useTO_input.checked,"exvb":opt_exportvb.checked,
  1794. "exirc":opt_exportirc.checked,"exreddit":opt_exportreddit.checked,"asc":sort_asc.checked, "dsc":sort_dsc.checked,
  1795. "theme":sufTheme};
  1796. //console.log(JSON.stringify(lastState));
  1797. localStorage["lastState_hit_scraper_with_export"] = JSON.stringify(lastState);
  1798. start_it();
  1799. }
  1800. else
  1801. {
  1802. global_run = false;
  1803. clearTimeout(wait_loop);
  1804. big_red_button.textContent = "Start";
  1805. set_progress_report("Stopped", true);
  1806. }
  1807. }
  1808.  
  1809. function start_it()
  1810. {
  1811. statusdetail_loop_finished = false;
  1812. big_red_button.textContent = "Stop";
  1813. found_key_list = [];
  1814. var ctime = new Date().getTime();
  1815. if (ctime - last_clear_time > save_results_time*666)
  1816. {
  1817. var last_history = scraper_history;
  1818. scraper_history = {};
  1819. for (var key in last_history)
  1820. {
  1821. if (last_history[key].new_result<save_results_time*1000)
  1822. {
  1823. scraper_history[key] = last_history[key];
  1824. if (last_history[key].found_this_time)
  1825. {
  1826. last_history[key].found_this_time = false;
  1827. if (last_history[key].new_result>save_new_results_time*1000)
  1828. last_history[key].initial_time = ctime-1000*save_new_results_time;
  1829. }
  1830. }
  1831.  
  1832. }
  1833. last_clear_time = ctime;
  1834. }
  1835. next_page = 1;
  1836. statusdetail_loop(initial_url);
  1837. }
  1838.  
  1839. function themeSwitchAux(th) {
  1840. var carr = $("[class$='"+sufTheme+"']");
  1841. for (var i=0; i<carr.length; i++) {
  1842. carr[i].className = carr[i].className.replace(/_[a-zA-Z]+/g, th);
  1843. }
  1844. sufTheme = th;
  1845. }
  1846.  
  1847. function show_interface()
  1848. {
  1849. //--- dynamic theme switching
  1850. var themeMenu = document.createElement("UL");
  1851. var thC2 = document.createElement("UL");
  1852. var thT1 = document.createElement("LI");
  1853. var thT2 = document.createElement("LI");
  1854. var thT3 = document.createElement("LI");
  1855. var thT4 = document.createElement("LI");
  1856. var thT5 = document.createElement("LI");
  1857. var thT6 = document.createElement("LI");
  1858. var thE0 = document.createElement("LI");
  1859. themeMenu.id = "thMenu";
  1860. thE0.innerHTML = " &#9670; ";
  1861. thE0.style.width = '20px';
  1862. thT1.textContent = "Solarium:Dark Theme";
  1863. thT2.textContent = "Solarium:Light Theme";
  1864. thT3.textContent = "Whisper Theme";
  1865. thT6.textContent = "Classic Theme";
  1866. thT4.textContent = "Custom Theme";
  1867. thT5.textContent = "I'm Feelin' Lucky";
  1868. thT1.addEventListener("click", function() { themeSwitchAux("_thSDark"); }, false);
  1869. thT2.addEventListener("click", function() { themeSwitchAux("_thSLight"); }, false);
  1870. thT3.addEventListener("click", function() { themeSwitchAux("_thWisp"); }, false);
  1871. thT4.addEventListener("click", function() { themeSwitchAux("_thCustom"); }, false);
  1872. thT5.addEventListener("click", function() { themeSwitchAux("_thRandom"); }, false);
  1873. thT6.addEventListener("click", function() { themeSwitchAux("_thClassic"); }, false);
  1874. thC2.appendChild(thT1);
  1875. thC2.appendChild(thT2);
  1876. thC2.appendChild(thT3);
  1877. thC2.appendChild(thT6);
  1878. thC2.appendChild(thT4);
  1879. thC2.appendChild(thT5);
  1880. thE0.appendChild(thC2);
  1881. themeMenu.appendChild(thE0);
  1882. control_panel.parentNode.insertBefore(themeMenu, control_panel);
  1883. //---
  1884.  
  1885. var spacer = document.createElement("SPAN");
  1886. spacer.className = "spacer"+sufTheme;
  1887. spacer.appendChild(document.createTextNode(SPACER_TEXT));
  1888.  
  1889. control_panel.appendChild(document.createElement("BR"));
  1890. control_panel.style.fontSize = "14px";
  1891. //control_panel.removeChild(big_red_button);
  1892. control_panel.appendChild(document.createTextNode("Auto-refresh delay: "));
  1893. time_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1894. time_input.title = "Enter search refresh delay in seconds.\n" + "Enter 0 for no auto-refresh.\n" + "Default is 0 (no auto-refresh).";
  1895. time_input.size = 3;
  1896. time_input.className = "cpInput"+sufTheme;
  1897. control_panel.appendChild(time_input);
  1898.  
  1899. control_panel.appendChild(document.createTextNode(" "));
  1900.  
  1901. control_panel.appendChild(spacer);
  1902. control_panel.appendChild(document.createTextNode("Pages to scrape: "));
  1903. page_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1904. page_input.title = "Enter number of pages to scrape. Default is 3.\n" + "Has no effect in a batch search (Most Available sort).";
  1905. page_input.size = 3;
  1906. page_input.className = "cpInput"+sufTheme;
  1907. control_panel.appendChild(page_input);
  1908.  
  1909. var skipspan = document.createElement("SPAN");
  1910. skipspan.appendChild(document.createTextNode("Correct for skips"));
  1911. skipspan.className += " cpSpans"+sufTheme;
  1912. skipspan.title = "Searches additional pages to get a more consistent number of results. Helpful if you're blocking a lot of items.";
  1913. if (correctForSkips.checked) skipspan.className += " cpSpansOn"+sufTheme;
  1914. skipspan.addEventListener("click", function() {
  1915. if (correctForSkips.checked) {
  1916. correctForSkips.checked = false;
  1917. skipspan.className = skipspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  1918. } else {
  1919. correctForSkips.checked = true;
  1920. skipspan.className += " cpSpansOn"+sufTheme;
  1921. } }, false);
  1922. //skipspan.appendChild(correctForSkips);
  1923. //correctForSkips.onclick = function () { if (correctForSkips.checked) skipspan.className += " cpSpansOn"+sufTheme; };
  1924. control_panel.appendChild(spacer.cloneNode(true));
  1925. control_panel.appendChild(skipspan);
  1926.  
  1927. control_panel.appendChild(document.createTextNode(" "));
  1928.  
  1929. control_panel.appendChild(spacer.cloneNode(true));
  1930. control_panel.appendChild(document.createTextNode("Minimum batch size: "));
  1931. min_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1932. min_input.title = "Enter minimum HITs for batch search (must Sort by Most Available).\n" + "Default is 100.";
  1933. min_input.size = 3;
  1934. min_input.className = "cpInput"+sufTheme;
  1935. control_panel.appendChild(min_input);
  1936. control_panel.appendChild(document.createTextNode(" "));
  1937.  
  1938. control_panel.appendChild(document.createElement("P"));
  1939. control_panel.appendChild(document.createTextNode("Minimum reward: "));
  1940. reward_input.size = 3;
  1941. reward_input.title = "Enter the minimum desired pay per HIT, such as 0.10";
  1942. reward_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1943. reward_input.className = "cpInput"+sufTheme;
  1944. control_panel.appendChild(reward_input);
  1945.  
  1946. var qualspan = document.createElement("SPAN");
  1947. qualspan.appendChild(document.createTextNode("Qualified"));
  1948. qualspan.className += " cpSpans"+sufTheme;
  1949. qualspan.title = "Only show HITs you're currently qualified for (must be logged in).";
  1950. if (qual_input.checked) qualspan.className += " cpSpansOn"+sufTheme;
  1951. qualspan.addEventListener("click", function() {
  1952. if (qual_input.checked) {
  1953. qual_input.checked = false;
  1954. qualspan.className = qualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  1955. } else {
  1956. qual_input.checked = true;
  1957. qualspan.className += " cpSpansOn"+sufTheme;
  1958. } }, false);
  1959. //qualspan.appendChild(qual_input);
  1960. //qual_input.onclick = function () { if (qual_input.checked) qualspan.className += " cpSpansOn"+sufTheme; };
  1961. control_panel.appendChild(spacer.cloneNode(true));
  1962. control_panel.appendChild(qualspan);
  1963.  
  1964. var forcemastersspan = document.createElement("SPAN");
  1965. forcemastersspan.appendChild(document.createTextNode("Masters Only"));
  1966. forcemastersspan.className += " cpSpans"+sufTheme;
  1967. forcemastersspan.title = "Only show HITs that require Masters qualifications.";
  1968. if (masters_input.checked) forcemastersspan.className += " cpSpansOn"+sufTheme;
  1969. forcemastersspan.addEventListener("click", function() {
  1970. if (masters_input.checked) {
  1971. masters_input.checked = false;
  1972. forcemastersspan.className = forcemastersspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  1973. } else {
  1974. masters_input.checked = true;
  1975. forcemastersspan.className += " cpSpansOn"+sufTheme;
  1976. } }, false);
  1977. //forcemastersspan.appendChild(masters_input);
  1978. //masters_input.onclick = function () { if (masters_input.checked) forcemastersspan.className += " cpSpansOn"+sufTheme; };
  1979. control_panel.appendChild(spacer.cloneNode(true));
  1980. control_panel.appendChild(forcemastersspan);
  1981.  
  1982. var mhidespan = document.createElement("SPAN");
  1983. mhidespan.appendChild(document.createTextNode("Hide Masters"));
  1984. mhidespan.className += " cpSpans"+sufTheme;
  1985. mhidespan.title = "Remove masters hits from the results if selected, otherwise display both masters and non-masters HITS.\nThe \"qualified\" setting superceedes this option.";
  1986. if (masters_hide.checked) mhidespan.className += " cpSpansOn"+sufTheme;
  1987. mhidespan.addEventListener("click", function() {
  1988. if (masters_hide.checked) {
  1989. masters_hide.checked = false;
  1990. mhidespan.className = mhidespan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  1991. } else {
  1992. masters_hide.checked = true;
  1993. mhidespan.className += " cpSpansOn"+sufTheme;
  1994. } }, false);
  1995. //mhidespan.appendChild(masters_hide);
  1996. //masters_hide.onclick = function () { if (masters_hide.checked) mhidespan.className += " cpSpansOn"+sufTheme; };
  1997. control_panel.appendChild(spacer.cloneNode(true));
  1998. control_panel.appendChild(mhidespan);
  1999.  
  2000. control_panel.appendChild(spacer.cloneNode(true));
  2001. control_panel.appendChild(document.createTextNode("Sort type: "));
  2002. control_panel.appendChild(sort_input);
  2003. sort_input.className += " cpInput"+sufTheme;
  2004. 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)";
  2005.  
  2006. var sinvertspan = document.createElement("SPAN");
  2007. sinvertspan.appendChild(document.createTextNode("Invert"));
  2008. sinvertspan.className += " cpSpans"+sufTheme;
  2009. 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)";
  2010. if (sort_input_invert.checked) sinvertspan.className += " cpSpansOn"+sufTheme;
  2011. sinvertspan.addEventListener("click", function() {
  2012. if (sort_input_invert.checked) {
  2013. sort_input_invert.checked = false;
  2014. sinvertspan.className = sinvertspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2015. } else {
  2016. sort_input_invert.checked = true;
  2017. sinvertspan.className += " cpSpansOn"+sufTheme;
  2018. } }, false);
  2019. //sinvertspan.appendChild(sort_input_invert);
  2020. //sort_input_invert.onclick = function () { if (sort_input_invert.checked) sinvertspan.className += " cpSpansOn"+sufTheme; };
  2021. control_panel.appendChild(spacer.cloneNode(true));
  2022. control_panel.appendChild(sinvertspan);
  2023.  
  2024. control_panel.appendChild(document.createElement("P"));
  2025. control_panel.appendChild(document.createTextNode("New HIT highlighting: "));
  2026. new_time_display_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  2027. new_time_display_input.title = "Enter time (in seconds) to keep new HITs highlighted.\n" + "Default is 300 (5 minutes).";
  2028. new_time_display_input.size = 6;
  2029. new_time_display_input.className = "cpInput"+sufTheme;
  2030. control_panel.appendChild(new_time_display_input);
  2031.  
  2032. var soundspan = document.createElement("SPAN");
  2033. soundspan.appendChild(document.createTextNode("Sound on new HIT "));
  2034. soundspan.className += " cpSpans"+sufTheme;
  2035. soundspan.title = "Play a sound when new results are found.";
  2036. if (friesAreDone.checked) soundspan.className += " cpSpansOn"+sufTheme;
  2037. audio_option.className += " cpInput"+sufTheme;
  2038. soundspan.addEventListener("click", function() {
  2039. if (friesAreDone.checked) {
  2040. friesAreDone.checked = false;
  2041. soundspan.className = soundspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2042. control_panel.removeChild(audio_option);
  2043. } else {
  2044. friesAreDone.checked = true;
  2045. soundspan.className += " cpSpansOn"+sufTheme;
  2046. control_panel.insertBefore(audio_option, soundspan.nextSibling);
  2047. } }, false);
  2048. //soundspan.appendChild(friesAreDone);
  2049. //friesAreDone.onclick = function () { if (friesAreDone.checked) soundspan.className += " cpSpansOn"+sufTheme; };
  2050. control_panel.appendChild(spacer.cloneNode(true));
  2051. control_panel.appendChild(soundspan);
  2052. audio_option.title = "Select which sound will be played.";
  2053. if (friesAreDone.checked) control_panel.insertBefore(audio_option, soundspan.nextSibling);
  2054.  
  2055. var ascspan = document.createElement("SPAN");
  2056. ascspan.className += " cpSpans"+sufTheme;
  2057. ascspan.style.fontSize = '15px';
  2058. if (sort_asc.checked) ascspan.className += " cpSpansOn"+sufTheme;
  2059. ascspan.title = "Sort results in ascending (low to high) order.";
  2060. var dscspan = document.createElement("SPAN");
  2061. dscspan.className += " cpSpans"+sufTheme;
  2062. dscspan.style.fontSize = '15px';
  2063. if (sort_dsc.checked) dscspan.className += " cpSpansOn"+sufTheme;
  2064. dscspan.title = "Sort results in descending (high to low) order.";
  2065. ascspan.addEventListener("click", function() {
  2066. if (!sort_asc.checked) {
  2067. sort_asc.checked = true;
  2068. sort_dsc.checked = false;
  2069. ascspan.className += " cpSpansOn"+sufTheme;
  2070. dscspan.className = dscspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2071. }
  2072. });
  2073. dscspan.addEventListener("click", function() {
  2074. if (!sort_dsc.checked) {
  2075. sort_dsc.checked = true;
  2076. sort_asc.checked = false;
  2077. dscspan.className += " cpSpansOn"+sufTheme;
  2078. ascspan.className = ascspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2079. }
  2080. });
  2081. var sortdiv = document.createElement("DIV");
  2082. sortdiv.className = "cpSortdiv"+sufTheme;
  2083. sortdiv.appendChild(document.createTextNode(" ("));
  2084. sortdiv.appendChild(ascspan);
  2085. sortdiv.appendChild(dscspan);
  2086. sortdiv.appendChild(document.createTextNode(")"));
  2087. ascspan.innerHTML = " &#9650; ";
  2088. //ascspan.appendChild(sort_asc);
  2089. dscspan.innerHTML = "&nbsp;&#9660; ";
  2090. //dscspan.appendChild(sort_dsc);
  2091.  
  2092. var topayspan = document.createElement("SPAN");
  2093. var toqualspan = document.createElement("SPAN");
  2094. topayspan.appendChild(document.createTextNode("Sort by TO pay"));
  2095. topayspan.className += " cpSpans"+sufTheme;
  2096. 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)";
  2097. if (sort_to.checked) {
  2098. topayspan.className += " cpSpansOn"+sufTheme;
  2099. } else topayspan.className = topayspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2100. topayspan.addEventListener("click", function() {
  2101. if (sort_to.checked) {
  2102. sort_to.checked = false;
  2103. control_panel.removeChild(sortdiv);
  2104. topayspan.className = topayspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2105. } else {
  2106. sort_to.checked = true;
  2107. sort_to2.checked = false;
  2108. control_panel.insertBefore(sortdiv, topayspan.nextSibling);
  2109. toqualspan.className = toqualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2110. topayspan.className += " cpSpansOn"+sufTheme;
  2111. } }, false);
  2112. //topayspan.appendChild(sort_to);
  2113. //sort_to.onclick = function () { if (sort_to.checked) topayspan.className += " cpSpansOn"+sufTheme;};
  2114. control_panel.appendChild(spacer.cloneNode(true));
  2115. control_panel.appendChild(topayspan);
  2116. if (sort_to.checked) control_panel.insertBefore(sortdiv, topayspan.nextSibling);
  2117.  
  2118. toqualspan.style.cursor = 'default';
  2119. toqualspan.appendChild(document.createTextNode("Sort by TO quality"));
  2120. toqualspan.className += " cpSpans"+sufTheme;
  2121. toqualspan.title = "After getting search results based on the other selected options,\n" + "re-sort the results by their overall Turkopticon rating. (Bayesian adjusted)";
  2122. if (sort_to2.checked) {
  2123. toqualspan.className += " cpSpansOn"+sufTheme;
  2124. } else toqualspan.className = toqualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2125. toqualspan.addEventListener("click", function() {
  2126. if (sort_to2.checked) {
  2127. sort_to2.checked = false;
  2128. control_panel.removeChild(sortdiv);
  2129. toqualspan.className = toqualspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2130. } else {
  2131. sort_to2.checked = true;
  2132. sort_to.checked = false;
  2133. control_panel.insertBefore(sortdiv, toqualspan.nextSibling);
  2134. topayspan.className = topayspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2135. toqualspan.className += " cpSpansOn"+sufTheme;
  2136. } }, false);
  2137. //toqualspan.appendChild(sort_to2);
  2138. //sort_to2.onclick = function () { if (sort_to2.checked) toqualspan.className += " cpSpansOn"+sufTheme;};
  2139. control_panel.appendChild(spacer.cloneNode(true));
  2140. control_panel.appendChild(toqualspan);
  2141. if (sort_to2.checked) control_panel.insertBefore(sortdiv, toqualspan.nextSibling);
  2142.  
  2143. control_panel.appendChild(document.createElement("P"));
  2144. control_panel.appendChild(document.createTextNode("Min pay TO: "));
  2145. control_panel.appendChild(min_to);
  2146. min_to.size = 3;
  2147. min_to.className = "cpInput"+sufTheme;
  2148. 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";
  2149.  
  2150. var hidenotospan = document.createElement("SPAN");
  2151. hidenotospan.appendChild(document.createTextNode("Hide no TO"));
  2152. hidenotospan.className += " cpSpans"+sufTheme;
  2153. hidenotospan.title = "After getting search results based on the other selected options,\n" + "hide any results that have no Turkopticon pay ratings yet.";
  2154. if (no_to_block.checked) hidenotospan.className += " cpSpansOn"+sufTheme;
  2155. hidenotospan.addEventListener("click", function() {
  2156. if (no_to_block.checked) {
  2157. no_to_block.checked = false;
  2158. hidenotospan.className = hidenotospan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2159. } else {
  2160. no_to_block.checked = true;
  2161. hidenotospan.className += " cpSpansOn"+sufTheme;
  2162. } }, false);
  2163. //hidenotospan.appendChild(no_to_block);
  2164. //no_to_block.onclick = function () { if (no_to_block.checked) hidenotospan.className += " cpSpansOn"+sufTheme; };
  2165. control_panel.appendChild(spacer.cloneNode(true));
  2166. control_panel.appendChild(hidenotospan);
  2167.  
  2168. var disabletospan = document.createElement("SPAN");
  2169. disabletospan.appendChild(document.createTextNode("Disable TO"));
  2170. disabletospan.className += " cpSpans"+sufTheme;
  2171. disabletospan.title = "Disable attempts to download ratings data from Turkopticon for the results table.";
  2172. if (useTO_input.checked) disabletospan.className += " cpSpansOn"+sufTheme;
  2173. disabletospan.addEventListener("click", function() {
  2174. if (useTO_input.checked) {
  2175. useTO_input.checked = false;
  2176. disabletospan.className = disabletospan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2177. } else {
  2178. useTO_input.checked = true;
  2179. disabletospan.className += " cpSpansOn"+sufTheme;
  2180. } }, false);
  2181. //disabletospan.appendChild(useTO_input);
  2182. //useTO_input.onclick = function () { if (useTO_input.checked) disabletospan.className += " cpSpansOn"+sufTheme; };
  2183. control_panel.appendChild(spacer.cloneNode(true));
  2184. control_panel.appendChild(disabletospan);
  2185.  
  2186. control_panel.appendChild(spacer.cloneNode(true));
  2187. control_panel.appendChild(document.createTextNode("Display export buttons: "));
  2188. var vbspan = document.createElement("SPAN");
  2189. vbspan.className += " cpSpans"+sufTheme;
  2190. vbspan.title = 'Show a button to export the specified HIT with vBulletin formatted text.';
  2191. if (opt_exportvb.checked) vbspan.className += " cpSpansOn"+sufTheme;
  2192. vbspan.addEventListener("click", function(){
  2193. var carr = document.getElementsByClassName('vbButton');
  2194. if (!opt_exportvb.checked) {
  2195. opt_exportvb.checked=true; vbspan.className += " cpSpansOn"+sufTheme;
  2196. for (var i=0; i<carr.length; i++)
  2197. carr[i].className = carr[i].className.replace(/taButtonsOff/, "taButtons");
  2198. } else {
  2199. for (var i=0; i<carr.length; i++)
  2200. carr[i].className = carr[i].className.replace(/taButtons/, "taButtonsOff");
  2201. opt_exportvb.checked=false; vbspan.className = vbspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2202. }
  2203. }, false);
  2204. vbspan.appendChild(document.createTextNode(" VB "));
  2205. //vbspan.appendChild(opt_exportvb);
  2206. control_panel.appendChild(vbspan);
  2207.  
  2208. var ircspan = document.createElement("SPAN");
  2209. ircspan.className += " cpSpans"+sufTheme;
  2210. ircspan.title = 'Show a button to export the specified HIT streamlined for IRC.';
  2211. if (opt_exportirc.checked) ircspan.className += " cpSpansOn"+sufTheme;
  2212. ircspan.addEventListener("click", function(){
  2213. var carr = document.getElementsByClassName('ircButton');
  2214. if (!opt_exportirc.checked) {
  2215. for (var i=0; i<carr.length; i++)
  2216. carr[i].className = carr[i].className.replace(/taButtonsOff/, "taButtons");
  2217. opt_exportirc.checked=true; ircspan.className += " cpSpansOn"+sufTheme;
  2218. } else {
  2219. for (var i=0; i<carr.length; i++)
  2220. carr[i].className = carr[i].className.replace(/taButtons/, "taButtonsOff");
  2221. opt_exportirc.checked=false; ircspan.className = ircspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2222. }
  2223. }, false);
  2224. ircspan.appendChild(document.createTextNode(" IRC "));
  2225. //ircspan.appendChild(opt_exportirc);
  2226. control_panel.appendChild(ircspan);
  2227.  
  2228. var redditspan = document.createElement("SPAN");
  2229. redditspan.className += " cpSpans"+sufTheme;
  2230. redditspan.title = 'Show a button to export the specified HIT formatted to r/HITsWorthTurkingFor standards .';
  2231. if (opt_exportreddit.checked) redditspan.className += " cpSpansOn"+sufTheme;
  2232. redditspan.addEventListener("click", function(){
  2233. var carr = document.getElementsByClassName('redditButton');
  2234. if (!opt_exportreddit.checked) {
  2235. for (var i=0; i<carr.length; i++)
  2236. carr[i].className = carr[i].className.replace(/taButtonsOff/, "taButtons");
  2237. opt_exportreddit.checked=true; redditspan.className += " cpSpansOn"+sufTheme;
  2238. } else {
  2239. for (var i=0; i<carr.length; i++)
  2240. carr[i].className = carr[i].className.replace(/taButtons/, "taButtonsOff");
  2241. opt_exportreddit.checked=false; redditspan.className = redditspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2242. }
  2243. }, false);
  2244. redditspan.appendChild(document.createTextNode(" HWTF "));
  2245. //redditspan.appendChild(opt_exportreddit);
  2246. control_panel.appendChild(redditspan);
  2247.  
  2248. control_panel.appendChild(document.createElement("P"));
  2249. control_panel.appendChild(document.createTextNode("Search Terms: "));
  2250. control_panel.appendChild(search_input);
  2251. search_input.size = 20;
  2252. search_input.title = "Enter keywords to search for.\n" + "Default is blank (no search terms).";
  2253. search_input.placeholder="Enter search terms here";
  2254. search_input.style.textAlign = "left";
  2255. search_input.className = "cpInput"+sufTheme;
  2256.  
  2257. var blockspan = document.createElement("SPAN");
  2258. blockspan.appendChild(document.createTextNode("Hide blocklisted"));
  2259. blockspan.className += " cpSpans"+sufTheme;
  2260. 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.";
  2261. if (useBlock.checked) blockspan.className += " cpSpansOn"+sufTheme;
  2262. blockspan.addEventListener("click", function() {
  2263. if (useBlock.checked) {
  2264. var mt = document.getElementsByClassName("scraperBlockedRow");
  2265. if (mt.length > 0) for (var i=mt.length-1; i>-1; i--)
  2266. mt[i].className = mt[i].className.replace(/(?:^| +)scraperBlockedRow/, "scraperBlockedRowOff");
  2267. useBlock.checked = false;
  2268. blockspan.className = blockspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2269. }
  2270. else {
  2271. var mt = document.getElementsByClassName("scraperBlockedRowOff");
  2272. if (mt.length > 0) for (var i=mt.length-1; i>-1; i--)
  2273. mt[i].className = mt[i].className.replace(/(?:^| +)scraperBlockedRowOff/, "scraperBlockedRow");
  2274. useBlock.checked = true;
  2275. blockspan.className += " cpSpansOn"+sufTheme;
  2276. } }, false);
  2277. //blockspan.appendChild(useBlock);
  2278. //useBlock.onclick = function() { if (useBlock.checked) blockspan.className += " cpSpansOn"+sufTheme; };
  2279. control_panel.appendChild(spacer.cloneNode(true));
  2280. control_panel.appendChild(blockspan);
  2281.  
  2282. var includelistspan = document.createElement("SPAN");
  2283. includelistspan.appendChild(document.createTextNode("Restrict to includelist"));
  2284. includelistspan.className += " cpSpans"+sufTheme;
  2285. includelistspan.title = "Show only HITs that match your includelist.\n" + "Be sure to edit your includelist first or no results will be displayed.";
  2286. if (matchOnly.checked) includelistspan.className += " cpSpansOn"+sufTheme;
  2287. includelistspan.addEventListener("click", function() {
  2288. if (matchOnly.checked) {
  2289. matchOnly.checked = false;
  2290. includelistspan.className = includelistspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2291. }
  2292. else {
  2293. matchOnly.checked = true;
  2294. includelistspan.className += " cpSpansOn"+sufTheme;
  2295. } }, false);
  2296. //includelistspan.appendChild(matchOnly);
  2297. //matchOnly.onclick = function() { if (matchOnly.checked) includelistspan.className += " cpSpansOn"+sufTheme; };
  2298. control_panel.appendChild(spacer.cloneNode(true));
  2299. control_panel.appendChild(includelistspan);
  2300.  
  2301. var highincspan = document.createElement("SPAN");
  2302. highincspan.appendChild(document.createTextNode("Highlight Includelisted"));
  2303. highincspan.className += " cpSpans"+sufTheme;
  2304. highincspan.title = "Outline HITs that match your includelist with a dashed green border.";
  2305. if (highlightIncludes_input.checked) highincspan.className += " cpSpansOn"+sufTheme;
  2306. highincspan.addEventListener("click", function() {
  2307. if (highlightIncludes_input.checked) {
  2308. var mt = document.getElementsByClassName("scraperIncludelistedRow");
  2309. if (mt.length > 0)
  2310. for (var i=mt.length-1; i>-1; i--) { mt[i].className = mt[i].className.replace(/(?:^| +)scraperIncludelistedRow/, "scraperIncludelistedRowOff"); }
  2311. highlightIncludes_input.checked = false;
  2312. highincspan.className = highincspan.className.replace(/(?:^| +)cpSpansOn_[a-zA-Z]+/g, "");
  2313. }
  2314. else {
  2315. var mt = document.getElementsByClassName("scraperIncludelistedRowOff");
  2316. if ( mt.length > 0)
  2317. for (var i=mt.length-1; i>-1; i--) { mt[i].className = mt[i].className.replace(/(?:^| +)scraperIncludelistedRowOff/, "scraperIncludelistedRow"); }
  2318. highlightIncludes_input.checked = true;
  2319. highincspan.className += " cpSpansOn"+sufTheme;
  2320. } }, false);
  2321. //highincspan.appendChild(highlightIncludes_input);
  2322. //highlightIncludes_input.onclick = function() { if (highlightIncludes_input.checked) highincspan.className += " cpSpansOn"+sufTheme; };
  2323. control_panel.appendChild(spacer.cloneNode(true));
  2324. control_panel.appendChild(highincspan);
  2325.  
  2326.  
  2327.  
  2328.  
  2329.  
  2330. big_red_button.textContent = "Start";
  2331. big_red_button.className += " cpButtons"+sufTheme;
  2332. big_red_button.onclick = function(){start_running();};
  2333.  
  2334. // open blocklist editor
  2335. reset_blocks.textContent = "Edit blocklist";
  2336. reset_blocks.className += " cpButtons"+sufTheme;
  2337. reset_blocks.setAttribute("id","blocklist_reset_button");
  2338. reset_blocks.onclick = function(){
  2339. //console.log("in");
  2340. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  2341. var textarea = $("#block_text");
  2342. var text = "";
  2343. for (var i = 0; i < ignore_list.length; i++){
  2344. text += ignore_list[i]+"^";
  2345. }
  2346. textarea.val(text.substring(0, text.length - 1));
  2347. $("#block_div").show();
  2348. };
  2349.  
  2350. // open includelist editor
  2351. include_button.textContent = "Edit includelist";
  2352. include_button.className += " cpButtons"+sufTheme;
  2353. include_button.id = "includes_reset_button";
  2354. include_button.onclick = function() {
  2355. include_list = GM_getValue("scraper_include_list").split('^');
  2356. var textarea = $("#include_text");
  2357. var text = "";
  2358. for (var i = 0; i < include_list.length; i++){
  2359. text += include_list[i]+"^";
  2360. }
  2361. textarea.val(text.substring(0, text.length - 1));
  2362. $("#include_div").show();
  2363. };
  2364.  
  2365. control_panel.appendChild(document.createElement("P"));
  2366. text_area.style.fontWeight = "normal";
  2367. 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';
  2368. var col_heads = ['Requester','Title','Reward','# Avail','TO pay','Accept HIT','M?'];
  2369. var row = text_area.createTHead().insertRow(0);
  2370. text_area.caption.style.fontWeight = 800;
  2371. text_area.caption.className = "tabhead"+sufTheme;
  2372. if (default_text_size > 10)
  2373. text_area.cellPadding=Math.min(Math.max(1,Math.floor((default_text_size-10)/2)),5);
  2374. //console.log(text_area.cellPadding);
  2375. //text_area.cellPadding=2;
  2376. text_area.caption.style.fontSize = "24px";
  2377. text_area.rows[0].style.fontWeight = 800;
  2378. for (var i=0; i<col_heads.length; i++)
  2379. {
  2380. var this_cell = row.insertCell(i);
  2381. this_cell.innerHTML = col_heads[i];
  2382. this_cell.style.fontSize = "14px";
  2383. this_cell.className = "tabhead"+sufTheme;
  2384. if (i > 1)
  2385. this_cell.style.textAlign = 'center';
  2386. }
  2387. text_area.style.width = '100%';
  2388. var table_div = document.createElement('div');
  2389. table_div.className = "bodytable"+sufTheme;
  2390. text_area.rows[0].className = "tabHead"+sufTheme;
  2391. table_div.style.fontSize = "14px";
  2392. document.body.insertBefore(table_div,control_panel.nextSibling);
  2393. var header_hide_button = document.createElement('button');
  2394. table_div.appendChild(big_red_button);
  2395. table_div.appendChild(document.createTextNode("   "));
  2396. table_div.appendChild(header_hide_button);
  2397. table_div.appendChild(document.createTextNode("   "));
  2398. table_div.appendChild(reset_blocks);
  2399. table_div.appendChild(document.createTextNode("   "));
  2400. table_div.appendChild(include_button);
  2401. table_div.appendChild(document.createTextNode("   "));
  2402. table_div.appendChild(showButton);
  2403. $("#show_TO_stuff_button").hide();
  2404.  
  2405. table_div.appendChild(document.createElement("P"));
  2406. var statusdiv = document.createElement("DIV");
  2407. statusdiv.className = "statusdiv"+sufTheme;
  2408. statusdiv.appendChild(progress_report);
  2409. statusdiv.appendChild(document.createElement("P"));
  2410. statusdiv.appendChild(document.createTextNode("Status messages: "));
  2411. statusdiv.appendChild(status_report);
  2412. table_div.appendChild(statusdiv);
  2413. table_div.appendChild(document.createElement("P"));
  2414. table_div.appendChild(text_area);
  2415. header_hide_button.textContent = "Hide Settings";
  2416. header_hide_button.className += " cpButtons"+sufTheme;
  2417. header_hide_button.onclick = function(){
  2418. if (header_hide_button.textContent == "Hide Settings"){
  2419. $('#control_panel').hide(300);
  2420. header_hide_button.textContent = "Show Settings";
  2421. }
  2422. else{
  2423. $('#control_panel').show(300);
  2424. header_hide_button.textContent = "Hide Settings";
  2425. }
  2426. };
  2427. }
  2428.  
  2429. /********HIT EXPORT ADDITIONS*****/
  2430.  
  2431. var EDIT = false;
  2432. var HIT;
  2433.  
  2434. var TO_BASE = "http://turkopticon.ucsd.edu/";
  2435. var API_BASE = "https://mturk-api.istrack.in/";
  2436. var API_URL = API_BASE + "multi-attrs.php?ids=";
  2437.  
  2438. var DEFAULT_TEMPLATE = '[table][tr][td][b]Title:[/b] [url={prev_link}][COLOR=blue]{title}[/COLOR][/url]\n';
  2439. DEFAULT_TEMPLATE += '[b]Requester:[/b] [url=https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId={rid}][COLOR=blue]{requester}[/COLOR][/url]';
  2440. DEFAULT_TEMPLATE += ' [{rid}] ([url='+TO_BASE+'{rid}][COLOR=blue]TO[/COLOR][/url])';
  2441. DEFAULT_TEMPLATE += '\n[b]TO Ratings:[/b]{to_stuff}';
  2442. DEFAULT_TEMPLATE += '\n[b]Description:[/b] {description}';
  2443. DEFAULT_TEMPLATE += '\n[b]Time:[/b] {time}';
  2444. DEFAULT_TEMPLATE += '\n[b]Hits Available:[/b] {hits}';
  2445. DEFAULT_TEMPLATE += '\n[b]Reward:[/b] [COLOR=green][b]{reward}[/b][/COLOR]';
  2446. DEFAULT_TEMPLATE += '\n[b]Qualifications:[/b] {quals}[/td][/tr][/table]';
  2447.  
  2448. var TEMPLATE;
  2449. var EASYLINK;
  2450.  
  2451. if (typeof GM_getValue === 'undefined')
  2452. TEMPLATE = null;
  2453. else {
  2454. TEMPLATE = GM_getValue('HITScraper Template');
  2455. EASYLINK = GM_getValue('HITScraper Easylink');
  2456. }
  2457. if (TEMPLATE == null) {
  2458. TEMPLATE = DEFAULT_TEMPLATE;
  2459. }
  2460.  
  2461. function buildXhrUrl(rai) {
  2462. var url = API_URL;
  2463. var ri = rai;
  2464. url += rai;
  2465. return url;
  2466. }
  2467.  
  2468. function makeXhrQuery(url) {
  2469. var xhr = new XMLHttpRequest();
  2470. try{
  2471. xhr.open('GET', url, false);
  2472. xhr.send(null);
  2473. return $.parseJSON(xhr.response);
  2474. }
  2475. catch(err){
  2476. return "TO DOWN";
  2477. }
  2478. }
  2479.  
  2480. function getNamesForEmptyResponses(rai, resp) {
  2481. for (var rid in rai) {
  2482. if (rai.hasOwnProperty(rid) && resp[rid] == "") {
  2483. resp[rid] = $.parseJSON('{"name": "' + rai[rid][0].innerHTML + '"}');
  2484. }
  2485. }
  2486. return resp;
  2487. }
  2488.  
  2489. function getKeys(obj) {
  2490. var keys = [];
  2491. for (var key in obj) {
  2492. keys.push(key);
  2493. }
  2494. return keys;
  2495. }
  2496.  
  2497. function export_sel_deleg(item,index,extype) {
  2498. //console.log(item);
  2499. if (extype == "vb") export_func(item);
  2500. else display(item, extype);
  2501. }
  2502.  
  2503. function block_deleg(item,index) {
  2504. //console.log(item);
  2505. block(item,index);
  2506. }
  2507.  
  2508. function block(hit,index){
  2509. var blockType = ["requester","title"];
  2510. var blockThis = hit[blockType[index]].replace(/\s+/g," ").toLowerCase().trim();
  2511. ignore_list.push(blockThis);
  2512. GM_setValue("scraper_ignore_list",ignore_list.join('^'));
  2513. //console.log(GM_getValue("scraper_ignore_list"));
  2514. alert("\""+blockThis+"\" ignored (added to blocklist). Re-scrape.");
  2515. }
  2516.  
  2517. function search_deleg(item,index) {
  2518. //console.log(item);
  2519. var searches = ["rid","title"];
  2520. search(item,searches[index]);
  2521. }
  2522.  
  2523. function hit_sort_func()
  2524. {
  2525. return function(a,b) {
  2526. if (a.date == b.date) {
  2527. if (a.requesterName < b.requesterName)
  2528. return -1;
  2529. if (a.requesterName > b.requesterName)
  2530. return 1;
  2531. if (a.title < b.title)
  2532. return -1;
  2533. if (a.title > b.title)
  2534. return 1;
  2535. if (a.status < b.status)
  2536. return -1;
  2537. if (a.status > b.status)
  2538. return 1;
  2539. }
  2540. if (a.date > b.date)
  2541. return 1;
  2542. if (a.date < b.date)
  2543. return -1;
  2544. };
  2545. }
  2546.  
  2547. function escapeRegExp(str) {
  2548. return str.replace(/[-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
  2549. }
  2550.  
  2551. function search(item,search_type){
  2552. //return true;/*
  2553.  
  2554. var request = indexedDB.open("HITDB", v);
  2555. request.onsuccess = function(e) {
  2556. HITStorage.indexedDB.db = e.target.result;
  2557. var db = HITStorage.indexedDB.db;
  2558. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  2559. var store = trans.objectStore("HIT");
  2560.  
  2561. var req;
  2562. var results = [];
  2563. var index;
  2564. var range;
  2565. req = store.openCursor();
  2566. req.onsuccess = function(event) {
  2567. var cursor = event.target.result;
  2568. if (cursor) {
  2569. hit = cursor.value;
  2570. var keys = ['title', 'requesterId'];
  2571. var re = new RegExp(escapeRegExp(item[search_type]),"ig");
  2572. for (var k in keys)
  2573. {
  2574. if (hit[keys[k]] != null && re.test(hit[keys[k]].trim())){
  2575. results.push(cursor.value);
  2576. }
  2577. }
  2578.  
  2579. cursor.continue();
  2580. }
  2581. else {
  2582. //console.log(results);
  2583. results.sort(hit_sort_func());
  2584. show_results(results);
  2585. }
  2586. db.close();
  2587. };
  2588. request.onerror = HITStorage.indexedDB.onerror;/**/
  2589. };
  2590. }
  2591.  
  2592. function format_hit_line (hit, odd, status_color, new_day)
  2593. {
  2594. var line = '<tr style="background-color:';
  2595. if (odd)
  2596. line += '#f1f3eb;';
  2597. else
  2598. line += 'white;';
  2599. line += ' valign=top;';
  2600. if (new_day)
  2601. line += ' border: 0px dotted #000000; border-width: 2px 0px 0px 0px">';
  2602. else
  2603. line += '">';
  2604.  
  2605. line += '<td>' + hit.date + '</td>';
  2606. if (hit.requesterLink != null)
  2607. line += '<td style="width:165px"><a href="' + hit.requesterLink + '" title="Contact this Requester">' + hit.requesterName + '</a></td>';
  2608. else
  2609. line += '<td style="width:165px">' + hit.requesterName + '</td>';
  2610. line += '<td style="width:213px">' + hit.title + '</td>';
  2611. line += '<td style="width:45px">$' + hit.reward.toFixed(2) + '</td>';
  2612. line += '<td style="color:' + status_color + '; width:55px">' + hit.status + '</td>';
  2613. line += '<td><div style="width:225px; overflow:hidden">' + hit.feedback + '</div></td>';
  2614. line += '</tr>\n';
  2615. return line;
  2616. }
  2617.  
  2618. function status_color (status)
  2619. {
  2620. var color = "green";
  2621.  
  2622. if (status.match("Pending Approval"))
  2623. color = "orange";
  2624. else if (status.match("Rejected"))
  2625. color = "red";
  2626.  
  2627. return color;
  2628. }
  2629.  
  2630. function show_results (results){
  2631. var resultsWindow = window.open();
  2632. resultsWindow.document.write("<html><head><title>Status Detail Search Results</title></head><body>\n");
  2633. resultsWindow.document.write("<h1>HITs matching your search:</h1>\n");
  2634. resultsWindow.document.write('<table style="border: 1px solid black;border-collapse:collapse;width:90%;margin-left:auto;margin-right:auto;">\n');
  2635. 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');
  2636.  
  2637. var odd = true;
  2638. var sum = 0;
  2639. var sum_rejected = 0;
  2640. var sum_approved = 0;
  2641. var sum_pending = 0;
  2642.  
  2643.  
  2644. var new_day = false;
  2645.  
  2646. for (var i=0; i<results.length; i++) {
  2647. odd = !odd;
  2648. sum += results[i].reward;
  2649. if (results[i].status == 'Rejected')
  2650. sum_rejected += results[i].reward;
  2651. else if (results[i].status == 'Pending Approval')
  2652. sum_pending += results[i].reward;
  2653. else
  2654. sum_approved += results[i].reward;
  2655.  
  2656. if (i>0 && (results[i-1].date != results[i].date))
  2657. new_day = true;
  2658. else
  2659. new_day = false;
  2660. resultsWindow.document.write(format_hit_line(results[i], odd, status_color(results[i].status), new_day ));
  2661. }
  2662.  
  2663. 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');
  2664. resultsWindow.document.write("</table>");
  2665. resultsWindow.document.write("<p>Found " + results.length + " matching HITs. $" + sum_approved.toFixed(2) + " approved, " +
  2666. "$" + sum_rejected.toFixed(2) + " rejected and $" + sum_pending.toFixed(2) + " pending.</p>");
  2667. resultsWindow.document.write("</body></html>");
  2668. resultsWindow.document.close();
  2669. }
  2670.  
  2671. // ---- vB
  2672.  
  2673. function export_func(item) {
  2674. HIT = item;
  2675. gedit_button.textContent = 'Edit Template';
  2676. apply_template(item);
  2677. gdiv.style.display = 'block';
  2678. gtextarea.select();
  2679. }
  2680.  
  2681. function apply_template(hit_data) {
  2682. var txt = TEMPLATE;
  2683.  
  2684. var vars = ['title', 'requester', 'rid', 'description', 'reward', 'quals', 'prev_link', 'time', 'hits', 'to_stuff', 'to_text'];
  2685.  
  2686. var resp = null;
  2687. if (txt.indexOf('{to_text}') >= 0 || txt.indexOf('{to_stuff}') >= 0){
  2688. var url = buildXhrUrl(hit_data["rid"]);
  2689. resp = getTOMulti(hit_data["rid"]);
  2690. //console.log(resp);
  2691. }
  2692. var toText = "";
  2693. var toStuff = "";
  2694. var toData = "";
  2695. var numResp = (resp == null || resp == "TO DOWN" ? "n/a" : resp[hit_data["rid"]].reviews);
  2696. if (resp == "TO DOWN"){
  2697. toStuff = " [URL=\""+TO_BASE+hit_data['rid']+"\"]TO down.[/URL]";
  2698. toText = toStuff;
  2699. }
  2700. else if (resp == null || resp[hit_data["rid"]].attrs == null && resp != "TO DOWN") {
  2701. toStuff = " No TO ";
  2702. toText = " No TO ";
  2703. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  2704. toStuff += "(Submit a new TO rating for this requester)[/URL]";
  2705. }
  2706. else {
  2707. for (var key in resp[hit_data["rid"]].attrs) {
  2708. //toText += "\n[*]"+key+": "+resp[hit_data["requesterId"]].attrs[key]+"\n";
  2709. var i = 0;
  2710. var color = "green";
  2711. var name = key;
  2712. var num = Math.floor(resp[hit_data["rid"]].attrs[key]);
  2713. switch (key){
  2714. case "comm":
  2715. name = "Communicativity";
  2716. break;
  2717. case "pay":
  2718. name = "Generosity";
  2719. break;
  2720. case "fast":
  2721. name = "Promptness";
  2722. break;
  2723. case "fair":
  2724. name = "Fairness";
  2725. break;
  2726. default:
  2727. name = key;
  2728. break;
  2729. }
  2730. switch (num){
  2731. case 0:
  2732. color = "red";
  2733. break;
  2734. case 1:
  2735. color = "red";
  2736. break;
  2737. case 2:
  2738. color = "orange";
  2739. break;
  2740. case 3:
  2741. color = "yellow";
  2742. break;
  2743. default:
  2744. break;
  2745. }
  2746. toText += (num > 0 ? "\n[color="+color+"]" : "\n");
  2747. for (i; i < num; i++){
  2748. toText += "[b]"+symbol+"[/b]";
  2749. }
  2750. toText += (num > 0 ? "[/color]" : "");
  2751. if (i < 5){
  2752. toText += "[color=white]";
  2753. for (i; i < 5; i++)
  2754. toText += "[b]"+symbol+"[/b]";
  2755. toText += "[/color]";
  2756. }
  2757. toText += " "+Number(resp[hit_data["rid"]].attrs[key]).toFixed(2)+" "+name;
  2758. toData += Number(resp[hit_data["rid"]].attrs[key]).toFixed(2) + ",";
  2759. }
  2760. //toText += "[/list]";
  2761. 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]");
  2762. toStuff = '\n[img]http://data.istrack.in/to/' + toData.slice(0,-1) + '.png[/img]';
  2763. toStuff += (txt.indexOf('{to_stuff}') >= 0 ? (txt.indexOf('{to_text}') >= 0 ? "" : toText) : "");
  2764. toStuff += "\nNumber of Reviews: "+numResp;
  2765. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  2766. toStuff += "\n(Submit a new TO rating for this requester)[/URL]";
  2767. }
  2768.  
  2769. for (var u = 0; u < vars.length; u++) {
  2770. var t = new RegExp('\{' + vars[u] + '\}', 'g');
  2771. if (vars[u] == "to_stuff") {
  2772. txt = txt.replace(t, toStuff);
  2773. }
  2774. else if (vars[u] == "to_text") {
  2775. txt = txt.replace(t, toText);
  2776. }
  2777. else if (vars[u] == "prev_link") {
  2778. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[u]]);
  2779. }
  2780. else if (vars[u] == "acc_link") {
  2781. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[u]]);
  2782. }
  2783. else
  2784. txt = txt.replace(t, hit_data[vars[u]]);
  2785. }
  2786. gtextarea.value = txt;
  2787. }
  2788.  
  2789. function hide_func(div) {
  2790. if (EDIT == false)
  2791. div.style.display = 'none';
  2792. }
  2793.  
  2794. function edit_func() {
  2795. if (EDIT == true) {
  2796. EDIT = false;
  2797. TEMPLATE = gtextarea.value;
  2798. gedit_button.textContent = 'Edit Template';
  2799. apply_template(HIT);
  2800. }
  2801. else {
  2802. //console.log("Editing");
  2803. EDIT = true;
  2804. gedit_button.textContent = 'Show Changes';
  2805. gsave_button.disabled = false;
  2806. gtextarea.value = TEMPLATE;
  2807. }
  2808. }
  2809.  
  2810. function default_func() {
  2811. GM_deleteValue('HITScraper Template');
  2812. TEMPLATE = DEFAULT_TEMPLATE;
  2813. EDIT = false;
  2814. gedit_button.textContent = 'Edit Template';
  2815. apply_template(HIT);
  2816. }
  2817.  
  2818. function save_func() {
  2819. if (EDIT)
  2820. TEMPLATE = gtextarea.value;
  2821. GM_setValue('HITScraper Template', TEMPLATE);
  2822. }
  2823.  
  2824. var gdiv = document.createElement('div');
  2825. var gtextarea = document.createElement('textarea');
  2826. var gdiv2 = document.createElement('label');
  2827.  
  2828. gdiv.style.position = 'fixed';
  2829. gdiv.style.width = '500px';
  2830. gdiv.style.height = '235px';
  2831. gdiv.style.left = '50%';
  2832. gdiv.style.right = '50%';
  2833. gdiv.style.margin = '-250px 0px 0px -250px';
  2834. gdiv.style.top = '300px';
  2835. gdiv.style.padding = '5px';
  2836. gdiv.style.border = '2px';
  2837. gdiv.style.backgroundColor = 'black';
  2838. gdiv.style.color = 'white';
  2839. gdiv.style.zIndex = '100';
  2840.  
  2841. gtextarea.style.padding = '2px';
  2842. gtextarea.style.width = '500px';
  2843. gtextarea.style.height = '200px';
  2844. 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}';
  2845.  
  2846. gdiv.textContent = 'Press Ctrl+C to copy to clipboard. Click textarea to close';
  2847. gdiv.style.fontSize = '12px';
  2848. gdiv.appendChild(gtextarea);
  2849.  
  2850. var gedit_button = document.createElement('button');
  2851. var gsave_button = document.createElement('button');
  2852. var gdefault_button = document.createElement('button');
  2853. var geasy_button = document.createElement('button');
  2854.  
  2855. gedit_button.textContent = 'Edit Template';
  2856. gedit_button.setAttribute('id', 'edit_button');
  2857. gedit_button.style.height = '18px';
  2858. gedit_button.style.width = '100px';
  2859. gedit_button.style.fontSize = '10px';
  2860. gedit_button.style.paddingLeft = '3px';
  2861. gedit_button.style.paddingRight = '3px';
  2862. gedit_button.style.backgroundColor = 'white';
  2863.  
  2864. gsave_button.textContent = 'Save Template';
  2865. gsave_button.setAttribute('id', 'save_button');
  2866. gsave_button.style.height = '18px';
  2867. gsave_button.style.width = '100px';
  2868. gsave_button.style.fontSize = '10px';
  2869. gsave_button.style.paddingLeft = '3px';
  2870. gsave_button.style.paddingRight = '3px';
  2871. gsave_button.style.backgroundColor = 'white';
  2872. gsave_button.style.marginLeft = '5px';
  2873.  
  2874. geasy_button.textContent = 'Change Adfly Url';
  2875. geasy_button.setAttribute('id', 'easy_button');
  2876. geasy_button.style.height = '18px';
  2877. geasy_button.style.width = '100px';
  2878. geasy_button.style.fontSize = '10px';
  2879. geasy_button.style.paddingLeft = '3px';
  2880.  
  2881. gdefault_button.textContent = ' D ';
  2882. gdefault_button.setAttribute('id', 'default_button');
  2883. gdefault_button.style.height = '18px';
  2884. gdefault_button.style.width = '20px';
  2885. gdefault_button.style.fontSize = '10px';
  2886. gdefault_button.style.paddingLeft = '3px';
  2887. gdefault_button.style.paddingRight = '3px';
  2888. gdefault_button.style.backgroundColor = 'white';
  2889. gdefault_button.style.marginLeft = '5px';
  2890. gdefault_button.title = 'Return default template';
  2891.  
  2892. gdiv.appendChild(gedit_button);
  2893. gdiv.appendChild(gsave_button);
  2894. gdiv.appendChild(gdefault_button);
  2895. gdiv.appendChild(geasy_button);
  2896. gsave_button.disabled = true;
  2897.  
  2898. gdiv.style.display = 'none';
  2899. gtextarea.addEventListener("click", function() {hide_func(gdiv);}, false);
  2900. gedit_button.addEventListener("click", function() {edit_func();}, false);
  2901. gsave_button.addEventListener("click", function() {save_func();}, false);
  2902. gdefault_button.addEventListener("click", function() {default_func();}, false);
  2903. document.body.insertBefore(gdiv, document.body.firstChild);
  2904.  
  2905. // ---- vB end
  2906.  
  2907. //Functions below were added for the irc export with the help of clickhappier and Cristo
  2908.  
  2909. function getTO(f){
  2910. var toComp = [];
  2911. var toUrl2 = 'https://mturk-api.istrack.in/multi-attrs.php?ids='+f;
  2912. var toUrl = 'https://turkopticon.ucsd.edu/api/multi-attrs.php?ids='+f;
  2913. var requestTO = new XMLHttpRequest();
  2914. try{ // first try Miku's TO mirror server (istrack.in)
  2915. requestTO.onreadystatechange = function () {
  2916. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  2917. if (requestTO.responseText.split(':').length > 2) {
  2918. var toInfo = requestTO.responseText.split('{')[3].split('}')[0].split(',');
  2919. for (var t = 0; t < 4; t++) {
  2920. var arrTo = toInfo[t].split(':');
  2921. toComp.push(arrTo[1].substring(1,4));
  2922. }
  2923. }
  2924. else { toComp = ['-','-','-','-']; }
  2925. }
  2926. };
  2927. requestTO.open('GET', toUrl, false);
  2928. requestTO.send(null);
  2929. return toComp;
  2930. }
  2931. catch(err){ // if mirror unavailable, try main TO server
  2932. try{
  2933. requestTO.onreadystatechange = function () {
  2934. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  2935. if (requestTO.responseText.split(':').length > 2) {
  2936. var toInfo = requestTO.responseText.split('{')[3].split('}')[0].split(',');
  2937. for (var t = 0; t < 4; t++) {
  2938. var arrTo = toInfo[t].split(':');
  2939. toComp.push(arrTo[1].substring(1,4));
  2940. }
  2941. }
  2942. else { toComp = ['-','-','-','-']; }
  2943. }
  2944. };
  2945. requestTO.open('GET', toUrl2, false);
  2946. requestTO.send(null);
  2947. return toComp;
  2948. }
  2949. catch(err){ // if both unavailable, return 'na's
  2950. toComp = ['na','na','na','na'];
  2951. return toComp;
  2952. }
  2953. }
  2954. }
  2955.  
  2956. function getTOMulti(f){
  2957. var toComp = {};
  2958. var toUrl2 = 'https://mturk-api.istrack.in/multi-attrs.php?ids='+f;
  2959. var toUrl = 'https://turkopticon.ucsd.edu/api/multi-attrs.php?ids='+f;
  2960. var rids = f.split(',');
  2961. var requestTO = new XMLHttpRequest();
  2962. try{ // first try Miku's TO mirror server (istrack.in)
  2963. requestTO.onreadystatechange = function () {
  2964. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  2965. if (requestTO.responseText.split(':').length > 2)
  2966. toComp = $.parseJSON(requestTO.responseText);
  2967. else
  2968. toComp = null;
  2969. }
  2970. };
  2971. requestTO.open('GET', toUrl, false);
  2972. requestTO.send(null);
  2973. return toComp;
  2974. }
  2975. catch(err){ // if mirror unavailable, try main TO server
  2976. try{
  2977. requestTO.onreadystatechange = function () {
  2978. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  2979. if (requestTO.responseText.split(':').length > 2)
  2980. toComp = $.parseJSON(requestTO.responseText);
  2981. else
  2982. toComp = null;
  2983. }
  2984. };
  2985. requestTO.open('GET', toUrl2, false);
  2986. requestTO.send(null);
  2987. return toComp;
  2988. }
  2989. catch(err){ // if both unavailable, return 'na's
  2990. toComp = "TO DOWN";
  2991. return toComp;
  2992. }
  2993. }
  2994. }
  2995.  
  2996. function sleep(ms){ // from http://www.digimantra.com/tutorials/sleep-or-wait-function-in-javascript/
  2997. var dt = new Date();
  2998. dt.setTime(dt.getTime() + ms);
  2999. while (new Date().getTime() < dt.getTime());
  3000. }
  3001. function ns4tShorten(url){ // mturk-only URL shortener on Tjololo's server ns4t.net
  3002. console.log("ns4tShorten function");
  3003. var shortRes;
  3004. var urlT = "https://ns4t.net/yourls-api.php" + "?action=shorturl&url=" + encodeURIComponent(url) + "&format=simple&title=MTurk&username=publicuser&password=publicpass";
  3005. var requestNs4t = new XMLHttpRequest();
  3006. try{
  3007. requestNs4t.onreadystatechange = function () {
  3008. if (requestNs4t.readyState == 4) {
  3009. if (requestNs4t.status == 200) {
  3010. shortRes = requestNs4t.responseText;
  3011. console.log("ns4t.net response: " + requestNs4t.status + " " + requestNs4t.statusText + " " + requestNs4t.responseText);
  3012. }
  3013. else {
  3014. console.log('ns4t.net unsuccessful: ' + requestNs4t.status + " " + requestNs4t.statusText);
  3015. }
  3016. }
  3017. };
  3018. requestNs4t.open('GET', urlT, false);
  3019. requestNs4t.send(null);
  3020. return shortRes;
  3021. }
  3022. catch(err){
  3023. return shortRes;
  3024. }
  3025. }
  3026. 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)
  3027. console.log("tnyimShorten function");
  3028. var shortRes;
  3029. var urlT = "https://tny.im/yourls-api.php" + "?action=shorturl&url=" + encodeURIComponent(url) + "&format=simple&title=MTurk";
  3030. var requestTnyim = new XMLHttpRequest();
  3031. try{
  3032. requestTnyim.onreadystatechange = function () {
  3033. if (requestTnyim.readyState == 4) {
  3034. if (requestTnyim.status == 200) {
  3035. shortRes = requestTnyim.responseText;
  3036. console.log("tny.im response: " + requestTnyim.status + " " + requestTnyim.statusText + " " + requestTnyim.responseText);
  3037. }
  3038. else {
  3039. console.log('tny.im unsuccessful: ' + requestTnyim.status + " " + requestTnyim.statusText);
  3040. }
  3041. }
  3042. };
  3043. requestTnyim.open('GET', urlT, false);
  3044. requestTnyim.send(null);
  3045. return shortRes;
  3046. }
  3047. catch(err){
  3048. return shortRes;
  3049. }
  3050. }
  3051. function googlShorten(url){ // Goo.gl URL Shortener
  3052. console.log("googlShorten function");
  3053. var shortRes;
  3054. var urlG = "https://www.googleapis.com/urlshortener/v1/url";
  3055. var requestGoogl = new XMLHttpRequest();
  3056. try{
  3057. requestGoogl.open("POST", urlG, false);
  3058. requestGoogl.setRequestHeader("Content-Type", "application/json");
  3059. requestGoogl.onreadystatechange = function() {
  3060. if (requestGoogl.readyState == 4) {
  3061. if (requestGoogl.status == 200) {
  3062. shortRes = JSON.parse(requestGoogl.response).id;
  3063. console.log("goo.gl response: " + requestGoogl.status + " " + requestGoogl.statusText + " " + JSON.parse(requestGoogl.response).id );
  3064. }
  3065. else {
  3066. console.log('goo.gl unsuccessful: ' + requestGoogl.status + " " + requestGoogl.statusText);
  3067. }
  3068. }
  3069. };
  3070. var data = new Object();
  3071. data.longUrl = url;
  3072. requestGoogl.send(JSON.stringify(data));
  3073. return shortRes;
  3074. }
  3075. catch(err){
  3076. return shortRes;
  3077. }
  3078. }
  3079. function shortenUrl(url){
  3080. sleep(500); // milliseconds delay - wait some milliseconds (currently half a second) between shortens to reduce chance of hitting usage limits
  3081. var shortRes;
  3082. shortRes = ns4tShorten(url);
  3083. if ( shortRes === undefined ) { // if you reached the ns4t.net URL shortener's temporary usage limits or the server is otherwise unavailable
  3084. shortRes = tnyimShorten(url);
  3085. if ( shortRes === undefined ) { // if you reached the tny.im URL shortener's temporary limits or the server is otherwise unavailable
  3086. shortRes = googlShorten(url);
  3087. if ( shortRes === undefined ) { // if you reached the Google URL shortener's temporary limits too or the server is otherwise unavailable
  3088. shortRes = "(x)";
  3089. }
  3090. }
  3091. }
  3092. return shortRes;
  3093. }
  3094.  
  3095. // output display box
  3096. // this is messy
  3097. // TODO: make it cleaner
  3098. var exportdiv = document.createElement('div');
  3099. var exporttextarea1 = document.createElement('textarea'); var exporttextarea2 = document.createElement('textarea');
  3100. var exporttextarea3 = document.createElement('textarea'); var exporttextarea4 = document.createElement('textarea');
  3101. var exporttextarea5 = document.createElement('textarea');
  3102. var exportclosebutton = document.createElement('button');var exporttitlebutton = document.createElement('button');var exporturlbutton = document.createElement('button');
  3103. var exportreqbutton = document.createElement('button');var exportpandabutton = document.createElement('button');var exporttobutton = document.createElement('button');
  3104. exportclosebutton.style.backgroundColor = 'black';
  3105. exportclosebutton.style.color = 'white';
  3106. exportclosebutton.style.border = 'none';
  3107. exportclosebutton.style.width = '505px';
  3108. exportclosebutton.style.align = 'center';
  3109. exportclosebutton.textContent='Close';
  3110. exporttitlebutton.style.backgroundColor = 'black';
  3111. exporttitlebutton.style.color = 'white';
  3112. exporttitlebutton.style.border = 'none';
  3113. exporttitlebutton.style.width = '101px';
  3114. exporttitlebutton.style.align = 'center';
  3115. exporttitlebutton.textContent='Title';
  3116. exporturlbutton.style.backgroundColor = 'black';
  3117. exporturlbutton.style.color = 'white';
  3118. exporturlbutton.style.border = 'none';
  3119. exporturlbutton.style.width = '101px';
  3120. exporturlbutton.style.align = 'center';
  3121. exporturlbutton.textContent='URL';
  3122. exportreqbutton.style.backgroundColor = 'black';
  3123. exportreqbutton.style.color = 'white';
  3124. exportreqbutton.style.border = 'none';
  3125. exportreqbutton.style.width = '101px';
  3126. exportreqbutton.style.align = 'center';
  3127. exportreqbutton.textContent='Req';
  3128. exportpandabutton.style.backgroundColor = 'black';
  3129. exportpandabutton.style.color = 'white';
  3130. exportpandabutton.style.border = 'none';
  3131. exportpandabutton.style.width = '101px';
  3132. exportpandabutton.style.align = 'center';
  3133. exportpandabutton.textContent='PandA';
  3134. exporttobutton.style.backgroundColor = 'black';
  3135. exporttobutton.style.color = 'white';
  3136. exporttobutton.style.border = 'none';
  3137. exporttobutton.style.width = '101px';
  3138. exporttobutton.style.align = 'center';
  3139. exporttobutton.textContent='TO';
  3140. exportdiv.style.position = 'fixed';
  3141. exportdiv.style.width = '505px';
  3142. exportdiv.style.height = '155px';
  3143. exportdiv.style.left = '50%';
  3144. exportdiv.style.right = '50%';
  3145. exportdiv.style.margin = '-250px 0px 0px -250px';
  3146. exportdiv.style.top = '300px';
  3147. exportdiv.style.padding = 'none'; // def 5px
  3148. exportdiv.style.border = 'none'; //def 2px
  3149. exportdiv.style.backgroundColor = 'black';
  3150. exportdiv.style.color = 'white';
  3151. exportdiv.style.zIndex = '1000';
  3152. exportdiv.setAttribute('id','export_div');
  3153. exportdiv.style.display = 'none';
  3154. exporttextarea1.style.padding = 'none';
  3155. exporttextarea1.style.width = '500px';
  3156. exporttextarea2.style.padding = 'none';
  3157. exporttextarea2.style.width = '500px';
  3158. exporttextarea2.style.height = '30px';
  3159. exporttextarea2.style.overflow = 'hidden';
  3160. exporttextarea2.setAttribute('id','hwtf_url');
  3161. exporttextarea3.style.padding = 'none';
  3162. exporttextarea3.style.width = '500px';
  3163. exporttextarea3.style.height = '30px';
  3164. exporttextarea3.style.overflow = 'hidden';
  3165. exporttextarea3.setAttribute('id','hwtf_req');
  3166. exporttextarea4.style.padding = 'none';
  3167. exporttextarea4.style.width = '500px';
  3168. exporttextarea4.style.height = '30px';
  3169. exporttextarea4.style.overflow = 'hidden';
  3170. exporttextarea4.setAttribute('id','hwtf_panda');
  3171. exporttextarea5.style.padding = '2px';
  3172. exporttextarea5.style.width = '500px';
  3173. exporttextarea5.style.height = '20px';
  3174. exporttextarea5.style.overflow = 'hidden';
  3175. exporttextarea5.setAttribute('id','hwtf_to');
  3176. exportdiv.style.fontSize = '12px';
  3177. exportdiv.style.display = 'block';
  3178.  
  3179.  
  3180. function display(hit, extype){
  3181. var capReq = hit["requester"];
  3182. var capReqId = hit["rid"];
  3183. var capTitle = hit["title"].replace(" (Requester link substituted)","");
  3184. var capRew = hit["reward"];
  3185. var capTime = hit["time"];
  3186. var capAvailable = hit["hits"];
  3187. var qualList = hit["quals"];
  3188. var qualColl = qualList.split(',');
  3189. var masterStat = '';
  3190. var locationStat = 'ICA';
  3191. var locationIndex = null;
  3192. for ( var m = 0; m < qualColl.length; m++ ) {
  3193. if ( qualColl[m].indexOf('Masters') > -1 ) {
  3194. masterStat = (extype == "irc") ? 'MASTERS • ' : ' [MASTERS]';
  3195. }
  3196. else if (qualColl[m].indexOf('is US') > -1) {
  3197. locationStat = 'US';
  3198. locationIndex = m;
  3199. }
  3200. else if (qualColl[m].match(/approval rate/)) {
  3201. qualColl[m] = qualColl[m].replace(/[A-Za-z ]+\(%\) is (?:not less than (\d+)|greater than (\d+))/, '>$1$2%');
  3202. }
  3203. else if (qualColl[m].match(/approved HITs/)) {
  3204. qualColl[m] = qualColl[m].replace(/Total approved HITs is (?:not less than (\d+)|greater than (\d+))/, '>$1$2');
  3205. qualColl[m] = qualColl[m].replace(/Total approved HITs is (?:not greater than (\d+)|less than (\d+))/, '<$1$2');
  3206. }
  3207. }
  3208. if (extype == "reddit" && locationIndex != null) qualColl.splice(locationIndex,1);
  3209. if (qualColl[0] == null) qualColl = "None";
  3210.  
  3211. //console.log('col: ' + qualColl);
  3212. //console.log('list: ' + qualList);
  3213.  
  3214. //URL SHORTENERS
  3215. var capUrl = (extype == "irc") ? shortenUrl(LINK_BASE+hit["prev_link"]) : LINK_BASE+hit["prev_link"];
  3216. var pandaUrl = (extype == "irc") ? shortenUrl(LINK_BASE+hit["acc_link"]) : LINK_BASE+hit["acc_link"];
  3217. var capReqUrl = (extype == "irc") ? shortenUrl(LINK_BASE+hit["req_link"]) : LINK_BASE+hit["req_link"];
  3218. var toLink = (extype == "irc") ? shortenUrl(TO_BASE+capReqId) : TO_BASE+capReqId;
  3219.  
  3220. var hitLinkUnav = '';
  3221. if (hit["prev_link"].indexOf("requesterId") > -1)
  3222. {
  3223. hitLinkUnav = " (preview link unavailable)";
  3224. pandaUrl = "";
  3225. }
  3226. var capToStats = getTO(capReqId);
  3227. var shortUrlUnav = '';
  3228. if ( capUrl == '(x)' && hit["prev_link"] != 'unavailable' )
  3229. {
  3230. shortUrlUnav = ' \r\n^ https://www.mturk.com'+hit["prev_link"];
  3231. } // add the full-length preview link at the end if both URL shortener attempts failed
  3232.  
  3233. var exIRC = masterStat + 'Requester: ' + capReq + ' ' + capReqUrl + ' • ' + 'HIT: ' + capTitle + ' ' + capUrl + hitLinkUnav + ' • ' + 'Pay: ' + capRew + ' • ' + 'Avail: ' + capAvailable + ' • ' + 'Time Limit: ' + capTime + ' • ' + 'TO: ' + 'Pay='+capToStats[1] + ' Fair='+capToStats[2] + ' Comm='+capToStats[0] + ' ' + toLink + ' • ' + 'PandA: ' + pandaUrl + hitLinkUnav + shortUrlUnav ;
  3234. var exTitle = locationStat + ' - ' + capTitle + ' - ' + capReq + ' - ' + capRew + '/' + 'COMTIME - ' + '(' + qualColl + ')' + masterStat;
  3235. var exUrl = 'URL: ' + capUrl +'\n\n';
  3236. var exReq = 'Req: ' + capReqUrl +'\n\n';
  3237. var exTO = 'TO: ' + toLink +'\n\n';
  3238. var exPanda = 'PandA: ' + pandaUrl +'\n\n';
  3239.  
  3240. if (extype == "irc") {
  3241. if (GM_setClipboard) { GM_setClipboard(exIRC); } // pre-copies irc string
  3242. exporttextarea1.value = exIRC;
  3243. exporttextarea1.style.height = '130px';
  3244. exporttextarea1.setAttribute('id', 'ircexport_text');
  3245. exportdiv.textContent = 'IRC Export: Press Ctrl+C to (re-)copy to clipboard. Click textarea to close.';
  3246. exportdiv.appendChild(exporttextarea1);
  3247. exporttextarea1.addEventListener("click", function(){ exportdiv.style.display = 'none'; }, false);
  3248. exportdiv.style.display = 'block';
  3249. exporttextarea1.select();
  3250. }
  3251. else if (extype == "reddit") {
  3252. exporttextarea1.value = exTitle;
  3253. exporttextarea2.value = exUrl;
  3254. exporttextarea3.value = exReq;
  3255. exporttextarea4.value = exPanda;
  3256. exporttextarea5.value = exTO;
  3257. exporttextarea1.style.height = '50px';
  3258. exporttextarea1.setAttribute('id','hwtf_title');
  3259. exportdiv.textContent = 'r/hwtf Export: Use the buttons for single-click copying.';
  3260. exportdiv.appendChild(exporttextarea1);
  3261. exportdiv.appendChild(exporttextarea2);
  3262. exportdiv.appendChild(exporttextarea3);
  3263. exportdiv.appendChild(exporttextarea4);
  3264. exportdiv.appendChild(exporttextarea5);
  3265. exportdiv.appendChild(exporttitlebutton);
  3266. exportdiv.appendChild(exporturlbutton);
  3267. exportdiv.appendChild(exportreqbutton);
  3268. exportdiv.appendChild(exportpandabutton);
  3269. exportdiv.appendChild(exporttobutton);
  3270. exportdiv.appendChild(exportclosebutton);
  3271. document.body.insertBefore(exportdiv, document.body.firstChild);
  3272. exportclosebutton.addEventListener("click", function(){ exportdiv.style.display = 'none'; }, false);
  3273. exporttitlebutton.addEventListener("click", function(){ exporttextarea1.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea1.value); } }, false);
  3274. exporturlbutton.addEventListener("click", function(){ exporttextarea2.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea2.value); } }, false);
  3275. exportreqbutton.addEventListener("click", function(){ exporttextarea3.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea3.value); } }, false);
  3276. exportpandabutton.addEventListener("click", function(){ exporttextarea4.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea4.value); } }, false);
  3277. exporttobutton.addEventListener("click", function(){ exporttextarea5.select(); if (GM_setClipboard) { GM_setClipboard(exporttextarea5.value); } }, false);
  3278. exportdiv.style.display = 'block';
  3279. //exporttextarea1.focus();
  3280. var tindex = exTitle.search(/COMTIME/);
  3281. exporttextarea1.setSelectionRange(tindex, tindex+7); // pre-select completion time
  3282. }
  3283. }