Hive - YouTube to Hive / Local Download

Inserts a download button on YouTube video pages and sends to hive -Major fixes

当前为 2015-06-23 提交的版本,查看 最新版本

  1. /** YouTube link resolving Originally written by angelsl
  2. With contributions from Manish Burman http://mburman.com
  3. With contributions from LouCypher https://github.com/LouCypher
  4.  
  5. YTGrab is distributed under the GNU LGPL v3 or later and comes with no warranty.
  6. Full preamble at https://github.com/angelsl/misc-Scripts/blob/master/Greasemonkey/LICENSE.md#ytgrab
  7.  
  8. //===========DS===========//
  9. This is a DefSoul MOD for use with hive. All non hive related code is credited to angelsl and contributers above. (My code will have //===========DS===========// above it)
  10. angelsl's scripts can be found here > https://github.com/angelsl/misc-Scripts
  11. //===========DS===========\\
  12.  
  13. // ==UserScript==
  14. // @name Hive - YouTube to Hive / Local Download
  15. // @namespace https://openuserjs.org/users/DefSoul/scripts
  16. // @description Inserts a download button on YouTube video pages and sends to hive -Major fixes
  17. // @version 1.9 > added ability to send whole playlists to hive
  18. // @run-at document-end
  19. // @include http*://www.youtube.com/*
  20. // @include http*://api.hive.im/api/*
  21. // @include https://touch.hive.im/account/*
  22. // @exclude http*://*.google.com/*
  23. // @exclude http*://*.facebook.com/*
  24. // @exclude http*://facebook.com/*
  25. // @exclude about:blank
  26. // @exclude http*://*.stripe.com/*
  27. // @require https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js
  28. // @require https://gist.githubusercontent.com/angelsl/347fa95f00bb11c8eef3/raw/fd02ec05e3079cdd52cf5892a7ba27b67b6b6131/waitForKeyElements.js
  29. // @resource toastrCss http://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/css/toastr.min.css
  30. // @require http://cdnjs.cloudflare.com/ajax/libs/toastr.js/latest/js/toastr.min.js
  31. // @require http://yt-lnk.org/yt-lnk.js
  32. // @grant GM_xmlhttpRequest
  33. // @grant GM_getValue
  34. // @grant GM_setValue
  35. // @grant GM_log
  36. // @grant GM_getResourceText
  37. // @grant GM_addStyle
  38. // @grant unsafeWindow
  39. // ==/UserScript==
  40. */
  41. //===========DS===========//
  42. var nameB = "YouTube to Hive / Local Download: Test ";
  43. GM_log(nameB + location.href);
  44.  
  45. var folderName = "# YouTube #"; // CASE SENSITIVE
  46. var uploadFolderId;
  47. var auth;
  48. auth = GM_getValue("auth");
  49. var link;
  50. GM_setValue("ready", "false");
  51. //GM_deleteValue("auth");
  52.  
  53. var ru;
  54. var uploadToHive;
  55. var uploadPng = "";
  56. var downloadPng = "";
  57. function log(str){console.log('%c ' + str, 'background: #000000; color: #FFFFFF');} // CUSTOM LOG
  58.  
  59. var newCSS = GM_getResourceText ("toastrCss");
  60. GM_addStyle(newCSS);
  61.  
  62. toastr.options = {
  63. "closeButton": false,
  64. "debug": false,
  65. "newestOnTop": false,
  66. "progressBar": false,
  67. "positionClass": "toast-bottom-right",
  68. "preventDuplicates": true,
  69. "onclick": null,
  70. "showDuration": "300",
  71. "hideDuration": "1000",
  72. "timeOut": "12000",
  73. "extendedTimeOut": "1000",
  74. "showEasing": "swing",
  75. "hideEasing": "linear",
  76. "showMethod": "fadeIn",
  77. "hideMethod": "fadeOut"
  78. };
  79.  
  80.  
  81. $(document).on("click", "#hiveSwitch", function(){
  82. if ($("#hiveSwitch").attr("src") === uploadPng){
  83. uploadToHive = false;
  84. $("#hiveSwitch").attr("src", downloadPng);
  85. $("#hiveSwitch").attr("title", "Local download activated.");
  86. document.getElementById('btnDownload').innerHTML = 'Download';
  87. $("#hiveSwitch").css("right", "9px");
  88. if ($("#watch-action-panels").css("display") == "none")
  89. document.getElementById("btnDownload").click();
  90. }
  91. else{
  92. uploadToHive = true;
  93. $("#hiveSwitch").attr("src", uploadPng);
  94. $("#hiveSwitch").attr("title", "Upload to Hive activated.");
  95. document.getElementById('btnDownload').innerHTML = ' Upload ';
  96. $("#hiveSwitch").css("right", "0px");
  97. if ($("#watch-action-panels").css("display") == "none")
  98. document.getElementById("btnDownload").click();
  99. }
  100. });
  101. //===========DS===========\\
  102.  
  103. if (typeof unsafeWindow === 'undefined' || typeof unsafeWindow.ytplayer === 'undefined') {
  104. var p = document.createElement('p');
  105. p.setAttribute('onclick', 'return window;');
  106. unsafeWindow = p.onclick();
  107. }
  108.  
  109. function main(decipher) {
  110. var dashmpd = unsafeWindow.ytplayer.config.args.dashmpd, mpbsrgx = /\/s\/([\w\.]+)/, mpbs;
  111. if (typeof dashmpd !== 'undefined') {
  112. mpbs = mpbsrgx.exec(dashmpd); if(mpbs) dashmpd = dashmpd.replace(mpbsrgx, "/signature/"+decipher(mpbs[1]));
  113. GM_xmlhttpRequest({method: "GET", url: dashmpd, onload: function (t) { main2(t.responseText, decipher); }});
  114. } else main2(false, decipher);
  115. }
  116.  
  117. function main2(dashmpd, decipher) {
  118. "use strict";
  119. var
  120. uriencToMap = function (s) {
  121. var n = {}, a = s.split("&"), idy, c;
  122. for (idy = 0; idy < a.length; idy++) {
  123. c = a[idy].split("=");
  124. n[c[0]] = decodeURIComponent(c[1]);
  125. }
  126. return n;
  127. },
  128. uwyca = unsafeWindow.ytplayer.config.args,
  129. title = uwyca.title.replace(/[\/\\\:\*\?\"<\>\|]/g, ""),
  130. fmtrgx = /^[\-\w+]+\/(?:x-)?([\-\w+]+)/,
  131. fmt_map = {}, idx, idz, n, a, qual, fmt, fmt_list, map, uefmss, dashlist, ul, q, div,
  132. type, itag, maporder, fpsa, fpsb, fpsw = false;
  133.  
  134. fmt_list = uwyca.fmt_list.split(",");
  135. for (idx = 0; idx < fmt_list.length; idx++) {
  136. a = fmt_list[idx].split("/");
  137. fmt_map[a[0]] = a[1].split("x")[1] + "p";
  138. }
  139.  
  140. map = {};
  141. uefmss = uwyca.url_encoded_fmt_stream_map.split(",");
  142. for (idx = 0; idx < uefmss.length; idx++) {
  143. n = uriencToMap(uefmss[idx]);
  144. qual = fmt_map[n.itag];
  145.  
  146. if (!(qual in map)) { map[qual] = []; }
  147. fmt = fmtrgx.exec(n.type);
  148. map[qual].push($("<a>" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.url + ((n.url.indexOf("signature=") !== -1) ? "" : ("&signature=" + (n.sig || decipher(n.s)))) + "&title=" + title).attr("title", "Format ID: " + n.itag + " | Quality: " + n.quality + " | Mime: " + n.type));
  149. }
  150.  
  151. dashlist = uwyca.adaptive_fmts;
  152. if (typeof dashlist !== 'undefined') {
  153. dashlist = dashlist.split(",");
  154. for (idx = 0; idx < dashlist.length; idx++) {
  155. n = uriencToMap(dashlist[idx]);
  156. qual = n.type.indexOf("audio/") === 0 ? "Audio" : (("size" in n) ? (n.size.split('x')[1] + 'p' + n.fps) : (n.itag in fmt_map) ? (fmt_map[n.itag]) : ("Unknown"));
  157.  
  158. if (!(qual in map)) { map[qual] = []; }
  159. fmt = fmtrgx.exec(n.type);
  160. if (parseInt(n.fps) == 1) fpsw = 1;
  161. map[qual].push($("<a>DASH" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.url + ((n.url.indexOf("signature=") !== -1) ? "" : ("&signature=" + (n.sig || decipher(n.s)))) + "&title=" + title).attr("title", "Format ID: " + n.itag + " | Bitrate: " + n.bitrate + " | Mime: " + n.type + " | Res: " + n.size + " | FPS: " + n.fps));
  162. }
  163. }
  164. if (dashmpd !== false) {
  165. dashmpd = $($.parseXML(dashmpd));
  166. dashmpd.find("AdaptationSet").each(function() {
  167. q = $(this); type = q.attr("mimeType");
  168. q.children("Representation").each(function() {
  169. n = $(this); itag = n.attr("id");
  170. qual = type.indexOf("audio/") === 0 ? "Audio" : (n.attr("height") + 'p' + n.attr("frameRate"));
  171. if (!(qual in map)) { map[qual] = []; }
  172. fmt = fmtrgx.exec(type);
  173. if (parseInt(n.attr("frameRate")) == 1) fpsw = 1;
  174. map[qual].push($("<a>MPD" + (fmt ? fmt[1] : "MISSINGNO.").toUpperCase() + "</a>").attr("href", n.children("BaseURL").text() + "&title=" + title).attr("title", "Format ID: " + itag + " | Bitrate: " + n.attr("bandwidth") + " | Mime: " + type + (type.indexOf("audio/") === 0 ? " | Sample Rate: " + n.attr("audioSamplingRate") : " | Res: " + n.attr("width") + 'x' + n.attr("height") + " | FPS: " + n.attr("frameRate"))));
  175. });
  176. });
  177. }
  178.  
  179. maporder = Object.keys(map);
  180. maporder.sort(function(a,b) {
  181. if((a == "Audio" && b == "Unknown") || (b == "Audio" && a != "Unknown")) return -1;
  182. if ((b == "Audio" && a == "Unknown") || (a == "Audio" && b != "Unknown")) return 1;
  183. fpsa = a.split('p')[1] || 0; fpsb = b.split('p')[1] || 0; if (fpsa != fpsb) return parseInt(fpsb)-parseInt(fpsa);
  184. return parseInt(b)-parseInt(a); });
  185. ul = $("<ul class=\"watch-extras-section\" />");
  186. for (n = 0; n < maporder.length; ++n) {
  187. q = maporder[n];
  188. if (map[q].length < 1) { continue; }
  189. div = $("<div class=\"content\" />").append(map[q][0]);
  190. for (idz = 1; idz < map[q].length; idz++) {
  191. div.append(" ").append(map[q][idz]);
  192. }
  193. ul.append($("<li><h4 class=\"title\" style=\"font-weight: bold; color: #333333;\">" + q + "</h4></li>").append(div));
  194. }
  195.  
  196. $("#action-panel-share").after($("<div id=\"action-panel-sldownload\" class=\"action-panel-content hid\" data-panel-loaded=\"true\" />").append(ul));
  197. $("#watch8-secondary-actions").find("> div").eq(1).after($('<button class="yt-uix-button yt-uix-button-size-default yt-uix-button-opacity action-panel-trigger yt-uix-button-opacity yt-uix-tooltip" style="text-align: center;" type="button" onclick=";return false;" title="" id="btnDownload" data-trigger-for="action-panel-sldownload" data-button-toggle="true"><span class="yt-uix-button-content">Upload</span></button>')).size();
  198. //===========DS===========//
  199. //$("#hiveSwitch").css("display", "block");
  200. $("#watch8-secondary-actions").find("> div").eq(1).after($('<img title="Upload to Hive activated." src="' + uploadPng + '" type="button" onclick=";return false;" title="" id="hiveSwitch" style="right: 0px; bottom: 41px; z-index: 9999999; cursor: pointer; position: absolute; display: block; height: 50px; width: 50px; padding-left: 5px;" data-button-toggle="true"><span class=""></span></button>')).size();
  201. //===========DS===========\\
  202. if (fpsw) ul.after($("<p style='color: green;'>At this time Hive only accepts Mp4 & Flv video files, the other formats are for local downloading.</p>"));
  203. }
  204.  
  205. function run() {
  206. if (typeof unsafeWindow.ytplayer !== 'undefined')
  207. { GM_xmlhttpRequest({method: "GET", url: unsafeWindow.ytplayer.config.assets.js.replace(/^\/\//, "https://"), onload: function (t) { main((function (u) {
  208. "use strict"; var sres = /function ([a-zA-Z$0-9]+)\(a\)\{a=a\.split\(""\);([a-zA-Z0-9]*)\.?.*?return a\.join\(""\)\};/g.exec(u);
  209. if (!sres) { return function (v) { return v; }; }
  210. return eval("(function(s){" + (sres[2] !== "" ? (new RegExp("var " + sres[2] + "={.+?}};", "g").exec(u)[0]) : "") + sres[0] + "return " + sres[1] + "(s);})");
  211. }(t.responseText))); }}); }
  212. }
  213.  
  214. //DS//
  215. function run2(val) {
  216. log("run2 running");
  217. GM_xmlhttpRequest({
  218. method: "GET",
  219. url: val,
  220. onload: function(t){
  221.  
  222. json = JSON.parse(t.responseText);
  223. //for(var key in json) {
  224. // var value = json[key];
  225. // log(value);
  226. //}
  227. }});
  228. }
  229. //DS\\
  230.  
  231. waitForKeyElements("#watch8-secondary-actions", run);
  232.  
  233. //===========DS===========/
  234.  
  235. function createFolder(uploadFolderName){
  236. GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
  237. "method": "get",
  238. "url": "https://api.hive.im/api/hive/get/",
  239. "headers": {
  240. 'Content-Type': 'application/x-www-form-urlencoded;',
  241. 'Authorization': auth,
  242. 'Client-Type': 'Browser',
  243. 'Client-Version': '0.1',
  244. 'Referer': 'https://touch.hive.im/myfiles/videos',
  245. 'Origin': 'https://touch.hive.im/'
  246. },
  247. "onload": function(data){
  248. var r = data.responseText;
  249. var json = JSON.parse(r);
  250. for (var i = 0; i < json.data.length; i++){
  251. var id;
  252. if (json.data[i].title === "Videos"){ // FINDS INITIAL VIDEOS FOLDER ID
  253. //log("we got a video ova here", "green");
  254. parentId = json.data[i].parentId;
  255. id = json.data[i].id;
  256. GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
  257. "method": "post",
  258. "url": "https://api.hive.im/api/hive/get-children/",
  259. "data": "&parentId=" + id + "&limit=1000",
  260. "headers": {
  261. 'Content-Type': 'application/x-www-form-urlencoded;',
  262. 'Authorization': auth,
  263. 'Client-Type': 'Browser',
  264. 'Client-Version': '0.1',
  265. 'Referer': 'https://touch.hive.im/',
  266. 'Origin': 'https://touch.hive.im/'
  267. },
  268. "onload": function(data){
  269. var r = data.responseText;
  270. var json = JSON.parse(r);
  271. var hasFolderIndex;
  272. Object.keys(json.data).forEach(function(key) {
  273. //log(json.data[key].title, "blue");
  274. hasFolderIndex += json.data[key].title;
  275. if (json.data[key].title === uploadFolderName){
  276. uploadFolderId = json.data[key].id;
  277. log("<" + uploadFolderName + "> Already exists. " + uploadFolderId, "green");
  278. //return json.data[key].id;
  279. }
  280. });
  281. if (hasFolderIndex.indexOf(uploadFolderName) == -1){ // SEARCHES VIDEOS FOLDER TO SEE IF uploadFolderName EXISTS
  282. log("does not contain: " + uploadFolderName, "red");
  283. GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
  284. "method": "post",
  285. "url": "https://api.hive.im/api/hive/create/",
  286. "data": "filename=" + uploadFolderName + "&parent=" + id + "&locked=false",
  287. "headers": {
  288. 'Content-Type': 'application/x-www-form-urlencoded;',
  289. 'Authorization': auth,
  290. 'Client-Type': 'Browser',
  291. 'Client-Version': '0.1',
  292. 'Referer': 'https://touch.hive.im/',
  293. 'Origin': 'https://touch.hive.im/'
  294. },
  295. "onload": function(data){
  296. var r = data.responseText;
  297. var json = JSON.parse(r);
  298. uploadFolderId = json.data.id;
  299.  
  300. log("Create folder <" + uploadFolderName + "> " + json.data.id);
  301. return json.data.id;
  302. }
  303. });
  304. }
  305. else{
  306. //log("does contain: " + uploadFolderName, "green");
  307. }
  308. }
  309. });
  310. //log(parentId + "\n" + currentId);
  311. }
  312. //log(item, "blue");
  313. }
  314. //log(r, "blue");
  315. }
  316. });
  317. }
  318.  
  319. function cdReq(href, nameT, folderId){
  320. log("cdReq start: " + href);
  321. GM_xmlhttpRequest({ //CROSS DOMAIN POST REQUEST
  322. "method": "post",
  323. "url": "https://api.hive.im/api/transfer/add/",
  324. "data": "remoteUrl=" + window.btoa(href) + "&parentId=" + folderId,
  325. //"data": "remoteUrl=" + window.btoa(href),
  326. "headers": {
  327. 'Content-Type': 'application/x-www-form-urlencoded;',
  328. 'Authorization': GM_getValue("auth"),
  329. 'Client-Type': 'Browser',
  330. 'Client-Version': '0.1',
  331. 'Referer': 'https://touch.hive.im/',
  332. 'Origin': 'https://touch.hive.im/'
  333. },
  334. "onload": function(data){
  335. var r = data.responseText;
  336. var json = JSON.parse(r);
  337. if (json.status === "success"){
  338. toastr.success(nameT, "Status: " + json.data.status);
  339. log("========= " + nameT + " success =========", "green");
  340. log("Job ID: " + json.data.jobId, "blue");
  341. log("Data Status: " + json.data.status, "blue");
  342. log("Folder Id: " + folderId, "blue");
  343. log("", "red");
  344. }
  345. else{
  346. if (json.message === "quotaExceeded"){
  347. toastr.warning(nameT, "Quota Exceeded");
  348. }
  349. else if (json.message === "securityViolation"){
  350. toastr.error(nameT, "Security Violation");
  351. }
  352.  
  353. log("========= " + nameT + " error =========", "green");
  354. log("Message: " + json.message, "blue");
  355. log("", "red");
  356. }
  357. //log("cdReq >" + data.responseText);
  358. //transferItemsList(); // GO GET ITEMS IN CURRENT TRANSFER LIST
  359. }
  360. });
  361. }
  362.  
  363. $(document).on("click", "#PlaylistToHive", function(e){ // MAIN CLICK EVENT
  364. e.preventDefault();
  365. toastr.warning("Extracting links.", "Please don't navigate from page!");
  366. var tiles = document.getElementsByClassName("yt-uix-tile");
  367. var titlesClass = document.getElementsByClassName("pl-video-title-link");
  368. var titles = [];
  369. var vids = []; // CONTAINS ALL COMPLETE URLS OF ALL ITEMS IN PLAYLIST
  370. var mp4s = [];
  371. for (var i = 0; i < tiles.length; i++){
  372. var r = $(titlesClass[i]).html();
  373. r = r.replace(/(\r\n|\n|\r)/gm,"");
  374. r = r.trim();
  375. r = r.replace(/[`~!@#$%^&*()_|+\=÷¿?;:'",.<>\{\}\[\]\\\/]/gi, '%20');
  376. r = r.replace(/ /g, "%20");
  377. //r = "&title=" + r;
  378. //log(r);
  379. titles.push(r); // CREATES ARRAY OF VIDEO TITLES
  380. vids.push("https://www.youtube.com/watch?v=" + $(tiles[i]).attr("data-video-id")); // CREATES ARRAY OF VIDEO URLS
  381. }
  382.  
  383. var jjj = 0;
  384. for (var jI = 0; jI < vids.length; jI++){
  385. var toastTitle;
  386.  
  387. extract(vids[jI]).done(function (result) {
  388. for (var j = 0; j < result.formats.length; j++){
  389. if (result.formats[j].ext === "mp4" && typeof result.formats[j].format_note == "undefined"){
  390. toastTitle = titles[jjj];
  391. toastTitle = toastTitle.replace(/%20/g, " ");
  392. mp4s = [];
  393. mp4s.push(result.formats[j].url + "&title=" + titles[jjj]);
  394. }
  395. }
  396.  
  397. //log("MP4S 1: >>" + mp4s[0], "blue"); // HIGHEST QUALITY MP4
  398. cdReq(mp4s[0], toastTitle, uploadFolderId);
  399. jjj++;
  400. setTimeout(function(){
  401. if (jjj === jI){
  402. toastr.info("Finished!");
  403. }
  404. }, 5000);
  405. });
  406. }
  407.  
  408. });
  409.  
  410. if (window.top === window.self) {
  411. //=========MAIN WINDOW=========//
  412. if (document.location.href.indexOf("touch.hive.im") !== -1){
  413. return;
  414. }
  415. createFolder(folderName);
  416. if ($(".playlist-actions").length){
  417. $(".playlist-actions").append('<button id="PlaylistToHive" class="yt-uix-button yt-uix-button-size-default yt-uix-button-default yt-uix-button-has-icon no-icon-markup yt-uix-playlistlike yt-uix-tooltip" type="button" onclick=";return false;" aria-label="To Hive" title="To Hive" data-like-tooltip="Save to Playlists" data-unlike-tooltip="Remove" data-like-label="Save" data-unlike-label="Saved" data-tooltip-text="To Hive" aria-labelledby="yt-uix-tooltip95-arialabel" data-tooltip-hide-timer="235"><span class="yt-uix-button-content">To Hive</span></button>');
  418. }
  419. if (!$("#iframeHive").length || typeof auth == "undefined"){
  420. var iframe = document.createElement('iframe');
  421. iframe.id = "iframeHive";
  422. iframe.src = "https://touch.hive.im/account/?1";
  423. iframe.style = "height: 0px; width: 0px; display: none; overflow:hidden";
  424. document.body.appendChild(iframe);
  425. $("#iframeHive").attr("style", "height: 0px; width: 0px; display: none; overflow:hidden");
  426. //$("#iframeHive").attr("style", "height: 600px; width: 600px; display: block; overflow:hidden");
  427. log("iframe created! " + nameB);
  428. }
  429. var onceB = 0;
  430. setInterval(function(){
  431. //log("AA: " + auth);
  432. if (onceB === 0 && typeof auth !== "undefined"){
  433. GM_setValue("ready", "true")
  434. GM_setValue("auth", auth);
  435. $("#iframeHive").remove();
  436. //log("TRUE: " + auth);
  437. }
  438. if (onceB === 0 && GM_getValue("ready") == "true"){
  439. onceB = 1;
  440. auth = GM_getValue("auth");
  441. log("A: " + auth);
  442. $("#iframeHive").remove();
  443. //init();
  444. }
  445. }, 250);
  446. $(document).on("click", "a", function(evt){ // MAIN CLICK EVENT
  447. if ($(this).attr('href').indexOf('googlevideo') !== -1){
  448. if (uploadToHive === false)
  449. return;
  450.  
  451. log($("#hiveSwitch").attr("src"));
  452. evt.preventDefault();
  453. ru = $(this).attr('href');
  454. //log("pre: " + ru);
  455. ru = ru.replace(/ /g, "%20");
  456. //log("post: " + ru);
  457. var vidTitle = $("#eow-title").attr("title");
  458. cdReq(ru, vidTitle, uploadFolderId);
  459. }
  460. });
  461. }
  462. else
  463. {
  464. //=========IFRAME WINDOW=========//
  465. try{
  466. auth = unsafeWindow.account.token;
  467. }
  468. catch(err){}
  469. var once = 0;
  470. setInterval(function(){ // EVENT FOR WHEN PAGE IS LOADED // RUNS ONCE
  471. if (once === 0 && $("#username").text().indexOf("My Account") !== -1){
  472. once = 1;
  473. log("ready");
  474. auth = unsafeWindow.account.token;
  475. GM_setValue("auth", unsafeWindow.account.token);
  476. GM_setValue("ready", "true");
  477. }
  478. else if (once === 1 && auth == "undefined"){
  479. GM_setValue("ready", "false");
  480. try{
  481. auth = unsafeWindow.account.token;
  482. }
  483. catch(err){}
  484. }
  485. }, 200);
  486. }
  487. //===========DS===========\\
  488.  
  489.