1 var resizePackagesNav; 2 var classesNav; 3 var devdocNav; 4 var sidenav; 5 var content; 6 var HEADER_HEIGHT = 117; 7 var cookie_namespace = 'android_developer'; 8 var NAV_PREF_TREE = "tree"; 9 var NAV_PREF_PANELS = "panels"; 10 var nav_pref; 11 var toRoot; 12 var isMobile = false; // true if mobile, so we can adjust some layout 13 var isIE6 = false; // true if IE6 14 15 // TODO: use $(document).ready instead 16 function addLoadEvent(newfun) { 17 var current = window.onload; 18 if (typeof window.onload != 'function') { 19 window.onload = newfun; 20 } else { 21 window.onload = function() { 22 current(); 23 newfun(); 24 } 25 } 26 } 27 28 var agent = navigator['userAgent'].toLowerCase(); 29 // If a mobile phone, set flag and do mobile setup 30 if ((agent.indexOf("mobile") != -1) || // android, iphone, ipod 31 (agent.indexOf("blackberry") != -1) || 32 (agent.indexOf("webos") != -1) || 33 (agent.indexOf("mini") != -1)) { // opera mini browsers 34 isMobile = true; 35 addLoadEvent(mobileSetup); 36 // If not a mobile browser, set the onresize event for IE6, and others 37 } else if (agent.indexOf("msie 6") != -1) { 38 isIE6 = true; 39 addLoadEvent(function() { 40 window.onresize = resizeAll; 41 }); 42 } else { 43 addLoadEvent(function() { 44 window.onresize = resizeHeight; 45 }); 46 } 47 48 function mobileSetup() { 49 $("body").css({'overflow':'auto'}); 50 $("html").css({'overflow':'auto'}); 51 $("#body-content").css({'position':'relative', 'top':'0'}); 52 $("#doc-content").css({'overflow':'visible', 'border-left':'3px solid #DDD'}); 53 $("#side-nav").css({'padding':'0'}); 54 $("#nav-tree").css({'overflow-y': 'auto'}); 55 } 56 57 /* loads the lists.js file to the page. 58 Loading this in the head was slowing page load time */ 59 addLoadEvent( function() { 60 var lists = document.createElement("script"); 61 lists.setAttribute("type","text/javascript"); 62 lists.setAttribute("src", toRoot+"reference/lists.js"); 63 document.getElementsByTagName("head")[0].appendChild(lists); 64 } ); 65 66 addLoadEvent( function() { 67 $("pre:not(.no-pretty-print)").addClass("prettyprint"); 68 prettyPrint(); 69 } ); 70 71 function setToRoot(root) { 72 toRoot = root; 73 // note: toRoot also used by carousel.js 74 } 75 76 function restoreWidth(navWidth) { 77 var windowWidth = $(window).width() + "px"; 78 content.css({marginLeft:parseInt(navWidth) + 6 + "px"}); //account for 6px-wide handle-bar 79 80 if (isIE6) { 81 content.css({width:parseInt(windowWidth) - parseInt(navWidth) - 6 + "px"}); // necessary in order for scrollbars to be visible 82 } 83 84 sidenav.css({width:navWidth}); 85 resizePackagesNav.css({width:navWidth}); 86 classesNav.css({width:navWidth}); 87 $("#packages-nav").css({width:navWidth}); 88 } 89 90 function restoreHeight(packageHeight) { 91 var windowHeight = ($(window).height() - HEADER_HEIGHT); 92 var swapperHeight = windowHeight - 13; 93 $("#swapper").css({height:swapperHeight + "px"}); 94 sidenav.css({height:windowHeight + "px"}); 95 content.css({height:windowHeight + "px"}); 96 resizePackagesNav.css({maxHeight:swapperHeight + "px", height:packageHeight}); 97 classesNav.css({height:swapperHeight - parseInt(packageHeight) + "px"}); 98 $("#packages-nav").css({height:parseInt(packageHeight) - 6 + "px"}); //move 6px to give space for the resize handle 99 devdocNav.css({height:sidenav.css("height")}); 100 $("#nav-tree").css({height:swapperHeight + "px"}); 101 } 102 103 function readCookie(cookie) { 104 var myCookie = cookie_namespace+"_"+cookie+"="; 105 if (document.cookie) { 106 var index = document.cookie.indexOf(myCookie); 107 if (index != -1) { 108 var valStart = index + myCookie.length; 109 var valEnd = document.cookie.indexOf(";", valStart); 110 if (valEnd == -1) { 111 valEnd = document.cookie.length; 112 } 113 var val = document.cookie.substring(valStart, valEnd); 114 return val; 115 } 116 } 117 return 0; 118 } 119 120 function writeCookie(cookie, val, section, expiration) { 121 if (val==undefined) return; 122 section = section == null ? "_" : "_"+section+"_"; 123 if (expiration == null) { 124 var date = new Date(); 125 date.setTime(date.getTime()+(10*365*24*60*60*1000)); // default expiration is one week 126 expiration = date.toGMTString(); 127 } 128 document.cookie = cookie_namespace + section + cookie + "=" + val + "; expires=" + expiration+"; path=/"; 129 } 130 131 function init() { 132 $("#side-nav").css({position:"absolute",left:0}); 133 content = $("#doc-content"); 134 resizePackagesNav = $("#resize-packages-nav"); 135 classesNav = $("#classes-nav"); 136 sidenav = $("#side-nav"); 137 devdocNav = $("#devdoc-nav"); 138 139 var cookiePath = ""; 140 if (location.href.indexOf("/reference/") != -1) { 141 cookiePath = "reference_"; 142 } else if (location.href.indexOf("/guide/") != -1) { 143 cookiePath = "guide_"; 144 } else if (location.href.indexOf("/sdk/") != -1) { 145 cookiePath = "sdk_"; 146 } else if ((location.href.indexOf("/resources/") != -1) || 147 (location.href.indexOf("/training/") != -1)) { 148 cookiePath = "resources_"; 149 } 150 151 if (!isMobile) { 152 $("#resize-packages-nav").resizable({handles: "s", resize: function(e, ui) { resizePackagesHeight(); } }); 153 $("#side-nav").resizable({handles: "e", resize: function(e, ui) { resizeWidth(); } }); 154 var cookieWidth = readCookie(cookiePath+'width'); 155 var cookieHeight = readCookie(cookiePath+'height'); 156 if (cookieWidth) { 157 restoreWidth(cookieWidth); 158 } else if ($("#side-nav").length) { 159 resizeWidth(); 160 } 161 if (cookieHeight) { 162 restoreHeight(cookieHeight); 163 } else { 164 resizeHeight(); 165 } 166 } 167 168 if (devdocNav.length) { // only dev guide, resources, and sdk 169 tryPopulateResourcesNav(); 170 highlightNav(location.href); 171 } 172 } 173 174 function tryPopulateResourcesNav() { 175 var sampleList = $('#devdoc-nav-sample-list'); 176 var articleList = $('#devdoc-nav-article-list'); 177 var tutorialList = $('#devdoc-nav-tutorial-list'); 178 var topicList = $('#devdoc-nav-topic-list'); 179 180 if (!topicList.length || !ANDROID_TAGS || !ANDROID_RESOURCES) 181 return; 182 183 var topics = []; 184 for (var topic in ANDROID_TAGS['topic']) { 185 topics.push({name:topic,title:ANDROID_TAGS['topic'][topic]}); 186 } 187 topics.sort(function(x,y){ return (x.title < y.title) ? -1 : 1; }); 188 for (var i = 0; i < topics.length; i++) { 189 topicList.append( 190 $('<li>').append( 191 $('<a>') 192 .attr('href', toRoot + "resources/browser.html?tag=" + topics[i].name) 193 .append($('<span>') 194 .addClass('en') 195 .html(topics[i].title) 196 ) 197 ) 198 ); 199 } 200 201 var _renderResourceList = function(tag, listNode) { 202 var resources = []; 203 var tags; 204 var resource; 205 var i, j; 206 for (i = 0; i < ANDROID_RESOURCES.length; i++) { 207 resource = ANDROID_RESOURCES[i]; 208 tags = resource.tags || []; 209 var hasTag = false; 210 for (j = 0; j < tags.length; j++) 211 if (tags[j] == tag) { 212 hasTag = true; 213 break; 214 } 215 if (!hasTag) 216 continue; 217 resources.push(resource); 218 } 219 //resources.sort(function(x,y){ return (x.title.en < y.title.en) ? -1 : 1; }); 220 for (i = 0; i < resources.length; i++) { 221 resource = resources[i]; 222 var listItemNode = $('<li>').append( 223 $('<a>') 224 .attr('href', toRoot + "resources/" + resource.path) 225 .append($('<span>') 226 .addClass('en') 227 .html(resource.title.en) 228 ) 229 ); 230 tags = resource.tags || []; 231 for (j = 0; j < tags.length; j++) { 232 if (tags[j] == 'new') { 233 listItemNode.get(0).innerHTML += ' <span class="new">new!</span>'; 234 break; 235 } else if (tags[j] == 'updated') { 236 listItemNode.get(0).innerHTML += ' <span class="new">updated!</span>'; 237 break; 238 } 239 } 240 listNode.append(listItemNode); 241 } 242 }; 243 244 _renderResourceList('sample', sampleList); 245 _renderResourceList('article', articleList); 246 _renderResourceList('tutorial', tutorialList); 247 } 248 249 function highlightNav(fullPageName) { 250 var lastSlashPos = fullPageName.lastIndexOf("/"); 251 var firstSlashPos; 252 if (fullPageName.indexOf("/guide/") != -1) { 253 firstSlashPos = fullPageName.indexOf("/guide/"); 254 } else if (fullPageName.indexOf("/sdk/") != -1) { 255 firstSlashPos = fullPageName.indexOf("/sdk/"); 256 } else if (fullPageName.indexOf("/resources/") != -1) { 257 firstSlashPos = fullPageName.indexOf("/resources/"); 258 } else if (fullPageName.indexOf("/training/") != -1) { 259 firstSlashPos = fullPageName.indexOf("/training/"); 260 } 261 if (lastSlashPos == (fullPageName.length - 1)) { // if the url ends in slash (add 'index.html') 262 fullPageName = fullPageName + "index.html"; 263 } 264 265 // get the path and page name from the URL (such as 'guide/topics/graphics/index.html') 266 var htmlPos = fullPageName.indexOf(".html"); 267 var pathPageName = fullPageName.slice(firstSlashPos, htmlPos + 5); // +5 advances past ".html" 268 // find instances of the page name in the side nav 269 var link = $("#devdoc-nav a[href$='"+ pathPageName+"']"); 270 // if there's no match, then let's backstep through the directory until we find an index.html 271 // page that matches our ancestor directories (only for dev guide and resources) 272 if ((link.length == 0) && ((fullPageName.indexOf("/guide/") != -1) || 273 (fullPageName.indexOf("/resources/") != -1))) { 274 lastBackstep = pathPageName.lastIndexOf("/"); 275 while (link.length == 0) { 276 backstepDirectory = pathPageName.lastIndexOf("/", lastBackstep); 277 link = $("#devdoc-nav a[href$='"+ pathPageName.slice(0, backstepDirectory + 278 1)+"index.html']"); 279 lastBackstep = pathPageName.lastIndexOf("/", lastBackstep - 1); 280 if (lastBackstep == 0) break; 281 } 282 } 283 284 // add 'selected' to the <li> or <div> that wraps this <a> 285 link.parent().addClass('selected'); 286 287 // if we're in a toggleable root link (<li class=toggle-list><div><a>) 288 if (link.parent().parent().hasClass('toggle-list')) { 289 toggle(link.parent().parent(), false); // open our own list 290 // then also check if we're in a third-level nested list that's toggleable 291 if (link.parent().parent().parent().is(':hidden')) { 292 toggle(link.parent().parent().parent().parent(), false); // open the super parent list 293 } 294 } 295 // if we're in a normal nav link (<li><a>) and the parent <ul> is hidden 296 else if (link.parent().parent().is(':hidden')) { 297 toggle(link.parent().parent().parent(), false); // open the parent list 298 // then also check if the parent list is also nested in a hidden list 299 if (link.parent().parent().parent().parent().is(':hidden')) { 300 toggle(link.parent().parent().parent().parent().parent(), false); // open the super parent list 301 } 302 } 303 } 304 305 /* Resize the height of the nav panels in the reference, 306 * and save the new size to a cookie */ 307 function resizePackagesHeight() { 308 var windowHeight = ($(window).height() - HEADER_HEIGHT); 309 var swapperHeight = windowHeight - 13; // move 13px for swapper link at the bottom 310 resizePackagesNav.css({maxHeight:swapperHeight + "px"}); 311 classesNav.css({height:swapperHeight - parseInt(resizePackagesNav.css("height")) + "px"}); 312 313 $("#swapper").css({height:swapperHeight + "px"}); 314 $("#packages-nav").css({height:parseInt(resizePackagesNav.css("height")) - 6 + "px"}); //move 6px for handle 315 316 var basePath = getBaseUri(location.pathname); 317 var section = basePath.substring(1,basePath.indexOf("/",1)); 318 writeCookie("height", resizePackagesNav.css("height"), section, null); 319 } 320 321 /* Resize the height of the side-nav and doc-content divs, 322 * which creates the frame effect */ 323 function resizeHeight() { 324 var docContent = $("#doc-content"); 325 326 // Get the window height and always resize the doc-content and side-nav divs 327 var windowHeight = ($(window).height() - HEADER_HEIGHT); 328 docContent.css({height:windowHeight + "px"}); 329 $("#side-nav").css({height:windowHeight + "px"}); 330 331 var href = location.href; 332 // If in the reference docs, also resize the "swapper", "classes-nav", and "nav-tree" divs 333 if (href.indexOf("/reference/") != -1) { 334 var swapperHeight = windowHeight - 13; 335 $("#swapper").css({height:swapperHeight + "px"}); 336 $("#classes-nav").css({height:swapperHeight - parseInt(resizePackagesNav.css("height")) + "px"}); 337 $("#nav-tree").css({height:swapperHeight + "px"}); 338 339 // Also resize the "devdoc-nav" div 340 } else if ($("#devdoc-nav").length) { 341 $("#devdoc-nav").css({height:sidenav.css("height")}); 342 } 343 344 // Hide the "Go to top" link if there's no vertical scroll 345 if ( parseInt($("#jd-content").css("height")) <= parseInt(docContent.css("height")) ) { 346 $("a[href='#top']").css({'display':'none'}); 347 } else { 348 $("a[href='#top']").css({'display':'inline'}); 349 } 350 } 351 352 /* Resize the width of the "side-nav" and the left margin of the "doc-content" div, 353 * which creates the resizable side bar */ 354 function resizeWidth() { 355 var windowWidth = $(window).width() + "px"; 356 var sidenav = $("#side-nav"); 357 if (sidenav.length) { 358 var sidenavWidth = sidenav.css("width"); 359 } else { 360 var sidenavWidth = 0; 361 } 362 content.css({marginLeft:parseInt(sidenavWidth) + 6 + "px"}); //account for 6px-wide handle-bar 363 364 if (isIE6) { 365 content.css({width:parseInt(windowWidth) - parseInt(sidenavWidth) - 6 + "px"}); // necessary in order to for scrollbars to be visible 366 } 367 368 resizePackagesNav.css({width:sidenavWidth}); 369 classesNav.css({width:sidenavWidth}); 370 $("#packages-nav").css({width:sidenavWidth}); 371 372 if (sidenav.length) { // Must check if the nav exists because IE6 calls resizeWidth() from resizeAll() for all pages 373 var basePath = getBaseUri(location.pathname); 374 var section = basePath.substring(1,basePath.indexOf("/",1)); 375 section = section.indexOf("training") != -1 ? "resources" : section; 376 writeCookie("width", sidenavWidth, section, null); 377 } 378 } 379 380 /* For IE6 only, 381 * because it can't properly perform auto width for "doc-content" div, 382 * avoiding this for all browsers provides better performance */ 383 function resizeAll() { 384 resizeHeight(); 385 resizeWidth(); 386 } 387 388 function getBaseUri(uri) { 389 var intlUrl = (uri.substring(0,6) == "/intl/"); 390 if (intlUrl) { 391 base = uri.substring(uri.indexOf('intl/')+5,uri.length); 392 base = base.substring(base.indexOf('/')+1, base.length); 393 //alert("intl, returning base url: /" + base); 394 return ("/" + base); 395 } else { 396 //alert("not intl, returning uri as found."); 397 return uri; 398 } 399 } 400 401 function requestAppendHL(uri) { 402 //append "?hl=<lang> to an outgoing request (such as to blog) 403 var lang = getLangPref(); 404 if (lang) { 405 var q = 'hl=' + lang; 406 uri += '?' + q; 407 window.location = uri; 408 return false; 409 } else { 410 return true; 411 } 412 } 413 414 function loadLast(cookiePath) { 415 var location = window.location.href; 416 if (location.indexOf("/"+cookiePath+"/") != -1) { 417 return true; 418 } 419 var lastPage = readCookie(cookiePath + "_lastpage"); 420 if (lastPage) { 421 window.location = lastPage; 422 return false; 423 } 424 return true; 425 } 426 427 $(window).unload(function(){ 428 var path = getBaseUri(location.pathname); 429 if (path.indexOf("/reference/") != -1) { 430 writeCookie("lastpage", path, "reference", null); 431 } else if (path.indexOf("/guide/") != -1) { 432 writeCookie("lastpage", path, "guide", null); 433 } else if ((path.indexOf("/resources/") != -1) || (path.indexOf("/training/") != -1)) { 434 writeCookie("lastpage", path, "resources", null); 435 } 436 }); 437 438 function toggle(obj, slide) { 439 var ul = $("ul:first", obj); 440 var li = ul.parent(); 441 if (li.hasClass("closed")) { 442 if (slide) { 443 ul.slideDown("fast"); 444 } else { 445 ul.show(); 446 } 447 li.removeClass("closed"); 448 li.addClass("open"); 449 $(".toggle-img", li).attr("title", "hide pages"); 450 } else { 451 ul.slideUp("fast"); 452 li.removeClass("open"); 453 li.addClass("closed"); 454 $(".toggle-img", li).attr("title", "show pages"); 455 } 456 } 457 458 function buildToggleLists() { 459 $(".toggle-list").each( 460 function(i) { 461 $("div:first", this).append("<a class='toggle-img' href='#' title='show pages' onClick='toggle(this.parentNode.parentNode, true); return false;'></a>"); 462 $(this).addClass("closed"); 463 }); 464 } 465 466 function getNavPref() { 467 var v = readCookie('reference_nav'); 468 if (v != NAV_PREF_TREE) { 469 v = NAV_PREF_PANELS; 470 } 471 return v; 472 } 473 474 function chooseDefaultNav() { 475 nav_pref = getNavPref(); 476 if (nav_pref == NAV_PREF_TREE) { 477 $("#nav-panels").toggle(); 478 $("#panel-link").toggle(); 479 $("#nav-tree").toggle(); 480 $("#tree-link").toggle(); 481 } 482 } 483 484 function swapNav() { 485 if (nav_pref == NAV_PREF_TREE) { 486 nav_pref = NAV_PREF_PANELS; 487 } else { 488 nav_pref = NAV_PREF_TREE; 489 init_default_navtree(toRoot); 490 } 491 var date = new Date(); 492 date.setTime(date.getTime()+(10*365*24*60*60*1000)); // keep this for 10 years 493 writeCookie("nav", nav_pref, "reference", date.toGMTString()); 494 495 $("#nav-panels").toggle(); 496 $("#panel-link").toggle(); 497 $("#nav-tree").toggle(); 498 $("#tree-link").toggle(); 499 500 if ($("#nav-tree").is(':visible')) scrollIntoView("nav-tree"); 501 else { 502 scrollIntoView("packages-nav"); 503 scrollIntoView("classes-nav"); 504 } 505 } 506 507 function scrollIntoView(nav) { 508 var navObj = $("#"+nav); 509 if (navObj.is(':visible')) { 510 var selected = $(".selected", navObj); 511 if (selected.length == 0) return; 512 if (selected.is("div")) selected = selected.parent(); // when the selected item is a parent 513 514 var scrolling = document.getElementById(nav); 515 var navHeight = navObj.height(); 516 var offsetTop = selected.position().top; 517 518 // handle nested items 519 if (selected.parent().parent().is(".toggle-list")) { 520 selected = selected.parent().parent(); 521 // handle second level nested items 522 if (selected.parent().parent().is(".toggle-list")) { 523 selected = selected.parent().parent(); 524 } 525 offsetTop += selected.position().top; 526 } 527 528 // 180px from the bottom of the list is the threshold 529 if(offsetTop > navHeight - 180) { 530 scrolling.scrollTop = offsetTop - navHeight + 180; 531 } 532 } 533 } 534 535 function changeTabLang(lang) { 536 var nodes = $("#header-tabs").find("."+lang); 537 for (i=0; i < nodes.length; i++) { // for each node in this language 538 var node = $(nodes[i]); 539 node.siblings().css("display","none"); // hide all siblings 540 if (node.not(":empty").length != 0) { //if this languages node has a translation, show it 541 node.css("display","inline"); 542 } else { //otherwise, show English instead 543 node.css("display","none"); 544 node.siblings().filter(".en").css("display","inline"); 545 } 546 } 547 } 548 549 function changeNavLang(lang) { 550 var nodes = $("#side-nav").find("."+lang); 551 for (i=0; i < nodes.length; i++) { // for each node in this language 552 var node = $(nodes[i]); 553 node.siblings().css("display","none"); // hide all siblings 554 if (node.not(":empty").length != 0) { // if this languages node has a translation, show it 555 node.css("display","inline"); 556 } else { // otherwise, show English instead 557 node.css("display","none"); 558 node.siblings().filter(".en").css("display","inline"); 559 } 560 } 561 } 562 563 function changeDocLang(lang) { 564 changeTabLang(lang); 565 changeNavLang(lang); 566 } 567 568 function changeLangPref(lang, refresh) { 569 var date = new Date(); 570 expires = date.toGMTString(date.setTime(date.getTime()+(10*365*24*60*60*1000))); // keep this for 50 years 571 //alert("expires: " + expires) 572 writeCookie("pref_lang", lang, null, expires); 573 //changeDocLang(lang); 574 if (refresh) { 575 l = getBaseUri(location.pathname); 576 window.location = l; 577 } 578 } 579 580 function loadLangPref() { 581 var lang = readCookie("pref_lang"); 582 if (lang != 0) { 583 $("#language").find("option[value='"+lang+"']").attr("selected",true); 584 } 585 } 586 587 function getLangPref() { 588 var lang = $("#language").find(":selected").attr("value"); 589 if (!lang) { 590 lang = readCookie("pref_lang"); 591 } 592 return (lang != 0) ? lang : 'en'; 593 } 594 595 596 /* Used to hide and reveal supplemental content, such as long code samples. 597 See the companion CSS in android-developer-docs.css */ 598 function toggleContent(obj) { 599 var div = $(obj.parentNode.parentNode); 600 var toggleMe = $(".toggle-content-toggleme",div); 601 if (div.hasClass("closed")) { // if it's closed, open it 602 toggleMe.slideDown(); 603 $(".toggle-content-text", obj).toggle(); 604 div.removeClass("closed").addClass("open"); 605 $(".toggle-content-img", div).attr("title", "hide").attr("src", toRoot + "assets/images/triangle-opened.png"); 606 } else { // if it's open, close it 607 toggleMe.slideUp('fast', function() { // Wait until the animation is done before closing arrow 608 $(".toggle-content-text", obj).toggle(); 609 div.removeClass("open").addClass("closed"); 610 $(".toggle-content-img", div).attr("title", "show").attr("src", toRoot + "assets/images/triangle-closed.png"); 611 }); 612 } 613 return false; 614 } 615