HIT Scraper WITH TC EXPORT

Snag HITs.

  1. // ==UserScript==
  2. // @name HIT Scraper WITH TC EXPORT
  3. // @author Kerek and TJ and jawz
  4. // @description Snag HITs.
  5. // Based in part on code from mmmturkeybacon Export Mturk History and mmmturkeybacon Color Coded Search with Checkpoints
  6. // @match https://www.mturk.com/mturk/findhits?match=true#hit_scraper*
  7. // @match https://www.mturk.com/mturk/findhits?match=true?hit_scraper*
  8. // @version 1.4.5
  9. // @grant GM_xmlhttpRequest
  10. // @grant GM_getValue
  11. // @grant GM_setValue
  12. // @grant GM_deleteValue
  13. // @grant GM_setClipboard
  14. // @require http://code.jquery.com/jquery-latest.min.js
  15. // @namespace https://greatest.deepsurf.us/users/1997
  16. // ==/UserScript==
  17.  
  18. //alter the requester ignore last as you desire, case insensitive
  19. var default_list = ["oscar smith", "Diamond Tip Research LLC", "jonathon weber", "jerry torres", "Crowdsource", "we-pay-you-fast", "turk experiment", "jon brelig"];
  20. var ignore_list = default_list;
  21. if (GM_getValue("scraper_ignore_list"))
  22. ignore_list = GM_getValue("scraper_ignore_list");
  23. else
  24. GM_setValue("scraper_ignore_list", default_list);
  25.  
  26. var include_list = [];
  27. if (GM_getValue("scraper_include_list"))
  28. include_list = GM_getValue("scraper_include_list");
  29.  
  30. //This is to update the hit export symbol
  31. var symbol = "☭";
  32.  
  33. //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.
  34. var correct_for_skips = true;
  35.  
  36. //weight the four TO ratings for the coloring. Default has pay twice as important as fairness and nothing for communication and fast.
  37. var COMM_WEIGHT = 0;
  38. var PAY_WEIGHT = 10;
  39. var FAIR_WEIGHT = 5;
  40. var FAST_WEIGHT = 0;
  41.  
  42. //Used for themeing, change the colors to change how scraper looks
  43. var GREEN = '#66CC66'; // > 4
  44. var LIGHTGREEN = '#ADFF2F'; // > 3 GREEN YELLOW
  45. var YELLOW = '#FFD700'; //Not used
  46. var ORANGE = '#FF9900'; // > 2
  47. var RED = '#FF3030'; // <= 2
  48. var BLUE = '#C0D9D9'; // no TO
  49. var GREY = 'lightGrey'; //TO down
  50. var BROWN = '#94704D'; //Font color
  51. var DARKGREY = '#9F9F9F'; //No HITDB, "Not Qualified" column
  52. var BACKGROUND_COLOR = "rgb(19, 19, 19)"; //Background of page
  53.  
  54. //display your hitdb records if applicable
  55. var check_hitDB = true;
  56.  
  57. //default text size
  58. var default_text_size=11;
  59.  
  60. //set to "true" to override checkbox setting and ding on new hits
  61. var newHitDing = false;
  62.  
  63. //DO NOT EDIT ANYTHING BELOW THIS LINE UNLESS YOU KNOW WHAT YOU ARE DOING!
  64.  
  65. var status_text = "None";
  66. var shouldDing = false;
  67. var audio = document.createElement('audio');
  68. document.body.appendChild(audio);
  69. audio.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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';
  70. audio.setAttribute("id", "ding_noise");
  71. audio.volume = 1;
  72.  
  73. function newHits(dingNoise) {
  74. console.log(dingNoise);
  75. if (dingNoise || newHitDing)
  76. document.getElementById("ding_noise").play();
  77. }
  78.  
  79. //For editing the blocklist
  80. var div = document.createElement('div');
  81. var textarea = document.createElement('textarea');
  82.  
  83. div.style.position = 'fixed';
  84. div.style.width = '500px';
  85. div.style.height = '235px';
  86. div.style.left = '50%';
  87. div.style.right = '50%';
  88. div.style.margin = '-250px 0px 0px -250px';
  89. div.style.top = '300px';
  90. div.style.padding = '5px';
  91. div.style.border = '2px';
  92. div.style.backgroundColor = 'black';
  93. div.style.color = 'white';
  94. div.style.zIndex = '100';
  95. div.setAttribute('id','block_div');
  96.  
  97. textarea.style.padding = '2px';
  98. textarea.style.width = '500px';
  99. textarea.style.height = '200px';
  100. textarea.title = 'Block list';
  101. textarea.setAttribute('id','block_text');
  102.  
  103. 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.';
  104. div.style.fontSize = '12px';
  105. div.appendChild(textarea);
  106.  
  107. var save_button = document.createElement('button');
  108.  
  109. save_button.textContent = 'Save';
  110. save_button.setAttribute('id', 'save_blocklist');
  111. save_button.style.height = '18px';
  112. save_button.style.width = '100px';
  113. save_button.style.fontSize = '10px';
  114. save_button.style.paddingLeft = '3px';
  115. save_button.style.paddingRight = '3px';
  116. save_button.style.backgroundColor = 'white';
  117. save_button.style.marginLeft = '5px';
  118.  
  119. div.appendChild(save_button);
  120.  
  121. $("#block_div").hide();
  122. save_button.addEventListener("click", function() {save_blocklist();}, false);
  123. document.body.insertBefore(div, document.body.firstChild);
  124.  
  125. //For editing the include list
  126. var shouldInclude = false;
  127. var div2 = document.createElement('div');
  128. var textarea2 = document.createElement('textarea');
  129.  
  130. div2.style.position = 'fixed';
  131. div2.style.width = '500px';
  132. div2.style.height = '235px';
  133. div2.style.left = '50%';
  134. div2.style.right = '50%';
  135. div2.style.margin = '-250px 0px 0px -250px';
  136. div2.style.top = '300px';
  137. div2.style.padding = '5px';
  138. div2.style.border = '2px';
  139. div2.style.backgroundColor = 'black';
  140. div2.style.color = 'white';
  141. div2.style.zIndex = '100';
  142. div2.setAttribute('id','include_div');
  143.  
  144. textarea2.style.padding = '2px';
  145. textarea2.style.width = '500px';
  146. textarea2.style.height = '200px';
  147. textarea2.title = 'include list';
  148. textarea2.setAttribute('id','include_text');
  149.  
  150. div2.textContent = 'Used if you only want to see certain requesters. Separator is ^. Only takes effect if \"Use includelist\" is checked.';
  151. div2.style.fontSize = '12px';
  152. div2.appendChild(textarea2);
  153.  
  154. var save_button2 = document.createElement('button');
  155.  
  156. save_button2.textContent = 'Save';
  157. save_button2.setAttribute('id', 'save_blocklist');
  158. save_button2.style.height = '18px';
  159. save_button2.style.width = '100px';
  160. save_button2.style.fontSize = '10px';
  161. save_button2.style.paddingLeft = '3px';
  162. save_button2.style.paddingRight = '3px';
  163. save_button2.style.backgroundColor = 'white';
  164. save_button2.style.marginLeft = '5px';
  165.  
  166. div2.appendChild(save_button2);
  167.  
  168. $("#include_div").hide();
  169. save_button2.addEventListener("click", function() {save_includelist();}, false);
  170. document.body.insertBefore(div2, document.body.firstChild);
  171.  
  172. function save_blocklist() {
  173. console.log("Save");
  174. var textarea = $("#block_text");
  175. var text = textarea.val();
  176. var block_list = text.split("^");
  177. console.log(block_list);
  178. var trimmed_list = [];
  179. for (var requester in block_list){
  180. if (block_list[requester].trim().length != 0)
  181. trimmed_list.push(block_list[requester].toLowerCase().trim());
  182. }
  183. GM_setValue("scraper_ignore_list",trimmed_list);
  184. ignore_list = GM_getValue("scraper_ignore_list");
  185. console.log(ignore_list);
  186. $("#block_div").hide();
  187. }
  188.  
  189. function save_includelist() {
  190. console.log("Save");
  191. var textarea = $("#include_text");
  192. var text = textarea.val();
  193. var includes = text.split("^");
  194. console.log(includes);
  195. var trimmed_list = [];
  196. for (var requester in includes){
  197. if (includes[requester].trim().length != 0)
  198. trimmed_list.push(includes[requester].toLowerCase().trim());
  199. }
  200. GM_setValue("scraper_include_list",trimmed_list);
  201. include_list = GM_getValue("scraper_include_list");
  202. console.log(include_list);
  203. $("#include_div").hide();
  204. }
  205.  
  206. var HITStorage = {};
  207. var indexedDB = window.indexedDB || window.webkitIndexedDB ||
  208. window.mozIndexedDB;
  209. window.IDBTransaction = window.IDBTransaction || window.webkitIDBTransaction || window.mozIDBTransaction;
  210. window.IDBKeyRange = window.IDBKeyRange || window.webkitIDBKeyRange || window.mozIDBKeyRange;
  211. HITStorage.IDBTransactionModes = { "READ_ONLY": "readonly", "READ_WRITE": "readwrite", "VERSION_CHANGE": "versionchange" };
  212. var IDBKeyRange = window.IDBKeyRange;
  213.  
  214. HITStorage.indexedDB = {};
  215. HITStorage.indexedDB = {};
  216. HITStorage.indexedDB.db = null;
  217.  
  218. HITStorage.indexedDB.onerror = function(e) {
  219. console.log(e);
  220. };
  221.  
  222. var v=4;
  223.  
  224. HITStorage.indexedDB.checkTitle = function(title,button) {
  225. var request = indexedDB.open("HITDB", v);
  226. request.onsuccess = function(e) {
  227. HITStorage.indexedDB.db = e.target.result;
  228. var db = HITStorage.indexedDB.db;
  229. if (!db.objectStoreNames.contains("HIT"))
  230. {
  231. db.close();
  232. return;
  233. }
  234. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  235. var store = trans.objectStore("HIT");
  236.  
  237. var index = store.index("title");
  238. index.get(title).onsuccess = function(event)
  239. {
  240. if (event.target.result === undefined)
  241. {
  242. console.log(title + ' not found');
  243. history[button].titledb=false;
  244. }
  245. else
  246. {
  247. console.log(title + ' found');
  248. history[button].titledb=true;
  249. }
  250. db.close();
  251. };
  252. };
  253. request.onerror = HITStorage.indexedDB.onerror;
  254. };
  255.  
  256. HITStorage.indexedDB.checkRequester = function(id,button) {
  257. var request = indexedDB.open("HITDB", v);
  258. request.onsuccess = function(e) {
  259. HITStorage.indexedDB.db = e.target.result;
  260. var db = HITStorage.indexedDB.db;
  261. if (!db.objectStoreNames.contains("HIT"))
  262. {
  263. db.close();
  264. return;
  265. }
  266. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  267. var store = trans.objectStore("HIT");
  268.  
  269. var index = store.index("requesterId");
  270. index.get(id).onsuccess = function(event)
  271. {
  272. if (event.target.result === undefined)
  273. {history[button].reqdb=false;
  274. console.log(id + ' not found');
  275. }
  276. else
  277. {
  278. history[button].reqdb=true;
  279. console.log(id + ' found');
  280. }
  281. db.close();
  282. };
  283. };
  284. request.onerror = HITStorage.indexedDB.onerror;
  285. };
  286.  
  287. var PAGES_TO_SCRAPE = 3;
  288. var MINIMUM_HITS = 100;
  289. var SEARCH_REFRESH=0;
  290. var URL_BASE = "/mturk/searchbar?searchWords=&selectedSearchType=hitgroups";
  291. var initial_url = URL_BASE;
  292. var TO_REQ_URL = "http://turkopticon.ucsd.edu/reports?id=";
  293. var found_key_list=[];
  294. var last_clear_time = new Date().getTime();
  295. var searched_once = false;
  296. var save_new_results_time = 120;
  297. var save_results_time = 3600;
  298. var default_type = 0;
  299. var cur_loc = window.location.href;
  300. var time_input = document.createElement("INPUT");
  301. time_input.value = 0;
  302. var page_input = document.createElement("INPUT");
  303. page_input.value = 3;
  304. var min_input = document.createElement("INPUT");
  305. var new_time_display_input = document.createElement("INPUT");
  306. new_time_display_input.value = 300;
  307. var reward_input = document.createElement("INPUT");
  308. var qual_input = document.createElement("INPUT");
  309. qual_input.type = "checkbox";
  310. qual_input.checked = true;
  311. var masters_input = document.createElement("INPUT");
  312. masters_input.type = "checkbox";
  313. var sort_input1 = document.createElement("INPUT");
  314. sort_input1.type = "radio";
  315. sort_input1.name = "sort_type";
  316. sort_input1.value = "latest";
  317. sort_input1.checked = true;
  318. var sort_input2 = document.createElement("INPUT");
  319. sort_input2.type = "radio";
  320. sort_input2.name = "sort_type";
  321. sort_input2.value = "most";
  322. var sort_input3 = document.createElement("INPUT");
  323. sort_input3.type = "radio";
  324. sort_input3.name = "sort_type";
  325. sort_input3.value = "amount";
  326. var sort_input4 = document.createElement("INPUT");
  327. sort_input4.type = "radio";
  328. sort_input4.name = "sort_type";
  329. sort_input4.value = "A-Z";
  330. var sort_input_invert = document.createElement("INPUT");
  331. sort_input_invert.type = "checkbox";
  332. var friesAreDone = document.createElement("INPUT");
  333. friesAreDone.type = "checkbox";
  334. var correctForSkips = document.createElement("INPUT");
  335. correctForSkips.type = "checkbox";
  336. correctForSkips.checked = true;
  337. var matchOnly = document.createElement("INPUT");
  338. matchOnly.type = "checkbox";
  339.  
  340. var search_input = document.createElement("INPUT");
  341.  
  342. var LINK_BASE = "https://www.mturk.com";
  343. var STATUSDETAIL_DELAY = 250;
  344. var MPRE_DELAY = 3000;
  345.  
  346. $('body').css('background', BACKGROUND_COLOR);
  347.  
  348. var next_page = 1;
  349.  
  350. var API_PROXY_BASE = 'https://mturk-api.istrack.in/';
  351. var API_MULTI_ATTRS_URL = API_PROXY_BASE + 'multi-attrs.php?ids=';
  352. var REVIEWS_BASE = 'http://turkopticon.ucsd.edu/';
  353.  
  354. var control_panel_HTML = '<div id="control_panel" style="margin: 0 auto 0 auto;' +
  355. 'border-bottom: 1px solid #000000; margin-bottom: 5px; ' +
  356. 'background-color: ' + BACKGROUND_COLOR + ';"></div>';
  357. $('body > :not(#control_panel)').hide(); //hide all nodes directly under the body
  358. $('body').prepend(control_panel_HTML);
  359.  
  360. var control_panel = document.getElementById("control_panel");
  361. var big_red_button = document.createElement("BUTTON");
  362. var reset_blocks = document.createElement("BUTTON");
  363. var include_button = document.createElement("BUTTON");
  364. var progress_report = document.createTextNode("Stopped");
  365. var status_report = document.createTextNode("None");
  366. var text_area = document.createElement("TABLE");
  367. big_red_button.textContent = "Show Interface";
  368. big_red_button.onclick = function(){show_interface();};
  369. control_panel.appendChild(big_red_button);
  370.  
  371. show_interface();
  372.  
  373. var global_run = false;
  374. var statusdetail_loop_finished = false;
  375. var date_header = "";
  376. var history = {};
  377. var wait_loop;
  378.  
  379. function set_progress_report(text, force)
  380. {
  381. if (global_run == true || force == true)
  382. {
  383. progress_report.textContent = text;
  384. status_report.textContent = status_text;
  385. }
  386. }
  387.  
  388. function get_progress_report()
  389. {
  390. return progress_report.textContent;
  391. }
  392.  
  393. function wait_until_stopped()
  394. {
  395. if (global_run == true)
  396. {
  397. if (statusdetail_loop_finished == true)
  398. {
  399. big_red_button.textContent = "Start";
  400. set_progress_report("Finished", false);
  401. }
  402. else
  403. {
  404. setTimeout(function(){wait_until_stopped();}, 500);
  405. }
  406. }
  407. }
  408.  
  409. function display_wait_time(wait_time)
  410. {
  411. if (global_run == true)
  412. {
  413. var current_progress = get_progress_report();
  414. if (current_progress.indexOf("Searching again in")!==-1)
  415. {
  416. set_progress_report(current_progress.replace(/Searching again in \d+ seconds/ , "Searching again in " + wait_time + " seconds"),false);
  417. }
  418. else
  419. set_progress_report(current_progress + " Searching again in " + wait_time + " seconds.", false);
  420. if (wait_time>1)
  421. setTimeout(function(){display_wait_time(wait_time-1);}, 1000);
  422. }
  423. }
  424.  
  425. function dispArr(ar)
  426. {
  427. var disp = "";
  428. for (var z = 0; z < ar.length; z++)
  429. {
  430. disp += "id " + z + " is " + ar[z] + " ";
  431. }
  432. console.log(disp);
  433. }
  434.  
  435. function scrape($src)
  436. {
  437. var $requester = $src.find('a[href^="/mturk/searchbar?selectedSearchType=hitgroups&requester"]');
  438. var $title = $src.find('a[class="capsulelink"]');
  439. var $reward = $src.find('span[class="reward"]');
  440. var $preview = $src.find('a[href^="/mturk/preview?"]');
  441. var $qualified = $src.find('a[href^="/mturk/notqualified?"]');
  442. var $times = $src.find('a[id^="duration_to_complete"]');
  443. var $descriptions = $src.find('a[id^="description"]');
  444. var not_qualified_group_IDs=[];
  445. var $quals = $src.find('a[id^="qualificationsRequired"]');
  446. $qualified.each(function(){
  447. var groupy = $(this).attr('href');
  448. groupy = groupy.replace("/mturk/notqualified?hitId=","");
  449. not_qualified_group_IDs.push(groupy);
  450. });
  451. var $mixed = $src.find('a[href^="/mturk/preview?"],a[href^="/mturk/notqualified?"]');
  452. var listy =[];
  453. $mixed.each(function(){
  454. var groupy = $(this).attr('href');
  455. groupy = groupy.replace("/mturk/notqualified?hitId=","");
  456. groupy = groupy.replace("/mturk/preview?groupId=","");
  457. listy.push(groupy);
  458. });
  459. listy = listy.filter(function(elem, pos) {
  460. return listy.indexOf(elem) == pos;
  461. });
  462.  
  463. for (var j = 0; j < $requester.length; j++)
  464. {
  465. var $hits = $requester.eq(j).parent().parent().parent().parent().parent().parent().find('td[class="capsule_field_text"]');
  466. var requester_name = $requester.eq(j).text().trim();
  467. var requester_link = $requester.eq(j).attr('href');
  468. var group_ID=listy[j];
  469. var preview_link = "/mturk/preview?groupId=" + group_ID;
  470. var title = $title.eq(j).text().trim();
  471. var reward = $reward.eq(j).text().trim();
  472. var hits = $hits.eq(4).text().trim();
  473. var time = $times.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  474. var description = $descriptions.eq(j).parent()[0].nextSibling.nextSibling.innerHTML;
  475. //console.log(description);
  476. var requester_id = requester_link.replace('/mturk/searchbar?selectedSearchType=hitgroups&requesterId=','');
  477. var accept_link;
  478. accept_link = preview_link.replace('preview','previewandaccept');
  479. /*HIT SCRAPER ADDITION*/
  480. var qElements = $quals.eq(j).parent().parent().parent().find('tr');
  481. //console.log(qElements);
  482.  
  483. var qualifications = [];
  484. for (var i = 1; i < qElements.length; i++) {
  485. 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, ' ')));
  486. }
  487. var qualList = (qualifications.join(', ') ? qualifications.join(', ') : "None");
  488.  
  489. key = requester_name+title+reward+group_ID;
  490. found_key_list.push(key);
  491. if (history[key] == undefined)
  492. {
  493. 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:""};
  494. history[key].req_link = requester_link;
  495. history[key].prev_link = preview_link;
  496. history[key].requester = requester_name;
  497. history[key].title = title;
  498. history[key].reward = reward;
  499. history[key].hits = hits;
  500. history[key].rid = requester_id;
  501. history[key].acc_link = accept_link;
  502. history[key].time = time;
  503. history[key].quals = qualList;
  504. history[key].description = description;
  505. HITStorage.indexedDB.checkRequester(requester_id,key);
  506. HITStorage.indexedDB.checkTitle(title,key);
  507. if (searched_once)
  508. {
  509. history[key].initial_time = new Date().getTime();//-1000*(save_new_results_time - SEARCH_REFRESH);
  510. history[key].new_result = 0;
  511. }
  512. else
  513. {
  514. history[key].initial_time = new Date().getTime()-1000*save_new_results_time;
  515. history[key].new_result = 1000*save_new_results_time;
  516. }
  517. if (not_qualified_group_IDs.indexOf(group_ID)!==-1)
  518. history[key].qualified = false;
  519. else
  520. history[key].qualified = true;
  521.  
  522. history[key].found_this_time = true;
  523. }
  524. else
  525. {
  526. history[key].new_result = new Date().getTime() - history[key].initial_time;
  527. history[key].found_this_time = true;
  528. history[key].hits = hits;
  529. }
  530. }
  531. }
  532.  
  533. function statusdetail_loop(next_URL)
  534. {
  535. if (global_run == true)
  536. {
  537. if (next_URL.length != 0)
  538. {
  539. $.get(next_URL, function(data)
  540. {
  541. var $src = $(data);
  542. var maxpagerate = $src.find('td[class="error_title"]:contains("You have exceeded the maximum allowed page request rate for this website.")');
  543. if (maxpagerate.length == 0)
  544. {
  545. if (next_page > PAGES_TO_SCRAPE)
  546. {
  547. if(status_text.indexOf("Correcting for skips") == -1)
  548. status_text += ". Correcting for skips";
  549. }
  550. set_progress_report("Processing page " + next_page, false);
  551. scrape($src);
  552. $next_URL = $src.find('a[href^="/mturk/viewsearchbar"]:contains("Next")');
  553. next_URL = ($next_URL.length != 0) ? $next_URL.attr("href") : "";
  554. next_page++;
  555. if (default_type == 1)
  556. {
  557. var hmin = MINIMUM_HITS+1;
  558. for (j = 0; j < found_key_list.length; j++)
  559. {
  560. console.log(history[found_key_list[j]]);
  561. if (history[found_key_list[j]].hits < hmin)
  562. {
  563. next_URL = "";
  564. next_page = -1;
  565. break;
  566. }
  567. }
  568. }
  569. else if (next_page > PAGES_TO_SCRAPE && correct_for_skips)
  570. {
  571. var skipped_hits = 0;
  572. var added_pages = 0;
  573. for (j = 0; j < found_key_list.length; j++)
  574. {
  575. var obj = history[found_key_list[j]];
  576. if (!ignore_check(obj.requester,obj.title))
  577. skipped_hits++;
  578. }
  579. added_pages = Math.floor(skipped_hits/10);
  580. if (skipped_hits%10 >6)
  581. added_pages++;
  582. if (next_page > PAGES_TO_SCRAPE + added_pages)
  583. {
  584. next_URL = "";
  585. next_page = -1;
  586. }
  587. }
  588. else if (next_page > PAGES_TO_SCRAPE)
  589. {
  590. next_URL = "";
  591. next_page = -1;
  592. }
  593. setTimeout(function(){statusdetail_loop(next_URL);}, STATUSDETAIL_DELAY);
  594. }
  595. else
  596. {
  597. console.log("MPRE");
  598. setTimeout(function(){statusdetail_loop(next_URL);}, MPRE_DELAY);
  599. }
  600. });
  601. }
  602. else
  603. {
  604. searched_once = true;
  605. var found_hits = found_key_list.length;
  606. var shown_hits = 0;
  607. var new_hits = 0;
  608. var url = API_MULTI_ATTRS_URL;
  609. var rids = [];
  610. var lastRow = text_area.rows.length - 1;
  611. for (i = lastRow; i>0; i--)
  612. text_area.deleteRow(i);
  613. for (j = 0; j < found_key_list.length; j++)
  614. {
  615. //(function(url,rids,j) {
  616. var obj = history[found_key_list[j]];
  617. if (ignore_check(obj.requester,obj.title) && obj.found_this_time){
  618. ++shown_hits;
  619. //console.log(obj);
  620. //hit export will update col_heads[1]
  621. 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>"];
  622. var row = text_area.insertRow(text_area.rows.length);
  623. url += obj.rid + ',';
  624. rids.push(obj.rid);
  625. if (check_hitDB)
  626. {
  627. col_heads.push("R");
  628. col_heads.push("T");
  629. }
  630. if (!obj.qualified)
  631. {
  632. col_heads.push("Not Qualified");
  633. }
  634. for (i=0; i<col_heads.length; i++)
  635. {
  636. var this_cell = row.insertCell(i);
  637. row.cells[i].style.fontSize = default_text_size;
  638. this_cell.innerHTML = col_heads[i];
  639. if(i>1)
  640. this_cell.style.textAlign = 'center';
  641. if (check_hitDB)
  642. {
  643. if (i==6)
  644. {
  645. if (obj.reqdb){
  646. this_cell.style.backgroundColor = GREEN;
  647. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,0);}})(obj));
  648. }
  649. else
  650. this_cell.style.backgroundColor = RED;
  651. }
  652. else if (i==7)
  653. {
  654. if (obj.titledb){
  655. this_cell.style.backgroundColor = GREEN;
  656. this_cell.addEventListener("click", (function (obj) { return function() {search_deleg(obj,1);}})(obj));
  657. }
  658. else
  659. this_cell.style.backgroundColor = RED;
  660. }
  661. else if (i==8)
  662. this_cell.style.backgroundColor = DARKGREY;
  663. }
  664. else if (i==6)
  665. this_cell.style.backgroundColor = DARKGREY;
  666. }
  667. if (Object.keys(history).length>0)
  668. {
  669. if (obj.new_result < 1000*save_new_results_time)
  670. {
  671. new_hits++;
  672. for (i in col_heads)
  673. {
  674. row.cells[i].style.fontSize = default_text_size + 1;
  675. row.cells[i].style.fontWeight = "bold";
  676. }
  677. }
  678. }
  679. button = document.createElement('button'); //HIT SCRAPER ADDITION
  680. button.textContent = 'vB';
  681. button.title = 'Export this HIT description as vBulletin formatted text';
  682. button.style.height = '14px';
  683. button.style.width = '30px';
  684. button.style.fontSize = '8px';
  685. button.style.border = '1px solid';
  686. button.style.padding = '0px';
  687. button.style.backgroundColor = 'transparent';
  688. button2 = document.createElement('button'); //BUTTON TO BLOCK REQUESTER
  689. button2.textContent = 'R';
  690. button2.title = 'Add requester to block list';
  691. button2.style.height = '14px';
  692. button2.style.width = '15px';
  693. button2.style.fontSize = '8px';
  694. button2.style.border = '1px solid';
  695. button2.style.padding = '0px';
  696. button2.style.backgroundColor = 'transparent';
  697. button3 = document.createElement('button'); //BUTTON TO BLOCK TITLE
  698. button3.textContent = 'T';
  699. button3.title = 'Add title to block list';
  700. button3.style.height = '14px';
  701. button3.style.width = '15px';
  702. button3.style.fontSize = '8px';
  703. button3.style.border = '1px solid';
  704. button3.style.padding = '0px';
  705. button3.style.backgroundColor = 'transparent';
  706. button5 = document.createElement('button'); //Tinychat
  707. button5.textContent = 'TC P';
  708. button5.title = 'Export this HIT description as vBulletin formatted text';
  709. button5.style.height = '14px';
  710. button5.style.width = '30px';
  711. button5.style.fontSize = '8px';
  712. button5.style.border = '1px solid';
  713. button5.style.padding = '0px';
  714. button5.style.backgroundColor = 'transparent';
  715. button6 = document.createElement('button'); //Tinychat
  716. button6.textContent = 'TC PA';
  717. button6.title = 'Export this HIT description as vBulletin formatted text';
  718. button6.style.height = '14px';
  719. button6.style.width = '30px';
  720. button6.style.fontSize = '8px';
  721. button6.style.border = '1px solid';
  722. button6.style.padding = '0px';
  723. button6.style.backgroundColor = 'transparent';
  724. //button.addEventListener("click", function() {export_func_deleg(j);}.bind(null,j), false);
  725. button.addEventListener("click", (function (obj,j) { return function() {export_func_deleg(obj,j);}})(obj,j));
  726. button5.addEventListener("click", (function (obj,j) { return function() {apply_template2(obj);}})(obj,j));
  727. button6.addEventListener("click", (function (obj,j) { return function() {apply_template2(obj,"A");}})(obj,j));
  728. row.cells[1].appendChild(document.createTextNode(" "));
  729. row.cells[1].appendChild(button);
  730. row.cells[1].appendChild(button5);
  731. row.cells[1].appendChild(button6);
  732. button2.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,0);}})(obj,j));
  733. row.cells[0].appendChild(document.createTextNode(" "));
  734. row.cells[0].appendChild(button2);
  735. button3.addEventListener("click", (function (obj,j) { return function() {block_deleg(obj,1);}})(obj,j));
  736. row.cells[0].appendChild(button3);
  737. }
  738. //});
  739. }
  740. set_progress_report("Scrape complete. " + shown_hits + " HITs found (" + new_hits + " new results). " + (found_hits - shown_hits) + " HITs ignored.", false);
  741. if (new_hits > 0){
  742. newHits(shouldDing);
  743. }
  744. url = url.substring(0,url.length - 1);
  745. //console.log(url);
  746. var success_flag = false;
  747. GM_xmlhttpRequest(
  748. {
  749. method: "GET",
  750. url: url,
  751. onload: function (results)
  752. {
  753. //console.log(results.responseText);
  754. rdata = $.parseJSON(results.responseText);
  755. for (i = 0; i < rids.length; i++)
  756. {
  757. text_area.rows[i+1].style.backgroundColor = GREY;
  758. if (rdata[rids[i]])
  759. {
  760. var pay = rdata[rids[i]].attrs.pay;
  761. var reviews = rdata[rids[i]].reviews;
  762. var average = 0;
  763. var sum = 0;
  764. var divisor = 0;
  765. var comm = rdata[rids[i]].attrs.comm;
  766. var fair = rdata[rids[i]].attrs.fair;
  767. var fast = rdata[rids[i]].attrs.fast;
  768. if (comm > 0)
  769. {
  770. sum += COMM_WEIGHT*comm;
  771. divisor += COMM_WEIGHT;
  772. }
  773. if (pay > 0)
  774. {
  775. sum += PAY_WEIGHT*pay;
  776. divisor += PAY_WEIGHT;
  777. }
  778. if (fair > 0)
  779. {
  780. sum += FAIR_WEIGHT*fair;
  781. divisor += FAIR_WEIGHT;
  782. }
  783. if (fast > 0)
  784. {
  785. sum += FAST_WEIGHT*fast;
  786. divisor += FAST_WEIGHT;
  787. }
  788. if (divisor > 0)
  789. {
  790. average = sum/divisor;
  791. }
  792. text_area.rows[i+1].cells[4].innerHTML = "<a href='"+ TO_REQ_URL+rids[i] +"' target='_blank'>" + pay + "</a>";
  793. if (reviews > 4)
  794. {
  795. if (average > 4.49)
  796. text_area.rows[i+1].style.backgroundColor = GREEN;
  797. else if (average > 3.49)
  798. text_area.rows[i+1].style.backgroundColor = LIGHTGREEN;
  799. //else if (average > 2.99)
  800. // text_area.rows[i+1].style.backgroundColor = YELLOW;
  801. else if (average > 1.99)
  802. text_area.rows[i+1].style.backgroundColor = ORANGE;
  803. else if (average > 0)
  804. text_area.rows[i+1].style.backgroundColor = RED;
  805. }
  806. }
  807. else
  808. {
  809. text_area.rows[i+1].cells[4].innerHTML = "No data";
  810. }
  811. }
  812. success_flag = true;
  813. }
  814. });
  815. if (!success_flag)
  816. for (i = 0; i < rids.length; i++) text_area.rows[i+1].style.backgroundColor = GREY;
  817. statusdetail_loop_finished = true;
  818. if (SEARCH_REFRESH>0)
  819. {
  820. wait_loop = setTimeout(function(){if (global_run) start_it();}, 1000*SEARCH_REFRESH);
  821. display_wait_time(SEARCH_REFRESH);
  822. }
  823. else
  824. {
  825. global_run = false;
  826. big_red_button.textContent = "Start";
  827. }
  828. }
  829. }
  830. }
  831.  
  832. function ignore_check(r,t){
  833. tempList = ignore_list.map(function(item) { return item.toLowerCase(); });
  834. foundR = -1;
  835. foundT = -1;
  836. foundR = tempList.indexOf(r.toLowerCase());
  837. foundT = tempList.indexOf(t.toLowerCase());
  838. if (shouldInclude){
  839. console.log(include_list);
  840. temp = include_list.map(function(item) { return item.toLowerCase(); }).indexOf(r.toLowerCase());
  841. console.log(temp);
  842. if (temp != -1)
  843. foundR = -1;
  844. else
  845. foundR = 0;
  846. }
  847. found = foundR == -1 && foundT == -1;
  848. //console.log("r: "+r+" t: "+t+" f: "+found);
  849. return found;
  850. //return -1 == ignore_list.map(function(item) { return item.toLowerCase(); }).indexOf(r.toLowerCase());
  851. }
  852.  
  853. function start_running()
  854. {
  855. if (big_red_button.textContent == "Start")
  856. {
  857. status_text="";
  858. ignore_list = GM_getValue("scraper_ignore_list");
  859. if (GM_getValue("scraper_include_list"))
  860. include_list = GM_getValue("scraper_include_list");
  861. global_run = true;
  862. initial_url = URL_BASE;
  863. if (search_input.value.length>0)
  864. {
  865. initial_url = initial_url.replace("searchWords=", "searchWords=" + search_input.value);
  866. }
  867. if (time_input.value.replace(/[^0-9]+/g,"") != "")
  868. {
  869. SEARCH_REFRESH = Number(time_input.value);
  870. }
  871. if (page_input.value.replace(/[^0-9]+/g,"") != "")
  872. {
  873. PAGES_TO_SCRAPE = Number(page_input.value);
  874. }
  875. if (min_input.value.replace(/[^0-9]+/g,"") != "")
  876. {
  877. if (!sort_input2.checked)
  878. status_text += " Minimum batch size selected but not sorting by most available";
  879. MINIMUM_HITS = Number(min_input.value);
  880. }
  881. if (new_time_display_input.value.replace(/[^0-9]+/g,"") != "")
  882. {
  883. save_new_results_time = Number(new_time_display_input.value);
  884. }
  885. if (reward_input.value.replace(/[^0-9]+/g,"") != "")
  886. {
  887. initial_url += "&minReward=" + reward_input.value;
  888. }
  889. else
  890. {
  891. initial_url += "&minReward=0.00";
  892. }
  893. if (qual_input.checked)
  894. {
  895. initial_url += "&qualifiedFor=on";
  896. }
  897. else
  898. {
  899. initial_url += "&qualifiedFor=off";
  900. }
  901. if (masters_input.checked)
  902. {
  903. initial_url += "&requiresMasterQual=on";
  904. }
  905. if (sort_input1.checked)
  906. {
  907. initial_url+= "&sortType=LastUpdatedTime%3A";
  908. default_type = 0;
  909. }
  910. else if (sort_input2.checked)
  911. {
  912. initial_url+= "&sortType=NumHITs%3A";
  913. default_type = 1;
  914. status_text += " Sorting by NumHITs ignores correct for skips in favor of minimum batch size";
  915. }
  916. else if (sort_input3.checked)
  917. {
  918. initial_url+= "&sortType=Reward%3A";
  919. default_type = 0;
  920. }
  921. else if (sort_input4.checked)
  922. {
  923. initial_url += "&sortType=Title%3A";
  924. }
  925. if (sort_input_invert.checked)
  926. {
  927. if (sort_input4.checked)
  928. initial_url += "1";
  929. else
  930. initial_url += "0";
  931. }
  932. else
  933. {
  934. if (sort_input4.checked)
  935. initial_url += "0";
  936. else
  937. initial_url += "1";
  938. }
  939. if (friesAreDone.checked)
  940. {
  941. shouldDing = true;
  942. }
  943. else {
  944. shouldDing = false;
  945. }
  946. if (correctForSkips.checked){
  947. if (matchOnly.checked)
  948. {
  949. status_text += " Match only checked, ignoring skip correction to prevent issues.";
  950. correct_for_skips = false;
  951. }
  952. else{
  953. correct_for_skips = true;
  954. }
  955. }
  956. else {
  957. correct_for_skips = false;
  958. }
  959. if (matchOnly.checked){
  960. if (include_list.length == 0){
  961. status_text += " No items in include list. Ignoring inclusion checkbox.";
  962. shouldInclude = false;
  963. }
  964. else
  965. shouldInclude = true;
  966. }
  967. else{
  968. shouldInclude = false;
  969. }
  970. if (status_text == "")
  971. status_text = "None";
  972. initial_url+="&pageNumber=1&searchSpec=HITGroupSearch"
  973. start_it();
  974. }
  975. else
  976. {
  977. global_run = false;
  978. clearTimeout(wait_loop);
  979. big_red_button.textContent = "Start";
  980. set_progress_report("Stopped", true);
  981. }
  982. }
  983.  
  984. function start_it()
  985. {
  986. statusdetail_loop_finished = false;
  987. big_red_button.textContent = "Stop";
  988. found_key_list=[];
  989. var ctime = new Date().getTime()
  990. if (ctime - last_clear_time > save_results_time*666)
  991. {
  992. var last_history=history;
  993. history = {};
  994. for (var key in last_history)
  995. {
  996. if (last_history[key].new_result<save_results_time*1000)
  997. {
  998. history[key]=last_history[key];
  999. if (last_history[key].found_this_time)
  1000. {
  1001. last_history[key].found_this_time = false;
  1002. if (last_history[key].new_result>save_new_results_time*1000)
  1003. last_history[key].initial_time = ctime-1000*save_new_results_time;
  1004. }
  1005. }
  1006.  
  1007. }
  1008. last_clear_time = ctime;
  1009. }
  1010. next_page = 1;
  1011. statusdetail_loop(initial_url);
  1012. }
  1013.  
  1014.  
  1015. function show_interface()
  1016. {
  1017. control_panel.style.color = BROWN;
  1018. control_panel.style.fontSize = 14;
  1019. control_panel.removeChild(big_red_button);
  1020. control_panel.appendChild(document.createTextNode("Auto-refresh delay: "));
  1021. time_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1022. time_input.title = "Enter search refresh delay in seconds\n" + "Enter 0 for no auto-refresh\n" + "Default is 0 (no auto-refresh)";
  1023. time_input.size = 3;
  1024. control_panel.appendChild(time_input);
  1025. control_panel.appendChild(document.createTextNode(" "));
  1026. control_panel.appendChild(document.createTextNode("| Pages to scrape: "));
  1027. page_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1028. page_input.title = "Enter number of pages to scrape\n" + "Default is 4";
  1029. page_input.size = 3;
  1030. control_panel.appendChild(page_input);
  1031. control_panel.appendChild(document.createTextNode(" Correct for skips: "));
  1032. correctForSkips.title = "Searches additional pages to get a consistent number of results. Helpful if you're blocking a lot of people";
  1033. control_panel.appendChild(correctForSkips);
  1034. control_panel.appendChild(document.createTextNode(" "));
  1035. control_panel.appendChild(document.createTextNode("| Minimum batch size: "));
  1036. min_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1037. min_input.title = "Enter minimum HITs for batch search\n" + "Default is 100";
  1038. min_input.size = 3;
  1039. control_panel.appendChild(min_input);
  1040. control_panel.appendChild(document.createTextNode(" "));
  1041. control_panel.appendChild(document.createTextNode("| New HIT highlighting: "));
  1042. new_time_display_input.onkeydown = function(event){if (event.keyCode == 13){start_running();}};
  1043. new_time_display_input.title = "Enter time (in seconds) to keep new HITs highlighted\n" + "Default is 300 (5 minutes)";
  1044. new_time_display_input.size = 6;
  1045. control_panel.appendChild(new_time_display_input);
  1046. control_panel.appendChild(document.createTextNode(" "));
  1047. control_panel.appendChild(document.createTextNode("| Ding on new hit: "));
  1048. control_panel.appendChild(friesAreDone);
  1049. control_panel.appendChild(document.createElement("P"));
  1050. control_panel.appendChild(document.createTextNode("Minimum reward: "));
  1051. reward_input.size = 6;
  1052. control_panel.appendChild(reward_input);
  1053. control_panel.appendChild(document.createTextNode(" "));
  1054.  
  1055. control_panel.appendChild(document.createTextNode("| Qualified: "));
  1056. control_panel.appendChild(qual_input);
  1057. control_panel.appendChild(document.createTextNode(" "));
  1058. control_panel.appendChild(document.createTextNode("| Masters: "));
  1059. control_panel.appendChild(masters_input);
  1060. control_panel.appendChild(document.createTextNode(" "));
  1061. control_panel.appendChild(document.createTextNode("| Sort types: "));
  1062. control_panel.appendChild(document.createTextNode(" Latest: "));
  1063. control_panel.appendChild(sort_input1);
  1064. control_panel.appendChild(document.createTextNode("| Most Available: "));
  1065. control_panel.appendChild(sort_input2);
  1066. control_panel.appendChild(document.createTextNode("| Amount: "));
  1067. control_panel.appendChild(sort_input3);
  1068. control_panel.appendChild(document.createTextNode("| A-Z: "));
  1069. control_panel.appendChild(sort_input4);
  1070. control_panel.appendChild(document.createTextNode("| Invert: "));
  1071. control_panel.appendChild(sort_input_invert);
  1072. control_panel.appendChild(document.createElement("P"));
  1073. control_panel.appendChild(search_input);
  1074. search_input.size = 20;
  1075. search_input.title = "Enter a search term to include\n" + "Default is blank (no included terms)";
  1076. search_input.placeholder="Enter search terms here";
  1077. control_panel.appendChild(document.createTextNode(" "));
  1078. control_panel.appendChild(document.createTextNode("| Use includelist: "));
  1079. control_panel.appendChild(document.createTextNode(" "));
  1080. matchOnly.title = "Be sure to edit the include list or nothing will be displayed.";
  1081. control_panel.appendChild(matchOnly);
  1082. big_red_button.textContent = "Start";
  1083. big_red_button.onclick = function(){start_running();};
  1084. reset_blocks.textContent = "Edit blocklist";
  1085. reset_blocks.onclick = function(){
  1086. console.log("in");
  1087. var div = $("#block_div");
  1088. var textarea = $("#block_text");
  1089. textarea.val(ignore_list.join('^'));
  1090. $("#block_div").show();
  1091. };
  1092. include_button.textContent = "Edit includes";
  1093. include_button.onclick = function() {
  1094. var div = $("#include_div");
  1095. var textarea = $("#include_text");
  1096. textarea.val(include_list.join('^'));
  1097. $("#include_div").show();
  1098. };
  1099. control_panel.appendChild(document.createTextNode(" | "));
  1100. control_panel.appendChild(big_red_button);
  1101. control_panel.appendChild(document.createTextNode(" "));
  1102. control_panel.appendChild(reset_blocks);
  1103. control_panel.appendChild(document.createTextNode(" "));
  1104. control_panel.appendChild(include_button);
  1105. control_panel.appendChild(document.createElement("P"));
  1106. control_panel.appendChild(progress_report);
  1107. control_panel.appendChild(document.createElement("P"));
  1108. control_panel.appendChild(document.createTextNode("Status messages: "));
  1109. control_panel.appendChild(status_report);
  1110. control_panel.appendChild(document.createElement("P"));
  1111. text_area.style.fontWeight = 400;
  1112. text_area.createCaption().innerHTML = "HITs";
  1113. var col_heads = ['Requester','Title','Reward','HITs Available','TO pay','Accept HIT'];
  1114. var row = text_area.createTHead().insertRow(0);
  1115. text_area.caption.style.fontWeight = 800;
  1116. text_area.caption.style.color = BROWN;
  1117. if (default_text_size > 10)
  1118. text_area.cellPadding=Math.min(Math.max(1,Math.floor((default_text_size-10)/2)),5);
  1119. //console.log(text_area.cellPadding);
  1120. //text_area.cellPadding=2;
  1121. text_area.caption.style.fontSize = 28;
  1122. text_area.rows[0].style.fontWeight = 800;
  1123. text_area.rows[0].style.color = BROWN;
  1124. for (i=0; i<col_heads.length; i++)
  1125. {
  1126. var this_cell = row.insertCell(i);
  1127. this_cell.innerHTML = col_heads[i];
  1128. this_cell.style.fontSize = 14;
  1129. if (i > 1)
  1130. this_cell.style.textAlign = 'center';
  1131. }
  1132. control_panel.appendChild(text_area);
  1133. }
  1134.  
  1135. /********HIT EXPORT ADDITIONS*****/
  1136.  
  1137. var EDIT = false;
  1138. var HIT;
  1139.  
  1140. var TO_BASE = "http://turkopticon.ucsd.edu/";
  1141. var API_BASE = "https://mturk-api.istrack.in/";
  1142. var API_URL = API_BASE + "multi-attrs.php?ids=";
  1143. DEFAULT_TEMPLATE = '[table][tr][td][b]Title:[/b] [url={prev_link}][COLOR=blue]{title}[/COLOR][/url]\n';
  1144. DEFAULT_TEMPLATE += '[b]Requester:[/b] [url=https://www.mturk.com/mturk/searchbar?selectedSearchType=hitgroups&requesterId={rid}][COLOR=blue]{requester}[/COLOR][/url]';
  1145. DEFAULT_TEMPLATE += ' [{rid}] ([url='+TO_BASE+'{rid}][COLOR=blue]TO[/COLOR][/url])';
  1146. DEFAULT_TEMPLATE += '\n[b]TO Ratings:[/b]{to_stuff}';
  1147. DEFAULT_TEMPLATE += '\n[b]Description:[/b] {description}';
  1148. DEFAULT_TEMPLATE += '\n[b]Time:[/b] {time}';
  1149. DEFAULT_TEMPLATE += '\n[b]Hits Available:[/b] {hits}';
  1150. DEFAULT_TEMPLATE += '\n[b]Reward:[/b] [COLOR=green][b]{reward}[/b][/COLOR]';
  1151. DEFAULT_TEMPLATE += '\n[b]Qualifications:[/b] {quals}[/td][/tr][/table]';
  1152.  
  1153. var TEMPLATE;
  1154. var EASYLINK;
  1155.  
  1156. if (typeof GM_getValue === 'undefined')
  1157. TEMPLATE = null;
  1158. else {
  1159. TEMPLATE = GM_getValue('HITScraper Template');
  1160. EASYLINK = GM_getValue('HITScraper Easylink');
  1161. }
  1162. if (TEMPLATE == null) {
  1163. TEMPLATE = DEFAULT_TEMPLATE;
  1164. }
  1165.  
  1166. function buildXhrUrl(rai) {
  1167. var url = API_URL;
  1168. var ri = rai;
  1169. url += rai;
  1170. return url;
  1171. }
  1172.  
  1173. function makeXhrQuery(url) {
  1174. var xhr = new XMLHttpRequest();
  1175. try{
  1176. xhr.open('GET', url, false);
  1177. xhr.send(null);
  1178. return $.parseJSON(xhr.response);
  1179. }
  1180. catch(err){
  1181. return "TO DOWN";
  1182. }
  1183. }
  1184.  
  1185. function getNamesForEmptyResponses(rai, resp) {
  1186. for (var rid in rai) {
  1187. if (rai.hasOwnProperty(rid) && resp[rid] == "") {
  1188. resp[rid] = $.parseJSON('{"name": "' + rai[rid][0].innerHTML + '"}');
  1189. }
  1190. }
  1191. return resp;
  1192. }
  1193.  
  1194. function getKeys(obj) {
  1195. var keys = [];
  1196. for (var key in obj) {
  1197. keys.push(key);
  1198. }
  1199. return keys;
  1200. }
  1201.  
  1202. function export_func_deleg(item,index) {
  1203. //console.log(item);
  1204. export_func(item);
  1205. }
  1206.  
  1207. function block_deleg(item,index) {
  1208. //console.log(item);
  1209. block(item,index);
  1210. }
  1211.  
  1212. function block(hit,index){
  1213. var blockType = ["requester","title"];
  1214. var blockThis = hit[blockType[index]];
  1215. ignore_list.push(blockThis);
  1216. GM_setValue("scraper_ignore_list",ignore_list);
  1217. //console.log(GM_getValue("scraper_ignore_list"));
  1218. alert("\""+blockThis+"\" ignored. Re-scrape");
  1219. }
  1220.  
  1221. function search_deleg(item,index) {
  1222. console.log(item);
  1223. var searches = ["rid","title"];
  1224. search(item,searches[index]);
  1225. }
  1226.  
  1227. function hit_sort_func()
  1228. {
  1229. return function(a,b) {
  1230. if (a.date == b.date) {
  1231. if (a.requesterName < b.requesterName)
  1232. return -1;
  1233. if (a.requesterName > b.requesterName)
  1234. return 1;
  1235. if (a.title < b.title)
  1236. return -1;
  1237. if (a.title > b.title)
  1238. return 1;
  1239. if (a.status < b.status)
  1240. return -1;
  1241. if (a.status > b.status)
  1242. return 1;
  1243. }
  1244. if (a.date > b.date)
  1245. return 1;
  1246. if (a.date < b.date)
  1247. return -1;
  1248. };
  1249. }
  1250.  
  1251. function escapeRegExp(str) {
  1252. return str.replace(/[-\[\]\/\{\}\(\)\*\+\?\.\\\^\$\|]/g, "\\$&");
  1253. }
  1254.  
  1255. function search(item,search_type){
  1256. //return true;/*
  1257. var request = indexedDB.open("HITDB", v);
  1258. request.onsuccess = function(e) {
  1259. HITStorage.indexedDB.db = e.target.result;
  1260. var db = HITStorage.indexedDB.db;
  1261. var trans = db.transaction(["HIT"], HITStorage.IDBTransactionModes.READ_ONLY);
  1262. var store = trans.objectStore("HIT");
  1263. var req;
  1264. var results = [];
  1265. var index;
  1266. var range;
  1267. req = store.openCursor();
  1268. req.onsuccess = function(event) {
  1269. var cursor = event.target.result;
  1270. if (cursor) {
  1271. hit = cursor.value;
  1272. var keys = ['title', 'requesterId'];
  1273. var re = new RegExp(escapeRegExp(item[search_type]),"ig");
  1274. for (var k in keys)
  1275. {
  1276. if (hit[keys[k]] != null && re.test(hit[keys[k]].trim())){
  1277. results.push(cursor.value);
  1278. }
  1279. }
  1280.  
  1281. cursor.continue();
  1282. }
  1283. else {
  1284. console.log(results);
  1285. results.sort(hit_sort_func());
  1286. show_results(results);
  1287. }
  1288. db.close();
  1289. };
  1290. request.onerror = HITStorage.indexedDB.onerror;/**/
  1291. }
  1292. }
  1293.  
  1294. function format_hit_line (hit, odd, status_color, new_day)
  1295. {
  1296. var line = '<tr style="background-color:';
  1297. if (odd)
  1298. line += '#f1f3eb;';
  1299. else
  1300. line += 'white;';
  1301. line += ' valign=top;';
  1302. if (new_day)
  1303. line += ' border: 0px dotted #000000; border-width: 2px 0px 0px 0px">';
  1304. else
  1305. line += '">';
  1306.  
  1307. line += '<td>' + hit.date + '</td>';
  1308. if (hit.requesterLink != null)
  1309. line += '<td style="width:165px"><a href="' + hit.requesterLink + '" title="Contact this Requester">' + hit.requesterName + '</a></td>';
  1310. else
  1311. line += '<td style="width:165px">' + hit.requesterName + '</td>';
  1312. line += '<td style="width:213px">' + hit.title + '</td>';
  1313. line += '<td style="width:45px">$' + hit.reward.toFixed(2) + '</td>';
  1314. line += '<td style="color:' + status_color + '; width:55px">' + hit.status + '</td>';
  1315. line += '<td><div style="width:225px; overflow:hidden">' + hit.feedback + '</div></td>';
  1316. line += '</tr>\n';
  1317. return line;
  1318. }
  1319.  
  1320. function status_color (status)
  1321. {
  1322. var color = "green";
  1323.  
  1324. if (status.match("Pending Approval"))
  1325. color = "orange";
  1326. else if (status.match("Rejected"))
  1327. color = "red";
  1328.  
  1329. return color;
  1330. }
  1331.  
  1332. function show_results (results){
  1333. resultsWindow = window.open();
  1334. resultsWindow.document.write("<html><head><title>Status Detail Search Results</title></head><body>\n");
  1335. resultsWindow.document.write("<h1>HITs matching your search:</h1>\n");
  1336. resultsWindow.document.write('<table style="border: 1px solid black;border-collapse:collapse;width:90%;margin-left:auto;margin-right:auto;">\n');
  1337. 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');
  1338.  
  1339. var odd = true;
  1340. var sum = 0;
  1341. var sum_rejected = 0;
  1342. var sum_approved = 0;
  1343. var sum_pending = 0;
  1344. var new_day = false;
  1345.  
  1346. for (var i=0; i<results.length; i++) {
  1347. odd = !odd;
  1348. sum += results[i].reward;
  1349. if (results[i].status == 'Rejected')
  1350. sum_rejected += results[i].reward;
  1351. else if (results[i].status == 'Pending Approval')
  1352. sum_pending += results[i].reward;
  1353. else
  1354. sum_approved += results[i].reward;
  1355.  
  1356. if (i>0 && (results[i-1].date != results[i].date))
  1357. new_day = true;
  1358. else
  1359. new_day = false;
  1360. resultsWindow.document.write(format_hit_line(results[i], odd, status_color(results[i].status), new_day ));
  1361. }
  1362.  
  1363. 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');
  1364. resultsWindow.document.write("</table>");
  1365. resultsWindow.document.write("<p>Found " + results.length + " matching HITs. $" + sum_approved.toFixed(2) + " approved, " +
  1366. "$" + sum_rejected.toFixed(2) + " rejected and $" + sum_pending.toFixed(2) + " pending.</p>");
  1367. resultsWindow.document.write("</body></html>")
  1368. resultsWindow.document.close();
  1369. }
  1370.  
  1371. function export_func(item) {
  1372. HIT = item;
  1373. edit_button.textContent = 'Edit Template';
  1374. apply_template(item);
  1375. div.style.display = 'block';
  1376. textarea.select();
  1377. }
  1378.  
  1379. function apply_template(hit_data) {
  1380. var txt = TEMPLATE;
  1381.  
  1382. var vars = ['title', 'requester', 'rid', 'description', 'reward', 'quals', 'prev_link', 'time', 'hits', 'to_stuff', 'to_text'];
  1383.  
  1384. var resp = null;
  1385. if (txt.indexOf('{to_text}') >= 0 || txt.indexOf('{to_stuff}') >= 0){
  1386. var url = buildXhrUrl(hit_data["rid"]);
  1387. resp = makeXhrQuery(url);
  1388. //console.log(resp);
  1389. }
  1390. var toText = "";
  1391. var toStuff = "";
  1392. var toData = "";
  1393. var numResp = (resp == null || resp == "TO DOWN" ? "n/a" : resp[hit_data["rid"]].reviews);
  1394. if (resp == "TO DOWN"){
  1395. toStuff = " [URL=\""+TO_BASE+hit_data['rid']+"\"]TO down.[/URL]";
  1396. toText = toStuff;
  1397. }
  1398. else if (resp == null || resp[hit_data["rid"]].attrs == null && resp != "TO DOWN") {
  1399. toStuff = " No TO ";
  1400. toText = " No TO ";
  1401. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  1402. toStuff += "(Submit a new TO rating for this requester)[/URL]";
  1403. }
  1404. else {
  1405. for (var key in resp[hit_data["rid"]].attrs) {
  1406. //toText += "\n[*]"+key+": "+resp[hit_data["requesterId"]].attrs[key]+"\n";
  1407. var i = 0;
  1408. var color = "green";
  1409. var name = key;
  1410. var num = Math.floor(resp[hit_data["rid"]].attrs[key]);
  1411. switch (key){
  1412. case "comm":
  1413. name = "Communicativity";
  1414. break;
  1415. case "pay":
  1416. name = "Generosity";
  1417. break;
  1418. case "fast":
  1419. name = "Promptness";
  1420. break;
  1421. case "fair":
  1422. name = "Fairness";
  1423. break;
  1424. default:
  1425. name = key;
  1426. break;
  1427. }
  1428. switch (num){
  1429. case 0:
  1430. color = "red";
  1431. break;
  1432. case 1:
  1433. color = "red";
  1434. break;
  1435. case 2:
  1436. color = "orange";
  1437. break;
  1438. case 3:
  1439. color = "yellow";
  1440. break;
  1441. default:
  1442. break;
  1443. }
  1444. toText += (num > 0 ? "\n[color="+color+"]" : "\n");
  1445. for (i; i < num; i++){
  1446. toText += "[b]"+symbol+"[/b]"
  1447. }
  1448. toText += (num > 0 ? "[/color]" : "")
  1449. if (i < 5){
  1450. toText += "[color=white]";
  1451. for (i; i < 5; i++)
  1452. toText += "[b]"+symbol+"[/b]";
  1453. toText += "[/color]";
  1454. }
  1455. toText += " "+Number(resp[hit_data["rid"]].attrs[key]).toFixed(2)+" "+name;
  1456. toData += Number(resp[hit_data["rid"]].attrs[key]).toFixed(2) + ",";
  1457. }
  1458. //toText += "[/list]";
  1459. 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]");
  1460. toStuff = '\n[img]http://data.istrack.in/to/' + toData.slice(0,-1) + '.png[/img]';
  1461. toStuff += (txt.indexOf('{to_stuff}') >= 0 ? (txt.indexOf('{to_text}') >= 0 ? "" : toText) : "");
  1462. toStuff += "\nNumber of Reviews: "+numResp;
  1463. toStuff += "[URL=\""+TO_BASE+"report?requester[amzn_id]=" + hit_data['rid'] + "&requester[amzn_name]=" + hit_data['requester'] + "\"]";
  1464. toStuff += "\n(Submit a new TO rating for this requester)[/URL]";
  1465. }
  1466. for (var i = 0; i < vars.length; i++) {
  1467. t = new RegExp('\{' + vars[i] + '\}', 'g');
  1468. if (vars[i] == "to_stuff") {
  1469. txt = txt.replace(t, toStuff);
  1470. }
  1471. else if (vars[i] == "to_text"){
  1472. txt = txt.replace(t, toText);
  1473. }
  1474. else if (vars[i] == "prev_link"){
  1475. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[i]]);
  1476. }
  1477. else if (vars[i] == "acc_link"){
  1478. txt = txt.replace(t,"https://www.mturk.com"+hit_data[vars[i]]);
  1479. }
  1480. else
  1481. txt = txt.replace(t, hit_data[vars[i]]);
  1482. }
  1483. textarea.value = txt;
  1484. }
  1485.  
  1486. function apply_template2(hit_data, link) {
  1487. var txt = TEMPLATE;
  1488. var masters = ""
  1489. var vars = ['title', 'requester', 'rid', 'description', 'reward', 'quals', 'prev_link', 'time', 'hits', 'to_stuff', 'to_text'];
  1490. var hit_link = hit_data['prev_link']
  1491. if (link == "A")
  1492. hit_link = hit_data['prev_link'].replace("preview?", "previewandaccept?")
  1493. var resp = null;
  1494. if (txt.indexOf('{to_text}') >= 0 || txt.indexOf('{to_stuff}') >= 0){
  1495. var url = buildXhrUrl(hit_data["rid"]);
  1496. resp = makeXhrQuery(url);
  1497. //console.log(resp);
  1498. }
  1499. var toData = "";
  1500. var numResp = (resp == null || resp == "TO DOWN" ? "n/a" : resp[hit_data["rid"]].reviews);
  1501. if (resp == "TO DOWN"){
  1502. toData = "TO Down";
  1503. }
  1504. else if (resp == null || resp[hit_data["rid"]].attrs == null && resp != "TO DOWN") {
  1505. toData = "No TO";
  1506. }
  1507. else {
  1508. toData = resp[hit_data["rid"]].attrs["pay"];
  1509. }
  1510. if (hit_data['quals'].indexOf("Masters") > -1)
  1511. masters = "Ma*****"
  1512. txt = masters + " " + hit_data['reward'] + " - " + hit_data['title'] + " - " + "https://www.mturk.com"+hit_link + " - " + hit_data['time'] + " - " + hit_data['requester'] + " - " + "TO: " + toData + " - " + "HITs: " + hit_data['hits'];
  1513. GM_setClipboard(txt);
  1514. }
  1515.  
  1516. function hide_func(div) {
  1517. if (EDIT == false)
  1518. div.style.display = 'none';
  1519. }
  1520.  
  1521. function edit_func() {
  1522. if (EDIT == true) {
  1523. EDIT = false;
  1524. TEMPLATE = textarea.value;
  1525. edit_button.textContent = 'Edit Template';
  1526. apply_template(HIT);
  1527. }
  1528. else {
  1529. console.log("Editing");
  1530. EDIT = true;
  1531. edit_button.textContent = 'Show Changes';
  1532. save_button.disabled = false;
  1533. textarea.value = TEMPLATE;
  1534. }
  1535. }
  1536.  
  1537. function default_func() {
  1538. GM_deleteValue('HITScraper Template');
  1539. TEMPLATE = DEFAULT_TEMPLATE;
  1540. EDIT = false;
  1541. edit_button.textContent = 'Edit Template';
  1542. apply_template(HIT);
  1543. }
  1544.  
  1545. function save_func() {
  1546. if (EDIT)
  1547. TEMPLATE = textarea.value;
  1548. GM_setValue('HITScraper Template', TEMPLATE);
  1549. }
  1550.  
  1551. var div = document.createElement('div');
  1552. var textarea = document.createElement('textarea');
  1553. var div2 = document.createElement('label');
  1554.  
  1555. div.style.position = 'fixed';
  1556. div.style.width = '500px';
  1557. div.style.height = '235px';
  1558. div.style.left = '50%';
  1559. div.style.right = '50%';
  1560. div.style.margin = '-250px 0px 0px -250px';
  1561. div.style.top = '300px';
  1562. div.style.padding = '5px';
  1563. div.style.border = '2px';
  1564. div.style.backgroundColor = 'black';
  1565. div.style.color = 'white';
  1566. div.style.zIndex = '100';
  1567.  
  1568. textarea.style.padding = '2px';
  1569. textarea.style.width = '500px';
  1570. textarea.style.height = '200px';
  1571. 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}';
  1572.  
  1573. div.textContent = 'Press Ctrl+C to copy to clipboard. Click textarea to close';
  1574. div.style.fontSize = '12px';
  1575. div.appendChild(textarea);
  1576.  
  1577. var edit_button = document.createElement('button');
  1578. var save_button = document.createElement('button');
  1579. var default_button = document.createElement('button');
  1580. var easy_button = document.createElement('button');
  1581.  
  1582. edit_button.textContent = 'Edit Template';
  1583. edit_button.setAttribute('id', 'edit_button');
  1584. edit_button.style.height = '18px';
  1585. edit_button.style.width = '100px';
  1586. edit_button.style.fontSize = '10px';
  1587. edit_button.style.paddingLeft = '3px';
  1588. edit_button.style.paddingRight = '3px';
  1589. edit_button.style.backgroundColor = 'white';
  1590.  
  1591. save_button.textContent = 'Save Template';
  1592. save_button.setAttribute('id', 'save_button');
  1593. save_button.style.height = '18px';
  1594. save_button.style.width = '100px';
  1595. save_button.style.fontSize = '10px';
  1596. save_button.style.paddingLeft = '3px';
  1597. save_button.style.paddingRight = '3px';
  1598. save_button.style.backgroundColor = 'white';
  1599. save_button.style.marginLeft = '5px';
  1600.  
  1601. easy_button.textContent = 'Change Adfly Url';
  1602. easy_button.setAttribute('id', 'easy_button');
  1603. easy_button.style.height = '18px';
  1604. easy_button.style.width = '100px';
  1605. easy_button.style.fontSize = '10px';
  1606. easy_button.style.paddingLeft = '3px';
  1607. default_button.textContent = ' D ';
  1608. default_button.setAttribute('id', 'default_button');
  1609. default_button.style.height = '18px';
  1610. default_button.style.width = '20px';
  1611. default_button.style.fontSize = '10px';
  1612. default_button.style.paddingLeft = '3px';
  1613. default_button.style.paddingRight = '3px';
  1614. default_button.style.backgroundColor = 'white';
  1615. default_button.style.marginLeft = '5px';
  1616. default_button.title = 'Return default template';
  1617. div.appendChild(edit_button);
  1618. div.appendChild(save_button);
  1619. div.appendChild(default_button);
  1620. div.appendChild(easy_button);
  1621. save_button.disabled = true;
  1622.  
  1623. div.style.display = 'none';
  1624. textarea.addEventListener("click", function() {hide_func(div);}, false);
  1625. edit_button.addEventListener("click", function() {edit_func();}, false);
  1626. save_button.addEventListener("click", function() {save_func();}, false);
  1627. default_button.addEventListener("click", function() {default_func();}, false);
  1628. document.body.insertBefore(div, document.body.firstChild);