HIT Scraper WITH EXPORT

Snag HITs.

当前为 2015-02-22 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name HIT Scraper WITH EXPORT
  3. // @author Kerek, TJ, and clickhappier
  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 1.6
  10. // @grant GM_xmlhttpRequest
  11. // @grant GM_setClipboard
  12. // @require http://code.jquery.com/jquery-latest.min.js
  13. // ==/UserScript==
  14.  
  15. //Fixes it soas not to rely on GM's storage schemes
  16. if (!this.GM_getValue || (this.GM_getValue.toString && this.GM_getValue.toString().indexOf("not supported")>-1)) {
  17. this.GM_getValue=function (key,def) {
  18. return localStorage[key] || def;
  19. };
  20. this.GM_setValue=function (key,value) {
  21. return localStorage[key]=value;
  22. };
  23. this.GM_deleteValue=function (key) {
  24. return localStorage.removeItem(key);
  25. };
  26. }
  27.  
  28. //alter the requester ignore last as you desire, case insensitive
  29. var default_list = ["oscar smith", "Diamond Tip Research LLC", "jonathon weber", "jerry torres", "Crowdsource", "we-pay-you-fast", "turk experiment", "jon brelig"];
  30. var ignore_list = default_list;
  31. if (GM_getValue("scraper_ignore_list"))
  32. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  33. else
  34. GM_setValue("scraper_ignore_list", default_list.join('^'));
  35.  
  36. var include_list = [];
  37. if (GM_getValue("scraper_include_list"))
  38. include_list = GM_getValue("scraper_include_list").split('^');
  39.  
  40. //This is to update the hit export symbol
  41. var symbol = "☭";
  42.  
  43. //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.
  44. var correct_for_skips = true;
  45.  
  46. //This will hide masters if selected to "hide"
  47. var hide_masters = false;
  48.  
  49. //This is to test sorting by TO
  50. var sort_TO = false;
  51.  
  52. //weight the four TO ratings for the coloring. Default has pay twice as important as fairness and nothing for communication and fast.
  53. var COMM_WEIGHT = 0;
  54. var PAY_WEIGHT = 10;
  55. var FAIR_WEIGHT = 5;
  56. var FAST_WEIGHT = 0;
  57.  
  58. //Used for themeing, change the colors to change how scraper looks
  59. var GREEN = '#66CC66'; // > 4
  60. var LIGHTGREEN = '#ADFF2F'; // > 3 GREEN YELLOW
  61. var YELLOW = '#FFD700'; //Not used
  62. var ORANGE = '#FF9900'; // > 2
  63. var RED = '#FF3030'; // <= 2
  64. var BLUE = '#C0D9D9'; // no TO
  65. var GREY = 'lightGrey'; //TO down
  66. var BROWN = '#94704D'; //Font color
  67. var DARKGREY = '#9F9F9F'; //No HITDB, "Not Qualified" column
  68. var BACKGROUND_COLOR = "rgb(19, 19, 19)"; //Background of page
  69.  
  70. //display your hitdb records if applicable
  71. var check_hitDB = true;
  72.  
  73. //default text size
  74. var default_text_size=11;
  75.  
  76. //set to "true" to override checkbox setting and ding on new hits
  77. var newHitDing = false;
  78.  
  79. //DO NOT EDIT ANYTHING BELOW THIS LINE UNLESS YOU KNOW WHAT YOU ARE DOING!
  80.  
  81. var status_text = "None";
  82. var shouldDing = false;
  83. var audio = document.createElement('audio');
  84. document.body.appendChild(audio);
  85. audio.src = 'data:audio/ogg;base64,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VVVVVVVVVVVVVQBqs0jMnBWmgKng1FRU32lmZmb+N6/V+w+MygnAXTBSU3/Ffjw4KGwvhYijgbg038TRQL8U3//xv5cO4FZf/z8bdj8V3CpuILkwNl5P8KE+53G3eCoQUAIQxdpeAGCps5MdqtGpNehUFBQFS2qx1bDEzNSDATAKKlIBjsJGEQBFZnmMUrStzKW+pXKVDOZ+hnQ2jloarUNK2iGVjPlUVcHRF4NCI9e2HcNCJ4NQ0eEQlnbmh0FYaUxBTUUzLjk5LjNVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVf/7kkScD/K0GL+QyBnCYEgHsjBlXAAAAaQAAAAgAAA0gAAABFVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVVU=';
  86. audio.setAttribute("id", "ding_noise");
  87. audio.volume = 1;
  88.  
  89. function newHits(dingNoise) {
  90. console.log(dingNoise);
  91. if (dingNoise || newHitDing)
  92. document.getElementById("ding_noise").play();
  93. }
  94.  
  95. //For editing the blocklist
  96. var div = document.createElement('div');
  97. var textarea = document.createElement('textarea');
  98.  
  99. div.style.position = 'fixed';
  100. div.style.width = '500px';
  101. div.style.height = '235px';
  102. div.style.left = '50%';
  103. div.style.right = '50%';
  104. div.style.margin = '-250px 0px 0px -250px';
  105. div.style.top = '300px';
  106. div.style.padding = '5px';
  107. div.style.border = '2px';
  108. div.style.backgroundColor = 'black';
  109. div.style.color = 'white';
  110. div.style.zIndex = '100';
  111. div.setAttribute('id','block_div');
  112.  
  113. textarea.style.padding = '2px';
  114. textarea.style.width = '500px';
  115. textarea.style.height = '180px';
  116. textarea.title = 'Block list';
  117. textarea.setAttribute('id','block_text');
  118.  
  119. div.textContent = 'Change the blocklist to be whatever you like, save to save it. Separate requesters with the ^ character. After clicking "Save", you\'ll need to scrape again to apply the changes.';
  120. div.style.fontSize = '12px';
  121. div.appendChild(textarea);
  122.  
  123. var save_button = document.createElement('button');
  124. var cancel_button = document.createElement('button');
  125.  
  126. save_button.textContent = 'Save';
  127. save_button.setAttribute('id', 'save_blocklist');
  128. save_button.style.height = '18px';
  129. save_button.style.width = '100px';
  130. save_button.style.fontSize = '10px';
  131. save_button.style.paddingLeft = '3px';
  132. save_button.style.paddingRight = '3px';
  133. save_button.style.backgroundColor = 'white';
  134. save_button.style.marginLeft = '5px';
  135.  
  136. cancel_button.textContent = 'Cancel';
  137. cancel_button.setAttribute('id', 'cancel_blocklist');
  138. cancel_button.style.height = '18px';
  139. cancel_button.style.width = '100px';
  140. cancel_button.style.fontSize = '10px';
  141. cancel_button.style.paddingLeft = '3px';
  142. cancel_button.style.paddingRight = '3px';
  143. cancel_button.style.backgroundColor = 'white';
  144. cancel_button.style.marginLeft = '5px';
  145.  
  146.  
  147. div.appendChild(save_button);
  148. div.appendChild(cancel_button);
  149.  
  150. $("#block_div").hide();
  151. save_button.addEventListener("click", function() {save_blocklist();}, false);
  152. cancel_button.addEventListener("click", function() {$("#block_div").hide();}, false);
  153. document.body.insertBefore(div, document.body.firstChild);
  154.  
  155. //For editing the include list
  156. var shouldInclude = false;
  157. var div2 = document.createElement('div');
  158. var textarea2 = document.createElement('textarea');
  159.  
  160. div2.style.position = 'fixed';
  161. div2.style.width = '500px';
  162. div2.style.height = '235px';
  163. div2.style.left = '50%';
  164. div2.style.right = '50%';
  165. div2.style.margin = '-250px 0px 0px -250px';
  166. div2.style.top = '300px';
  167. div2.style.padding = '5px';
  168. div2.style.border = '2px';
  169. div2.style.backgroundColor = 'black';
  170. div2.style.color = 'white';
  171. div2.style.zIndex = '100';
  172. div2.setAttribute('id','include_div');
  173.  
  174. textarea2.style.padding = '2px';
  175. textarea2.style.width = '500px';
  176. textarea2.style.height = '180px';
  177. textarea2.title = 'include list';
  178. textarea2.setAttribute('id','include_text');
  179.  
  180. div2.textContent = 'Used if you only want to see certain requesters. Separator is ^. Only takes effect if \"Use includelist\" is checked.';
  181. div2.style.fontSize = '12px';
  182. div2.appendChild(textarea2);
  183.  
  184. var save_button2 = document.createElement('button');
  185. var cancel_button2 = document.createElement('button');
  186.  
  187. save_button2.textContent = 'Save';
  188. save_button2.setAttribute('id', 'save_includelist');
  189. save_button2.style.height = '18px';
  190. save_button2.style.width = '100px';
  191. save_button2.style.fontSize = '10px';
  192. save_button2.style.paddingLeft = '3px';
  193. save_button2.style.paddingRight = '3px';
  194. save_button2.style.backgroundColor = 'white';
  195. save_button2.style.marginLeft = '5px';
  196.  
  197. cancel_button2.textContent = 'Cancel';
  198. cancel_button2.setAttribute('id', 'cancel_includelist');
  199. cancel_button2.style.height = '18px';
  200. cancel_button2.style.width = '100px';
  201. cancel_button2.style.fontSize = '10px';
  202. cancel_button2.style.paddingLeft = '3px';
  203. cancel_button2.style.paddingRight = '3px';
  204. cancel_button2.style.backgroundColor = 'white';
  205. cancel_button2.style.marginLeft = '5px';
  206.  
  207. div2.appendChild(save_button2);
  208. div2.appendChild(cancel_button2);
  209.  
  210. $("#include_div").hide();
  211. save_button2.addEventListener("click", function() {save_includelist();}, false);
  212. cancel_button2.addEventListener("click", function() {$("#include_div").hide();}, false);
  213. document.body.insertBefore(div2, document.body.firstChild);
  214.  
  215. function save_blocklist() {
  216. console.log("Save");
  217. var textarea = $("#block_text");
  218. var text = textarea.val();
  219. var block_list = text.split("^");
  220. var trimmed_list = [];
  221. for (var requester in block_list){
  222. if (block_list[requester].trim().length != 0)
  223. trimmed_list.push(block_list[requester].toLowerCase().trim());
  224. }
  225. console.log(trimmed_list);
  226. GM_setValue("scraper_ignore_list",trimmed_list.join('^'));
  227. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  228. console.log("Save complete: ");
  229. console.log(ignore_list);
  230. $("#block_div").hide();
  231. }
  232.  
  233. function save_includelist() {
  234. console.log("Save");
  235. var textarea = $("#include_text");
  236. var text = textarea.val();
  237. var includes = text.split("^");
  238. console.log(includes);
  239. var trimmed_list = [];
  240. for (var requester in includes){
  241. if (includes[requester].trim().length != 0)
  242. trimmed_list.push(includes[requester].toLowerCase().trim());
  243. }
  244. GM_setValue("scraper_include_list",trimmed_list.join('^'));
  245. include_list = GM_getValue("scraper_include_list").split('^');
  246. console.log(include_list);
  247. $("#include_div").hide();
  248. }
  249.  
  250. var HITStorage = {};
  251. var indexedDB = window.indexedDB || window.webkitIndexedDB ||
  252. window.mozIndexedDB;
  253. window.IDBTransaction = window.IDBTransaction || window.webkitIDBTransaction || window.mozIDBTransaction;
  254. window.IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange || window.mozIDBKeyRange;
  255. HITStorage.IDBTransactionModes = { "READ_ONLY": "readonly", "READ_WRITE": "readwrite", "VERSION_CHANGE": "versionchange" };
  256. var IDBKeyRange = window.IDBKeyRange;
  257.  
  258. HITStorage.indexedDB = {};
  259. HITStorage.indexedDB = {};
  260. HITStorage.indexedDB.db = null;
  261.  
  262. HITStorage.indexedDB.onerror = function(e) {
  263. console.log(e);
  264. };
  265.  
  266. var v=4;
  267.  
  268. HITStorage.indexedDB.checkTitle = function(title,button) {
  269. var request = indexedDB.open("HITDB", v);
  270. request.onsuccess = function(e) {
  271. HITStorage.indexedDB.db = e.target.result;
  272. var db = HITStorage.indexedDB.db;
  273. if (!db.objectStoreNames.contains("HIT"))
  274. {
  275. db.close();
  276. return;
  277. }
  278. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  279. var store = trans.objectStore("HIT");
  280. var index = store.index("title");
  281. index.get(title).onsuccess = function(event)
  282. {
  283. if (event.target.result === undefined)
  284. {
  285. console.log(title + ' not found');
  286. history[button].titledb=false;
  287. }
  288. else
  289. {
  290. console.log(title + ' found');
  291. history[button].titledb=true;
  292. }
  293. db.close();
  294. };
  295. };
  296. request.onerror = HITStorage.indexedDB.onerror;
  297. };
  298.  
  299. HITStorage.indexedDB.checkRequester = function(id,button) {
  300. var request = indexedDB.open("HITDB", v);
  301. request.onsuccess = function(e) {
  302. HITStorage.indexedDB.db = e.target.result;
  303. var db = HITStorage.indexedDB.db;
  304. if (!db.objectStoreNames.contains("HIT"))
  305. {
  306. db.close();
  307. return;
  308. }
  309. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  310. var store = trans.objectStore("HIT");
  311. var index = store.index("requesterId");
  312. index.get(id).onsuccess = function(event)
  313. {
  314. if (event.target.result === undefined)
  315. {history[button].reqdb=false;
  316. console.log(id + ' not found');
  317. }
  318. else
  319. {
  320. history[button].reqdb=true;
  321. console.log(id + ' found');
  322. }
  323. db.close();
  324. };
  325. };
  326. request.onerror = HITStorage.indexedDB.onerror;
  327. };
  328.  
  329. var PAGES_TO_SCRAPE = 3;
  330. var MINIMUM_HITS = 100;
  331. var SEARCH_REFRESH=0;
  332. var MINIMUM_TO = -1;
  333. var URL_BASE = "/mturk/searchbar?searchWords=&selectedSearchType=hitgroups";
  334. var initial_url = URL_BASE;
  335. var TO_REQ_URL = "http://turkopticon.ucsd.edu/reports?id=";
  336. var found_key_list=[];
  337. var last_clear_time = new Date().getTime();
  338. var searched_once = false;
  339. var save_new_results_time = 120;
  340. var save_results_time = 3600;
  341. var default_type = 0;
  342. var cur_loc = window.location.href;
  343. var time_input = document.createElement("INPUT");
  344. time_input.value = 0;
  345. var page_input = document.createElement("INPUT");
  346. page_input.value = 3;
  347. var min_input = document.createElement("INPUT");
  348. var new_time_display_input = document.createElement("INPUT");
  349. new_time_display_input.value = 300;
  350. var reward_input = document.createElement("INPUT");
  351. var qual_input = document.createElement("INPUT");
  352. qual_input.type = "checkbox";
  353. if (!document.getElementById('lnkWorkerSignin'))
  354. qual_input.checked = true;
  355. var masters_input = document.createElement("INPUT");
  356. masters_input.type = "checkbox";
  357. var masters_dropdown = document.createElement("SELECT");
  358. var myes = document.createElement("OPTION");
  359. var mno = document.createElement("OPTION");
  360. myes.value = "y";
  361. mno.value = "n";
  362. myes.text = "Show";
  363. mno.text = "Hide";
  364. masters_dropdown.appendChild(myes);
  365. masters_dropdown.appendChild(mno);
  366. var sort_input = document.createElement("SELECT");
  367. var sort_input1 = document.createElement("OPTION");
  368. sort_input1.text = "Latest";
  369. sort_input1.value = "late";
  370. var sort_input2 = document.createElement("OPTION");
  371. sort_input2.text = "Most Available";
  372. sort_input2.value = "most";
  373. var sort_input3 = document.createElement("OPTION");
  374. sort_input3.text = "Reward";
  375. sort_input3.value = "amount";
  376. var sort_input4 = document.createElement("OPTION");
  377. sort_input4.text = "Title";
  378. sort_input4.value = "alpha";
  379. sort_input.appendChild(sort_input1);
  380. sort_input.appendChild(sort_input2);
  381. sort_input.appendChild(sort_input3);
  382. sort_input.appendChild(sort_input4);
  383. var sort_input_invert = document.createElement("INPUT");
  384. sort_input_invert.type = "checkbox";
  385. var sort_to = document.createElement("INPUT");
  386. sort_to.type = "checkbox";
  387. var min_to = document.createElement("INPUT");
  388. var friesAreDone = document.createElement("INPUT");
  389. friesAreDone.type = "checkbox";
  390. var correctForSkips = document.createElement("INPUT");
  391. correctForSkips.type = "checkbox";
  392. correctForSkips.checked = true;
  393. var matchOnly = document.createElement("INPUT");
  394. matchOnly.type = "checkbox";
  395.  
  396. var search_input = document.createElement("INPUT");
  397.  
  398. var LINK_BASE = "https://www.mturk.com";
  399. var STATUSDETAIL_DELAY = 250;
  400. var MPRE_DELAY = 3000;
  401.  
  402. $('body').css('background', BACKGROUND_COLOR);
  403.  
  404. var next_page = 1;
  405.  
  406. var API_PROXY_BASE = 'https://mturk-api.istrack.in/';
  407. var API_MULTI_ATTRS_URL = API_PROXY_BASE + 'multi-attrs.php?ids=';
  408. var REVIEWS_BASE = 'http://turkopticon.ucsd.edu/';
  409.  
  410. var control_panel_HTML = '<div id="control_panel" style="margin: 0 auto 0 auto;' +
  411. 'border-bottom: 1px solid #000000; margin-bottom: 5px; ' +
  412. 'background-color: ' + BACKGROUND_COLOR + ';"></div>';
  413. $('body > :not(#control_panel)').hide(); //hide all nodes directly under the body
  414. $('body').prepend(control_panel_HTML);
  415.  
  416. var control_panel = document.getElementById("control_panel");
  417. var big_red_button = document.createElement("BUTTON");
  418. var reset_blocks = document.createElement("BUTTON");
  419. var include_button = document.createElement("BUTTON");
  420. var progress_report = document.createTextNode("Stopped");
  421. var status_report = document.createTextNode("None");
  422. var text_area = document.createElement("TABLE");
  423. text_area.id = "body_table";
  424. big_red_button.textContent = "Show Interface";
  425. big_red_button.onclick = function(){show_interface();};
  426. control_panel.appendChild(big_red_button);
  427.  
  428. var showButton = document.createElement("BUTTON");
  429. showButton.onclick = function() { console.log("Clicked"); show_TO_rows(); };
  430. showButton.textContent = "Show hits below TO threshold";
  431. showButton.id = "show_TO_stuff_button";
  432.  
  433. function show_TO_rows() {
  434. $("#body_table tr:hidden:gt(0)").each(function () { $(this).show(); });
  435. }
  436.  
  437. show_interface();
  438.  
  439. var global_run = false;
  440. var statusdetail_loop_finished = false;
  441. var date_header = "";
  442. var history = {};
  443. var wait_loop;
  444.  
  445. function set_progress_report(text, force)
  446. {
  447. if (global_run == true || force == true)
  448. {
  449. progress_report.textContent = text;
  450. status_report.textContent = status_text;
  451. }
  452. }
  453.  
  454. function get_progress_report()
  455. {
  456. return progress_report.textContent;
  457. }
  458.  
  459. function wait_until_stopped()
  460. {
  461. if (global_run == true)
  462. {
  463. if (statusdetail_loop_finished == true)
  464. {
  465. big_red_button.textContent = "Start";
  466. set_progress_report("Finished", false);
  467. }
  468. else
  469. {
  470. setTimeout(function(){wait_until_stopped();}, 500);
  471. }
  472. }
  473. }
  474.  
  475. function display_wait_time(wait_time)
  476. {
  477. if (global_run == true)
  478. {
  479. var current_progress = get_progress_report();
  480. if (current_progress.indexOf("Searching again in")!==-1)
  481. {
  482. set_progress_report(current_progress.replace(/Searching again in \d+ seconds/ , "Searching again in " + wait_time + " seconds"),false);
  483. }
  484. else
  485. set_progress_report(current_progress + " Searching again in " + wait_time + " seconds.", false);
  486. if (wait_time>1)
  487. setTimeout(function(){display_wait_time(wait_time-1);}, 1000);
  488. }
  489. }
  490.  
  491. function dispArr(ar)
  492. {
  493. var disp = "";
  494. for (var z = 0; z < ar.length; z++)
  495. {
  496. disp += "id " + z + " is " + ar[z] + " ";
  497. }
  498. console.log(disp);
  499. }
  500.  
  501. function scrape($src)
  502. {
  503. var $requester = $src.find('a[href^="/mturk/searchbar?selectedSearchType=hitgroups&requester"]');
  504. var $title = $src.find('a[class="capsulelink"]');
  505. var $reward = $src.find('span[class="reward"]');
  506. var $preview = $src.find('a[href^="/mturk/preview?"]');
  507. var $qualified = $src.find('a[href^="/mturk/notqualified?"],a[id^="private_hit"]');
  508. var $times = $src.find('a[id^="duration_to_complete"]');
  509. var $descriptions = $src.find('a[id^="description"]');
  510. var not_qualified_group_IDs=[];
  511. var $quals = $src.find('a[id^="qualificationsRequired"]');
  512. $qualified.each(function(){
  513. var groupy = $(this).attr('href');
  514. if (groupy){
  515. groupy = groupy.replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1");
  516. groupy = groupy.replace("/mturk/notqualified?hit","");
  517. }
  518. not_qualified_group_IDs.push(groupy);
  519. });
  520. var $mixed = $src.find('a[href^="/mturk/preview?"],a[href^="/mturk/notqualified?"],a[id^="private_hit"]');
  521. console.log($mixed);
  522. var listy =[];
  523. $mixed.each(function(){
  524. var groupy = $(this).attr('href');
  525. if (groupy)
  526. {
  527. groupy = groupy.replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1");
  528. groupy = groupy.replace("/mturk/notqualified?hit","");
  529. groupy = groupy.replace("/mturk/preview?groupId=","");
  530. }
  531. listy.push(groupy);
  532. });
  533. //console.log(listy);
  534. //listy = listy.filter(function(elem, pos) {
  535. // return listy.indexOf(elem) == pos;
  536. //});
  537. //console.log(listy);
  538. for (var j = 0; j < $requester.length; j++)
  539. {
  540. var $hits = $requester.eq(j).parent().parent().parent().parent().parent().parent().find('td[class="capsule_field_text"]');
  541. var requester_name = $requester.eq(j).text().trim();
  542. var requester_link = $requester.eq(j).attr('href');
  543. var group_ID=(listy[j] ? listy[j].replace(/\/mturk\/notqualified\?hitGroupId=([A-Z0-9]+)(&.+)?/,"$1") : "");
  544. if (listy[j] && listy[j].indexOf("notqualified") > -1)
  545. noqual = true;
  546. group_ID=group_ID.replace("/mturk/notqualified?hit","");
  547. var masters = false;
  548. var title = $title.eq(j).text().trim();
  549. var preview_link = "/mturk/preview?groupId=" + group_ID;
  550. console.log(listy[j]);
  551. console.log(group_ID);
  552. if (!group_ID || group_ID.length == 0){
  553. preview_link = requester_link;
  554. title += " (Requester link substituted)";
  555. }
  556. var reward = $reward.eq(j).text().trim();
  557. var hits = $hits.eq(4).text().trim();
  558. var time = $times.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  559. var description = $descriptions.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  560. //console.log(description);
  561. var requester_id = requester_link.replace('/mturk/searchbar?selectedSearchType=hitgroups&requesterId=','');
  562. var accept_link;
  563. accept_link = preview_link.replace('preview','previewandaccept');
  564. /*HIT SCRAPER ADDITION*/
  565. var qElements = $quals.eq(j).parent().parent().parent().find('tr');
  566. //console.log(qElements);
  567. var qualifications = [];
  568. for (var i = 1; i < qElements.length; i++) {
  569. 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, ' ')));
  570. if (qElements[i].childNodes[1].textContent.trim().replace(/\s+/g, ' ').indexOf("Masters") != -1)
  571. masters=true;
  572. }
  573. var qualList = (qualifications.join(', ') ? qualifications.join(', ') : "None");
  574. key = requester_name+title+reward+group_ID;
  575. if (found_key_list.indexOf(key) == -1)
  576. found_key_list.push(key);
  577. else
  578. {
  579. console.log("DUPE: "+key);
  580. continue;
  581. }
  582. if (history[key] == undefined)
  583. {
  584. history[key] = {requester:"", title:"", description:"", reward:"", hits:"", req_link:"", quals:"", prev_link:"", rid:"", acc_link:"", new_result:"", qualified:"", found_this_time:"", initial_time:"", reqdb:"",titledb:"",time:"",masters:false};
  585. history[key].req_link = requester_link;
  586. history[key].prev_link = preview_link;
  587. history[key].requester = requester_name;
  588. history[key].title = title;
  589. history[key].reward = reward;
  590. history[key].hits = hits;
  591. history[key].rid = requester_id;
  592. history[key].acc_link = accept_link;
  593. history[key].time = time;
  594. history[key].quals = qualList;
  595. history[key].description = description;
  596. history[key].masters = masters;
  597. HITStorage.indexedDB.checkRequester(requester_id,key);
  598. HITStorage.indexedDB.checkTitle(title,key);
  599. if (searched_once)
  600. {
  601. history[key].initial_time = new Date().getTime();//-1000*(save_new_results_time - SEARCH_REFRESH);
  602. history[key].new_result = 0;
  603. }
  604. else
  605. {
  606. history[key].initial_time = new Date().getTime()-1000*save_new_results_time;
  607. history[key].new_result = 1000*save_new_results_time;
  608. }
  609. if (not_qualified_group_IDs.indexOf(group_ID)!==-1)
  610. history[key].qualified = false;
  611. else
  612. history[key].qualified = true;
  613. history[key].found_this_time = true;
  614. }
  615. else
  616. {
  617. history[key].new_result = new Date().getTime() - history[key].initial_time;
  618. history[key].found_this_time = true;
  619. history[key].hits = hits;
  620. }
  621. }
  622. }
  623.  
  624. function statusdetail_loop(next_URL)
  625. {
  626. if (global_run == true)
  627. {
  628. if (next_URL.length != 0)
  629. {
  630. $.get(next_URL, function(data)
  631. {
  632. var $src = $(data);
  633. var maxpagerate = $src.find('td[class="error_title"]:contains("You have exceeded the maximum allowed page request rate for this website.")');
  634. if (maxpagerate.length == 0)
  635. {
  636. if (next_page > PAGES_TO_SCRAPE)
  637. {
  638. if(status_text.indexOf("Correcting for skips") == -1)
  639. status_text += ". Correcting for skips";
  640. }
  641. set_progress_report("Processing page " + next_page, false);
  642. scrape($src);
  643. $next_URL = $src.find('a[href^="/mturk/viewsearchbar"]:contains("Next")');
  644. next_URL = ($next_URL.length != 0) ? $next_URL.attr("href") : "";
  645. next_page++;
  646. if (default_type == 1)
  647. {
  648. var hmin = MINIMUM_HITS+1;
  649. for (j = 0; j < found_key_list.length; j++)
  650. {
  651. console.log(history[found_key_list[j]]);
  652. if (history[found_key_list[j]].hits < hmin)
  653. {
  654. next_URL = "";
  655. next_page = -1;
  656. break;
  657. }
  658. }
  659. }
  660. else if (next_page > PAGES_TO_SCRAPE && correct_for_skips)
  661. {
  662. var skipped_hits = 0;
  663. var added_pages = 0;
  664. for (j = 0; j < found_key_list.length; j++)
  665. {
  666. var obj = history[found_key_list[j]];
  667. if (!ignore_check(obj.requester,obj.title))
  668. skipped_hits++;
  669. }
  670. added_pages = Math.floor(skipped_hits/10);
  671. if (skipped_hits%10 >6)
  672. added_pages++;
  673. if (next_page > PAGES_TO_SCRAPE + added_pages)
  674. {
  675. next_URL = "";
  676. next_page = -1;
  677. }
  678. }
  679. else if (next_page > PAGES_TO_SCRAPE)
  680. {
  681. next_URL = "";
  682. next_page = -1;
  683. }
  684. setTimeout(function(){statusdetail_loop(next_URL);}, STATUSDETAIL_DELAY);
  685. }
  686. else
  687. {
  688. console.log("MPRE");
  689. setTimeout(function(){statusdetail_loop(next_URL);}, MPRE_DELAY);
  690. }
  691. });
  692. }
  693. else
  694. {
  695. searched_once = true;
  696. var found_hits = found_key_list.length;
  697. var shown_hits = 0;
  698. var new_hits = 0;
  699. var url = API_MULTI_ATTRS_URL;
  700. var rids = [];
  701. var lastRow = text_area.rows.length - 1;
  702. for (i = lastRow; i>0; i--)
  703. text_area.deleteRow(i);
  704. for (j = 0; j < found_key_list.length; j++)
  705. {
  706. //(function(url,rids,j) {
  707. var obj = history[found_key_list[j]];
  708. if (!(obj.masters == true && hide_masters) && (ignore_check(obj.requester,obj.title) && obj.found_this_time)){
  709. ++shown_hits;
  710. //console.log(obj);
  711. //hit export will update col_heads[1]
  712. var col_heads = ["<a href='"+ LINK_BASE+obj.req_link +"' target='_blank'>" + obj.requester + "</a>","<a href='"+ LINK_BASE+obj.prev_link +"' target='_blank' title='"+ obj.description +"'>" + obj.title + "</a>",obj.reward,obj.hits,"TO down","<a href='"+ LINK_BASE+obj.acc_link +"' target='_blank'>Accept</a>","M"];
  713. var row = text_area.insertRow(text_area.rows.length);
  714. url += obj.rid + ',';
  715. rids.push(obj.rid);
  716. if (check_hitDB)
  717. {
  718. col_heads.push("R");
  719. col_heads.push("T");
  720. }
  721. if (!obj.qualified)
  722. {
  723. col_heads.push("Not Qualified");
  724. }
  725. for (i=0; i<col_heads.length; i++)
  726. {
  727. var this_cell = row.insertCell(i);
  728. row.cells[i].style.fontSize = default_text_size;
  729. this_cell.innerHTML = col_heads[i];
  730. if(i>1)
  731. this_cell.style.textAlign = 'center';
  732. if (i==6)
  733. {
  734. var listOfQuals = obj.quals.replace(/,/g,'\n');
  735. listOfQuals = listOfQuals.replace(/\[(.*?)\]/g,'');
  736. this_cell.title=listOfQuals;
  737. if (obj.masters)
  738. {
  739. this_cell.style.backgroundColor="#551A8B";
  740. this_cell.innerHTML="Y";
  741. }
  742. else
  743. {
  744. this_cell.style.backgroundColor="#0066CC";
  745. this_cell.innerHTML="N";
  746. }
  747. }
  748. if (check_hitDB)
  749. {
  750. if (i==7)
  751. {
  752. if (obj.reqdb){
  753. this_cell.style.backgroundColor = GREEN;
  754. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,0);}})(obj));
  755. }
  756. else
  757. this_cell.style.backgroundColor = RED;
  758. }
  759. else if (i==8)
  760. {
  761. if (obj.titledb){
  762. this_cell.style.backgroundColor = GREEN;
  763. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,1);}})(obj));
  764. }
  765. else
  766. this_cell.style.backgroundColor = RED;
  767. }
  768. else if (i==9)
  769. this_cell.style.backgroundColor = DARKGREY;
  770. }
  771. else if (i==7)
  772. this_cell.style.backgroundColor = DARKGREY;
  773. }
  774. if (Object.keys(history).length>0)
  775. {
  776. if (obj.new_result < 1000*save_new_results_time)
  777. {
  778. new_hits++;
  779. for (i in col_heads)
  780. {
  781. row.cells[i].style.fontSize = default_text_size + 1;
  782. row.cells[i].style.fontWeight = "bold";
  783. }
  784. }
  785. }
  786. button = document.createElement('button'); //HIT SCRAPER ADDITION
  787. button.textContent = 'vB';
  788. button.title = 'Export this HIT description as vBulletin formatted text';
  789. button.style.height = '14px';
  790. button.style.width = '30px';
  791. button.style.fontSize = '8px';
  792. button.style.border = '1px solid';
  793. button.style.padding = '0px';
  794. button.style.backgroundColor = 'transparent';
  795. button2 = document.createElement('button'); //BUTTON TO BLOCK REQUESTER
  796. button2.textContent = 'R';
  797. button2.title = 'Add requester to block list';
  798. button2.style.height = '14px';
  799. button2.style.width = '15px';
  800. button2.style.fontSize = '8px';
  801. button2.style.border = '1px solid';
  802. button2.style.padding = '0px';
  803. button2.style.backgroundColor = 'transparent';
  804. button3 = document.createElement('button'); //BUTTON TO BLOCK TITLE
  805. button3.textContent = 'T';
  806. button3.title = 'Add title to block list';
  807. button3.style.height = '14px';
  808. button3.style.width = '15px';
  809. button3.style.fontSize = '8px';
  810. button3.style.border = '1px solid';
  811. button3.style.padding = '0px';
  812. button3.style.backgroundColor = 'transparent';
  813. button4 = document.createElement('button');
  814. button4.textContent = 'IRC';
  815. button4.style.height = '14px';
  816. button4.style.width = '30px';
  817. button4.style.fontSize = '8px';
  818. button4.style.border = '1px solid';
  819. button4.style.padding = '0px';
  820. button4.style.backgroundColor = 'transparent';
  821. button4.title = 'Click to save Hit information to your clipboard';
  822. //button.addEventListener("click", function() {export_func_deleg(j);}.bind(null,j), false);
  823. button.addEventListener("click", (function (obj,j) { return function() {export_func_deleg(obj,j);}})(obj,j));
  824. row.cells[1].appendChild(document.createTextNode(" "));
  825. row.cells[1].appendChild(button);
  826. button4.addEventListener("click", (function (obj,j) { return function() {export_irc_deleg(obj,j);}})(obj,j));
  827. row.cells[1].appendChild(document.createTextNode(" "));
  828. row.cells[1].appendChild(button4);
  829. button2.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,0);}})(obj,j));
  830. row.cells[0].appendChild(document.createTextNode(" "));
  831. row.cells[0].appendChild(button2);
  832. button3.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,1);}})(obj,j));
  833. row.cells[0].appendChild(button3);
  834. }
  835. //});
  836. }
  837. set_progress_report("Scrape complete. " + shown_hits + " HITs found (" + new_hits + " new results). " + (found_hits - shown_hits) + " HITs ignored.", false);
  838. if (new_hits > 0){
  839. newHits(shouldDing);
  840. }
  841. url = url.substring(0,url.length - 1);
  842. //console.log(url);
  843. var success_flag = false;
  844. GM_xmlhttpRequest(
  845. {
  846. method: "GET",
  847. url: url,
  848. onload: function (results)
  849. {
  850. //console.log(results.responseText);
  851. rdata = $.parseJSON(results.responseText);
  852. for (i = 0; i < rids.length; i++)
  853. {
  854. text_area.rows[i+1].style.backgroundColor = GREY;
  855. if (rdata[rids[i]])
  856. {
  857. var pay = rdata[rids[i]].attrs.pay;
  858. var reviews = rdata[rids[i]].reviews;
  859. var average = 0;
  860. var sum = 0;
  861. var divisor = 0;
  862. console.log(rdata[rids[i]]);
  863. var comm = rdata[rids[i]].attrs.comm.trim();
  864. var fair = rdata[rids[i]].attrs.fair.trim();
  865. var fast = rdata[rids[i]].attrs.fast.trim();
  866. var reviews = rdata[rids[i]].reviews;
  867. var tos = rdata[rids[i]].tos_flags;
  868. var titleText = "comm: "+comm+"\nfair: "+fair+"\nfast: "+fast+"\npay: "+pay+"\nReviews: "+reviews+"\nTOS: "+tos;
  869. if (comm > 0)
  870. {
  871. sum += COMM_WEIGHT*comm;
  872. divisor += COMM_WEIGHT;
  873. }
  874. if (pay > 0)
  875. {
  876. sum += PAY_WEIGHT*pay;
  877. divisor += PAY_WEIGHT;
  878. }
  879. if (fair > 0)
  880. {
  881. sum += FAIR_WEIGHT*fair;
  882. divisor += FAIR_WEIGHT;
  883. }
  884. if (fast > 0)
  885. {
  886. sum += FAST_WEIGHT*fast;
  887. divisor += FAST_WEIGHT;
  888. }
  889. if (divisor > 0)
  890. {
  891. average = sum/divisor;
  892. }
  893. text_area.rows[i+1].cells[4].innerHTML = "<a href='"+ TO_REQ_URL+rids[i] +"' target='_blank'>" + pay + "</a>";
  894. text_area.rows[i+1].cells[4].setAttribute("title", titleText);
  895. if (reviews > 4)
  896. {
  897. if (average > 4.49)
  898. text_area.rows[i+1].style.backgroundColor = GREEN;
  899. else if (average > 3.49)
  900. text_area.rows[i+1].style.backgroundColor = LIGHTGREEN;
  901. //else if (average > 2.99)
  902. // text_area.rows[i+1].style.backgroundColor = YELLOW;
  903. else if (average > 1.99)
  904. text_area.rows[i+1].style.backgroundColor = ORANGE;
  905. else if (average > 0)
  906. text_area.rows[i+1].style.backgroundColor = RED;
  907. }
  908. if (sort_TO)
  909. {
  910. table_sort();
  911. }
  912. if (MINIMUM_TO > -1)
  913. {
  914. table_TO_trim();
  915. }
  916. }
  917. else
  918. {
  919. text_area.rows[i+1].cells[4].innerHTML = "No data";
  920. }
  921. }
  922. success_flag = true;
  923. }
  924. });
  925. if (!success_flag)
  926. for (i = 0; i < rids.length; i++) text_area.rows[i+1].style.backgroundColor = GREY;
  927. statusdetail_loop_finished = true;
  928. if (SEARCH_REFRESH>0)
  929. {
  930. wait_loop = setTimeout(function(){if (global_run) start_it();}, 1000*SEARCH_REFRESH);
  931. display_wait_time(SEARCH_REFRESH);
  932. }
  933. else
  934. {
  935. global_run = false;
  936. big_red_button.textContent = "Start";
  937. }
  938. }
  939. }
  940. }
  941.  
  942. function table_TO_trim(){
  943. $("#body_table tr:gt(0)").each(function () {
  944. var TO = $(this).find("td:eq(4)").text();
  945. if (TO != "No data" && TO != "TO down")
  946. {
  947. if (Number(TO) < MINIMUM_TO)
  948. {
  949. $(this).hide();
  950. }
  951. }
  952. });
  953. }
  954.  
  955. function table_sort() {
  956. //--- Get the table we want to sort.
  957. var jTableToSort = $("#body_table");
  958. //--- Get the rows to sort, but skip the first row, since it contains column titles.
  959. var jRowsToSort = jTableToSort.find ("tr:gt(0)");
  960. //--- Sort the rows in place.
  961. jRowsToSort.sort(SortByFirstColumnTextAscending).appendTo(jTableToSort);
  962. }
  963.  
  964. function SortByFirstColumnTextAscending (zA, zB)
  965. {
  966. var ValA_Text = $(zA).find("td:eq(4)").text();
  967. var ValB_Text = $(zB).find("td:eq(4)").text();
  968. if (ValA_Text > ValB_Text)
  969. return 1;
  970. else if (ValA_Text < ValB_Text)
  971. return -1;
  972. else
  973. return 0;
  974. }/**/
  975.  
  976. function ignore_check(r,t){
  977. tempList = ignore_list.map(function(item) { return item.toLowerCase(); });
  978. foundR = -1;
  979. foundT = -1;
  980. foundR = tempList.indexOf(r.toLowerCase());
  981. foundT = tempList.indexOf(t.toLowerCase());
  982. if (shouldInclude){
  983. console.log(include_list);
  984. temp = include_list.map(function(item) { return item.toLowerCase(); }).indexOf(r.toLowerCase());
  985. console.log(temp);
  986. if (temp != -1)
  987. foundR = -1;
  988. else
  989. foundR = 0;
  990. }
  991. found = foundR == -1 && foundT == -1;
  992. //console.log("r: "+r+" t: "+t+" f: "+found);
  993. return found;
  994. //return -1 == ignore_list.map(function(item) { return item.toLowerCase(); }).indexOf(r.toLowerCase());
  995. }
  996.  
  997. function start_running()
  998. {
  999. if (big_red_button.textContent == "Start")
  1000. {
  1001. status_text="";
  1002. ignore_list = GM_getValue("scraper_ignore_list").split('^');
  1003. if (GM_getValue("scraper_include_list"))
  1004. include_list = GM_getValue("scraper_include_list").split('^');
  1005. global_run = true;
  1006. initial_url = URL_BASE;
  1007. if (search_input.value.length>0)
  1008. {
  1009. initial_url = initial_url.replace("searchWords=", "searchWords=" + search_input.value);
  1010. }
  1011. if (time_input.value.replace(/[^0-9]+/g,"") != "")
  1012. {
  1013. SEARCH_REFRESH = Number(time_input.value);
  1014. }
  1015. if (page_input.value.replace(/[^0-9]+/g,"") != "")
  1016. {
  1017. PAGES_TO_SCRAPE = Number(page_input.value);
  1018. }
  1019. if (min_input.value.replace(/[^0-9]+/g,"") != "")
  1020. {
  1021. if (!sort_input2.checked)
  1022. status_text += " Minimum batch size selected but not sorting by most available";
  1023. MINIMUM_HITS = Number(min_input.value);
  1024. }
  1025. if (new_time_display_input.value.replace(/[^0-9]+/g,"") != "")
  1026. {
  1027. save_new_results_time = Number(new_time_display_input.value);
  1028. }
  1029. if (reward_input.value.replace(/[^0-9]+/g,"") != "")
  1030. {
  1031. initial_url += "&minReward=" + reward_input.value;
  1032. }
  1033. else
  1034. {
  1035. initial_url += "&minReward=0.00";
  1036. }
  1037. if (qual_input.checked)
  1038. {
  1039. initial_url += "&qualifiedFor=on";
  1040. }
  1041. else
  1042. {
  1043. initial_url += "&qualifiedFor=off";
  1044. }
  1045. if (masters_input.checked)
  1046. {
  1047. initial_url += "&requiresMasterQual=on";
  1048. }
  1049. if (masters_dropdown[masters_dropdown.selectedIndex].value == "n")
  1050. {
  1051. hide_masters=true;
  1052. status_text += "Masters hits hidden";
  1053. }
  1054. switch (sort_input[sort_input.selectedIndex].value)
  1055. {
  1056. case "late":
  1057. initial_url+= "&sortType=LastUpdatedTime%3A";
  1058. default_type = 0;
  1059. break;
  1060. case "most":
  1061. initial_url+= "&sortType=NumHITs%3A";
  1062. default_type = 1;
  1063. status_text += " Sorting by NumHITs ignores correct for skips in favor of minimum batch size";
  1064. break;
  1065. case "amount":
  1066. initial_url+= "&sortType=Reward%3A";
  1067. default_type = 0;
  1068. break;
  1069. case "alpha":
  1070. initial_url += "&sortType=Title%3A";
  1071. break;
  1072. default:
  1073. alert("I don't know how you did it, but you broke it. Good job.");
  1074. }
  1075. if (sort_input_invert.checked)
  1076. {
  1077. if (sort_input[sort_input.selectedIndex].value == "alpha")
  1078. initial_url += "1";
  1079. else
  1080. initial_url += "0";
  1081. }
  1082. else
  1083. {
  1084. if (sort_input[sort_input.selectedIndex].value == "alpha")
  1085. initial_url += "0";
  1086. else
  1087. initial_url += "1";
  1088. }
  1089. if (sort_to.checked)
  1090. {
  1091. sort_TO = true;
  1092. status_text += "Sorting by TO is still in testing";
  1093. }
  1094. if (min_to.value.replace(/[^0-9]+/g,"") != "")
  1095. {
  1096. MINIMUM_TO = Number(min_to.value);
  1097. $("#show_TO_stuff_button").show();
  1098. }
  1099. else
  1100. {
  1101. MINIMUM_TO = -1;
  1102. $("#show_TO_stuff_button").hide();
  1103. }
  1104. if (friesAreDone.checked)
  1105. {
  1106. shouldDing = true;
  1107. }
  1108. else {
  1109. shouldDing = false;
  1110. }
  1111. if (correctForSkips.checked){
  1112. if (matchOnly.checked)
  1113. {
  1114. status_text += " Match only checked, ignoring skip correction to prevent issues.";
  1115. correct_for_skips = false;
  1116. }
  1117. else{
  1118. correct_for_skips = true;
  1119. }
  1120. }
  1121. else {
  1122. correct_for_skips = false;
  1123. }
  1124. if (matchOnly.checked){
  1125. if (include_list.length == 0){
  1126. status_text += " No items in include list. Ignoring inclusion checkbox.";
  1127. shouldInclude = false;
  1128. }
  1129. else
  1130. shouldInclude = true;
  1131. }
  1132. else{
  1133. shouldInclude = false;
  1134. }
  1135. if (status_text == "")
  1136. status_text = "None";
  1137. initial_url+="&pageNumber=1&searchSpec=HITGroupSearch"
  1138. start_it();
  1139. }
  1140. else
  1141. {
  1142. global_run = false;
  1143. clearTimeout(wait_loop);
  1144. big_red_button.textContent = "Start";
  1145. set_progress_report("Stopped", true);
  1146. }
  1147. }
  1148.  
  1149. function start_it()
  1150. {
  1151. statusdetail_loop_finished = false;
  1152. big_red_button.textContent = "Stop";
  1153. found_key_list=[];
  1154. var ctime = new Date().getTime()
  1155. if (ctime - last_clear_time > save_results_time*666)
  1156. {
  1157. var last_history=history;
  1158. history = {};
  1159. for (var key in last_history)
  1160. {
  1161. if (last_history[key].new_result<save_results_time*1000)
  1162. {
  1163. history[key]=last_history[key];
  1164. if (last_history[key].found_this_time)
  1165. {
  1166. last_history[key].found_this_time = false;
  1167. if (last_history[key].new_result>save_new_results_time*1000)
  1168. last_history[key].initial_time = ctime-1000*save_new_results_time;
  1169. }
  1170. }
  1171. }
  1172. last_clear_time = ctime;
  1173. }
  1174. next_page = 1;
  1175. statusdetail_loop(initial_url);
  1176. }
  1177.  
  1178.  
  1179. function show_interface()
  1180. {
  1181. control_panel.style.color = BROWN;
  1182. control_panel.style.fontSize = 14;
  1183. control_panel.removeChild(big_red_button);
  1184. control_panel.appendChild(document.createTextNode("Auto-refresh delay: "));
  1185. time_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1186. time_input.title = "Enter search refresh delay in seconds\n" + "Enter 0 for no auto-refresh\n" + "Default is 0 (no auto-refresh)";
  1187. time_input.size = 3;
  1188. control_panel.appendChild(time_input);
  1189. control_panel.appendChild(document.createTextNode(" "));
  1190. control_panel.appendChild(document.createTextNode("| Pages to scrape: "));
  1191. page_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1192. page_input.title = "Enter number of pages to scrape\n" + "Default is 3";
  1193. page_input.size = 3;
  1194. control_panel.appendChild(page_input);
  1195. control_panel.appendChild(document.createTextNode(" Correct for skips: "));
  1196. correctForSkips.title = "Searches additional pages to get a consistent number of results. Helpful if you're blocking a lot of people";
  1197. control_panel.appendChild(correctForSkips);
  1198. control_panel.appendChild(document.createTextNode(" "));
  1199. control_panel.appendChild(document.createTextNode("| Minimum batch size: "));
  1200. min_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1201. min_input.title = "Enter minimum HITs for batch search\n" + "Default is 100";
  1202. min_input.size = 3;
  1203. control_panel.appendChild(min_input);
  1204. control_panel.appendChild(document.createTextNode(" "));
  1205. control_panel.appendChild(document.createTextNode("| New HIT highlighting: "));
  1206. new_time_display_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1207. new_time_display_input.title = "Enter time (in seconds) to keep new HITs highlighted\n" + "Default is 300 (5 minutes)";
  1208. new_time_display_input.size = 6;
  1209. control_panel.appendChild(new_time_display_input);
  1210. control_panel.appendChild(document.createTextNode(" "));
  1211. control_panel.appendChild(document.createTextNode("| Ding on new hit: "));
  1212. control_panel.appendChild(friesAreDone);
  1213. control_panel.appendChild(document.createElement("P"));
  1214. control_panel.appendChild(document.createTextNode("Minimum reward: "));
  1215. reward_input.size = 6;
  1216. control_panel.appendChild(reward_input);
  1217. control_panel.appendChild(document.createTextNode(" "));
  1218. control_panel.appendChild(document.createTextNode("| Qualified: "));
  1219. control_panel.appendChild(qual_input);
  1220. control_panel.appendChild(document.createTextNode(" "));
  1221. control_panel.appendChild(document.createTextNode("| Masters: Require "));
  1222. control_panel.appendChild(masters_input);
  1223. control_panel.appendChild(document.createTextNode(" Show "));
  1224. control_panel.appendChild(masters_dropdown);
  1225. control_panel.appendChild(document.createTextNode("| Sort types: "));
  1226. control_panel.appendChild(sort_input);
  1227. /*control_panel.appendChild(document.createTextNode(" Latest: "));
  1228. control_panel.appendChild(sort_input1);
  1229. control_panel.appendChild(document.createTextNode("| Most Available: "));
  1230. control_panel.appendChild(sort_input2);
  1231. control_panel.appendChild(document.createTextNode("| Amount: "));
  1232. control_panel.appendChild(sort_input3);
  1233. control_panel.appendChild(document.createTextNode("| A-Z: "));
  1234. control_panel.appendChild(sort_input4);*/
  1235. control_panel.appendChild(document.createElement("P"));
  1236. control_panel.appendChild(document.createTextNode("Invert: "));
  1237. control_panel.appendChild(sort_input_invert);
  1238. control_panel.appendChild(document.createTextNode("| TO: "));
  1239. control_panel.appendChild(sort_to);
  1240. control_panel.appendChild(document.createTextNode("| Minimum TO: "));
  1241. control_panel.appendChild(min_to);
  1242. control_panel.appendChild(document.createElement("P"));
  1243. control_panel.appendChild(search_input);
  1244. search_input.size = 20;
  1245. search_input.title = "Enter a search term to include\n" + "Default is blank (no included terms)";
  1246. search_input.placeholder="Enter search terms here";
  1247. control_panel.appendChild(document.createTextNode(" "));
  1248. control_panel.appendChild(document.createTextNode("| Use includelist: "));
  1249. control_panel.appendChild(document.createTextNode(" "));
  1250. matchOnly.title = "Be sure to edit the include list or nothing will be displayed.";
  1251. control_panel.appendChild(matchOnly);
  1252. big_red_button.textContent = "Start";
  1253. big_red_button.onclick = function(){start_running();};
  1254. reset_blocks.textContent = "Edit blocklist";
  1255. reset_blocks.setAttribute("id","blocklist_reset_button");
  1256. reset_blocks.onclick = function(){
  1257. console.log("in");
  1258. var textarea = $("#block_text");
  1259. var text = "";
  1260. for (var i in ignore_list){
  1261. text += ignore_list[i]+"^";
  1262. }
  1263. textarea.text(text.substring(0, text.length - 1));
  1264. $("#block_div").show();
  1265. };
  1266. include_button.textContent = "Edit includes";
  1267. include_button.id = "includes_reset_button";
  1268. include_button.onclick = function() {
  1269. var div = $("#include_div");
  1270. var textarea = $("#include_text");
  1271. var text = "";
  1272. for (var i in include_list){
  1273. text += include_list[i]+"^";
  1274. }
  1275. textarea.text(text.substring(0, text.length - 1));
  1276. $("#include_div").show();
  1277. };
  1278. control_panel.appendChild(document.createTextNode(" | "));
  1279. control_panel.appendChild(big_red_button);
  1280. control_panel.appendChild(document.createTextNode(" "));
  1281. control_panel.appendChild(reset_blocks);
  1282. control_panel.appendChild(document.createTextNode(" "));
  1283. control_panel.appendChild(include_button);
  1284. control_panel.appendChild(document.createElement("P"));
  1285. control_panel.appendChild(progress_report);
  1286. control_panel.appendChild(document.createElement("P"));
  1287. control_panel.appendChild(document.createTextNode("Status messages: "));
  1288. control_panel.appendChild(status_report);
  1289. control_panel.appendChild(document.createElement("P"));
  1290. text_area.style.fontWeight = 400;
  1291. text_area.createCaption().innerHTML = "HITs";
  1292. var col_heads = ['Requester','Title','Reward','HITs Available','TO pay','Accept HIT','M?'];
  1293. var row = text_area.createTHead().insertRow(0);
  1294. text_area.caption.style.fontWeight = 800;
  1295. text_area.caption.style.color = BROWN;
  1296. if (default_text_size > 10)
  1297. text_area.cellPadding=Math.min(Math.max(1,Math.floor((default_text_size-10)/2)),5);
  1298. //console.log(text_area.cellPadding);
  1299. //text_area.cellPadding=2;
  1300. text_area.caption.style.fontSize = 28;
  1301. text_area.rows[0].style.fontWeight = 800;
  1302. text_area.rows[0].style.color = BROWN;
  1303. for (i=0; i<col_heads.length; i++)
  1304. {
  1305. var this_cell = row.insertCell(i);
  1306. this_cell.innerHTML = col_heads[i];
  1307. this_cell.style.fontSize = 14;
  1308. if (i > 1)
  1309. this_cell.style.textAlign = 'center';
  1310. }
  1311. var header_hide_button = document.createElement('button');
  1312. document.body.insertBefore(header_hide_button,control_panel.nextSibling);
  1313. document.body.insertBefore(text_area,header_hide_button.nextSibling);
  1314. document.body.insertBefore(showButton,header_hide_button.nextSibling);
  1315. document.body.insertBefore(document.createTextNode(" "),header_hide_button.nextSibling);
  1316. $("#show_TO_stuff_button").hide();
  1317. header_hide_button.textContent = "Hide";
  1318. header_hide_button.onclick = function(){
  1319. if (header_hide_button.textContent == "Hide"){
  1320. $('#control_panel').hide();
  1321. header_hide_button.textContent = "Show";
  1322. }
  1323. else{
  1324. $('#control_panel').show();
  1325. header_hide_button.textContent = "Hide";
  1326. }
  1327. };
  1328. }
  1329.  
  1330. /********HIT EXPORT ADDITIONS*****/
  1331.  
  1332. var EDIT = false;
  1333. var HIT;
  1334.  
  1335. var TO_BASE = "http://turkopticon.ucsd.edu/";
  1336. var API_BASE = "https://mturk-api.istrack.in/";
  1337. var API_URL = API_BASE + "multi-attrs.php?ids=";
  1338.  
  1339. DEFAULT_TEMPLATE = '[table][tr][td][b]Title:[/b] [url={prev_link}][COLOR=blue]{title}[/COLOR][/url]\n';
  1340. DEFAULT_TEMPLATE += '[b]Requester:[/b] [url=https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId={rid}][COLOR=blue]{requester}[/COLOR][/url]';
  1341. DEFAULT_TEMPLATE += ' [{rid}] ([url='+TO_BASE+'{rid}][COLOR=blue]TO[/COLOR][/url])';
  1342. DEFAULT_TEMPLATE += '\n[b]TO Ratings:[/b]{to_stuff}';
  1343. DEFAULT_TEMPLATE += '\n[b]Description:[/b] {description}';
  1344. DEFAULT_TEMPLATE += '\n[b]Time:[/b] {time}';
  1345. DEFAULT_TEMPLATE += '\n[b]Hits Available:[/b] {hits}';
  1346. DEFAULT_TEMPLATE += '\n[b]Reward:[/b] [COLOR=green][b]{reward}[/b][/COLOR]';
  1347. DEFAULT_TEMPLATE += '\n[b]Qualifications:[/b] {quals}[/td][/tr][/table]';
  1348.  
  1349. var TEMPLATE;
  1350. var EASYLINK;
  1351.  
  1352. if (typeof GM_getValue === 'undefined')
  1353. TEMPLATE = null;
  1354. else {
  1355. TEMPLATE = GM_getValue('HITScraper Template');
  1356. EASYLINK = GM_getValue('HITScraper Easylink');
  1357. }
  1358. if (TEMPLATE == null) {
  1359. TEMPLATE = DEFAULT_TEMPLATE;
  1360. }
  1361.  
  1362. function buildXhrUrl(rai) {
  1363. var url = API_URL;
  1364. var ri = rai;
  1365. url += rai;
  1366. return url;
  1367. }
  1368.  
  1369. function makeXhrQuery(url) {
  1370. var xhr = new XMLHttpRequest();
  1371. try{
  1372. xhr.open('GET', url, false);
  1373. xhr.send(null);
  1374. return $.parseJSON(xhr.response);
  1375. }
  1376. catch(err){
  1377. return "TO DOWN";
  1378. }
  1379. }
  1380.  
  1381. function getNamesForEmptyResponses(rai, resp) {
  1382. for (var rid in rai) {
  1383. if (rai.hasOwnProperty(rid) && resp[rid] == "") {
  1384. resp[rid] = $.parseJSON('{"name": "' + rai[rid][0].innerHTML + '"}');
  1385. }
  1386. }
  1387. return resp;
  1388. }
  1389.  
  1390. function getKeys(obj) {
  1391. var keys = [];
  1392. for (var key in obj) {
  1393. keys.push(key);
  1394. }
  1395. return keys;
  1396. }
  1397.  
  1398. function export_func_deleg(item,index) {
  1399. //console.log(item);
  1400. export_func(item);
  1401. }
  1402.  
  1403. function export_irc_deleg(item,index) {
  1404. //console.log(item);
  1405. export_irc(item);
  1406. }
  1407.  
  1408. function block_deleg(item,index) {
  1409. //console.log(item);
  1410. block(item,index);
  1411. }
  1412.  
  1413. function block(hit,index){
  1414. var blockType = ["requester","title"];
  1415. var blockThis = hit[blockType[index]];
  1416. ignore_list.push(blockThis);
  1417. GM_setValue("scraper_ignore_list",ignore_list.join('^'));
  1418. //console.log(GM_getValue("scraper_ignore_list"));
  1419. alert("\""+blockThis+"\" ignored. Re-scrape");
  1420. }
  1421.  
  1422. function search_deleg(item,index) {
  1423. console.log(item);
  1424. var searches = ["rid","title"];
  1425. search(item,searches[index]);
  1426. }
  1427.  
  1428. function hit_sort_func()
  1429. {
  1430. return function(a,b) {
  1431. if (a.date == b.date) {
  1432. if (a.requesterName < b.requesterName)
  1433. return -1;
  1434. if (a.requesterName > b.requesterName)
  1435. return 1;
  1436. if (a.title < b.title)
  1437. return -1;
  1438. if (a.title > b.title)
  1439. return 1;
  1440. if (a.status < b.status)
  1441. return -1;
  1442. if (a.status > b.status)
  1443. return 1;
  1444. }
  1445. if (a.date > b.date)
  1446. return 1;
  1447. if (a.date < b.date)
  1448. return -1;
  1449. };
  1450. }
  1451.  
  1452. function escapeRegExp(str) {
  1453. return str.replace(/[-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
  1454. }
  1455.  
  1456. function search(item,search_type){
  1457. //return true;/*
  1458. var request = indexedDB.open("HITDB", v);
  1459. request.onsuccess = function(e) {
  1460. HITStorage.indexedDB.db = e.target.result;
  1461. var db = HITStorage.indexedDB.db;
  1462. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  1463. var store = trans.objectStore("HIT");
  1464. var req;
  1465. var results = [];
  1466. var index;
  1467. var range;
  1468. req = store.openCursor();
  1469. req.onsuccess = function(event) {
  1470. var cursor = event.target.result;
  1471. if (cursor) {
  1472. hit = cursor.value;
  1473. var keys = ['title', 'requesterId'];
  1474. var re = new RegExp(escapeRegExp(item[search_type]),"ig");
  1475. for (var k in keys)
  1476. {
  1477. if (hit[keys[k]] != null && re.test(hit[keys[k]].trim())){
  1478. results.push(cursor.value);
  1479. }
  1480. }
  1481. cursor.continue();
  1482. }
  1483. else {
  1484. console.log(results);
  1485. results.sort(hit_sort_func());
  1486. show_results(results);
  1487. }
  1488. db.close();
  1489. };
  1490. request.onerror = HITStorage.indexedDB.onerror;/**/
  1491. }
  1492. }
  1493.  
  1494. function format_hit_line (hit, odd, status_color, new_day)
  1495. {
  1496. var line = '<tr style="background-color:';
  1497. if (odd)
  1498. line += '#f1f3eb;';
  1499. else
  1500. line += 'white;';
  1501. line += ' valign=top;';
  1502. if (new_day)
  1503. line += ' border: 0px dotted #000000; border-width: 2px 0px 0px 0px">';
  1504. else
  1505. line += '">';
  1506. line += '<td>' + hit.date + '</td>';
  1507. if (hit.requesterLink != null)
  1508. line += '<td style="width:165px"><a href="' + hit.requesterLink + '" title="Contact this Requester">' + hit.requesterName + '</a></td>';
  1509. else
  1510. line += '<td style="width:165px">' + hit.requesterName + '</td>';
  1511. line += '<td style="width:213px">' + hit.title + '</td>';
  1512. line += '<td style="width:45px">$' + hit.reward.toFixed(2) + '</td>';
  1513. line += '<td style="color:' + status_color + '; width:55px">' + hit.status + '</td>';
  1514. line += '<td><div style="width:225px; overflow:hidden">' + hit.feedback + '</div></td>';
  1515. line += '</tr>\n';
  1516. return line;
  1517. }
  1518.  
  1519. function status_color (status)
  1520. {
  1521. var color = "green";
  1522. if (status.match("Pending Approval"))
  1523. color = "orange";
  1524. else if (status.match("Rejected"))
  1525. color = "red";
  1526. return color;
  1527. }
  1528.  
  1529. function show_results (results){
  1530. resultsWindow = window.open();
  1531. resultsWindow.document.write("<html><head><title>Status Detail Search Results</title></head><body>\n");
  1532. resultsWindow.document.write("<h1>HITs matching your search:</h1>\n");
  1533. resultsWindow.document.write('<table style="border: 1px solid black;border-collapse:collapse;width:90%;margin-left:auto;margin-right:auto;">\n');
  1534. 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');
  1535. var odd = true;
  1536. var sum = 0;
  1537. var sum_rejected = 0;
  1538. var sum_approved = 0;
  1539. var sum_pending = 0;
  1540. var new_day = false;
  1541. for (var i=0; i<results.length; i++) {
  1542. odd = !odd;
  1543. sum += results[i].reward;
  1544. if (results[i].status == 'Rejected')
  1545. sum_rejected += results[i].reward;
  1546. else if (results[i].status == 'Pending Approval')
  1547. sum_pending += results[i].reward;
  1548. else
  1549. sum_approved += results[i].reward;
  1550. if (i>0 && (results[i-1].date != results[i].date))
  1551. new_day = true;
  1552. else
  1553. new_day = false;
  1554. resultsWindow.document.write(format_hit_line(results[i], odd, status_color(results[i].status), new_day ));
  1555. }
  1556. 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');
  1557. resultsWindow.document.write("</table>");
  1558. resultsWindow.document.write("<p>Found " + results.length + " matching HITs. $" + sum_approved.toFixed(2) + " approved, " +
  1559. "$" + sum_rejected.toFixed(2) + " rejected and $" + sum_pending.toFixed(2) + " pending.</p>");
  1560. resultsWindow.document.write("</body></html>")
  1561. resultsWindow.document.close();
  1562. }
  1563.  
  1564.  
  1565. function export_func(item) {
  1566. HIT = item;
  1567. edit_button.textContent = 'Edit Template';
  1568. apply_template(item);
  1569. div.style.display = 'block';
  1570. textarea.select();
  1571. }
  1572.  
  1573. function export_irc(item) {
  1574. display(item);
  1575. }
  1576.  
  1577. function apply_template(hit_data) {
  1578. var txt = TEMPLATE;
  1579. var vars = ['title', 'requester', 'rid', 'description', 'reward', 'quals', 'prev_link', 'time', 'hits', 'to_stuff', 'to_text'];
  1580. var resp = null;
  1581. if (txt.indexOf('{to_text}') >= 0 || txt.indexOf('{to_stuff}') >= 0){
  1582. var url = buildXhrUrl(hit_data["rid"]);
  1583. resp = makeXhrQuery(url);
  1584. //console.log(resp);
  1585. }
  1586. var toText = "";
  1587. var toStuff = "";
  1588. var toData = "";
  1589. var numResp = (resp == null || resp == "TO DOWN" ? "n/a" : resp[hit_data["rid"]].reviews);
  1590. if (resp == "TO DOWN"){
  1591. toStuff = " [URL=\""+TO_BASE+hit_data['rid']+"\"]TO down.[/URL]";
  1592. toText = toStuff;
  1593. }
  1594. else if (resp == null || resp[hit_data["rid"]].attrs == null && resp != "TO DOWN") {
  1595. toStuff = " No TO ";
  1596. toText = " No TO ";
  1597. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  1598. toStuff += "(Submit a new TO rating for this requester)[/URL]";
  1599. }
  1600. else {
  1601. for (var key in resp[hit_data["rid"]].attrs) {
  1602. //toText += "\n[*]"+key+": "+resp[hit_data["requesterId"]].attrs[key]+"\n";
  1603. var i = 0;
  1604. var color = "green";
  1605. var name = key;
  1606. var num = Math.floor(resp[hit_data["rid"]].attrs[key]);
  1607. switch (key){
  1608. case "comm":
  1609. name = "Communicativity";
  1610. break;
  1611. case "pay":
  1612. name = "Generosity";
  1613. break;
  1614. case "fast":
  1615. name = "Promptness";
  1616. break;
  1617. case "fair":
  1618. name = "Fairness";
  1619. break;
  1620. default:
  1621. name = key;
  1622. break;
  1623. }
  1624. switch (num){
  1625. case 0:
  1626. color = "red";
  1627. break;
  1628. case 1:
  1629. color = "red";
  1630. break;
  1631. case 2:
  1632. color = "orange";
  1633. break;
  1634. case 3:
  1635. color = "yellow";
  1636. break;
  1637. default:
  1638. break;
  1639. }
  1640. toText += (num > 0 ? "\n[color="+color+"]" : "\n");
  1641. for (i; i < num; i++){
  1642. toText += "[b]"+symbol+"[/b]"
  1643. }
  1644. toText += (num > 0 ? "[/color]" : "")
  1645. if (i < 5){
  1646. toText += "[color=white]";
  1647. for (i; i < 5; i++)
  1648. toText += "[b]"+symbol+"[/b]";
  1649. toText += "[/color]";
  1650. }
  1651. toText += " "+Number(resp[hit_data["rid"]].attrs[key]).toFixed(2)+" "+name;
  1652. toData += Number(resp[hit_data["rid"]].attrs[key]).toFixed(2) + ",";
  1653. }
  1654. //toText += "[/list]";
  1655. 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]");
  1656. toStuff = '\n[img]http://data.istrack.in/to/' + toData.slice(0,-1) + '.png[/img]';
  1657. toStuff += (txt.indexOf('{to_stuff}') >= 0 ? (txt.indexOf('{to_text}') >= 0 ? "" : toText) : "");
  1658. toStuff += "\nNumber of Reviews: "+numResp;
  1659. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  1660. toStuff += "\n(Submit a new TO rating for this requester)[/URL]";
  1661. }
  1662. for (var i = 0; i < vars.length; i++) {
  1663. t = new RegExp('\{' + vars[i] + '\}', 'g');
  1664. if (vars[i] == "to_stuff") {
  1665. txt = txt.replace(t, toStuff);
  1666. }
  1667. else if (vars[i] == "to_text") {
  1668. txt = txt.replace(t, toText);
  1669. }
  1670. else if (vars[i] == "prev_link") {
  1671. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[i]]);
  1672. }
  1673. else if (vars[i] == "acc_link") {
  1674. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[i]]);
  1675. }
  1676. else
  1677. txt = txt.replace(t, hit_data[vars[i]]);
  1678. }
  1679. textarea.value = txt;
  1680. }
  1681. function hide_func(div) {
  1682. if (EDIT == false)
  1683. div.style.display = 'none';
  1684. }
  1685. function edit_func() {
  1686. if (EDIT == true) {
  1687. EDIT = false;
  1688. TEMPLATE = textarea.value;
  1689. edit_button.textContent = 'Edit Template';
  1690. apply_template(HIT);
  1691. }
  1692. else {
  1693. console.log("Editing");
  1694. EDIT = true;
  1695. edit_button.textContent = 'Show Changes';
  1696. save_button.disabled = false;
  1697. textarea.value = TEMPLATE;
  1698. }
  1699. }
  1700. function default_func() {
  1701. GM_deleteValue('HITScraper Template');
  1702. TEMPLATE = DEFAULT_TEMPLATE;
  1703. EDIT = false;
  1704. edit_button.textContent = 'Edit Template';
  1705. apply_template(HIT);
  1706. }
  1707. function save_func() {
  1708. if (EDIT)
  1709. TEMPLATE = textarea.value;
  1710. GM_setValue('HITScraper Template', TEMPLATE);
  1711. }
  1712. var div = document.createElement('div');
  1713. var textarea = document.createElement('textarea');
  1714. var div2 = document.createElement('label');
  1715. div.style.position = 'fixed';
  1716. div.style.width = '500px';
  1717. div.style.height = '235px';
  1718. div.style.left = '50%';
  1719. div.style.right = '50%';
  1720. div.style.margin = '-250px 0px 0px -250px';
  1721. div.style.top = '300px';
  1722. div.style.padding = '5px';
  1723. div.style.border = '2px';
  1724. div.style.backgroundColor = 'black';
  1725. div.style.color = 'white';
  1726. div.style.zIndex = '100';
  1727. textarea.style.padding = '2px';
  1728. textarea.style.width = '500px';
  1729. textarea.style.height = '200px';
  1730. textarea.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}';
  1731. div.textContent = 'Press Ctrl+C to copy to clipboard. Click textarea to close';
  1732. div.style.fontSize = '12px';
  1733. div.appendChild(textarea);
  1734. var edit_button = document.createElement('button');
  1735. var save_button = document.createElement('button');
  1736. var default_button = document.createElement('button');
  1737. var easy_button = document.createElement('button');
  1738. edit_button.textContent = 'Edit Template';
  1739. edit_button.setAttribute('id', 'edit_button');
  1740. edit_button.style.height = '18px';
  1741. edit_button.style.width = '100px';
  1742. edit_button.style.fontSize = '10px';
  1743. edit_button.style.paddingLeft = '3px';
  1744. edit_button.style.paddingRight = '3px';
  1745. edit_button.style.backgroundColor = 'white';
  1746. save_button.textContent = 'Save Template';
  1747. save_button.setAttribute('id', 'save_button');
  1748. save_button.style.height = '18px';
  1749. save_button.style.width = '100px';
  1750. save_button.style.fontSize = '10px';
  1751. save_button.style.paddingLeft = '3px';
  1752. save_button.style.paddingRight = '3px';
  1753. save_button.style.backgroundColor = 'white';
  1754. save_button.style.marginLeft = '5px';
  1755. easy_button.textContent = 'Change Adfly Url';
  1756. easy_button.setAttribute('id', 'easy_button');
  1757. easy_button.style.height = '18px';
  1758. easy_button.style.width = '100px';
  1759. easy_button.style.fontSize = '10px';
  1760. easy_button.style.paddingLeft = '3px';
  1761. default_button.textContent = ' D ';
  1762. default_button.setAttribute('id', 'default_button');
  1763. default_button.style.height = '18px';
  1764. default_button.style.width = '20px';
  1765. default_button.style.fontSize = '10px';
  1766. default_button.style.paddingLeft = '3px';
  1767. default_button.style.paddingRight = '3px';
  1768. default_button.style.backgroundColor = 'white';
  1769. default_button.style.marginLeft = '5px';
  1770. default_button.title = 'Return default template';
  1771. div.appendChild(edit_button);
  1772. div.appendChild(save_button);
  1773. div.appendChild(default_button);
  1774. div.appendChild(easy_button);
  1775. save_button.disabled = true;
  1776. div.style.display = 'none';
  1777. textarea.addEventListener("click", function() {hide_func(div);}, false);
  1778. edit_button.addEventListener("click", function() {edit_func();}, false);
  1779. save_button.addEventListener("click", function() {save_func();}, false);
  1780. default_button.addEventListener("click", function() {default_func();}, false);
  1781. document.body.insertBefore(div, document.body.firstChild);
  1782.  
  1783. //Functions below were added for the irc export with the help of clickhappier and Cristo
  1784.  
  1785. function getTO(f){
  1786. var toComp = [];
  1787. var toUrl = 'https://mturk-api.istrack.in/multi-attrs.php?ids='+f;
  1788. requestTO = new XMLHttpRequest();
  1789. requestTO.onreadystatechange = function () {
  1790. if ((requestTO.readyState ===4) && (requestTO.status ===200)) {
  1791. if(requestTO.responseText.split(':').length > 2){
  1792. var toInfo = requestTO.responseText.split('{')[3].split('}')[0].split(',');
  1793. for (var t = 0; t < 3; t++){
  1794. var arrTo = toInfo[t].split(':');
  1795. toComp.push(arrTo[1].substring(1,4));
  1796. }
  1797. } else {
  1798. toComp = ['-','-','-'];
  1799. }
  1800. }
  1801. };
  1802. requestTO.open('GET', toUrl, false);
  1803. requestTO.send(null);
  1804. return toComp;
  1805. }
  1806.  
  1807. function sleep(ms){ // from http://www.digimantra.com/tutorials/sleep-or-wait-function-in-javascript/
  1808. var dt = new Date();
  1809. dt.setTime(dt.getTime() + ms);
  1810. while (new Date().getTime() < dt.getTime());
  1811. }
  1812. function ns4tShorten(url){ // mturk-only URL shortener on Tjololo's server ns4t.net
  1813. console.log("ns4tShorten function");
  1814. var shortRes;
  1815. var urlT = "https://ns4t.net/yourls-api.php" + "?action=shorturl&url=" + encodeURIComponent(url) + "&format=simple&title=MTurk&username=publicuser&password=publicpass";
  1816. requestNs4t = new XMLHttpRequest();
  1817. requestNs4t.onreadystatechange = function () {
  1818. if (requestNs4t.readyState == 4) {
  1819. if (requestNs4t.status == 200) {
  1820. shortRes = requestNs4t.responseText;
  1821. console.log("ns4t.net response: " + requestNs4t.status + " " + requestNs4t.statusText + " " + requestNs4t.responseText);
  1822. }
  1823. else {
  1824. console.log('ns4t.net unsuccessful: ' + requestNs4t.status + " " + requestNs4t.statusText);
  1825. }
  1826. }
  1827. };
  1828. requestNs4t.open('GET', urlT, false);
  1829. requestNs4t.send(null);
  1830. return shortRes;
  1831. }
  1832. 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)
  1833. console.log("tnyimShorten function");
  1834. var shortRes;
  1835. var urlT = "https://tny.im/yourls-api.php" + "?action=shorturl&url=" + encodeURIComponent(url) + "&format=simple&title=MTurk";
  1836. requestTnyim = new XMLHttpRequest();
  1837. requestTnyim.onreadystatechange = function () {
  1838. if (requestTnyim.readyState == 4) {
  1839. if (requestTnyim.status == 200) {
  1840. shortRes = requestTnyim.responseText;
  1841. console.log("tny.im response: " + requestTnyim.status + " " + requestTnyim.statusText + " " + requestTnyim.responseText);
  1842. }
  1843. else {
  1844. console.log('tny.im unsuccessful: ' + requestTnyim.status + " " + requestTnyim.statusText);
  1845. }
  1846. }
  1847. };
  1848. requestTnyim.open('GET', urlT, false);
  1849. requestTnyim.send(null);
  1850. return shortRes;
  1851. }
  1852. function googlShorten(url){ // Goo.gl URL Shortener
  1853. console.log("googlShorten function");
  1854. var shortRes;
  1855. var urlG = "https://www.googleapis.com/urlshortener/v1/url";
  1856. var requestGoogl = new XMLHttpRequest();
  1857. requestGoogl.open("POST", urlG, false);
  1858. requestGoogl.setRequestHeader("Content-Type", "application/json");
  1859. requestGoogl.onreadystatechange = function() {
  1860. if (requestGoogl.readyState == 4) {
  1861. if (requestGoogl.status == 200) {
  1862. shortRes = JSON.parse(requestGoogl.response).id;
  1863. console.log("goo.gl response: " + requestGoogl.status + " " + requestGoogl.statusText + " " + JSON.parse(requestGoogl.response).id );
  1864. }
  1865. else {
  1866. console.log('goo.gl unsuccessful: ' + requestGoogl.status + " " + requestGoogl.statusText);
  1867. }
  1868. }
  1869. };
  1870. var data = new Object();
  1871. data.longUrl = url;
  1872. requestGoogl.send(JSON.stringify(data));
  1873.  
  1874. return shortRes;
  1875. }
  1876. function shortenUrl(url){
  1877. sleep(300); // milliseconds delay - wait some milliseconds (currently less than 1/3rd of a second) between shortens to reduce chance of hitting usage limits
  1878. var shortRes;
  1879. shortRes = ns4tShorten(url);
  1880. if ( shortRes === undefined ) { // if you reached the Google URL shortener's temporary usage limits or the server is otherwise unavailable
  1881. shortRes = googlShorten(url);
  1882. if ( shortRes === undefined ) { // if you reached the ns4t.net URL shortener's temporary limits or the server is otherwise unavailable
  1883. shortRes = tnyimShorten(url);
  1884. if ( shortRes === undefined ) { // if you reached the tny.im URL shortener's temporary limits too or the server is otherwise unavailable
  1885. shortRes = "(x)";
  1886. }
  1887. }
  1888. }
  1889. return shortRes;
  1890. }
  1891.  
  1892. // output display box
  1893. var ircexportdiv = document.createElement('div');
  1894. var ircexporttextarea = document.createElement('textarea');
  1895. ircexportdiv.style.position = 'fixed';
  1896. ircexportdiv.style.width = '500px';
  1897. ircexportdiv.style.height = '155px';
  1898. ircexportdiv.style.left = '50%';
  1899. ircexportdiv.style.right = '50%';
  1900. ircexportdiv.style.margin = '-250px 0px 0px -250px';
  1901. ircexportdiv.style.top = '300px';
  1902. ircexportdiv.style.padding = '5px';
  1903. ircexportdiv.style.border = '2px';
  1904. ircexportdiv.style.backgroundColor = 'black';
  1905. ircexportdiv.style.color = 'white';
  1906. ircexportdiv.style.zIndex = '100';
  1907. ircexportdiv.setAttribute('id','ircexport_div');
  1908. ircexportdiv.style.display = 'none';
  1909. ircexporttextarea.style.padding = '2px';
  1910. ircexporttextarea.style.width = '500px';
  1911. ircexporttextarea.style.height = '130px';
  1912. ircexporttextarea.title = 'IRC Export Output';
  1913. ircexporttextarea.setAttribute('id','ircexport_text');
  1914. ircexportdiv.textContent = 'IRC Export: Press Ctrl+C to (re-)copy to clipboard. Click textarea to close.';
  1915. ircexportdiv.style.fontSize = '12px';
  1916. ircexportdiv.appendChild(ircexporttextarea);
  1917. document.body.insertBefore(ircexportdiv, document.body.firstChild);
  1918. ircexporttextarea.addEventListener("click", function(){ ircexportdiv.style.display = 'none'; }, false);
  1919.  
  1920. function display(hit){
  1921. //var theButton = e.target;
  1922. //theButton.style.backgroundColor = '#CC0000';
  1923. //var capHand = document.getElementById('capsule'+theButton.getAttribute("place")+'-0');
  1924. //var tBodies = capHand.parentNode.parentNode.parentNode.parentNode.parentNode.parentNode.parentNode;
  1925. console.log(hit);
  1926. //return 0;
  1927. var capReq = hit["requester"];
  1928. var capReqId = hit["rid"];
  1929. var capTitle = hit["title"];
  1930. var capRew = hit["reward"];
  1931.  
  1932. var capTime = hit["time"];
  1933.  
  1934. var capAvailable = hit["time"];
  1935.  
  1936. var qualList = hit["quals"];
  1937. var qualColl = qualList.split(',');
  1938. var masterStat = '';
  1939. for ( var m = 0; m < qualColl.length; m++ ) {
  1940. if ( qualColl[m].indexOf('Masters') > -1 ) {
  1941. masterStat = 'MASTERS • ';
  1942. }
  1943. }
  1944. var capUrl = shortenUrl('https://www.mturk.com'+hit["prev_link"]);
  1945. var capReqUrl = shortenUrl('https://www.mturk.com'+hit["req_link"]);
  1946. var hitLinkUnav = '';
  1947. //if ( capGId == 'unavailable' ) { capUrl = capReqUrl; hitLinkUnav = " (preview link unavailable)"; } // handle logged-out export requests for HITs with no preview/notqualified links
  1948. if (capUrl.indexOf("requesterId") > -1)
  1949. {
  1950. hitLinkUnav = " (preview link unavailable)";
  1951. }
  1952. var toLink = shortenUrl('http://turkopticon.ucsd.edu/'+capReqId);
  1953. var capToStats = getTO(capReqId);
  1954. var shortUrlUnav = '';
  1955. if ( capUrl == '(x)' && capGId != 'unavailable' )
  1956. {
  1957. shortUrlUnav = ' \r\n^ https://www.mturk.com'+hit["prev_link"];
  1958. } // add the full-length preview link at the end if both URL shortener attempts failed
  1959. var exString = 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 + shortUrlUnav ;
  1960. GM_setClipboard(exString);
  1961. ircexporttextarea.textContent = exString;
  1962. ircexportdiv.style.display = 'block';
  1963. ircexporttextarea.select();
  1964. }