Home | History | Annotate | Download | only in js
      1 var classesNav;
      2 var devdocNav;
      3 var sidenav;
      4 var cookie_namespace = 'android_developer';
      5 var NAV_PREF_TREE = "tree";
      6 var NAV_PREF_PANELS = "panels";
      7 var nav_pref;
      8 var isMobile = false; // true if mobile, so we can adjust some layout
      9 var mPagePath; // initialized in ready() function
     10 
     11 var basePath = getBaseUri(location.pathname);
     12 var SITE_ROOT = toRoot + basePath.substring(1,basePath.indexOf("/",1));
     13 var GOOGLE_DATA; // combined data for google service apis, used for search suggest
     14 
     15 // Ensure that all ajax getScript() requests allow caching
     16 $.ajaxSetup({
     17   cache: true
     18 });
     19 
     20 /******  ON LOAD SET UP STUFF *********/
     21 
     22 var navBarIsFixed = false;
     23 $(document).ready(function() {
     24 
     25   // load json file for JD doc search suggestions
     26   $.getScript(toRoot + 'reference/jd_lists.js');
     27   // load json file for Android API search suggestions
     28   $.getScript(toRoot + 'reference/lists.js');
     29   // load json files for Google services API suggestions
     30   $.getScript(toRoot + 'reference/gcm_lists.js', function(data, textStatus, jqxhr) {
     31       // once the GCM json (GCM_DATA) is loaded, load the GMS json (GMS_DATA) and merge the data
     32       if(jqxhr.status === 200) {
     33           $.getScript(toRoot + 'reference/gms_lists.js', function(data, textStatus, jqxhr) {
     34               if(jqxhr.status === 200) {
     35                   // combine GCM and GMS data
     36                   GOOGLE_DATA = GMS_DATA;
     37                   var start = GOOGLE_DATA.length;
     38                   for (var i=0; i<GCM_DATA.length; i++) {
     39                       GOOGLE_DATA.push({id:start+i, label:GCM_DATA[i].label,
     40                               link:GCM_DATA[i].link, type:GCM_DATA[i].type});
     41                   }
     42               }
     43           });
     44       }
     45   });
     46 
     47   // setup keyboard listener for search shortcut
     48   $('body').keyup(function(event) {
     49     if (event.which == 191) {
     50       $('#search_autocomplete').focus();
     51     }
     52   });
     53 
     54   // init the fullscreen toggle click event
     55   $('#nav-swap .fullscreen').click(function(){
     56     if ($(this).hasClass('disabled')) {
     57       toggleFullscreen(true);
     58     } else {
     59       toggleFullscreen(false);
     60     }
     61   });
     62 
     63   // initialize the divs with custom scrollbars
     64   $('.scroll-pane').jScrollPane( {verticalGutter:0} );
     65 
     66   // add HRs below all H2s (except for a few other h2 variants)
     67   $('h2').not('#qv h2').not('#tb h2').not('.sidebox h2').not('#devdoc-nav h2').not('h2.norule').css({marginBottom:0}).after('<hr/>');
     68 
     69   // set up the search close button
     70   $('.search .close').click(function() {
     71     $searchInput = $('#search_autocomplete');
     72     $searchInput.attr('value', '');
     73     $(this).addClass("hide");
     74     $("#search-container").removeClass('active');
     75     $("#search_autocomplete").blur();
     76     search_focus_changed($searchInput.get(), false);
     77     hideResults();
     78   });
     79 
     80   // Set up quicknav
     81   var quicknav_open = false;
     82   $("#btn-quicknav").click(function() {
     83     if (quicknav_open) {
     84       $(this).removeClass('active');
     85       quicknav_open = false;
     86       collapse();
     87     } else {
     88       $(this).addClass('active');
     89       quicknav_open = true;
     90       expand();
     91     }
     92   })
     93 
     94   var expand = function() {
     95    $('#header-wrap').addClass('quicknav');
     96    $('#quicknav').stop().show().animate({opacity:'1'});
     97   }
     98 
     99   var collapse = function() {
    100     $('#quicknav').stop().animate({opacity:'0'}, 100, function() {
    101       $(this).hide();
    102       $('#header-wrap').removeClass('quicknav');
    103     });
    104   }
    105 
    106 
    107   //Set up search
    108   $("#search_autocomplete").focus(function() {
    109     $("#search-container").addClass('active');
    110   })
    111   $("#search-container").mouseover(function() {
    112     $("#search-container").addClass('active');
    113     $("#search_autocomplete").focus();
    114   })
    115   $("#search-container").mouseout(function() {
    116     if ($("#search_autocomplete").is(":focus")) return;
    117     if ($("#search_autocomplete").val() == '') {
    118       setTimeout(function(){
    119         $("#search-container").removeClass('active');
    120         $("#search_autocomplete").blur();
    121       },250);
    122     }
    123   })
    124   $("#search_autocomplete").blur(function() {
    125     if ($("#search_autocomplete").val() == '') {
    126       $("#search-container").removeClass('active');
    127     }
    128   })
    129 
    130 
    131   // prep nav expandos
    132   var pagePath = document.location.pathname;
    133   // account for intl docs by removing the intl/*/ path
    134   if (pagePath.indexOf("/intl/") == 0) {
    135     pagePath = pagePath.substr(pagePath.indexOf("/",6)); // start after intl/ to get last /
    136   }
    137 
    138   if (pagePath.indexOf(SITE_ROOT) == 0) {
    139     if (pagePath == '' || pagePath.charAt(pagePath.length - 1) == '/') {
    140       pagePath += 'index.html';
    141     }
    142   }
    143 
    144   // Need a copy of the pagePath before it gets changed in the next block;
    145   // it's needed to perform proper tab highlighting in offline docs (see rootDir below)
    146   var pagePathOriginal = pagePath;
    147   if (SITE_ROOT.match(/\.\.\//) || SITE_ROOT == '') {
    148     // If running locally, SITE_ROOT will be a relative path, so account for that by
    149     // finding the relative URL to this page. This will allow us to find links on the page
    150     // leading back to this page.
    151     var pathParts = pagePath.split('/');
    152     var relativePagePathParts = [];
    153     var upDirs = (SITE_ROOT.match(/(\.\.\/)+/) || [''])[0].length / 3;
    154     for (var i = 0; i < upDirs; i++) {
    155       relativePagePathParts.push('..');
    156     }
    157     for (var i = 0; i < upDirs; i++) {
    158       relativePagePathParts.push(pathParts[pathParts.length - (upDirs - i) - 1]);
    159     }
    160     relativePagePathParts.push(pathParts[pathParts.length - 1]);
    161     pagePath = relativePagePathParts.join('/');
    162   } else {
    163     // Otherwise the page path is already an absolute URL
    164   }
    165 
    166   // Highlight the header tabs...
    167   // highlight Design tab
    168   if ($("body").hasClass("design")) {
    169     $("#header li.design a").addClass("selected");
    170 
    171   // highlight Develop tab
    172   } else if ($("body").hasClass("develop") || $("body").hasClass("google")) {
    173     $("#header li.develop a").addClass("selected");
    174     // In Develop docs, also highlight appropriate sub-tab
    175     var rootDir = pagePathOriginal.substring(1,pagePathOriginal.indexOf('/', 1));
    176     if (rootDir == "training") {
    177       $("#nav-x li.training a").addClass("selected");
    178     } else if (rootDir == "guide") {
    179       $("#nav-x li.guide a").addClass("selected");
    180     } else if (rootDir == "reference") {
    181       // If the root is reference, but page is also part of Google Services, select Google
    182       if ($("body").hasClass("google")) {
    183         $("#nav-x li.google a").addClass("selected");
    184       } else {
    185         $("#nav-x li.reference a").addClass("selected");
    186       }
    187     } else if ((rootDir == "tools") || (rootDir == "sdk")) {
    188       $("#nav-x li.tools a").addClass("selected");
    189     } else if ($("body").hasClass("google")) {
    190       $("#nav-x li.google a").addClass("selected");
    191     } else if ($("body").hasClass("samples")) {
    192       $("#nav-x li.samples a").addClass("selected");
    193     }
    194 
    195   // highlight Distribute tab
    196   } else if ($("body").hasClass("distribute")) {
    197     $("#header li.distribute a").addClass("selected");
    198   }
    199 
    200   // set global variable so we can highlight the sidenav a bit later (such as for google reference)
    201   // and highlight the sidenav
    202   mPagePath = pagePath;
    203   highlightSidenav();
    204 
    205   // set up prev/next links if they exist
    206   var $selNavLink = $('#nav').find('a[href="' + pagePath + '"]');
    207   var $selListItem;
    208   if ($selNavLink.length) {
    209     $selListItem = $selNavLink.closest('li');
    210 
    211     // set up prev links
    212     var $prevLink = [];
    213     var $prevListItem = $selListItem.prev('li');
    214 
    215     var crossBoundaries = ($("body.design").length > 0) || ($("body.guide").length > 0) ? true :
    216 false; // navigate across topic boundaries only in design docs
    217     if ($prevListItem.length) {
    218       if ($prevListItem.hasClass('nav-section')) {
    219         // jump to last topic of previous section
    220         $prevLink = $prevListItem.find('a:last');
    221       } else if (!$selListItem.hasClass('nav-section')) {
    222         // jump to previous topic in this section
    223         $prevLink = $prevListItem.find('a:eq(0)');
    224       }
    225     } else {
    226       // jump to this section's index page (if it exists)
    227       var $parentListItem = $selListItem.parents('li');
    228       $prevLink = $selListItem.parents('li').find('a');
    229 
    230       // except if cross boundaries aren't allowed, and we're at the top of a section already
    231       // (and there's another parent)
    232       if (!crossBoundaries && $parentListItem.hasClass('nav-section')
    233                            && $selListItem.hasClass('nav-section')) {
    234         $prevLink = [];
    235       }
    236     }
    237 
    238     // set up next links
    239     var $nextLink = [];
    240     var startClass = false;
    241     var training = $(".next-class-link").length; // decides whether to provide "next class" link
    242     var isCrossingBoundary = false;
    243 
    244     if ($selListItem.hasClass('nav-section') && $selListItem.children('div.empty').length == 0) {
    245       // we're on an index page, jump to the first topic
    246       $nextLink = $selListItem.find('ul:eq(0)').find('a:eq(0)');
    247 
    248       // if there aren't any children, go to the next section (required for About pages)
    249       if($nextLink.length == 0) {
    250         $nextLink = $selListItem.next('li').find('a');
    251       } else if ($('.topic-start-link').length) {
    252         // as long as there's a child link and there is a "topic start link" (we're on a landing)
    253         // then set the landing page "start link" text to be the first doc title
    254         $('.topic-start-link').text($nextLink.text().toUpperCase());
    255       }
    256 
    257       // If the selected page has a description, then it's a class or article homepage
    258       if ($selListItem.find('a[description]').length) {
    259         // this means we're on a class landing page
    260         startClass = true;
    261       }
    262     } else {
    263       // jump to the next topic in this section (if it exists)
    264       $nextLink = $selListItem.next('li').find('a:eq(0)');
    265       if ($nextLink.length == 0) {
    266         isCrossingBoundary = true;
    267         // no more topics in this section, jump to the first topic in the next section
    268         $nextLink = $selListItem.parents('li:eq(0)').next('li.nav-section').find('a:eq(0)');
    269         if (!$nextLink.length) {  // Go up another layer to look for next page (lesson > class > course)
    270           $nextLink = $selListItem.parents('li:eq(1)').next('li.nav-section').find('a:eq(0)');
    271           if ($nextLink.length == 0) {
    272             // if that doesn't work, we're at the end of the list, so disable NEXT link
    273             $('.next-page-link').attr('href','').addClass("disabled")
    274                                 .click(function() { return false; });
    275           }
    276         }
    277       }
    278     }
    279 
    280     if (startClass) {
    281       $('.start-class-link').attr('href', $nextLink.attr('href')).removeClass("hide");
    282 
    283       // if there's no training bar (below the start button),
    284       // then we need to add a bottom border to button
    285       if (!$("#tb").length) {
    286         $('.start-class-link').css({'border-bottom':'1px solid #DADADA'});
    287       }
    288     } else if (isCrossingBoundary && !$('body.design').length) {  // Design always crosses boundaries
    289       $('.content-footer.next-class').show();
    290       $('.next-page-link').attr('href','')
    291                           .removeClass("hide").addClass("disabled")
    292                           .click(function() { return false; });
    293       if ($nextLink.length) {
    294         $('.next-class-link').attr('href',$nextLink.attr('href'))
    295                              .removeClass("hide").append($nextLink.html());
    296         $('.next-class-link').find('.new').empty();
    297       }
    298     } else {
    299       $('.next-page-link').attr('href', $nextLink.attr('href')).removeClass("hide");
    300     }
    301 
    302     if (!startClass && $prevLink.length) {
    303       var prevHref = $prevLink.attr('href');
    304       if (prevHref == SITE_ROOT + 'index.html') {
    305         // Don't show Previous when it leads to the homepage
    306       } else {
    307         $('.prev-page-link').attr('href', $prevLink.attr('href')).removeClass("hide");
    308       }
    309     }
    310 
    311     // If this is a training 'article', there should be no prev/next nav
    312     // ... if the grandparent is the "nav" ... and it has no child list items...
    313     if (training && $selListItem.parents('ul').eq(1).is('[id="nav"]') &&
    314         !$selListItem.find('li').length) {
    315       $('.next-page-link,.prev-page-link').attr('href','').addClass("disabled")
    316                           .click(function() { return false; });
    317     }
    318 
    319   }
    320 
    321 
    322 
    323   // Set up the course landing pages for Training with class names and descriptions
    324   if ($('body.trainingcourse').length) {
    325     var $classLinks = $selListItem.find('ul li a').not('#nav .nav-section .nav-section ul a');
    326     var $classDescriptions = $classLinks.attr('description');
    327 
    328     var $olClasses  = $('<ol class="class-list"></ol>');
    329     var $liClass;
    330     var $imgIcon;
    331     var $h2Title;
    332     var $pSummary;
    333     var $olLessons;
    334     var $liLesson;
    335     $classLinks.each(function(index) {
    336       $liClass  = $('<li></li>');
    337       $h2Title  = $('<a class="title" href="'+$(this).attr('href')+'"><h2>' + $(this).html()+'</h2><span></span></a>');
    338       $pSummary = $('<p class="description">' + $(this).attr('description') + '</p>');
    339 
    340       $olLessons  = $('<ol class="lesson-list"></ol>');
    341 
    342       $lessons = $(this).closest('li').find('ul li a');
    343 
    344       if ($lessons.length) {
    345         $imgIcon = $('<img src="'+toRoot+'assets/images/resource-tutorial.png" '
    346             + ' width="64" height="64" alt=""/>');
    347         $lessons.each(function(index) {
    348           $olLessons.append('<li><a href="'+$(this).attr('href')+'">' + $(this).html()+'</a></li>');
    349         });
    350       } else {
    351         $imgIcon = $('<img src="'+toRoot+'assets/images/resource-article.png" '
    352             + ' width="64" height="64" alt=""/>');
    353         $pSummary.addClass('article');
    354       }
    355 
    356       $liClass.append($h2Title).append($imgIcon).append($pSummary).append($olLessons);
    357       $olClasses.append($liClass);
    358     });
    359     $('.jd-descr').append($olClasses);
    360   }
    361 
    362   // Set up expand/collapse behavior
    363   initExpandableNavItems("#nav");
    364 
    365 
    366   $(".scroll-pane").scroll(function(event) {
    367       event.preventDefault();
    368       return false;
    369   });
    370 
    371   /* Resize nav height when window height changes */
    372   $(window).resize(function() {
    373     if ($('#side-nav').length == 0) return;
    374     var stylesheet = $('link[rel="stylesheet"][class="fullscreen"]');
    375     setNavBarLeftPos(); // do this even if sidenav isn't fixed because it could become fixed
    376     // make sidenav behave when resizing the window and side-scolling is a concern
    377     if (navBarIsFixed) {
    378       if ((stylesheet.attr("disabled") == "disabled") || stylesheet.length == 0) {
    379         updateSideNavPosition();
    380       } else {
    381         updateSidenavFullscreenWidth();
    382       }
    383     }
    384     resizeNav();
    385   });
    386 
    387 
    388   // Set up fixed navbar
    389   var prevScrollLeft = 0; // used to compare current position to previous position of horiz scroll
    390   $(window).scroll(function(event) {
    391     if ($('#side-nav').length == 0) return;
    392     if (event.target.nodeName == "DIV") {
    393       // Dump scroll event if the target is a DIV, because that means the event is coming
    394       // from a scrollable div and so there's no need to make adjustments to our layout
    395       return;
    396     }
    397     var scrollTop = $(window).scrollTop();
    398     var headerHeight = $('#header').outerHeight();
    399     var subheaderHeight = $('#nav-x').outerHeight();
    400     var searchResultHeight = $('#searchResults').is(":visible") ?
    401                              $('#searchResults').outerHeight() : 0;
    402     var totalHeaderHeight = headerHeight + subheaderHeight + searchResultHeight;
    403     // we set the navbar fixed when the scroll position is beyond the height of the site header...
    404     var navBarShouldBeFixed = scrollTop > totalHeaderHeight;
    405     // ... except if the document content is shorter than the sidenav height.
    406     // (this is necessary to avoid crazy behavior on OSX Lion due to overscroll bouncing)
    407     if ($("#doc-col").height() < $("#side-nav").height()) {
    408       navBarShouldBeFixed = false;
    409     }
    410 
    411     var scrollLeft = $(window).scrollLeft();
    412     // When the sidenav is fixed and user scrolls horizontally, reposition the sidenav to match
    413     if (navBarIsFixed && (scrollLeft != prevScrollLeft)) {
    414       updateSideNavPosition();
    415       prevScrollLeft = scrollLeft;
    416     }
    417 
    418     // Don't continue if the header is sufficently far away
    419     // (to avoid intensive resizing that slows scrolling)
    420     if (navBarIsFixed && navBarShouldBeFixed) {
    421       return;
    422     }
    423 
    424     if (navBarIsFixed != navBarShouldBeFixed) {
    425       if (navBarShouldBeFixed) {
    426         // make it fixed
    427         var width = $('#devdoc-nav').width();
    428         $('#devdoc-nav')
    429             .addClass('fixed')
    430             .css({'width':width+'px'})
    431             .prependTo('#body-content');
    432         // add neato "back to top" button
    433         $('#devdoc-nav a.totop').css({'display':'block','width':$("#nav").innerWidth()+'px'});
    434 
    435         // update the sidenaav position for side scrolling
    436         updateSideNavPosition();
    437       } else {
    438         // make it static again
    439         $('#devdoc-nav')
    440             .removeClass('fixed')
    441             .css({'width':'auto','margin':''})
    442             .prependTo('#side-nav');
    443         $('#devdoc-nav a.totop').hide();
    444       }
    445       navBarIsFixed = navBarShouldBeFixed;
    446     }
    447 
    448     resizeNav(250); // pass true in order to delay the scrollbar re-initialization for performance
    449   });
    450 
    451 
    452   var navBarLeftPos;
    453   if ($('#devdoc-nav').length) {
    454     setNavBarLeftPos();
    455   }
    456 
    457 
    458   // Set up play-on-hover <video> tags.
    459   $('video.play-on-hover').bind('click', function(){
    460     $(this).get(0).load(); // in case the video isn't seekable
    461     $(this).get(0).play();
    462   });
    463 
    464   // Set up tooltips
    465   var TOOLTIP_MARGIN = 10;
    466   $('acronym,.tooltip-link').each(function() {
    467     var $target = $(this);
    468     var $tooltip = $('<div>')
    469         .addClass('tooltip-box')
    470         .append($target.attr('title'))
    471         .hide()
    472         .appendTo('body');
    473     $target.removeAttr('title');
    474 
    475     $target.hover(function() {
    476       // in
    477       var targetRect = $target.offset();
    478       targetRect.width = $target.width();
    479       targetRect.height = $target.height();
    480 
    481       $tooltip.css({
    482         left: targetRect.left,
    483         top: targetRect.top + targetRect.height + TOOLTIP_MARGIN
    484       });
    485       $tooltip.addClass('below');
    486       $tooltip.show();
    487     }, function() {
    488       // out
    489       $tooltip.hide();
    490     });
    491   });
    492 
    493   // Set up <h2> deeplinks
    494   $('h2').click(function() {
    495     var id = $(this).attr('id');
    496     if (id) {
    497       document.location.hash = id;
    498     }
    499   });
    500 
    501   //Loads the +1 button
    502   var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
    503   po.src = 'https://apis.google.com/js/plusone.js';
    504   var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
    505 
    506 
    507   // Revise the sidenav widths to make room for the scrollbar
    508   // which avoids the visible width from changing each time the bar appears
    509   var $sidenav = $("#side-nav");
    510   var sidenav_width = parseInt($sidenav.innerWidth());
    511 
    512   $("#devdoc-nav  #nav").css("width", sidenav_width - 4 + "px"); // 4px is scrollbar width
    513 
    514 
    515   $(".scroll-pane").removeAttr("tabindex"); // get rid of tabindex added by jscroller
    516 
    517   if ($(".scroll-pane").length > 1) {
    518     // Check if there's a user preference for the panel heights
    519     var cookieHeight = readCookie("reference_height");
    520     if (cookieHeight) {
    521       restoreHeight(cookieHeight);
    522     }
    523   }
    524 
    525   resizeNav();
    526 
    527   /* init the language selector based on user cookie for lang */
    528   loadLangPref();
    529   changeNavLang(getLangPref());
    530 
    531   /* setup event handlers to ensure the overflow menu is visible while picking lang */
    532   $("#language select")
    533       .mousedown(function() {
    534         $("div.morehover").addClass("hover"); })
    535       .blur(function() {
    536         $("div.morehover").removeClass("hover"); });
    537 
    538   /* some global variable setup */
    539   resizePackagesNav = $("#resize-packages-nav");
    540   classesNav = $("#classes-nav");
    541   devdocNav = $("#devdoc-nav");
    542 
    543   var cookiePath = "";
    544   if (location.href.indexOf("/reference/") != -1) {
    545     cookiePath = "reference_";
    546   } else if (location.href.indexOf("/guide/") != -1) {
    547     cookiePath = "guide_";
    548   } else if (location.href.indexOf("/tools/") != -1) {
    549     cookiePath = "tools_";
    550   } else if (location.href.indexOf("/training/") != -1) {
    551     cookiePath = "training_";
    552   } else if (location.href.indexOf("/design/") != -1) {
    553     cookiePath = "design_";
    554   } else if (location.href.indexOf("/distribute/") != -1) {
    555     cookiePath = "distribute_";
    556   }
    557 
    558 });
    559 // END of the onload event
    560 
    561 
    562 function initExpandableNavItems(rootTag) {
    563   $(rootTag + ' li.nav-section .nav-section-header').click(function() {
    564     var section = $(this).closest('li.nav-section');
    565     if (section.hasClass('expanded')) {
    566     /* hide me and descendants */
    567       section.find('ul').slideUp(250, function() {
    568         // remove 'expanded' class from my section and any children
    569         section.closest('li').removeClass('expanded');
    570         $('li.nav-section', section).removeClass('expanded');
    571         resizeNav();
    572       });
    573     } else {
    574     /* show me */
    575       // first hide all other siblings
    576       var $others = $('li.nav-section.expanded', $(this).closest('ul')).not('.sticky');
    577       $others.removeClass('expanded').children('ul').slideUp(250);
    578 
    579       // now expand me
    580       section.closest('li').addClass('expanded');
    581       section.children('ul').slideDown(250, function() {
    582         resizeNav();
    583       });
    584     }
    585   });
    586 
    587   // Stop expand/collapse behavior when clicking on nav section links
    588   // (since we're navigating away from the page)
    589   // This selector captures the first instance of <a>, but not those with "#" as the href.
    590   $('.nav-section-header').find('a:eq(0)').not('a[href="#"]').click(function(evt) {
    591     window.location.href = $(this).attr('href');
    592     return false;
    593   });
    594 }
    595 
    596 /** Highlight the current page in sidenav, expanding children as appropriate */
    597 function highlightSidenav() {
    598   // if something is already highlighted, undo it. This is for dynamic navigation (Samples index)
    599   if ($("ul#nav li.selected").length) {
    600     unHighlightSidenav();
    601   }
    602   // look for URL in sidenav, including the hash
    603   var $selNavLink = $('#nav').find('a[href="' + mPagePath + location.hash + '"]');
    604 
    605   // If the selNavLink is still empty, look for it without the hash
    606   if ($selNavLink.length == 0) {
    607     $selNavLink = $('#nav').find('a[href="' + mPagePath + '"]');
    608   }
    609 
    610   var $selListItem;
    611   if ($selNavLink.length) {
    612     // Find this page's <li> in sidenav and set selected
    613     $selListItem = $selNavLink.closest('li');
    614     $selListItem.addClass('selected');
    615 
    616     // Traverse up the tree and expand all parent nav-sections
    617     $selNavLink.parents('li.nav-section').each(function() {
    618       $(this).addClass('expanded');
    619       $(this).children('ul').show();
    620     });
    621   }
    622 }
    623 
    624 function unHighlightSidenav() {
    625   $("ul#nav li.selected").removeClass("selected");
    626   $('ul#nav li.nav-section.expanded').removeClass('expanded').children('ul').hide();
    627 }
    628 
    629 function toggleFullscreen(enable) {
    630   var delay = 20;
    631   var enabled = true;
    632   var stylesheet = $('link[rel="stylesheet"][class="fullscreen"]');
    633   if (enable) {
    634     // Currently NOT USING fullscreen; enable fullscreen
    635     stylesheet.removeAttr('disabled');
    636     $('#nav-swap .fullscreen').removeClass('disabled');
    637     $('#devdoc-nav').css({left:''});
    638     setTimeout(updateSidenavFullscreenWidth,delay); // need to wait a moment for css to switch
    639     enabled = true;
    640   } else {
    641     // Currently USING fullscreen; disable fullscreen
    642     stylesheet.attr('disabled', 'disabled');
    643     $('#nav-swap .fullscreen').addClass('disabled');
    644     setTimeout(updateSidenavFixedWidth,delay); // need to wait a moment for css to switch
    645     enabled = false;
    646   }
    647   writeCookie("fullscreen", enabled, null, null);
    648   setNavBarLeftPos();
    649   resizeNav(delay);
    650   updateSideNavPosition();
    651   setTimeout(initSidenavHeightResize,delay);
    652 }
    653 
    654 
    655 function setNavBarLeftPos() {
    656   navBarLeftPos = $('#body-content').offset().left;
    657 }
    658 
    659 
    660 function updateSideNavPosition() {
    661   var newLeft = $(window).scrollLeft() - navBarLeftPos;
    662   $('#devdoc-nav').css({left: -newLeft});
    663   $('#devdoc-nav .totop').css({left: -(newLeft - parseInt($('#side-nav').css('margin-left')))});
    664 }
    665 
    666 // TODO: use $(document).ready instead
    667 function addLoadEvent(newfun) {
    668   var current = window.onload;
    669   if (typeof window.onload != 'function') {
    670     window.onload = newfun;
    671   } else {
    672     window.onload = function() {
    673       current();
    674       newfun();
    675     }
    676   }
    677 }
    678 
    679 var agent = navigator['userAgent'].toLowerCase();
    680 // If a mobile phone, set flag and do mobile setup
    681 if ((agent.indexOf("mobile") != -1) ||      // android, iphone, ipod
    682     (agent.indexOf("blackberry") != -1) ||
    683     (agent.indexOf("webos") != -1) ||
    684     (agent.indexOf("mini") != -1)) {        // opera mini browsers
    685   isMobile = true;
    686 }
    687 
    688 
    689 $(document).ready(function() {
    690   $("pre:not(.no-pretty-print)").addClass("prettyprint");
    691   prettyPrint();
    692 });
    693 
    694 
    695 
    696 
    697 /* ######### RESIZE THE SIDENAV HEIGHT ########## */
    698 
    699 function resizeNav(delay) {
    700   var $nav = $("#devdoc-nav");
    701   var $window = $(window);
    702   var navHeight;
    703 
    704   // Get the height of entire window and the total header height.
    705   // Then figure out based on scroll position whether the header is visible
    706   var windowHeight = $window.height();
    707   var scrollTop = $window.scrollTop();
    708   var headerHeight = $('#header').outerHeight();
    709   var subheaderHeight = $('#nav-x').outerHeight();
    710   var headerVisible = (scrollTop < (headerHeight + subheaderHeight));
    711 
    712   // get the height of space between nav and top of window.
    713   // Could be either margin or top position, depending on whether the nav is fixed.
    714   var topMargin = (parseInt($nav.css('margin-top')) || parseInt($nav.css('top'))) + 1;
    715   // add 1 for the #side-nav bottom margin
    716 
    717   // Depending on whether the header is visible, set the side nav's height.
    718   if (headerVisible) {
    719     // The sidenav height grows as the header goes off screen
    720     navHeight = windowHeight - (headerHeight + subheaderHeight - scrollTop) - topMargin;
    721   } else {
    722     // Once header is off screen, the nav height is almost full window height
    723     navHeight = windowHeight - topMargin;
    724   }
    725 
    726 
    727 
    728   $scrollPanes = $(".scroll-pane");
    729   if ($scrollPanes.length > 1) {
    730     // subtract the height of the api level widget and nav swapper from the available nav height
    731     navHeight -= ($('#api-nav-header').outerHeight(true) + $('#nav-swap').outerHeight(true));
    732 
    733     $("#swapper").css({height:navHeight + "px"});
    734     if ($("#nav-tree").is(":visible")) {
    735       $("#nav-tree").css({height:navHeight});
    736     }
    737 
    738     var classesHeight = navHeight - parseInt($("#resize-packages-nav").css("height")) - 10 + "px";
    739     //subtract 10px to account for drag bar
    740 
    741     // if the window becomes small enough to make the class panel height 0,
    742     // then the package panel should begin to shrink
    743     if (parseInt(classesHeight) <= 0) {
    744       $("#resize-packages-nav").css({height:navHeight - 10}); //subtract 10px for drag bar
    745       $("#packages-nav").css({height:navHeight - 10});
    746     }
    747 
    748     $("#classes-nav").css({'height':classesHeight, 'margin-top':'10px'});
    749     $("#classes-nav .jspContainer").css({height:classesHeight});
    750 
    751 
    752   } else {
    753     $nav.height(navHeight);
    754   }
    755 
    756   if (delay) {
    757     updateFromResize = true;
    758     delayedReInitScrollbars(delay);
    759   } else {
    760     reInitScrollbars();
    761   }
    762 
    763 }
    764 
    765 var updateScrollbars = false;
    766 var updateFromResize = false;
    767 
    768 /* Re-initialize the scrollbars to account for changed nav size.
    769  * This method postpones the actual update by a 1/4 second in order to optimize the
    770  * scroll performance while the header is still visible, because re-initializing the
    771  * scroll panes is an intensive process.
    772  */
    773 function delayedReInitScrollbars(delay) {
    774   // If we're scheduled for an update, but have received another resize request
    775   // before the scheduled resize has occured, just ignore the new request
    776   // (and wait for the scheduled one).
    777   if (updateScrollbars && updateFromResize) {
    778     updateFromResize = false;
    779     return;
    780   }
    781 
    782   // We're scheduled for an update and the update request came from this method's setTimeout
    783   if (updateScrollbars && !updateFromResize) {
    784     reInitScrollbars();
    785     updateScrollbars = false;
    786   } else {
    787     updateScrollbars = true;
    788     updateFromResize = false;
    789     setTimeout('delayedReInitScrollbars()',delay);
    790   }
    791 }
    792 
    793 /* Re-initialize the scrollbars to account for changed nav size. */
    794 function reInitScrollbars() {
    795   var pane = $(".scroll-pane").each(function(){
    796     var api = $(this).data('jsp');
    797     if (!api) { setTimeout(reInitScrollbars,300); return;}
    798     api.reinitialise( {verticalGutter:0} );
    799   });
    800   $(".scroll-pane").removeAttr("tabindex"); // get rid of tabindex added by jscroller
    801 }
    802 
    803 
    804 /* Resize the height of the nav panels in the reference,
    805  * and save the new size to a cookie */
    806 function saveNavPanels() {
    807   var basePath = getBaseUri(location.pathname);
    808   var section = basePath.substring(1,basePath.indexOf("/",1));
    809   writeCookie("height", resizePackagesNav.css("height"), section, null);
    810 }
    811 
    812 
    813 
    814 function restoreHeight(packageHeight) {
    815     $("#resize-packages-nav").height(packageHeight);
    816     $("#packages-nav").height(packageHeight);
    817   //  var classesHeight = navHeight - packageHeight;
    818  //   $("#classes-nav").css({height:classesHeight});
    819   //  $("#classes-nav .jspContainer").css({height:classesHeight});
    820 }
    821 
    822 
    823 
    824 /* ######### END RESIZE THE SIDENAV HEIGHT ########## */
    825 
    826 
    827 
    828 
    829 
    830 /** Scroll the jScrollPane to make the currently selected item visible
    831     This is called when the page finished loading. */
    832 function scrollIntoView(nav) {
    833   var $nav = $("#"+nav);
    834   var element = $nav.jScrollPane({/* ...settings... */});
    835   var api = element.data('jsp');
    836 
    837   if ($nav.is(':visible')) {
    838     var $selected = $(".selected", $nav);
    839     if ($selected.length == 0) {
    840       // If no selected item found, exit
    841       return;
    842     }
    843     // get the selected item's offset from its container nav by measuring the item's offset
    844     // relative to the document then subtract the container nav's offset relative to the document
    845     var selectedOffset = $selected.offset().top - $nav.offset().top;
    846     if (selectedOffset > $nav.height() * .8) { // multiply nav height by .8 so we move up the item
    847                                                // if it's more than 80% down the nav
    848       // scroll the item up by an amount equal to 80% the container nav's height
    849       api.scrollTo(0, selectedOffset - ($nav.height() * .8), false);
    850     }
    851   }
    852 }
    853 
    854 
    855 
    856 
    857 
    858 
    859 /* Show popup dialogs */
    860 function showDialog(id) {
    861   $dialog = $("#"+id);
    862   $dialog.prepend('<div class="box-border"><div class="top"> <div class="left"></div> <div class="right"></div></div><div class="bottom"> <div class="left"></div> <div class="right"></div> </div> </div>');
    863   $dialog.wrapInner('<div/>');
    864   $dialog.removeClass("hide");
    865 }
    866 
    867 
    868 
    869 
    870 
    871 /* #########    COOKIES!     ########## */
    872 
    873 function readCookie(cookie) {
    874   var myCookie = cookie_namespace+"_"+cookie+"=";
    875   if (document.cookie) {
    876     var index = document.cookie.indexOf(myCookie);
    877     if (index != -1) {
    878       var valStart = index + myCookie.length;
    879       var valEnd = document.cookie.indexOf(";", valStart);
    880       if (valEnd == -1) {
    881         valEnd = document.cookie.length;
    882       }
    883       var val = document.cookie.substring(valStart, valEnd);
    884       return val;
    885     }
    886   }
    887   return 0;
    888 }
    889 
    890 function writeCookie(cookie, val, section, expiration) {
    891   if (val==undefined) return;
    892   section = section == null ? "_" : "_"+section+"_";
    893   if (expiration == null) {
    894     var date = new Date();
    895     date.setTime(date.getTime()+(10*365*24*60*60*1000)); // default expiration is one week
    896     expiration = date.toGMTString();
    897   }
    898   var cookieValue = cookie_namespace + section + cookie + "=" + val
    899                     + "; expires=" + expiration+"; path=/";
    900   document.cookie = cookieValue;
    901 }
    902 
    903 /* #########     END COOKIES!     ########## */
    904 
    905 
    906 
    907 
    908 
    909 
    910 
    911 
    912 
    913 
    914 
    915 
    916 
    917 
    918 
    919 
    920 
    921 
    922 
    923 /*      MISC LIBRARY FUNCTIONS     */
    924 
    925 
    926 
    927 
    928 
    929 function toggle(obj, slide) {
    930   var ul = $("ul:first", obj);
    931   var li = ul.parent();
    932   if (li.hasClass("closed")) {
    933     if (slide) {
    934       ul.slideDown("fast");
    935     } else {
    936       ul.show();
    937     }
    938     li.removeClass("closed");
    939     li.addClass("open");
    940     $(".toggle-img", li).attr("title", "hide pages");
    941   } else {
    942     ul.slideUp("fast");
    943     li.removeClass("open");
    944     li.addClass("closed");
    945     $(".toggle-img", li).attr("title", "show pages");
    946   }
    947 }
    948 
    949 
    950 function buildToggleLists() {
    951   $(".toggle-list").each(
    952     function(i) {
    953       $("div:first", this).append("<a class='toggle-img' href='#' title='show pages' onClick='toggle(this.parentNode.parentNode, true); return false;'></a>");
    954       $(this).addClass("closed");
    955     });
    956 }
    957 
    958 
    959 
    960 function hideNestedItems(list, toggle) {
    961   $list = $(list);
    962   // hide nested lists
    963   if($list.hasClass('showing')) {
    964     $("li ol", $list).hide('fast');
    965     $list.removeClass('showing');
    966   // show nested lists
    967   } else {
    968     $("li ol", $list).show('fast');
    969     $list.addClass('showing');
    970   }
    971   $(".more,.less",$(toggle)).toggle();
    972 }
    973 
    974 
    975 
    976 
    977 
    978 
    979 
    980 
    981 
    982 
    983 
    984 
    985 
    986 
    987 
    988 
    989 
    990 
    991 
    992 
    993 
    994 
    995 
    996 
    997 
    998 
    999 
   1000 
   1001 /*      REFERENCE NAV SWAP     */
   1002 
   1003 
   1004 function getNavPref() {
   1005   var v = readCookie('reference_nav');
   1006   if (v != NAV_PREF_TREE) {
   1007     v = NAV_PREF_PANELS;
   1008   }
   1009   return v;
   1010 }
   1011 
   1012 function chooseDefaultNav() {
   1013   nav_pref = getNavPref();
   1014   if (nav_pref == NAV_PREF_TREE) {
   1015     $("#nav-panels").toggle();
   1016     $("#panel-link").toggle();
   1017     $("#nav-tree").toggle();
   1018     $("#tree-link").toggle();
   1019   }
   1020 }
   1021 
   1022 function swapNav() {
   1023   if (nav_pref == NAV_PREF_TREE) {
   1024     nav_pref = NAV_PREF_PANELS;
   1025   } else {
   1026     nav_pref = NAV_PREF_TREE;
   1027     init_default_navtree(toRoot);
   1028   }
   1029   var date = new Date();
   1030   date.setTime(date.getTime()+(10*365*24*60*60*1000)); // keep this for 10 years
   1031   writeCookie("nav", nav_pref, "reference", date.toGMTString());
   1032 
   1033   $("#nav-panels").toggle();
   1034   $("#panel-link").toggle();
   1035   $("#nav-tree").toggle();
   1036   $("#tree-link").toggle();
   1037 
   1038   resizeNav();
   1039 
   1040   // Gross nasty hack to make tree view show up upon first swap by setting height manually
   1041   $("#nav-tree .jspContainer:visible")
   1042       .css({'height':$("#nav-tree .jspContainer .jspPane").height() +'px'});
   1043   // Another nasty hack to make the scrollbar appear now that we have height
   1044   resizeNav();
   1045 
   1046   if ($("#nav-tree").is(':visible')) {
   1047     scrollIntoView("nav-tree");
   1048   } else {
   1049     scrollIntoView("packages-nav");
   1050     scrollIntoView("classes-nav");
   1051   }
   1052 }
   1053 
   1054 
   1055 
   1056 /* ############################################ */
   1057 /* ##########     LOCALIZATION     ############ */
   1058 /* ############################################ */
   1059 
   1060 function getBaseUri(uri) {
   1061   var intlUrl = (uri.substring(0,6) == "/intl/");
   1062   if (intlUrl) {
   1063     base = uri.substring(uri.indexOf('intl/')+5,uri.length);
   1064     base = base.substring(base.indexOf('/')+1, base.length);
   1065       //alert("intl, returning base url: /" + base);
   1066     return ("/" + base);
   1067   } else {
   1068       //alert("not intl, returning uri as found.");
   1069     return uri;
   1070   }
   1071 }
   1072 
   1073 function requestAppendHL(uri) {
   1074 //append "?hl=<lang> to an outgoing request (such as to blog)
   1075   var lang = getLangPref();
   1076   if (lang) {
   1077     var q = 'hl=' + lang;
   1078     uri += '?' + q;
   1079     window.location = uri;
   1080     return false;
   1081   } else {
   1082     return true;
   1083   }
   1084 }
   1085 
   1086 
   1087 function changeNavLang(lang) {
   1088   var $links = $("#devdoc-nav,#header,#nav-x,.training-nav-top,.content-footer").find("a["+lang+"-lang]");
   1089   $links.each(function(i){ // for each link with a translation
   1090     var $link = $(this);
   1091     if (lang != "en") { // No need to worry about English, because a language change invokes new request
   1092       // put the desired language from the attribute as the text
   1093       $link.text($link.attr(lang+"-lang"))
   1094     }
   1095   });
   1096 }
   1097 
   1098 function changeLangPref(lang, submit) {
   1099   var date = new Date();
   1100   expires = date.toGMTString(date.setTime(date.getTime()+(10*365*24*60*60*1000)));
   1101   // keep this for 50 years
   1102   //alert("expires: " + expires)
   1103   writeCookie("pref_lang", lang, null, expires);
   1104 
   1105   //  #######  TODO:  Remove this condition once we're stable on devsite #######
   1106   //  This condition is only needed if we still need to support legacy GAE server
   1107   if (devsite) {
   1108     // Switch language when on Devsite server
   1109     if (submit) {
   1110       $("#setlang").submit();
   1111     }
   1112   } else {
   1113     // Switch language when on legacy GAE server
   1114     if (submit) {
   1115       window.location = getBaseUri(location.pathname);
   1116     }
   1117   }
   1118 }
   1119 
   1120 function loadLangPref() {
   1121   var lang = readCookie("pref_lang");
   1122   if (lang != 0) {
   1123     $("#language").find("option[value='"+lang+"']").attr("selected",true);
   1124   }
   1125 }
   1126 
   1127 function getLangPref() {
   1128   var lang = $("#language").find(":selected").attr("value");
   1129   if (!lang) {
   1130     lang = readCookie("pref_lang");
   1131   }
   1132   return (lang != 0) ? lang : 'en';
   1133 }
   1134 
   1135 /* ##########     END LOCALIZATION     ############ */
   1136 
   1137 
   1138 
   1139 
   1140 
   1141 
   1142 /* Used to hide and reveal supplemental content, such as long code samples.
   1143    See the companion CSS in android-developer-docs.css */
   1144 function toggleContent(obj) {
   1145   var div = $(obj).closest(".toggle-content");
   1146   var toggleMe = $(".toggle-content-toggleme:eq(0)",div);
   1147   if (div.hasClass("closed")) { // if it's closed, open it
   1148     toggleMe.slideDown();
   1149     $(".toggle-content-text:eq(0)", obj).toggle();
   1150     div.removeClass("closed").addClass("open");
   1151     $(".toggle-content-img:eq(0)", div).attr("title", "hide").attr("src", toRoot
   1152                   + "assets/images/triangle-opened.png");
   1153   } else { // if it's open, close it
   1154     toggleMe.slideUp('fast', function() {  // Wait until the animation is done before closing arrow
   1155       $(".toggle-content-text:eq(0)", obj).toggle();
   1156       div.removeClass("open").addClass("closed");
   1157       div.find(".toggle-content").removeClass("open").addClass("closed")
   1158               .find(".toggle-content-toggleme").hide();
   1159       $(".toggle-content-img", div).attr("title", "show").attr("src", toRoot
   1160                   + "assets/images/triangle-closed.png");
   1161     });
   1162   }
   1163   return false;
   1164 }
   1165 
   1166 
   1167 /* New version of expandable content */
   1168 function toggleExpandable(link,id) {
   1169   if($(id).is(':visible')) {
   1170     $(id).slideUp();
   1171     $(link).removeClass('expanded');
   1172   } else {
   1173     $(id).slideDown();
   1174     $(link).addClass('expanded');
   1175   }
   1176 }
   1177 
   1178 function hideExpandable(ids) {
   1179   $(ids).slideUp();
   1180   $(ids).prev('h4').find('a.expandable').removeClass('expanded');
   1181 }
   1182 
   1183 
   1184 
   1185 
   1186 
   1187 /*
   1188  *  Slideshow 1.0
   1189  *  Used on /index.html and /develop/index.html for carousel
   1190  *
   1191  *  Sample usage:
   1192  *  HTML -
   1193  *  <div class="slideshow-container">
   1194  *   <a href="" class="slideshow-prev">Prev</a>
   1195  *   <a href="" class="slideshow-next">Next</a>
   1196  *   <ul>
   1197  *       <li class="item"><img src="images/marquee1.jpg"></li>
   1198  *       <li class="item"><img src="images/marquee2.jpg"></li>
   1199  *       <li class="item"><img src="images/marquee3.jpg"></li>
   1200  *       <li class="item"><img src="images/marquee4.jpg"></li>
   1201  *   </ul>
   1202  *  </div>
   1203  *
   1204  *   <script type="text/javascript">
   1205  *   $('.slideshow-container').dacSlideshow({
   1206  *       auto: true,
   1207  *       btnPrev: '.slideshow-prev',
   1208  *       btnNext: '.slideshow-next'
   1209  *   });
   1210  *   </script>
   1211  *
   1212  *  Options:
   1213  *  btnPrev:    optional identifier for previous button
   1214  *  btnNext:    optional identifier for next button
   1215  *  btnPause:   optional identifier for pause button
   1216  *  auto:       whether or not to auto-proceed
   1217  *  speed:      animation speed
   1218  *  autoTime:   time between auto-rotation
   1219  *  easing:     easing function for transition
   1220  *  start:      item to select by default
   1221  *  scroll:     direction to scroll in
   1222  *  pagination: whether or not to include dotted pagination
   1223  *
   1224  */
   1225 
   1226  (function($) {
   1227  $.fn.dacSlideshow = function(o) {
   1228 
   1229      //Options - see above
   1230      o = $.extend({
   1231          btnPrev:   null,
   1232          btnNext:   null,
   1233          btnPause:  null,
   1234          auto:      true,
   1235          speed:     500,
   1236          autoTime:  12000,
   1237          easing:    null,
   1238          start:     0,
   1239          scroll:    1,
   1240          pagination: true
   1241 
   1242      }, o || {});
   1243 
   1244      //Set up a carousel for each
   1245      return this.each(function() {
   1246 
   1247          var running = false;
   1248          var animCss = o.vertical ? "top" : "left";
   1249          var sizeCss = o.vertical ? "height" : "width";
   1250          var div = $(this);
   1251          var ul = $("ul", div);
   1252          var tLi = $("li", ul);
   1253          var tl = tLi.size();
   1254          var timer = null;
   1255 
   1256          var li = $("li", ul);
   1257          var itemLength = li.size();
   1258          var curr = o.start;
   1259 
   1260          li.css({float: o.vertical ? "none" : "left"});
   1261          ul.css({margin: "0", padding: "0", position: "relative", "list-style-type": "none", "z-index": "1"});
   1262          div.css({position: "relative", "z-index": "2", left: "0px"});
   1263 
   1264          var liSize = o.vertical ? height(li) : width(li);
   1265          var ulSize = liSize * itemLength;
   1266          var divSize = liSize;
   1267 
   1268          li.css({width: li.width(), height: li.height()});
   1269          ul.css(sizeCss, ulSize+"px").css(animCss, -(curr*liSize));
   1270 
   1271          div.css(sizeCss, divSize+"px");
   1272 
   1273          //Pagination
   1274          if (o.pagination) {
   1275              var pagination = $("<div class='pagination'></div>");
   1276              var pag_ul = $("<ul></ul>");
   1277              if (tl > 1) {
   1278                for (var i=0;i<tl;i++) {
   1279                     var li = $("<li>"+i+"</li>");
   1280                     pag_ul.append(li);
   1281                     if (i==o.start) li.addClass('active');
   1282                         li.click(function() {
   1283                         go(parseInt($(this).text()));
   1284                     })
   1285                 }
   1286                 pagination.append(pag_ul);
   1287                 div.append(pagination);
   1288              }
   1289          }
   1290 
   1291          //Previous button
   1292          if(o.btnPrev)
   1293              $(o.btnPrev).click(function(e) {
   1294                  e.preventDefault();
   1295                  return go(curr-o.scroll);
   1296              });
   1297 
   1298          //Next button
   1299          if(o.btnNext)
   1300              $(o.btnNext).click(function(e) {
   1301                  e.preventDefault();
   1302                  return go(curr+o.scroll);
   1303              });
   1304 
   1305          //Pause button
   1306          if(o.btnPause)
   1307              $(o.btnPause).click(function(e) {
   1308                  e.preventDefault();
   1309                  if ($(this).hasClass('paused')) {
   1310                      startRotateTimer();
   1311                  } else {
   1312                      pauseRotateTimer();
   1313                  }
   1314              });
   1315 
   1316          //Auto rotation
   1317          if(o.auto) startRotateTimer();
   1318 
   1319          function startRotateTimer() {
   1320              clearInterval(timer);
   1321              timer = setInterval(function() {
   1322                   if (curr == tl-1) {
   1323                     go(0);
   1324                   } else {
   1325                     go(curr+o.scroll);
   1326                   }
   1327               }, o.autoTime);
   1328              $(o.btnPause).removeClass('paused');
   1329          }
   1330 
   1331          function pauseRotateTimer() {
   1332              clearInterval(timer);
   1333              $(o.btnPause).addClass('paused');
   1334          }
   1335 
   1336          //Go to an item
   1337          function go(to) {
   1338              if(!running) {
   1339 
   1340                  if(to<0) {
   1341                     to = itemLength-1;
   1342                  } else if (to>itemLength-1) {
   1343                     to = 0;
   1344                  }
   1345                  curr = to;
   1346 
   1347                  running = true;
   1348 
   1349                  ul.animate(
   1350                      animCss == "left" ? { left: -(curr*liSize) } : { top: -(curr*liSize) } , o.speed, o.easing,
   1351                      function() {
   1352                          running = false;
   1353                      }
   1354                  );
   1355 
   1356                  $(o.btnPrev + "," + o.btnNext).removeClass("disabled");
   1357                  $( (curr-o.scroll<0 && o.btnPrev)
   1358                      ||
   1359                     (curr+o.scroll > itemLength && o.btnNext)
   1360                      ||
   1361                     []
   1362                   ).addClass("disabled");
   1363 
   1364 
   1365                  var nav_items = $('li', pagination);
   1366                  nav_items.removeClass('active');
   1367                  nav_items.eq(to).addClass('active');
   1368 
   1369 
   1370              }
   1371              if(o.auto) startRotateTimer();
   1372              return false;
   1373          };
   1374      });
   1375  };
   1376 
   1377  function css(el, prop) {
   1378      return parseInt($.css(el[0], prop)) || 0;
   1379  };
   1380  function width(el) {
   1381      return  el[0].offsetWidth + css(el, 'marginLeft') + css(el, 'marginRight');
   1382  };
   1383  function height(el) {
   1384      return el[0].offsetHeight + css(el, 'marginTop') + css(el, 'marginBottom');
   1385  };
   1386 
   1387  })(jQuery);
   1388 
   1389 
   1390 /*
   1391  *  dacSlideshow 1.0
   1392  *  Used on develop/index.html for side-sliding tabs
   1393  *
   1394  *  Sample usage:
   1395  *  HTML -
   1396  *  <div class="slideshow-container">
   1397  *   <a href="" class="slideshow-prev">Prev</a>
   1398  *   <a href="" class="slideshow-next">Next</a>
   1399  *   <ul>
   1400  *       <li class="item"><img src="images/marquee1.jpg"></li>
   1401  *       <li class="item"><img src="images/marquee2.jpg"></li>
   1402  *       <li class="item"><img src="images/marquee3.jpg"></li>
   1403  *       <li class="item"><img src="images/marquee4.jpg"></li>
   1404  *   </ul>
   1405  *  </div>
   1406  *
   1407  *   <script type="text/javascript">
   1408  *   $('.slideshow-container').dacSlideshow({
   1409  *       auto: true,
   1410  *       btnPrev: '.slideshow-prev',
   1411  *       btnNext: '.slideshow-next'
   1412  *   });
   1413  *   </script>
   1414  *
   1415  *  Options:
   1416  *  btnPrev:    optional identifier for previous button
   1417  *  btnNext:    optional identifier for next button
   1418  *  auto:       whether or not to auto-proceed
   1419  *  speed:      animation speed
   1420  *  autoTime:   time between auto-rotation
   1421  *  easing:     easing function for transition
   1422  *  start:      item to select by default
   1423  *  scroll:     direction to scroll in
   1424  *  pagination: whether or not to include dotted pagination
   1425  *
   1426  */
   1427  (function($) {
   1428  $.fn.dacTabbedList = function(o) {
   1429 
   1430      //Options - see above
   1431      o = $.extend({
   1432          speed : 250,
   1433          easing: null,
   1434          nav_id: null,
   1435          frame_id: null
   1436      }, o || {});
   1437 
   1438      //Set up a carousel for each
   1439      return this.each(function() {
   1440 
   1441          var curr = 0;
   1442          var running = false;
   1443          var animCss = "margin-left";
   1444          var sizeCss = "width";
   1445          var div = $(this);
   1446 
   1447          var nav = $(o.nav_id, div);
   1448          var nav_li = $("li", nav);
   1449          var nav_size = nav_li.size();
   1450          var frame = div.find(o.frame_id);
   1451          var content_width = $(frame).find('ul').width();
   1452          //Buttons
   1453          $(nav_li).click(function(e) {
   1454            go($(nav_li).index($(this)));
   1455          })
   1456 
   1457          //Go to an item
   1458          function go(to) {
   1459              if(!running) {
   1460                  curr = to;
   1461                  running = true;
   1462 
   1463                  frame.animate({ 'margin-left' : -(curr*content_width) }, o.speed, o.easing,
   1464                      function() {
   1465                          running = false;
   1466                      }
   1467                  );
   1468 
   1469 
   1470                  nav_li.removeClass('active');
   1471                  nav_li.eq(to).addClass('active');
   1472 
   1473 
   1474              }
   1475              return false;
   1476          };
   1477      });
   1478  };
   1479 
   1480  function css(el, prop) {
   1481      return parseInt($.css(el[0], prop)) || 0;
   1482  };
   1483  function width(el) {
   1484      return  el[0].offsetWidth + css(el, 'marginLeft') + css(el, 'marginRight');
   1485  };
   1486  function height(el) {
   1487      return el[0].offsetHeight + css(el, 'marginTop') + css(el, 'marginBottom');
   1488  };
   1489 
   1490  })(jQuery);
   1491 
   1492 
   1493 
   1494 
   1495 
   1496 /* ######################################################## */
   1497 /* ################  SEARCH SUGGESTIONS  ################## */
   1498 /* ######################################################## */
   1499 
   1500 
   1501 
   1502 var gSelectedIndex = -1;  // the index position of currently highlighted suggestion
   1503 var gSelectedColumn = -1;  // which column of suggestion lists is currently focused
   1504 
   1505 var gMatches = new Array();
   1506 var gLastText = "";
   1507 var gInitialized = false;
   1508 var ROW_COUNT_FRAMEWORK = 20;       // max number of results in list
   1509 var gListLength = 0;
   1510 
   1511 
   1512 var gGoogleMatches = new Array();
   1513 var ROW_COUNT_GOOGLE = 15;          // max number of results in list
   1514 var gGoogleListLength = 0;
   1515 
   1516 var gDocsMatches = new Array();
   1517 var ROW_COUNT_DOCS = 100;          // max number of results in list
   1518 var gDocsListLength = 0;
   1519 
   1520 function onSuggestionClick(link) {
   1521   // When user clicks a suggested document, track it
   1522   _gaq.push(['_trackEvent', 'Suggestion Click', 'clicked: ' + $(link).text(),
   1523             'from: ' + $("#search_autocomplete").val()]);
   1524 }
   1525 
   1526 function set_item_selected($li, selected)
   1527 {
   1528     if (selected) {
   1529         $li.attr('class','jd-autocomplete jd-selected');
   1530     } else {
   1531         $li.attr('class','jd-autocomplete');
   1532     }
   1533 }
   1534 
   1535 function set_item_values(toroot, $li, match)
   1536 {
   1537     var $link = $('a',$li);
   1538     $link.html(match.__hilabel || match.label);
   1539     $link.attr('href',toroot + match.link);
   1540 }
   1541 
   1542 function new_suggestion($list) {
   1543     var $li = $("<li class='jd-autocomplete'></li>");
   1544     $list.append($li);
   1545 
   1546     $li.mousedown(function() {
   1547         window.location = this.firstChild.getAttribute("href");
   1548     });
   1549     $li.mouseover(function() {
   1550         $('.search_filtered_wrapper li').removeClass('jd-selected');
   1551         $(this).addClass('jd-selected');
   1552         gSelectedColumn = $(".search_filtered:visible").index($(this).closest('.search_filtered'));
   1553         gSelectedIndex = $("li", $(".search_filtered:visible")[gSelectedColumn]).index(this);
   1554     });
   1555     $li.append("<a onclick='onSuggestionClick(this)'></a>");
   1556     $li.attr('class','show-item');
   1557     return $li;
   1558 }
   1559 
   1560 function sync_selection_table(toroot)
   1561 {
   1562     var $li; //list item jquery object
   1563     var i; //list item iterator
   1564 
   1565     // if there are NO results at all, hide all columns
   1566     if (!(gMatches.length > 0) && !(gGoogleMatches.length > 0) && !(gDocsMatches.length > 0)) {
   1567         $('.suggest-card').hide(300);
   1568         return;
   1569     }
   1570 
   1571     // if there are api results
   1572     if ((gMatches.length > 0) || (gGoogleMatches.length > 0)) {
   1573       // reveal suggestion list
   1574       $('.suggest-card.dummy').show();
   1575       $('.suggest-card.reference').show();
   1576       var listIndex = 0; // list index position
   1577 
   1578       // reset the lists
   1579       $(".search_filtered_wrapper.reference li").remove();
   1580 
   1581       // ########### ANDROID RESULTS #############
   1582       if (gMatches.length > 0) {
   1583 
   1584           // determine android results to show
   1585           gListLength = gMatches.length < ROW_COUNT_FRAMEWORK ?
   1586                         gMatches.length : ROW_COUNT_FRAMEWORK;
   1587           for (i=0; i<gListLength; i++) {
   1588               var $li = new_suggestion($(".suggest-card.reference ul"));
   1589               set_item_values(toroot, $li, gMatches[i]);
   1590               set_item_selected($li, i == gSelectedIndex);
   1591           }
   1592       }
   1593 
   1594       // ########### GOOGLE RESULTS #############
   1595       if (gGoogleMatches.length > 0) {
   1596           // show header for list
   1597           $(".suggest-card.reference ul").append("<li class='header'>in Google Services:</li>");
   1598 
   1599           // determine google results to show
   1600           gGoogleListLength = gGoogleMatches.length < ROW_COUNT_GOOGLE ? gGoogleMatches.length : ROW_COUNT_GOOGLE;
   1601           for (i=0; i<gGoogleListLength; i++) {
   1602               var $li = new_suggestion($(".suggest-card.reference ul"));
   1603               set_item_values(toroot, $li, gGoogleMatches[i]);
   1604               set_item_selected($li, i == gSelectedIndex);
   1605           }
   1606       }
   1607     } else {
   1608       $('.suggest-card.reference').hide();
   1609       $('.suggest-card.dummy').hide();
   1610     }
   1611 
   1612     // ########### JD DOC RESULTS #############
   1613     if (gDocsMatches.length > 0) {
   1614         // reset the lists
   1615         $(".search_filtered_wrapper.docs li").remove();
   1616 
   1617         // determine google results to show
   1618         gDocsListLength = gDocsMatches.length < ROW_COUNT_DOCS ? gDocsMatches.length : ROW_COUNT_DOCS;
   1619         for (i=0; i<gDocsListLength; i++) {
   1620             var sugg = gDocsMatches[i];
   1621             var $li;
   1622             if (sugg.type == "design") {
   1623                 $li = new_suggestion($(".suggest-card.design ul"));
   1624             } else
   1625             if (sugg.type == "distribute") {
   1626                 $li = new_suggestion($(".suggest-card.distribute ul"));
   1627             } else
   1628             if (sugg.type == "training") {
   1629                 $li = new_suggestion($(".suggest-card.develop .child-card.training"));
   1630             } else
   1631             if (sugg.type == "guide"||"google") {
   1632                 $li = new_suggestion($(".suggest-card.develop .child-card.guides"));
   1633             } else {
   1634               continue;
   1635             }
   1636 
   1637             set_item_values(toroot, $li, sugg);
   1638             set_item_selected($li, i == gSelectedIndex);
   1639         }
   1640 
   1641         // add heading and show or hide card
   1642         if ($(".suggest-card.design li").length > 0) {
   1643           $(".suggest-card.design ul").prepend("<li class='header'>Design:</li>");
   1644           $(".suggest-card.design").show(300);
   1645         } else {
   1646           $('.suggest-card.design').hide(300);
   1647         }
   1648         if ($(".suggest-card.distribute li").length > 0) {
   1649           $(".suggest-card.distribute ul").prepend("<li class='header'>Distribute:</li>");
   1650           $(".suggest-card.distribute").show(300);
   1651         } else {
   1652           $('.suggest-card.distribute').hide(300);
   1653         }
   1654         if ($(".child-card.guides li").length > 0) {
   1655           $(".child-card.guides").prepend("<li class='header'>Guides:</li>");
   1656           $(".child-card.guides li").appendTo(".suggest-card.develop ul");
   1657         }
   1658         if ($(".child-card.training li").length > 0) {
   1659           $(".child-card.training").prepend("<li class='header'>Training:</li>");
   1660           $(".child-card.training li").appendTo(".suggest-card.develop ul");
   1661         }
   1662 
   1663         if ($(".suggest-card.develop li").length > 0) {
   1664           $(".suggest-card.develop").show(300);
   1665         } else {
   1666           $('.suggest-card.develop').hide(300);
   1667         }
   1668 
   1669     } else {
   1670       $('.search_filtered_wrapper.docs .suggest-card:not(.dummy)').hide(300);
   1671     }
   1672 }
   1673 
   1674 /** Called by the search input's onkeydown and onkeyup events.
   1675   * Handles navigation with keyboard arrows, Enter key to invoke search,
   1676   * otherwise invokes search suggestions on key-up event.
   1677   * @param e       The JS event
   1678   * @param kd      True if the event is key-down
   1679   * @param toroot  A string for the site's root path
   1680   * @returns       True if the event should bubble up
   1681   */
   1682 function search_changed(e, kd, toroot)
   1683 {
   1684     var search = document.getElementById("search_autocomplete");
   1685     var text = search.value.replace(/(^ +)|( +$)/g, '');
   1686     // get the ul hosting the currently selected item
   1687     gSelectedColumn = gSelectedColumn >= 0 ? gSelectedColumn :  0;
   1688     var $columns = $(".search_filtered_wrapper").find(".search_filtered:visible");
   1689     var $selectedUl = $columns[gSelectedColumn];
   1690 
   1691     // show/hide the close button
   1692     if (text != '') {
   1693         $(".search .close").removeClass("hide");
   1694     } else {
   1695         $(".search .close").addClass("hide");
   1696     }
   1697     // 27 = esc
   1698     if (e.keyCode == 27) {
   1699         // close all search results
   1700         if (kd) $('.search .close').trigger('click');
   1701         return true;
   1702     }
   1703     // 13 = enter
   1704     else if (e.keyCode == 13) {
   1705         if (gSelectedIndex < 0) {
   1706             $('.suggest-card').hide();
   1707             if ($("#searchResults").is(":hidden") && (search.value != "")) {
   1708               // if results aren't showing (and text not empty), return true to allow search to execute
   1709               return true;
   1710             } else {
   1711               // otherwise, results are already showing, so allow ajax to auto refresh the results
   1712               // and ignore this Enter press to avoid the reload.
   1713               return false;
   1714             }
   1715         } else if (kd && gSelectedIndex >= 0) {
   1716             // click the link corresponding to selected item
   1717             $("a",$("li",$selectedUl)[gSelectedIndex]).get()[0].click();
   1718             return false;
   1719         }
   1720     }
   1721     // Stop here if Google results are showing
   1722     else if ($("#searchResults").is(":visible")) {
   1723         return true;
   1724     }
   1725     // 38 UP ARROW
   1726     else if (kd && (e.keyCode == 38)) {
   1727         // if the next item is a header, skip it
   1728         if ($($("li", $selectedUl)[gSelectedIndex-1]).hasClass("header")) {
   1729             gSelectedIndex--;
   1730         }
   1731         if (gSelectedIndex >= 0) {
   1732             $('li', $selectedUl).removeClass('jd-selected');
   1733             gSelectedIndex--;
   1734             $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
   1735             // If user reaches top, reset selected column
   1736             if (gSelectedIndex < 0) {
   1737               gSelectedColumn = -1;
   1738             }
   1739         }
   1740         return false;
   1741     }
   1742     // 40 DOWN ARROW
   1743     else if (kd && (e.keyCode == 40)) {
   1744         // if the next item is a header, skip it
   1745         if ($($("li", $selectedUl)[gSelectedIndex+1]).hasClass("header")) {
   1746             gSelectedIndex++;
   1747         }
   1748         if ((gSelectedIndex < $("li", $selectedUl).length-1) ||
   1749                         ($($("li", $selectedUl)[gSelectedIndex+1]).hasClass("header"))) {
   1750             $('li', $selectedUl).removeClass('jd-selected');
   1751             gSelectedIndex++;
   1752             $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
   1753         }
   1754         return false;
   1755     }
   1756     // Consider left/right arrow navigation
   1757     // NOTE: Order of suggest columns are reverse order (index position 0 is on right)
   1758     else if (kd && $columns.length > 1 && gSelectedColumn >= 0) {
   1759       // 37 LEFT ARROW
   1760       // go left only if current column is not left-most column (last column)
   1761       if (e.keyCode == 37 && gSelectedColumn < $columns.length - 1) {
   1762         $('li', $selectedUl).removeClass('jd-selected');
   1763         gSelectedColumn++;
   1764         $selectedUl = $columns[gSelectedColumn];
   1765         // keep or reset the selected item to last item as appropriate
   1766         gSelectedIndex = gSelectedIndex >
   1767                 $("li", $selectedUl).length-1 ?
   1768                 $("li", $selectedUl).length-1 : gSelectedIndex;
   1769         // if the corresponding item is a header, move down
   1770         if ($($("li", $selectedUl)[gSelectedIndex]).hasClass("header")) {
   1771           gSelectedIndex++;
   1772         }
   1773         // set item selected
   1774         $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
   1775         return false;
   1776       }
   1777       // 39 RIGHT ARROW
   1778       // go right only if current column is not the right-most column (first column)
   1779       else if (e.keyCode == 39 && gSelectedColumn > 0) {
   1780         $('li', $selectedUl).removeClass('jd-selected');
   1781         gSelectedColumn--;
   1782         $selectedUl = $columns[gSelectedColumn];
   1783         // keep or reset the selected item to last item as appropriate
   1784         gSelectedIndex = gSelectedIndex >
   1785                 $("li", $selectedUl).length-1 ?
   1786                 $("li", $selectedUl).length-1 : gSelectedIndex;
   1787         // if the corresponding item is a header, move down
   1788         if ($($("li", $selectedUl)[gSelectedIndex]).hasClass("header")) {
   1789           gSelectedIndex++;
   1790         }
   1791         // set item selected
   1792         $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
   1793         return false;
   1794       }
   1795     }
   1796 
   1797     // if key-up event and not arrow down/up,
   1798     // read the search query and add suggestsions to gMatches
   1799     else if (!kd && (e.keyCode != 40)
   1800                  && (e.keyCode != 38)
   1801                  && (e.keyCode != 37)
   1802                  && (e.keyCode != 39)) {
   1803         gSelectedIndex = -1;
   1804         gMatches = new Array();
   1805         matchedCount = 0;
   1806         gGoogleMatches = new Array();
   1807         matchedCountGoogle = 0;
   1808         gDocsMatches = new Array();
   1809         matchedCountDocs = 0;
   1810 
   1811         // Search for Android matches
   1812         for (var i=0; i<DATA.length; i++) {
   1813             var s = DATA[i];
   1814             if (text.length != 0 &&
   1815                   s.label.toLowerCase().indexOf(text.toLowerCase()) != -1) {
   1816                 gMatches[matchedCount] = s;
   1817                 matchedCount++;
   1818             }
   1819         }
   1820         rank_autocomplete_api_results(text, gMatches);
   1821         for (var i=0; i<gMatches.length; i++) {
   1822             var s = gMatches[i];
   1823         }
   1824 
   1825 
   1826         // Search for Google matches
   1827         for (var i=0; i<GOOGLE_DATA.length; i++) {
   1828             var s = GOOGLE_DATA[i];
   1829             if (text.length != 0 &&
   1830                   s.label.toLowerCase().indexOf(text.toLowerCase()) != -1) {
   1831                 gGoogleMatches[matchedCountGoogle] = s;
   1832                 matchedCountGoogle++;
   1833             }
   1834         }
   1835         rank_autocomplete_api_results(text, gGoogleMatches);
   1836         for (var i=0; i<gGoogleMatches.length; i++) {
   1837             var s = gGoogleMatches[i];
   1838         }
   1839 
   1840         highlight_autocomplete_result_labels(text);
   1841 
   1842 
   1843 
   1844         // Search for JD docs
   1845         if (text.length >= 3) {
   1846           for (var i=0; i<JD_DATA.length; i++) {
   1847             // Regex to match only the beginning of a word
   1848             var textRegex = new RegExp("\\b" + text.toLowerCase(), "g");
   1849             // current search comparison, with counters for tag and title,
   1850             // used later to improve ranking
   1851             var s = JD_DATA[i];
   1852             s.matched_tag = 0;
   1853             s.matched_title = 0;
   1854             var matched = false;
   1855 
   1856             // Check if query matches any tags; work backwards toward 1 to assist ranking
   1857             for (var j = s.tags.length - 1; j >= 0; j--) {
   1858               // it matches a tag
   1859               if (s.tags[j].toLowerCase().match(textRegex)) {
   1860                 matched = true;
   1861                 s.matched_tag = j + 1; // add 1 to index position
   1862               }
   1863             }
   1864             // Don't consider doc title for lessons (only for class landing pages)
   1865             // ...it is not a training lesson (or is but has matched a tag)
   1866             if (!(s.type == "training" && s.link.indexOf("index.html") == -1) || matched) {
   1867               // it matches the doc title
   1868               if (s.label.toLowerCase().match(textRegex)) {
   1869                 matched = true;
   1870                 s.matched_title = 1;
   1871               }
   1872             }
   1873             if (matched) {
   1874               gDocsMatches[matchedCountDocs] = s;
   1875               matchedCountDocs++;
   1876             }
   1877           }
   1878           rank_autocomplete_doc_results(text, gDocsMatches);
   1879         }
   1880 
   1881         // draw the suggestions
   1882         sync_selection_table(toroot);
   1883         return true; // allow the event to bubble up to the search api
   1884     }
   1885 }
   1886 
   1887 /* Order the jd doc result list based on match quality */
   1888 function rank_autocomplete_doc_results(query, matches) {
   1889     query = query || '';
   1890     if (!matches || !matches.length)
   1891       return;
   1892 
   1893     var _resultScoreFn = function(match) {
   1894         var score = 1.0;
   1895 
   1896         // if the query matched a tag
   1897         if (match.matched_tag > 0) {
   1898           // multiply score by factor relative to position in tags list (max of 3)
   1899           score *= 3 / match.matched_tag;
   1900 
   1901           // if it also matched the title
   1902           if (match.matched_title > 0) {
   1903             score *= 2;
   1904           }
   1905         } else if (match.matched_title > 0) {
   1906           score *= 3;
   1907         }
   1908 
   1909         return score;
   1910     };
   1911 
   1912     for (var i=0; i<matches.length; i++) {
   1913         matches[i].__resultScore = _resultScoreFn(matches[i]);
   1914     }
   1915 
   1916     matches.sort(function(a,b){
   1917         var n = b.__resultScore - a.__resultScore;
   1918         if (n == 0) // lexicographical sort if scores are the same
   1919             n = (a.label < b.label) ? -1 : 1;
   1920         return n;
   1921     });
   1922 }
   1923 
   1924 /* Order the result list based on match quality */
   1925 function rank_autocomplete_api_results(query, matches) {
   1926     query = query || '';
   1927     if (!matches || !matches.length)
   1928       return;
   1929 
   1930     // helper function that gets the last occurence index of the given regex
   1931     // in the given string, or -1 if not found
   1932     var _lastSearch = function(s, re) {
   1933       if (s == '')
   1934         return -1;
   1935       var l = -1;
   1936       var tmp;
   1937       while ((tmp = s.search(re)) >= 0) {
   1938         if (l < 0) l = 0;
   1939         l += tmp;
   1940         s = s.substr(tmp + 1);
   1941       }
   1942       return l;
   1943     };
   1944 
   1945     // helper function that counts the occurrences of a given character in
   1946     // a given string
   1947     var _countChar = function(s, c) {
   1948       var n = 0;
   1949       for (var i=0; i<s.length; i++)
   1950         if (s.charAt(i) == c) ++n;
   1951       return n;
   1952     };
   1953 
   1954     var queryLower = query.toLowerCase();
   1955     var queryAlnum = (queryLower.match(/\w+/) || [''])[0];
   1956     var partPrefixAlnumRE = new RegExp('\\b' + queryAlnum);
   1957     var partExactAlnumRE = new RegExp('\\b' + queryAlnum + '\\b');
   1958 
   1959     var _resultScoreFn = function(result) {
   1960         // scores are calculated based on exact and prefix matches,
   1961         // and then number of path separators (dots) from the last
   1962         // match (i.e. favoring classes and deep package names)
   1963         var score = 1.0;
   1964         var labelLower = result.label.toLowerCase();
   1965         var t;
   1966         t = _lastSearch(labelLower, partExactAlnumRE);
   1967         if (t >= 0) {
   1968             // exact part match
   1969             var partsAfter = _countChar(labelLower.substr(t + 1), '.');
   1970             score *= 200 / (partsAfter + 1);
   1971         } else {
   1972             t = _lastSearch(labelLower, partPrefixAlnumRE);
   1973             if (t >= 0) {
   1974                 // part prefix match
   1975                 var partsAfter = _countChar(labelLower.substr(t + 1), '.');
   1976                 score *= 20 / (partsAfter + 1);
   1977             }
   1978         }
   1979 
   1980         return score;
   1981     };
   1982 
   1983     for (var i=0; i<matches.length; i++) {
   1984         // if the API is deprecated, default score is 0; otherwise, perform scoring
   1985         if (matches[i].deprecated == "true") {
   1986           matches[i].__resultScore = 0;
   1987         } else {
   1988           matches[i].__resultScore = _resultScoreFn(matches[i]);
   1989         }
   1990     }
   1991 
   1992     matches.sort(function(a,b){
   1993         var n = b.__resultScore - a.__resultScore;
   1994         if (n == 0) // lexicographical sort if scores are the same
   1995             n = (a.label < b.label) ? -1 : 1;
   1996         return n;
   1997     });
   1998 }
   1999 
   2000 /* Add emphasis to part of string that matches query */
   2001 function highlight_autocomplete_result_labels(query) {
   2002     query = query || '';
   2003     if ((!gMatches || !gMatches.length) && (!gGoogleMatches || !gGoogleMatches.length))
   2004       return;
   2005 
   2006     var queryLower = query.toLowerCase();
   2007     var queryAlnumDot = (queryLower.match(/[\w\.]+/) || [''])[0];
   2008     var queryRE = new RegExp(
   2009         '(' + queryAlnumDot.replace(/\./g, '\\.') + ')', 'ig');
   2010     for (var i=0; i<gMatches.length; i++) {
   2011         gMatches[i].__hilabel = gMatches[i].label.replace(
   2012             queryRE, '<b>$1</b>');
   2013     }
   2014     for (var i=0; i<gGoogleMatches.length; i++) {
   2015         gGoogleMatches[i].__hilabel = gGoogleMatches[i].label.replace(
   2016             queryRE, '<b>$1</b>');
   2017     }
   2018 }
   2019 
   2020 function search_focus_changed(obj, focused)
   2021 {
   2022     if (!focused) {
   2023         if(obj.value == ""){
   2024           $(".search .close").addClass("hide");
   2025         }
   2026         $(".suggest-card").hide();
   2027     }
   2028 }
   2029 
   2030 function submit_search() {
   2031   var query = document.getElementById('search_autocomplete').value;
   2032   location.hash = 'q=' + query;
   2033   loadSearchResults();
   2034   $("#searchResults").slideDown('slow');
   2035   return false;
   2036 }
   2037 
   2038 
   2039 function hideResults() {
   2040   $("#searchResults").slideUp();
   2041   $(".search .close").addClass("hide");
   2042   location.hash = '';
   2043 
   2044   $("#search_autocomplete").val("").blur();
   2045 
   2046   // reset the ajax search callback to nothing, so results don't appear unless ENTER
   2047   searchControl.setSearchStartingCallback(this, function(control, searcher, query) {});
   2048 
   2049   // forcefully regain key-up event control (previously jacked by search api)
   2050   $("#search_autocomplete").keyup(function(event) {
   2051     return search_changed(event, false, toRoot);
   2052   });
   2053 
   2054   return false;
   2055 }
   2056 
   2057 
   2058 
   2059 /* ########################################################## */
   2060 /* ################  CUSTOM SEARCH ENGINE  ################## */
   2061 /* ########################################################## */
   2062 
   2063 var searchControl;
   2064 google.load('search', '1', {"callback" : function() {
   2065             searchControl = new google.search.SearchControl();
   2066           } });
   2067 
   2068 function loadSearchResults() {
   2069   document.getElementById("search_autocomplete").style.color = "#000";
   2070 
   2071   searchControl = new google.search.SearchControl();
   2072 
   2073   // use our existing search form and use tabs when multiple searchers are used
   2074   drawOptions = new google.search.DrawOptions();
   2075   drawOptions.setDrawMode(google.search.SearchControl.DRAW_MODE_TABBED);
   2076   drawOptions.setInput(document.getElementById("search_autocomplete"));
   2077 
   2078   // configure search result options
   2079   searchOptions = new google.search.SearcherOptions();
   2080   searchOptions.setExpandMode(GSearchControl.EXPAND_MODE_OPEN);
   2081 
   2082   // configure each of the searchers, for each tab
   2083   devSiteSearcher = new google.search.WebSearch();
   2084   devSiteSearcher.setUserDefinedLabel("All");
   2085   devSiteSearcher.setSiteRestriction("001482626316274216503:zu90b7s047u");
   2086 
   2087   designSearcher = new google.search.WebSearch();
   2088   designSearcher.setUserDefinedLabel("Design");
   2089   designSearcher.setSiteRestriction("http://developer.android.com/design/");
   2090 
   2091   trainingSearcher = new google.search.WebSearch();
   2092   trainingSearcher.setUserDefinedLabel("Training");
   2093   trainingSearcher.setSiteRestriction("http://developer.android.com/training/");
   2094 
   2095   guidesSearcher = new google.search.WebSearch();
   2096   guidesSearcher.setUserDefinedLabel("Guides");
   2097   guidesSearcher.setSiteRestriction("http://developer.android.com/guide/");
   2098 
   2099   referenceSearcher = new google.search.WebSearch();
   2100   referenceSearcher.setUserDefinedLabel("Reference");
   2101   referenceSearcher.setSiteRestriction("http://developer.android.com/reference/");
   2102 
   2103   googleSearcher = new google.search.WebSearch();
   2104   googleSearcher.setUserDefinedLabel("Google Services");
   2105   googleSearcher.setSiteRestriction("http://developer.android.com/google/");
   2106 
   2107   blogSearcher = new google.search.WebSearch();
   2108   blogSearcher.setUserDefinedLabel("Blog");
   2109   blogSearcher.setSiteRestriction("http://android-developers.blogspot.com");
   2110 
   2111   // add each searcher to the search control
   2112   searchControl.addSearcher(devSiteSearcher, searchOptions);
   2113   searchControl.addSearcher(designSearcher, searchOptions);
   2114   searchControl.addSearcher(trainingSearcher, searchOptions);
   2115   searchControl.addSearcher(guidesSearcher, searchOptions);
   2116   searchControl.addSearcher(referenceSearcher, searchOptions);
   2117   searchControl.addSearcher(googleSearcher, searchOptions);
   2118   searchControl.addSearcher(blogSearcher, searchOptions);
   2119 
   2120   // configure result options
   2121   searchControl.setResultSetSize(google.search.Search.LARGE_RESULTSET);
   2122   searchControl.setLinkTarget(google.search.Search.LINK_TARGET_SELF);
   2123   searchControl.setTimeoutInterval(google.search.SearchControl.TIMEOUT_SHORT);
   2124   searchControl.setNoResultsString(google.search.SearchControl.NO_RESULTS_DEFAULT_STRING);
   2125 
   2126   // upon ajax search, refresh the url and search title
   2127   searchControl.setSearchStartingCallback(this, function(control, searcher, query) {
   2128     updateResultTitle(query);
   2129     var query = document.getElementById('search_autocomplete').value;
   2130     location.hash = 'q=' + query;
   2131   });
   2132 
   2133   // once search results load, set up click listeners
   2134   searchControl.setSearchCompleteCallback(this, function(control, searcher, query) {
   2135     addResultClickListeners();
   2136   });
   2137 
   2138   // draw the search results box
   2139   searchControl.draw(document.getElementById("leftSearchControl"), drawOptions);
   2140 
   2141   // get query and execute the search
   2142   searchControl.execute(decodeURI(getQuery(location.hash)));
   2143 
   2144   document.getElementById("search_autocomplete").focus();
   2145   addTabListeners();
   2146 }
   2147 // End of loadSearchResults
   2148 
   2149 
   2150 google.setOnLoadCallback(function(){
   2151   if (location.hash.indexOf("q=") == -1) {
   2152     // if there's no query in the url, don't search and make sure results are hidden
   2153     $('#searchResults').hide();
   2154     return;
   2155   } else {
   2156     // first time loading search results for this page
   2157     $('#searchResults').slideDown('slow');
   2158     $(".search .close").removeClass("hide");
   2159     loadSearchResults();
   2160   }
   2161 }, true);
   2162 
   2163 // when an event on the browser history occurs (back, forward, load) requery hash and do search
   2164 $(window).hashchange( function(){
   2165   // Exit if the hash isn't a search query or there's an error in the query
   2166   if ((location.hash.indexOf("q=") == -1) || (query == "undefined")) {
   2167     // If the results pane is open, close it.
   2168     if (!$("#searchResults").is(":hidden")) {
   2169       hideResults();
   2170     }
   2171     return;
   2172   }
   2173 
   2174   // Otherwise, we have a search to do
   2175   var query = decodeURI(getQuery(location.hash));
   2176   searchControl.execute(query);
   2177   $('#searchResults').slideDown('slow');
   2178   $("#search_autocomplete").focus();
   2179   $(".search .close").removeClass("hide");
   2180 
   2181   updateResultTitle(query);
   2182 });
   2183 
   2184 function updateResultTitle(query) {
   2185   $("#searchTitle").html("Results for <em>" + escapeHTML(query) + "</em>");
   2186 }
   2187 
   2188 // forcefully regain key-up event control (previously jacked by search api)
   2189 $("#search_autocomplete").keyup(function(event) {
   2190   return search_changed(event, false, toRoot);
   2191 });
   2192 
   2193 // add event listeners to each tab so we can track the browser history
   2194 function addTabListeners() {
   2195   var tabHeaders = $(".gsc-tabHeader");
   2196   for (var i = 0; i < tabHeaders.length; i++) {
   2197     $(tabHeaders[i]).attr("id",i).click(function() {
   2198     /*
   2199       // make a copy of the page numbers for the search left pane
   2200       setTimeout(function() {
   2201         // remove any residual page numbers
   2202         $('#searchResults .gsc-tabsArea .gsc-cursor-box.gs-bidi-start-align').remove();
   2203         // move the page numbers to the left position; make a clone,
   2204         // because the element is drawn to the DOM only once
   2205         // and because we're going to remove it (previous line),
   2206         // we need it to be available to move again as the user navigates
   2207         $('#searchResults .gsc-webResult .gsc-cursor-box.gs-bidi-start-align:visible')
   2208                         .clone().appendTo('#searchResults .gsc-tabsArea');
   2209         }, 200);
   2210       */
   2211     });
   2212   }
   2213   setTimeout(function(){$(tabHeaders[0]).click()},200);
   2214 }
   2215 
   2216 // add analytics tracking events to each result link
   2217 function addResultClickListeners() {
   2218   $("#searchResults a.gs-title").each(function(index, link) {
   2219     // When user clicks enter for Google search results, track it
   2220     $(link).click(function() {
   2221       _gaq.push(['_trackEvent', 'Google Click', 'clicked: ' + $(this).text(),
   2222                 'from: ' + $("#search_autocomplete").val()]);
   2223     });
   2224   });
   2225 }
   2226 
   2227 
   2228 function getQuery(hash) {
   2229   var queryParts = hash.split('=');
   2230   return queryParts[1];
   2231 }
   2232 
   2233 /* returns the given string with all HTML brackets converted to entities
   2234     TODO: move this to the site's JS library */
   2235 function escapeHTML(string) {
   2236   return string.replace(/</g,"&lt;")
   2237                 .replace(/>/g,"&gt;");
   2238 }
   2239 
   2240 
   2241 
   2242 
   2243 
   2244 
   2245 
   2246 /* ######################################################## */
   2247 /* #################  JAVADOC REFERENCE ################### */
   2248 /* ######################################################## */
   2249 
   2250 /* Initialize some droiddoc stuff, but only if we're in the reference */
   2251 if (location.pathname.indexOf("/reference") == 0) {
   2252   if(!(location.pathname.indexOf("/reference-gms/packages.html") == 0)
   2253     && !(location.pathname.indexOf("/reference-gcm/packages.html") == 0)
   2254     && !(location.pathname.indexOf("/reference/com/google") == 0)) {
   2255     $(document).ready(function() {
   2256       // init available apis based on user pref
   2257       changeApiLevel();
   2258       initSidenavHeightResize()
   2259       });
   2260   }
   2261 }
   2262 
   2263 var API_LEVEL_COOKIE = "api_level";
   2264 var minLevel = 1;
   2265 var maxLevel = 1;
   2266 
   2267 /******* SIDENAV DIMENSIONS ************/
   2268 
   2269   function initSidenavHeightResize() {
   2270     // Change the drag bar size to nicely fit the scrollbar positions
   2271     var $dragBar = $(".ui-resizable-s");
   2272     $dragBar.css({'width': $dragBar.parent().width() - 5 + "px"});
   2273 
   2274     $( "#resize-packages-nav" ).resizable({
   2275       containment: "#nav-panels",
   2276       handles: "s",
   2277       alsoResize: "#packages-nav",
   2278       resize: function(event, ui) { resizeNav(); }, /* resize the nav while dragging */
   2279       stop: function(event, ui) { saveNavPanels(); } /* once stopped, save the sizes to cookie  */
   2280       });
   2281 
   2282   }
   2283 
   2284 function updateSidenavFixedWidth() {
   2285   if (!navBarIsFixed) return;
   2286   $('#devdoc-nav').css({
   2287     'width' : $('#side-nav').css('width'),
   2288     'margin' : $('#side-nav').css('margin')
   2289   });
   2290   $('#devdoc-nav a.totop').css({'display':'block','width':$("#nav").innerWidth()+'px'});
   2291 
   2292   initSidenavHeightResize();
   2293 }
   2294 
   2295 function updateSidenavFullscreenWidth() {
   2296   if (!navBarIsFixed) return;
   2297   $('#devdoc-nav').css({
   2298     'width' : $('#side-nav').css('width'),
   2299     'margin' : $('#side-nav').css('margin')
   2300   });
   2301   $('#devdoc-nav .totop').css({'left': 'inherit'});
   2302 
   2303   initSidenavHeightResize();
   2304 }
   2305 
   2306 function buildApiLevelSelector() {
   2307   maxLevel = SINCE_DATA.length;
   2308   var userApiLevel = parseInt(readCookie(API_LEVEL_COOKIE));
   2309   userApiLevel = userApiLevel == 0 ? maxLevel : userApiLevel; // If there's no cookie (zero), use the max by default
   2310 
   2311   minLevel = parseInt($("#doc-api-level").attr("class"));
   2312   // Handle provisional api levels; the provisional level will always be the highest possible level
   2313   // Provisional api levels will also have a length; other stuff that's just missing a level won't,
   2314   // so leave those kinds of entities at the default level of 1 (for example, the R.styleable class)
   2315   if (isNaN(minLevel) && minLevel.length) {
   2316     minLevel = maxLevel;
   2317   }
   2318   var select = $("#apiLevelSelector").html("").change(changeApiLevel);
   2319   for (var i = maxLevel-1; i >= 0; i--) {
   2320     var option = $("<option />").attr("value",""+SINCE_DATA[i]).append(""+SINCE_DATA[i]);
   2321   //  if (SINCE_DATA[i] < minLevel) option.addClass("absent"); // always false for strings (codenames)
   2322     select.append(option);
   2323   }
   2324 
   2325   // get the DOM element and use setAttribute cuz IE6 fails when using jquery .attr('selected',true)
   2326   var selectedLevelItem = $("#apiLevelSelector option[value='"+userApiLevel+"']").get(0);
   2327   selectedLevelItem.setAttribute('selected',true);
   2328 }
   2329 
   2330 function changeApiLevel() {
   2331   maxLevel = SINCE_DATA.length;
   2332   var selectedLevel = maxLevel;
   2333 
   2334   selectedLevel = parseInt($("#apiLevelSelector option:selected").val());
   2335   toggleVisisbleApis(selectedLevel, "body");
   2336 
   2337   var date = new Date();
   2338   date.setTime(date.getTime()+(10*365*24*60*60*1000)); // keep this for 10 years
   2339   var expiration = date.toGMTString();
   2340   writeCookie(API_LEVEL_COOKIE, selectedLevel, null, expiration);
   2341 
   2342   if (selectedLevel < minLevel) {
   2343     var thing = ($("#jd-header").html().indexOf("package") != -1) ? "package" : "class";
   2344     $("#naMessage").show().html("<div><p><strong>This " + thing
   2345               + " requires API level " + minLevel + " or higher.</strong></p>"
   2346               + "<p>This document is hidden because your selected API level for the documentation is "
   2347               + selectedLevel + ". You can change the documentation API level with the selector "
   2348               + "above the left navigation.</p>"
   2349               + "<p>For more information about specifying the API level your app requires, "
   2350               + "read <a href='" + toRoot + "training/basics/supporting-devices/platforms.html'"
   2351               + ">Supporting Different Platform Versions</a>.</p>"
   2352               + "<input type='button' value='OK, make this page visible' "
   2353               + "title='Change the API level to " + minLevel + "' "
   2354               + "onclick='$(\"#apiLevelSelector\").val(\"" + minLevel + "\");changeApiLevel();' />"
   2355               + "</div>");
   2356   } else {
   2357     $("#naMessage").hide();
   2358   }
   2359 }
   2360 
   2361 function toggleVisisbleApis(selectedLevel, context) {
   2362   var apis = $(".api",context);
   2363   apis.each(function(i) {
   2364     var obj = $(this);
   2365     var className = obj.attr("class");
   2366     var apiLevelIndex = className.lastIndexOf("-")+1;
   2367     var apiLevelEndIndex = className.indexOf(" ", apiLevelIndex);
   2368     apiLevelEndIndex = apiLevelEndIndex != -1 ? apiLevelEndIndex : className.length;
   2369     var apiLevel = className.substring(apiLevelIndex, apiLevelEndIndex);
   2370     if (apiLevel.length == 0) { // for odd cases when the since data is actually missing, just bail
   2371       return;
   2372     }
   2373     apiLevel = parseInt(apiLevel);
   2374 
   2375     // Handle provisional api levels; if this item's level is the provisional one, set it to the max
   2376     var selectedLevelNum = parseInt(selectedLevel)
   2377     var apiLevelNum = parseInt(apiLevel);
   2378     if (isNaN(apiLevelNum)) {
   2379         apiLevelNum = maxLevel;
   2380     }
   2381 
   2382     // Grey things out that aren't available and give a tooltip title
   2383     if (apiLevelNum > selectedLevelNum) {
   2384       obj.addClass("absent").attr("title","Requires API Level \""
   2385             + apiLevel + "\" or higher");
   2386     }
   2387     else obj.removeClass("absent").removeAttr("title");
   2388   });
   2389 }
   2390 
   2391 
   2392 
   2393 
   2394 /* #################  SIDENAV TREE VIEW ################### */
   2395 
   2396 function new_node(me, mom, text, link, children_data, api_level)
   2397 {
   2398   var node = new Object();
   2399   node.children = Array();
   2400   node.children_data = children_data;
   2401   node.depth = mom.depth + 1;
   2402 
   2403   node.li = document.createElement("li");
   2404   mom.get_children_ul().appendChild(node.li);
   2405 
   2406   node.label_div = document.createElement("div");
   2407   node.label_div.className = "label";
   2408   if (api_level != null) {
   2409     $(node.label_div).addClass("api");
   2410     $(node.label_div).addClass("api-level-"+api_level);
   2411   }
   2412   node.li.appendChild(node.label_div);
   2413 
   2414   if (children_data != null) {
   2415     node.expand_toggle = document.createElement("a");
   2416     node.expand_toggle.href = "javascript:void(0)";
   2417     node.expand_toggle.onclick = function() {
   2418           if (node.expanded) {
   2419             $(node.get_children_ul()).slideUp("fast");
   2420             node.plus_img.src = me.toroot + "assets/images/triangle-closed-small.png";
   2421             node.expanded = false;
   2422           } else {
   2423             expand_node(me, node);
   2424           }
   2425        };
   2426     node.label_div.appendChild(node.expand_toggle);
   2427 
   2428     node.plus_img = document.createElement("img");
   2429     node.plus_img.src = me.toroot + "assets/images/triangle-closed-small.png";
   2430     node.plus_img.className = "plus";
   2431     node.plus_img.width = "8";
   2432     node.plus_img.border = "0";
   2433     node.expand_toggle.appendChild(node.plus_img);
   2434 
   2435     node.expanded = false;
   2436   }
   2437 
   2438   var a = document.createElement("a");
   2439   node.label_div.appendChild(a);
   2440   node.label = document.createTextNode(text);
   2441   a.appendChild(node.label);
   2442   if (link) {
   2443     a.href = me.toroot + link;
   2444   } else {
   2445     if (children_data != null) {
   2446       a.className = "nolink";
   2447       a.href = "javascript:void(0)";
   2448       a.onclick = node.expand_toggle.onclick;
   2449       // This next line shouldn't be necessary.  I'll buy a beer for the first
   2450       // person who figures out how to remove this line and have the link
   2451       // toggle shut on the first try. --joeo (a] android.com
   2452       node.expanded = false;
   2453     }
   2454   }
   2455 
   2456 
   2457   node.children_ul = null;
   2458   node.get_children_ul = function() {
   2459       if (!node.children_ul) {
   2460         node.children_ul = document.createElement("ul");
   2461         node.children_ul.className = "children_ul";
   2462         node.children_ul.style.display = "none";
   2463         node.li.appendChild(node.children_ul);
   2464       }
   2465       return node.children_ul;
   2466     };
   2467 
   2468   return node;
   2469 }
   2470 
   2471 
   2472 
   2473 
   2474 function expand_node(me, node)
   2475 {
   2476   if (node.children_data && !node.expanded) {
   2477     if (node.children_visited) {
   2478       $(node.get_children_ul()).slideDown("fast");
   2479     } else {
   2480       get_node(me, node);
   2481       if ($(node.label_div).hasClass("absent")) {
   2482         $(node.get_children_ul()).addClass("absent");
   2483       }
   2484       $(node.get_children_ul()).slideDown("fast");
   2485     }
   2486     node.plus_img.src = me.toroot + "assets/images/triangle-opened-small.png";
   2487     node.expanded = true;
   2488 
   2489     // perform api level toggling because new nodes are new to the DOM
   2490     var selectedLevel = $("#apiLevelSelector option:selected").val();
   2491     toggleVisisbleApis(selectedLevel, "#side-nav");
   2492   }
   2493 }
   2494 
   2495 function get_node(me, mom)
   2496 {
   2497   mom.children_visited = true;
   2498   for (var i in mom.children_data) {
   2499     var node_data = mom.children_data[i];
   2500     mom.children[i] = new_node(me, mom, node_data[0], node_data[1],
   2501         node_data[2], node_data[3]);
   2502   }
   2503 }
   2504 
   2505 function this_page_relative(toroot)
   2506 {
   2507   var full = document.location.pathname;
   2508   var file = "";
   2509   if (toroot.substr(0, 1) == "/") {
   2510     if (full.substr(0, toroot.length) == toroot) {
   2511       return full.substr(toroot.length);
   2512     } else {
   2513       // the file isn't under toroot.  Fail.
   2514       return null;
   2515     }
   2516   } else {
   2517     if (toroot != "./") {
   2518       toroot = "./" + toroot;
   2519     }
   2520     do {
   2521       if (toroot.substr(toroot.length-3, 3) == "../" || toroot == "./") {
   2522         var pos = full.lastIndexOf("/");
   2523         file = full.substr(pos) + file;
   2524         full = full.substr(0, pos);
   2525         toroot = toroot.substr(0, toroot.length-3);
   2526       }
   2527     } while (toroot != "" && toroot != "/");
   2528     return file.substr(1);
   2529   }
   2530 }
   2531 
   2532 function find_page(url, data)
   2533 {
   2534   var nodes = data;
   2535   var result = null;
   2536   for (var i in nodes) {
   2537     var d = nodes[i];
   2538     if (d[1] == url) {
   2539       return new Array(i);
   2540     }
   2541     else if (d[2] != null) {
   2542       result = find_page(url, d[2]);
   2543       if (result != null) {
   2544         return (new Array(i).concat(result));
   2545       }
   2546     }
   2547   }
   2548   return null;
   2549 }
   2550 
   2551 function init_default_navtree(toroot) {
   2552   // load json file for navtree data
   2553   $.getScript(toRoot + 'navtree_data.js', function(data, textStatus, jqxhr) {
   2554       // when the file is loaded, initialize the tree
   2555       if(jqxhr.status === 200) {
   2556           init_navtree("tree-list", toroot, NAVTREE_DATA);
   2557       }
   2558   });
   2559 
   2560   // perform api level toggling because because the whole tree is new to the DOM
   2561   var selectedLevel = $("#apiLevelSelector option:selected").val();
   2562   toggleVisisbleApis(selectedLevel, "#side-nav");
   2563 }
   2564 
   2565 function init_navtree(navtree_id, toroot, root_nodes)
   2566 {
   2567   var me = new Object();
   2568   me.toroot = toroot;
   2569   me.node = new Object();
   2570 
   2571   me.node.li = document.getElementById(navtree_id);
   2572   me.node.children_data = root_nodes;
   2573   me.node.children = new Array();
   2574   me.node.children_ul = document.createElement("ul");
   2575   me.node.get_children_ul = function() { return me.node.children_ul; };
   2576   //me.node.children_ul.className = "children_ul";
   2577   me.node.li.appendChild(me.node.children_ul);
   2578   me.node.depth = 0;
   2579 
   2580   get_node(me, me.node);
   2581 
   2582   me.this_page = this_page_relative(toroot);
   2583   me.breadcrumbs = find_page(me.this_page, root_nodes);
   2584   if (me.breadcrumbs != null && me.breadcrumbs.length != 0) {
   2585     var mom = me.node;
   2586     for (var i in me.breadcrumbs) {
   2587       var j = me.breadcrumbs[i];
   2588       mom = mom.children[j];
   2589       expand_node(me, mom);
   2590     }
   2591     mom.label_div.className = mom.label_div.className + " selected";
   2592     addLoadEvent(function() {
   2593       scrollIntoView("nav-tree");
   2594       });
   2595   }
   2596 }
   2597 
   2598 
   2599 
   2600 
   2601 
   2602 
   2603 
   2604 
   2605 /* TODO: eliminate redundancy with non-google functions */
   2606 function init_google_navtree(navtree_id, toroot, root_nodes)
   2607 {
   2608   var me = new Object();
   2609   me.toroot = toroot;
   2610   me.node = new Object();
   2611 
   2612   me.node.li = document.getElementById(navtree_id);
   2613   me.node.children_data = root_nodes;
   2614   me.node.children = new Array();
   2615   me.node.children_ul = document.createElement("ul");
   2616   me.node.get_children_ul = function() { return me.node.children_ul; };
   2617   //me.node.children_ul.className = "children_ul";
   2618   me.node.li.appendChild(me.node.children_ul);
   2619   me.node.depth = 0;
   2620 
   2621   get_google_node(me, me.node);
   2622 }
   2623 
   2624 function new_google_node(me, mom, text, link, children_data, api_level)
   2625 {
   2626   var node = new Object();
   2627   var child;
   2628   node.children = Array();
   2629   node.children_data = children_data;
   2630   node.depth = mom.depth + 1;
   2631   node.get_children_ul = function() {
   2632       if (!node.children_ul) {
   2633         node.children_ul = document.createElement("ul");
   2634         node.children_ul.className = "tree-list-children";
   2635         node.li.appendChild(node.children_ul);
   2636       }
   2637       return node.children_ul;
   2638     };
   2639   node.li = document.createElement("li");
   2640 
   2641   mom.get_children_ul().appendChild(node.li);
   2642 
   2643 
   2644   if(link) {
   2645     child = document.createElement("a");
   2646 
   2647   }
   2648   else {
   2649     child = document.createElement("span");
   2650     child.className = "tree-list-subtitle";
   2651 
   2652   }
   2653   if (children_data != null) {
   2654     node.li.className="nav-section";
   2655     node.label_div = document.createElement("div");
   2656     node.label_div.className = "nav-section-header-ref";
   2657     node.li.appendChild(node.label_div);
   2658     get_google_node(me, node);
   2659     node.label_div.appendChild(child);
   2660   }
   2661   else {
   2662     node.li.appendChild(child);
   2663   }
   2664   if(link) {
   2665     child.href = me.toroot + link;
   2666   }
   2667   node.label = document.createTextNode(text);
   2668   child.appendChild(node.label);
   2669 
   2670   node.children_ul = null;
   2671 
   2672   return node;
   2673 }
   2674 
   2675 function get_google_node(me, mom)
   2676 {
   2677   mom.children_visited = true;
   2678   var linkText;
   2679   for (var i in mom.children_data) {
   2680     var node_data = mom.children_data[i];
   2681     linkText = node_data[0];
   2682 
   2683     if(linkText.match("^"+"com.google.android")=="com.google.android"){
   2684       linkText = linkText.substr(19, linkText.length);
   2685     }
   2686       mom.children[i] = new_google_node(me, mom, linkText, node_data[1],
   2687           node_data[2], node_data[3]);
   2688   }
   2689 }
   2690 
   2691 
   2692 
   2693 
   2694 
   2695 
   2696 /****** NEW version of script to build google and sample navs dynamically ******/
   2697 // TODO: update Google reference docs to tolerate this new implementation
   2698 
   2699 var NODE_NAME = 0;
   2700 var NODE_HREF = 1;
   2701 var NODE_GROUP = 2;
   2702 var NODE_TAGS = 3;
   2703 var NODE_CHILDREN = 4;
   2704 
   2705 function init_google_navtree2(navtree_id, data)
   2706 {
   2707   var $containerUl = $("#"+navtree_id);
   2708   for (var i in data) {
   2709     var node_data = data[i];
   2710     $containerUl.append(new_google_node2(node_data));
   2711   }
   2712 
   2713   // Make all third-generation list items 'sticky' to prevent them from collapsing
   2714   $containerUl.find('li li li.nav-section').addClass('sticky');
   2715 
   2716   initExpandableNavItems("#"+navtree_id);
   2717 }
   2718 
   2719 function new_google_node2(node_data)
   2720 {
   2721   var linkText = node_data[NODE_NAME];
   2722   if(linkText.match("^"+"com.google.android")=="com.google.android"){
   2723     linkText = linkText.substr(19, linkText.length);
   2724   }
   2725   var $li = $('<li>');
   2726   var $a;
   2727   if (node_data[NODE_HREF] != null) {
   2728     $a = $('<a href="' + toRoot + node_data[NODE_HREF] + '" title="' + linkText + '" >'
   2729         + linkText + '</a>');
   2730   } else {
   2731     $a = $('<a href="#" onclick="return false;" title="' + linkText + '" >'
   2732         + linkText + '/</a>');
   2733   }
   2734   var $childUl = $('<ul>');
   2735   if (node_data[NODE_CHILDREN] != null) {
   2736     $li.addClass("nav-section");
   2737     $a = $('<div class="nav-section-header">').append($a);
   2738     if (node_data[NODE_HREF] == null) $a.addClass('empty');
   2739 
   2740     for (var i in node_data[NODE_CHILDREN]) {
   2741       var child_node_data = node_data[NODE_CHILDREN][i];
   2742       $childUl.append(new_google_node2(child_node_data));
   2743     }
   2744     $li.append($childUl);
   2745   }
   2746   $li.prepend($a);
   2747 
   2748   return $li;
   2749 }
   2750 
   2751 
   2752 
   2753 
   2754 
   2755 
   2756 
   2757 
   2758 
   2759 
   2760 
   2761 function showGoogleRefTree() {
   2762   init_default_google_navtree(toRoot);
   2763   init_default_gcm_navtree(toRoot);
   2764 }
   2765 
   2766 function init_default_google_navtree(toroot) {
   2767   // load json file for navtree data
   2768   $.getScript(toRoot + 'gms_navtree_data.js', function(data, textStatus, jqxhr) {
   2769       // when the file is loaded, initialize the tree
   2770       if(jqxhr.status === 200) {
   2771           init_google_navtree("gms-tree-list", toroot, GMS_NAVTREE_DATA);
   2772           highlightSidenav();
   2773           resizeNav();
   2774       }
   2775   });
   2776 }
   2777 
   2778 function init_default_gcm_navtree(toroot) {
   2779   // load json file for navtree data
   2780   $.getScript(toRoot + 'gcm_navtree_data.js', function(data, textStatus, jqxhr) {
   2781       // when the file is loaded, initialize the tree
   2782       if(jqxhr.status === 200) {
   2783           init_google_navtree("gcm-tree-list", toroot, GCM_NAVTREE_DATA);
   2784           highlightSidenav();
   2785           resizeNav();
   2786       }
   2787   });
   2788 }
   2789 
   2790 function showSamplesRefTree() {
   2791   init_default_samples_navtree(toRoot);
   2792 }
   2793 
   2794 function init_default_samples_navtree(toroot) {
   2795   // load json file for navtree data
   2796   $.getScript(toRoot + 'samples_navtree_data.js', function(data, textStatus, jqxhr) {
   2797       // when the file is loaded, initialize the tree
   2798       if(jqxhr.status === 200) {
   2799           // hack to remove the "about the samples" link then put it back in
   2800           // after we nuke the list to remove the dummy static list of samples
   2801           var $firstLi = $("#nav.samples-nav > li:first-child").clone();
   2802           $("#nav.samples-nav").empty();
   2803           $("#nav.samples-nav").append($firstLi);
   2804 
   2805           init_google_navtree2("nav.samples-nav", SAMPLES_NAVTREE_DATA);
   2806           highlightSidenav();
   2807           resizeNav();
   2808           if ($("#jd-content #samples").length) {
   2809             showSamples();
   2810           }
   2811       }
   2812   });
   2813 }
   2814 
   2815 /* TOGGLE INHERITED MEMBERS */
   2816 
   2817 /* Toggle an inherited class (arrow toggle)
   2818  * @param linkObj  The link that was clicked.
   2819  * @param expand  'true' to ensure it's expanded. 'false' to ensure it's closed.
   2820  *                'null' to simply toggle.
   2821  */
   2822 function toggleInherited(linkObj, expand) {
   2823     var base = linkObj.getAttribute("id");
   2824     var list = document.getElementById(base + "-list");
   2825     var summary = document.getElementById(base + "-summary");
   2826     var trigger = document.getElementById(base + "-trigger");
   2827     var a = $(linkObj);
   2828     if ( (expand == null && a.hasClass("closed")) || expand ) {
   2829         list.style.display = "none";
   2830         summary.style.display = "block";
   2831         trigger.src = toRoot + "assets/images/triangle-opened.png";
   2832         a.removeClass("closed");
   2833         a.addClass("opened");
   2834     } else if ( (expand == null && a.hasClass("opened")) || (expand == false) ) {
   2835         list.style.display = "block";
   2836         summary.style.display = "none";
   2837         trigger.src = toRoot + "assets/images/triangle-closed.png";
   2838         a.removeClass("opened");
   2839         a.addClass("closed");
   2840     }
   2841     return false;
   2842 }
   2843 
   2844 /* Toggle all inherited classes in a single table (e.g. all inherited methods)
   2845  * @param linkObj  The link that was clicked.
   2846  * @param expand  'true' to ensure it's expanded. 'false' to ensure it's closed.
   2847  *                'null' to simply toggle.
   2848  */
   2849 function toggleAllInherited(linkObj, expand) {
   2850   var a = $(linkObj);
   2851   var table = $(a.parent().parent().parent()); // ugly way to get table/tbody
   2852   var expandos = $(".jd-expando-trigger", table);
   2853   if ( (expand == null && a.text() == "[Expand]") || expand ) {
   2854     expandos.each(function(i) {
   2855       toggleInherited(this, true);
   2856     });
   2857     a.text("[Collapse]");
   2858   } else if ( (expand == null && a.text() == "[Collapse]") || (expand == false) ) {
   2859     expandos.each(function(i) {
   2860       toggleInherited(this, false);
   2861     });
   2862     a.text("[Expand]");
   2863   }
   2864   return false;
   2865 }
   2866 
   2867 /* Toggle all inherited members in the class (link in the class title)
   2868  */
   2869 function toggleAllClassInherited() {
   2870   var a = $("#toggleAllClassInherited"); // get toggle link from class title
   2871   var toggles = $(".toggle-all", $("#body-content"));
   2872   if (a.text() == "[Expand All]") {
   2873     toggles.each(function(i) {
   2874       toggleAllInherited(this, true);
   2875     });
   2876     a.text("[Collapse All]");
   2877   } else {
   2878     toggles.each(function(i) {
   2879       toggleAllInherited(this, false);
   2880     });
   2881     a.text("[Expand All]");
   2882   }
   2883   return false;
   2884 }
   2885 
   2886 /* Expand all inherited members in the class. Used when initiating page search */
   2887 function ensureAllInheritedExpanded() {
   2888   var toggles = $(".toggle-all", $("#body-content"));
   2889   toggles.each(function(i) {
   2890     toggleAllInherited(this, true);
   2891   });
   2892   $("#toggleAllClassInherited").text("[Collapse All]");
   2893 }
   2894 
   2895 
   2896 /* HANDLE KEY EVENTS
   2897  * - Listen for Ctrl+F (Cmd on Mac) and expand all inherited members (to aid page search)
   2898  */
   2899 var agent = navigator['userAgent'].toLowerCase();
   2900 var mac = agent.indexOf("macintosh") != -1;
   2901 
   2902 $(document).keydown( function(e) {
   2903 var control = mac ? e.metaKey && !e.ctrlKey : e.ctrlKey; // get ctrl key
   2904   if (control && e.which == 70) {  // 70 is "F"
   2905     ensureAllInheritedExpanded();
   2906   }
   2907 });
   2908 
   2909 
   2910 
   2911 
   2912 
   2913 
   2914 /* On-demand functions */
   2915 
   2916 /** Move sample code line numbers out of PRE block and into non-copyable column */
   2917 function initCodeLineNumbers() {
   2918   var numbers = $("#codesample-block a.number");
   2919   if (numbers.length) {
   2920     $("#codesample-line-numbers").removeClass("hidden").append(numbers);
   2921   }
   2922 
   2923   $(document).ready(function() {
   2924     // select entire line when clicked
   2925     $("span.code-line").click(function() {
   2926       if (!shifted) {
   2927         selectText(this);
   2928       }
   2929     });
   2930     // invoke line link on double click
   2931     $(".code-line").dblclick(function() {
   2932       document.location.hash = $(this).attr('id');
   2933     });
   2934     // highlight the line when hovering on the number
   2935     $("#codesample-line-numbers a.number").mouseover(function() {
   2936       var id = $(this).attr('href');
   2937       $(id).css('background','#e7e7e7');
   2938     });
   2939     $("#codesample-line-numbers a.number").mouseout(function() {
   2940       var id = $(this).attr('href');
   2941       $(id).css('background','none');
   2942     });
   2943   });
   2944 }
   2945 
   2946 // create SHIFT key binder to avoid the selectText method when selecting multiple lines
   2947 var shifted = false;
   2948 $(document).bind('keyup keydown', function(e){shifted = e.shiftKey; return true;} );
   2949 
   2950 // courtesy of jasonedelman.com
   2951 function selectText(element) {
   2952     var doc = document
   2953         , range, selection
   2954     ;
   2955     if (doc.body.createTextRange) { //ms
   2956         range = doc.body.createTextRange();
   2957         range.moveToElementText(element);
   2958         range.select();
   2959     } else if (window.getSelection) { //all others
   2960         selection = window.getSelection();
   2961         range = doc.createRange();
   2962         range.selectNodeContents(element);
   2963         selection.removeAllRanges();
   2964         selection.addRange(range);
   2965     }
   2966 }
   2967 
   2968 
   2969 
   2970 
   2971 /** Display links and other information about samples that match the
   2972     group specified by the URL */
   2973 function showSamples() {
   2974   var group = $("#samples").attr('class');
   2975   $("#samples").html("<p>Here are some samples for <b>" + group + "</b> apps:</p>");
   2976 
   2977   var $ul = $("<ul>");
   2978   $selectedLi = $("#nav li.selected");
   2979 
   2980   $selectedLi.children("ul").children("li").each(function() {
   2981       var $li = $("<li>").append($(this).find("a").first().clone());
   2982       $ul.append($li);
   2983   });
   2984 
   2985   $("#samples").append($ul);
   2986 
   2987 }
   2988