docs.js revision 641c2c2fdaf68fa48b044177fef7c890a42c8a11
1var classesNav;
2var devdocNav;
3var sidenav;
4var cookie_namespace = 'android_developer';
5var NAV_PREF_TREE = "tree";
6var NAV_PREF_PANELS = "panels";
7var nav_pref;
8var isMobile = false; // true if mobile, so we can adjust some layout
9var mPagePath; // initialized in ready() function
10
11var basePath = getBaseUri(location.pathname);
12var SITE_ROOT = toRoot + basePath.substring(1,basePath.indexOf("/",1));
13var 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
22var 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 :
216false; // 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
562function 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 */
597function 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
624function unHighlightSidenav() {
625  $("ul#nav li.selected").removeClass("selected");
626  $('ul#nav li.nav-section.expanded').removeClass('expanded').children('ul').hide();
627}
628
629function 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
655function setNavBarLeftPos() {
656  navBarLeftPos = $('#body-content').offset().left;
657}
658
659
660function 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
667function 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
679var agent = navigator['userAgent'].toLowerCase();
680// If a mobile phone, set flag and do mobile setup
681if ((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
699function 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
765var updateScrollbars = false;
766var 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 */
773function 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. */
794function 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 */
806function 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
814function 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. */
832function 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 */
860function 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
873function 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
890function 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
929function 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
950function 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
960function 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
1004function getNavPref() {
1005  var v = readCookie('reference_nav');
1006  if (v != NAV_PREF_TREE) {
1007    v = NAV_PREF_PANELS;
1008  }
1009  return v;
1010}
1011
1012function 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
1022function 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
1060function 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
1073function 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
1087function 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
1098function 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
1120function loadLangPref() {
1121  var lang = readCookie("pref_lang");
1122  if (lang != 0) {
1123    $("#language").find("option[value='"+lang+"']").attr("selected",true);
1124  }
1125}
1126
1127function 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 */
1144function 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 */
1168function 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
1178function 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
1502var gSelectedIndex = -1;  // the index position of currently highlighted suggestion
1503var gSelectedColumn = -1;  // which column of suggestion lists is currently focused
1504
1505var gMatches = new Array();
1506var gLastText = "";
1507var gInitialized = false;
1508var ROW_COUNT_FRAMEWORK = 20;       // max number of results in list
1509var gListLength = 0;
1510
1511
1512var gGoogleMatches = new Array();
1513var ROW_COUNT_GOOGLE = 15;          // max number of results in list
1514var gGoogleListLength = 0;
1515
1516var gDocsMatches = new Array();
1517var ROW_COUNT_DOCS = 100;          // max number of results in list
1518var gDocsListLength = 0;
1519
1520function 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
1526function 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
1535function 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
1542function 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
1560function 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  */
1682function 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 */
1888function 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 */
1925function 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 */
2001function 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
2020function 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
2030function 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
2039function 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
2063var searchControl;
2064google.load('search', '1', {"callback" : function() {
2065            searchControl = new google.search.SearchControl();
2066          } });
2067
2068function 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
2150google.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
2184function 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
2194function 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
2217function 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
2228function 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 */
2235function 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 */
2251if (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
2263var API_LEVEL_COOKIE = "api_level";
2264var minLevel = 1;
2265var 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
2284function 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
2295function 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
2306function 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
2330function 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
2361function 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. To reveal, change the target API level "
2386              + "above the left navigation.");
2387    }
2388    else obj.removeClass("absent").removeAttr("title");
2389  });
2390}
2391
2392
2393
2394
2395/* #################  SIDENAV TREE VIEW ################### */
2396
2397function new_node(me, mom, text, link, children_data, api_level)
2398{
2399  var node = new Object();
2400  node.children = Array();
2401  node.children_data = children_data;
2402  node.depth = mom.depth + 1;
2403
2404  node.li = document.createElement("li");
2405  mom.get_children_ul().appendChild(node.li);
2406
2407  node.label_div = document.createElement("div");
2408  node.label_div.className = "label";
2409  if (api_level != null) {
2410    $(node.label_div).addClass("api");
2411    $(node.label_div).addClass("api-level-"+api_level);
2412  }
2413  node.li.appendChild(node.label_div);
2414
2415  if (children_data != null) {
2416    node.expand_toggle = document.createElement("a");
2417    node.expand_toggle.href = "javascript:void(0)";
2418    node.expand_toggle.onclick = function() {
2419          if (node.expanded) {
2420            $(node.get_children_ul()).slideUp("fast");
2421            node.plus_img.src = me.toroot + "assets/images/triangle-closed-small.png";
2422            node.expanded = false;
2423          } else {
2424            expand_node(me, node);
2425          }
2426       };
2427    node.label_div.appendChild(node.expand_toggle);
2428
2429    node.plus_img = document.createElement("img");
2430    node.plus_img.src = me.toroot + "assets/images/triangle-closed-small.png";
2431    node.plus_img.className = "plus";
2432    node.plus_img.width = "8";
2433    node.plus_img.border = "0";
2434    node.expand_toggle.appendChild(node.plus_img);
2435
2436    node.expanded = false;
2437  }
2438
2439  var a = document.createElement("a");
2440  node.label_div.appendChild(a);
2441  node.label = document.createTextNode(text);
2442  a.appendChild(node.label);
2443  if (link) {
2444    a.href = me.toroot + link;
2445  } else {
2446    if (children_data != null) {
2447      a.className = "nolink";
2448      a.href = "javascript:void(0)";
2449      a.onclick = node.expand_toggle.onclick;
2450      // This next line shouldn't be necessary.  I'll buy a beer for the first
2451      // person who figures out how to remove this line and have the link
2452      // toggle shut on the first try. --joeo@android.com
2453      node.expanded = false;
2454    }
2455  }
2456
2457
2458  node.children_ul = null;
2459  node.get_children_ul = function() {
2460      if (!node.children_ul) {
2461        node.children_ul = document.createElement("ul");
2462        node.children_ul.className = "children_ul";
2463        node.children_ul.style.display = "none";
2464        node.li.appendChild(node.children_ul);
2465      }
2466      return node.children_ul;
2467    };
2468
2469  return node;
2470}
2471
2472
2473
2474
2475function expand_node(me, node)
2476{
2477  if (node.children_data && !node.expanded) {
2478    if (node.children_visited) {
2479      $(node.get_children_ul()).slideDown("fast");
2480    } else {
2481      get_node(me, node);
2482      if ($(node.label_div).hasClass("absent")) {
2483        $(node.get_children_ul()).addClass("absent");
2484      }
2485      $(node.get_children_ul()).slideDown("fast");
2486    }
2487    node.plus_img.src = me.toroot + "assets/images/triangle-opened-small.png";
2488    node.expanded = true;
2489
2490    // perform api level toggling because new nodes are new to the DOM
2491    var selectedLevel = $("#apiLevelSelector option:selected").val();
2492    toggleVisisbleApis(selectedLevel, "#side-nav");
2493  }
2494}
2495
2496function get_node(me, mom)
2497{
2498  mom.children_visited = true;
2499  for (var i in mom.children_data) {
2500    var node_data = mom.children_data[i];
2501    mom.children[i] = new_node(me, mom, node_data[0], node_data[1],
2502        node_data[2], node_data[3]);
2503  }
2504}
2505
2506function this_page_relative(toroot)
2507{
2508  var full = document.location.pathname;
2509  var file = "";
2510  if (toroot.substr(0, 1) == "/") {
2511    if (full.substr(0, toroot.length) == toroot) {
2512      return full.substr(toroot.length);
2513    } else {
2514      // the file isn't under toroot.  Fail.
2515      return null;
2516    }
2517  } else {
2518    if (toroot != "./") {
2519      toroot = "./" + toroot;
2520    }
2521    do {
2522      if (toroot.substr(toroot.length-3, 3) == "../" || toroot == "./") {
2523        var pos = full.lastIndexOf("/");
2524        file = full.substr(pos) + file;
2525        full = full.substr(0, pos);
2526        toroot = toroot.substr(0, toroot.length-3);
2527      }
2528    } while (toroot != "" && toroot != "/");
2529    return file.substr(1);
2530  }
2531}
2532
2533function find_page(url, data)
2534{
2535  var nodes = data;
2536  var result = null;
2537  for (var i in nodes) {
2538    var d = nodes[i];
2539    if (d[1] == url) {
2540      return new Array(i);
2541    }
2542    else if (d[2] != null) {
2543      result = find_page(url, d[2]);
2544      if (result != null) {
2545        return (new Array(i).concat(result));
2546      }
2547    }
2548  }
2549  return null;
2550}
2551
2552function init_default_navtree(toroot) {
2553  // load json file for navtree data
2554  $.getScript(toRoot + 'navtree_data.js', function(data, textStatus, jqxhr) {
2555      // when the file is loaded, initialize the tree
2556      if(jqxhr.status === 200) {
2557          init_navtree("tree-list", toroot, NAVTREE_DATA);
2558      }
2559  });
2560
2561  // perform api level toggling because because the whole tree is new to the DOM
2562  var selectedLevel = $("#apiLevelSelector option:selected").val();
2563  toggleVisisbleApis(selectedLevel, "#side-nav");
2564}
2565
2566function init_navtree(navtree_id, toroot, root_nodes)
2567{
2568  var me = new Object();
2569  me.toroot = toroot;
2570  me.node = new Object();
2571
2572  me.node.li = document.getElementById(navtree_id);
2573  me.node.children_data = root_nodes;
2574  me.node.children = new Array();
2575  me.node.children_ul = document.createElement("ul");
2576  me.node.get_children_ul = function() { return me.node.children_ul; };
2577  //me.node.children_ul.className = "children_ul";
2578  me.node.li.appendChild(me.node.children_ul);
2579  me.node.depth = 0;
2580
2581  get_node(me, me.node);
2582
2583  me.this_page = this_page_relative(toroot);
2584  me.breadcrumbs = find_page(me.this_page, root_nodes);
2585  if (me.breadcrumbs != null && me.breadcrumbs.length != 0) {
2586    var mom = me.node;
2587    for (var i in me.breadcrumbs) {
2588      var j = me.breadcrumbs[i];
2589      mom = mom.children[j];
2590      expand_node(me, mom);
2591    }
2592    mom.label_div.className = mom.label_div.className + " selected";
2593    addLoadEvent(function() {
2594      scrollIntoView("nav-tree");
2595      });
2596  }
2597}
2598
2599
2600
2601
2602
2603
2604
2605
2606/* TODO: eliminate redundancy with non-google functions */
2607function init_google_navtree(navtree_id, toroot, root_nodes)
2608{
2609  var me = new Object();
2610  me.toroot = toroot;
2611  me.node = new Object();
2612
2613  me.node.li = document.getElementById(navtree_id);
2614  me.node.children_data = root_nodes;
2615  me.node.children = new Array();
2616  me.node.children_ul = document.createElement("ul");
2617  me.node.get_children_ul = function() { return me.node.children_ul; };
2618  //me.node.children_ul.className = "children_ul";
2619  me.node.li.appendChild(me.node.children_ul);
2620  me.node.depth = 0;
2621
2622  get_google_node(me, me.node);
2623}
2624
2625function new_google_node(me, mom, text, link, children_data, api_level)
2626{
2627  var node = new Object();
2628  var child;
2629  node.children = Array();
2630  node.children_data = children_data;
2631  node.depth = mom.depth + 1;
2632  node.get_children_ul = function() {
2633      if (!node.children_ul) {
2634        node.children_ul = document.createElement("ul");
2635        node.children_ul.className = "tree-list-children";
2636        node.li.appendChild(node.children_ul);
2637      }
2638      return node.children_ul;
2639    };
2640  node.li = document.createElement("li");
2641
2642  mom.get_children_ul().appendChild(node.li);
2643
2644
2645  if(link) {
2646    child = document.createElement("a");
2647
2648  }
2649  else {
2650    child = document.createElement("span");
2651    child.className = "tree-list-subtitle";
2652
2653  }
2654  if (children_data != null) {
2655    node.li.className="nav-section";
2656    node.label_div = document.createElement("div");
2657    node.label_div.className = "nav-section-header-ref";
2658    node.li.appendChild(node.label_div);
2659    get_google_node(me, node);
2660    node.label_div.appendChild(child);
2661  }
2662  else {
2663    node.li.appendChild(child);
2664  }
2665  if(link) {
2666    child.href = me.toroot + link;
2667  }
2668  node.label = document.createTextNode(text);
2669  child.appendChild(node.label);
2670
2671  node.children_ul = null;
2672
2673  return node;
2674}
2675
2676function get_google_node(me, mom)
2677{
2678  mom.children_visited = true;
2679  var linkText;
2680  for (var i in mom.children_data) {
2681    var node_data = mom.children_data[i];
2682    linkText = node_data[0];
2683
2684    if(linkText.match("^"+"com.google.android")=="com.google.android"){
2685      linkText = linkText.substr(19, linkText.length);
2686    }
2687      mom.children[i] = new_google_node(me, mom, linkText, node_data[1],
2688          node_data[2], node_data[3]);
2689  }
2690}
2691
2692
2693
2694
2695
2696
2697/****** NEW version of script to build google and sample navs dynamically ******/
2698// TODO: update Google reference docs to tolerate this new implementation
2699
2700var NODE_NAME = 0;
2701var NODE_HREF = 1;
2702var NODE_GROUP = 2;
2703var NODE_TAGS = 3;
2704var NODE_CHILDREN = 4;
2705
2706function init_google_navtree2(navtree_id, data)
2707{
2708  var $containerUl = $("#"+navtree_id);
2709  for (var i in data) {
2710    var node_data = data[i];
2711    $containerUl.append(new_google_node2(node_data));
2712  }
2713
2714  // Make all third-generation list items 'sticky' to prevent them from collapsing
2715  $containerUl.find('li li li.nav-section').addClass('sticky');
2716
2717  initExpandableNavItems("#"+navtree_id);
2718}
2719
2720function new_google_node2(node_data)
2721{
2722  var linkText = node_data[NODE_NAME];
2723  if(linkText.match("^"+"com.google.android")=="com.google.android"){
2724    linkText = linkText.substr(19, linkText.length);
2725  }
2726  var $li = $('<li>');
2727  var $a;
2728  if (node_data[NODE_HREF] != null) {
2729    $a = $('<a href="' + toRoot + node_data[NODE_HREF] + '" title="' + linkText + '" >'
2730        + linkText + '</a>');
2731  } else {
2732    $a = $('<a href="#" onclick="return false;" title="' + linkText + '" >'
2733        + linkText + '/</a>');
2734  }
2735  var $childUl = $('<ul>');
2736  if (node_data[NODE_CHILDREN] != null) {
2737    $li.addClass("nav-section");
2738    $a = $('<div class="nav-section-header">').append($a);
2739    if (node_data[NODE_HREF] == null) $a.addClass('empty');
2740
2741    for (var i in node_data[NODE_CHILDREN]) {
2742      var child_node_data = node_data[NODE_CHILDREN][i];
2743      $childUl.append(new_google_node2(child_node_data));
2744    }
2745    $li.append($childUl);
2746  }
2747  $li.prepend($a);
2748
2749  return $li;
2750}
2751
2752
2753
2754
2755
2756
2757
2758
2759
2760
2761
2762function showGoogleRefTree() {
2763  init_default_google_navtree(toRoot);
2764  init_default_gcm_navtree(toRoot);
2765}
2766
2767function init_default_google_navtree(toroot) {
2768  // load json file for navtree data
2769  $.getScript(toRoot + 'gms_navtree_data.js', function(data, textStatus, jqxhr) {
2770      // when the file is loaded, initialize the tree
2771      if(jqxhr.status === 200) {
2772          init_google_navtree("gms-tree-list", toroot, GMS_NAVTREE_DATA);
2773          highlightSidenav();
2774          resizeNav();
2775      }
2776  });
2777}
2778
2779function init_default_gcm_navtree(toroot) {
2780  // load json file for navtree data
2781  $.getScript(toRoot + 'gcm_navtree_data.js', function(data, textStatus, jqxhr) {
2782      // when the file is loaded, initialize the tree
2783      if(jqxhr.status === 200) {
2784          init_google_navtree("gcm-tree-list", toroot, GCM_NAVTREE_DATA);
2785          highlightSidenav();
2786          resizeNav();
2787      }
2788  });
2789}
2790
2791function showSamplesRefTree() {
2792  init_default_samples_navtree(toRoot);
2793}
2794
2795function init_default_samples_navtree(toroot) {
2796  // load json file for navtree data
2797  $.getScript(toRoot + 'samples_navtree_data.js', function(data, textStatus, jqxhr) {
2798      // when the file is loaded, initialize the tree
2799      if(jqxhr.status === 200) {
2800          // hack to remove the "about the samples" link then put it back in
2801          // after we nuke the list to remove the dummy static list of samples
2802          var $firstLi = $("#nav.samples-nav > li:first-child").clone();
2803          $("#nav.samples-nav").empty();
2804          $("#nav.samples-nav").append($firstLi);
2805
2806          init_google_navtree2("nav.samples-nav", SAMPLES_NAVTREE_DATA);
2807          highlightSidenav();
2808          resizeNav();
2809          if ($("#jd-content #samples").length) {
2810            showSamples();
2811          }
2812      }
2813  });
2814}
2815
2816/* TOGGLE INHERITED MEMBERS */
2817
2818/* Toggle an inherited class (arrow toggle)
2819 * @param linkObj  The link that was clicked.
2820 * @param expand  'true' to ensure it's expanded. 'false' to ensure it's closed.
2821 *                'null' to simply toggle.
2822 */
2823function toggleInherited(linkObj, expand) {
2824    var base = linkObj.getAttribute("id");
2825    var list = document.getElementById(base + "-list");
2826    var summary = document.getElementById(base + "-summary");
2827    var trigger = document.getElementById(base + "-trigger");
2828    var a = $(linkObj);
2829    if ( (expand == null && a.hasClass("closed")) || expand ) {
2830        list.style.display = "none";
2831        summary.style.display = "block";
2832        trigger.src = toRoot + "assets/images/triangle-opened.png";
2833        a.removeClass("closed");
2834        a.addClass("opened");
2835    } else if ( (expand == null && a.hasClass("opened")) || (expand == false) ) {
2836        list.style.display = "block";
2837        summary.style.display = "none";
2838        trigger.src = toRoot + "assets/images/triangle-closed.png";
2839        a.removeClass("opened");
2840        a.addClass("closed");
2841    }
2842    return false;
2843}
2844
2845/* Toggle all inherited classes in a single table (e.g. all inherited methods)
2846 * @param linkObj  The link that was clicked.
2847 * @param expand  'true' to ensure it's expanded. 'false' to ensure it's closed.
2848 *                'null' to simply toggle.
2849 */
2850function toggleAllInherited(linkObj, expand) {
2851  var a = $(linkObj);
2852  var table = $(a.parent().parent().parent()); // ugly way to get table/tbody
2853  var expandos = $(".jd-expando-trigger", table);
2854  if ( (expand == null && a.text() == "[Expand]") || expand ) {
2855    expandos.each(function(i) {
2856      toggleInherited(this, true);
2857    });
2858    a.text("[Collapse]");
2859  } else if ( (expand == null && a.text() == "[Collapse]") || (expand == false) ) {
2860    expandos.each(function(i) {
2861      toggleInherited(this, false);
2862    });
2863    a.text("[Expand]");
2864  }
2865  return false;
2866}
2867
2868/* Toggle all inherited members in the class (link in the class title)
2869 */
2870function toggleAllClassInherited() {
2871  var a = $("#toggleAllClassInherited"); // get toggle link from class title
2872  var toggles = $(".toggle-all", $("#body-content"));
2873  if (a.text() == "[Expand All]") {
2874    toggles.each(function(i) {
2875      toggleAllInherited(this, true);
2876    });
2877    a.text("[Collapse All]");
2878  } else {
2879    toggles.each(function(i) {
2880      toggleAllInherited(this, false);
2881    });
2882    a.text("[Expand All]");
2883  }
2884  return false;
2885}
2886
2887/* Expand all inherited members in the class. Used when initiating page search */
2888function ensureAllInheritedExpanded() {
2889  var toggles = $(".toggle-all", $("#body-content"));
2890  toggles.each(function(i) {
2891    toggleAllInherited(this, true);
2892  });
2893  $("#toggleAllClassInherited").text("[Collapse All]");
2894}
2895
2896
2897/* HANDLE KEY EVENTS
2898 * - Listen for Ctrl+F (Cmd on Mac) and expand all inherited members (to aid page search)
2899 */
2900var agent = navigator['userAgent'].toLowerCase();
2901var mac = agent.indexOf("macintosh") != -1;
2902
2903$(document).keydown( function(e) {
2904var control = mac ? e.metaKey && !e.ctrlKey : e.ctrlKey; // get ctrl key
2905  if (control && e.which == 70) {  // 70 is "F"
2906    ensureAllInheritedExpanded();
2907  }
2908});
2909
2910
2911
2912
2913
2914
2915/* On-demand functions */
2916
2917/** Move sample code line numbers out of PRE block and into non-copyable column */
2918function initCodeLineNumbers() {
2919  var numbers = $("#codesample-block a.number");
2920  if (numbers.length) {
2921    $("#codesample-line-numbers").removeClass("hidden").append(numbers);
2922  }
2923
2924  $(document).ready(function() {
2925    // select entire line when clicked
2926    $("span.code-line").click(function() {
2927      if (!shifted) {
2928        selectText(this);
2929      }
2930    });
2931    // invoke line link on double click
2932    $(".code-line").dblclick(function() {
2933      document.location.hash = $(this).attr('id');
2934    });
2935    // highlight the line when hovering on the number
2936    $("#codesample-line-numbers a.number").mouseover(function() {
2937      var id = $(this).attr('href');
2938      $(id).css('background','#e7e7e7');
2939    });
2940    $("#codesample-line-numbers a.number").mouseout(function() {
2941      var id = $(this).attr('href');
2942      $(id).css('background','none');
2943    });
2944  });
2945}
2946
2947// create SHIFT key binder to avoid the selectText method when selecting multiple lines
2948var shifted = false;
2949$(document).bind('keyup keydown', function(e){shifted = e.shiftKey; return true;} );
2950
2951// courtesy of jasonedelman.com
2952function selectText(element) {
2953    var doc = document
2954        , range, selection
2955    ;
2956    if (doc.body.createTextRange) { //ms
2957        range = doc.body.createTextRange();
2958        range.moveToElementText(element);
2959        range.select();
2960    } else if (window.getSelection) { //all others
2961        selection = window.getSelection();
2962        range = doc.createRange();
2963        range.selectNodeContents(element);
2964        selection.removeAllRanges();
2965        selection.addRange(range);
2966    }
2967}
2968
2969
2970
2971
2972/** Display links and other information about samples that match the
2973    group specified by the URL */
2974function showSamples() {
2975  var group = $("#samples").attr('class');
2976  $("#samples").html("<p>Here are some samples for <b>" + group + "</b> apps:</p>");
2977
2978  var $ul = $("<ul>");
2979  $selectedLi = $("#nav li.selected");
2980
2981  $selectedLi.children("ul").children("li").each(function() {
2982      var $li = $("<li>").append($(this).find("a").first().clone());
2983      $ul.append($li);
2984  });
2985
2986  $("#samples").append($ul);
2987
2988}
2989