blob: ba65e44295ccce19bbcb1403eecf29d8e40d2837 [file] [log] [blame]
Scott Maine4d8f1b2012-06-21 18:03:05 -07001var classesNav;
2var devdocNav;
3var sidenav;
4var cookie_namespace = 'android_developer';
5var NAV_PREF_TREE = "tree";
6var NAV_PREF_PANELS = "panels";
7var nav_pref;
Scott Maine4d8f1b2012-06-21 18:03:05 -07008var isMobile = false; // true if mobile, so we can adjust some layout
Scott Mainf6145542013-04-01 16:38:11 -07009var mPagePath; // initialized in ready() function
Scott Maine4d8f1b2012-06-21 18:03:05 -070010
Scott Main1b3db112012-07-03 14:06:22 -070011var basePath = getBaseUri(location.pathname);
12var SITE_ROOT = toRoot + basePath.substring(1,basePath.indexOf("/",1));
Scott Main7e447ed2013-02-19 17:22:37 -080013var GOOGLE_DATA; // combined data for google service apis, used for search suggest
Scott Main1b3db112012-07-03 14:06:22 -070014
Scott Main25e73002013-03-27 15:24:06 -070015// Ensure that all ajax getScript() requests allow caching
16$.ajaxSetup({
17 cache: true
18});
Scott Maine4d8f1b2012-06-21 18:03:05 -070019
20/****** ON LOAD SET UP STUFF *********/
21
22var navBarIsFixed = false;
23$(document).ready(function() {
Scott Main7e447ed2013-02-19 17:22:37 -080024
Scott Main0e76e7e2013-03-12 10:24:07 -070025 // load json file for JD doc search suggestions
26 $.getScript(toRoot + 'reference/jd_lists.js');
Scott Main7e447ed2013-02-19 17:22:37 -080027 // load json file for Android API search suggestions
28 $.getScript(toRoot + 'reference/lists.js');
29 // load json files for Google services API suggestions
Scott Main9f2971d2013-02-26 13:07:41 -080030 $.getScript(toRoot + 'reference/gcm_lists.js', function(data, textStatus, jqxhr) {
Scott Main7e447ed2013-02-19 17:22:37 -080031 // once the GCM json (GCM_DATA) is loaded, load the GMS json (GMS_DATA) and merge the data
32 if(jqxhr.status === 200) {
Scott Main9f2971d2013-02-26 13:07:41 -080033 $.getScript(toRoot + 'reference/gms_lists.js', function(data, textStatus, jqxhr) {
Scott Main7e447ed2013-02-19 17:22:37 -080034 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
Scott Main0e76e7e2013-03-12 10:24:07 -070047 // setup keyboard listener for search shortcut
48 $('body').keyup(function(event) {
49 if (event.which == 191) {
50 $('#search_autocomplete').focus();
51 }
52 });
Scott Main015d6162013-01-29 09:01:52 -080053
Scott Maine4d8f1b2012-06-21 18:03:05 -070054 // 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)
Scott Maindb3678b2012-10-23 14:13:41 -070067 $('h2').not('#qv h2').not('#tb h2').not('.sidebox h2').not('#devdoc-nav h2').not('h2.norule').css({marginBottom:0}).after('<hr/>');
Scott Maine4d8f1b2012-06-21 18:03:05 -070068
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();
Scott Main0e76e7e2013-03-12 10:24:07 -070076 search_focus_changed($searchInput.get(), false);
77 hideResults();
Scott Maine4d8f1b2012-06-21 18:03:05 -070078 });
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 }
Scott Mainac2aef52013-02-12 14:15:23 -0800137
Scott Maine4d8f1b2012-06-21 18:03:05 -0700138 if (pagePath.indexOf(SITE_ROOT) == 0) {
139 if (pagePath == '' || pagePath.charAt(pagePath.length - 1) == '/') {
140 pagePath += 'index.html';
141 }
142 }
143
Scott Main01a25452013-02-12 17:32:27 -0800144 // 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;
Scott Maine4d8f1b2012-06-21 18:03:05 -0700147 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
Scott Mainac2aef52013-02-12 14:15:23 -0800166 // 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");
Scott Mainac2aef52013-02-12 14:15:23 -0800174 // In Develop docs, also highlight appropriate sub-tab
Scott Main01a25452013-02-12 17:32:27 -0800175 var rootDir = pagePathOriginal.substring(1,pagePathOriginal.indexOf('/', 1));
Scott Mainac2aef52013-02-12 14:15:23 -0800176 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 }
192
193 // highlight Distribute tab
194 } else if ($("body").hasClass("distribute")) {
195 $("#header li.distribute a").addClass("selected");
196 }
197
Scott Mainf6145542013-04-01 16:38:11 -0700198 // set global variable so we can highlight the sidenav a bit later (such as for google reference)
199 // and highlight the sidenav
200 mPagePath = pagePath;
201 highlightSidenav();
Scott Mainac2aef52013-02-12 14:15:23 -0800202
Scott Mainf6145542013-04-01 16:38:11 -0700203 // set up prev/next links if they exist
Scott Maine4d8f1b2012-06-21 18:03:05 -0700204 var $selNavLink = $('#nav').find('a[href="' + pagePath + '"]');
Scott Main5a1123e2012-09-26 12:51:28 -0700205 var $selListItem;
Scott Maine4d8f1b2012-06-21 18:03:05 -0700206 if ($selNavLink.length) {
Scott Mainac2aef52013-02-12 14:15:23 -0800207 $selListItem = $selNavLink.closest('li');
Scott Maine4d8f1b2012-06-21 18:03:05 -0700208
209 // set up prev links
210 var $prevLink = [];
211 var $prevListItem = $selListItem.prev('li');
212
213 var crossBoundaries = ($("body.design").length > 0) || ($("body.guide").length > 0) ? true :
214false; // navigate across topic boundaries only in design docs
215 if ($prevListItem.length) {
216 if ($prevListItem.hasClass('nav-section')) {
Scott Main5a1123e2012-09-26 12:51:28 -0700217 // jump to last topic of previous section
218 $prevLink = $prevListItem.find('a:last');
219 } else if (!$selListItem.hasClass('nav-section')) {
Scott Maine4d8f1b2012-06-21 18:03:05 -0700220 // jump to previous topic in this section
221 $prevLink = $prevListItem.find('a:eq(0)');
222 }
223 } else {
224 // jump to this section's index page (if it exists)
225 var $parentListItem = $selListItem.parents('li');
226 $prevLink = $selListItem.parents('li').find('a');
227
228 // except if cross boundaries aren't allowed, and we're at the top of a section already
229 // (and there's another parent)
230 if (!crossBoundaries && $parentListItem.hasClass('nav-section')
231 && $selListItem.hasClass('nav-section')) {
232 $prevLink = [];
233 }
234 }
235
Scott Maine4d8f1b2012-06-21 18:03:05 -0700236 // set up next links
237 var $nextLink = [];
Scott Maine4d8f1b2012-06-21 18:03:05 -0700238 var startClass = false;
239 var training = $(".next-class-link").length; // decides whether to provide "next class" link
240 var isCrossingBoundary = false;
241
242 if ($selListItem.hasClass('nav-section')) {
243 // we're on an index page, jump to the first topic
Scott Mainb505ca62012-07-26 18:00:14 -0700244 $nextLink = $selListItem.find('ul:eq(0)').find('a:eq(0)');
Scott Maine4d8f1b2012-06-21 18:03:05 -0700245
246 // if there aren't any children, go to the next section (required for About pages)
247 if($nextLink.length == 0) {
248 $nextLink = $selListItem.next('li').find('a');
Scott Mainb505ca62012-07-26 18:00:14 -0700249 } else if ($('.topic-start-link').length) {
250 // as long as there's a child link and there is a "topic start link" (we're on a landing)
251 // then set the landing page "start link" text to be the first doc title
252 $('.topic-start-link').text($nextLink.text().toUpperCase());
Scott Maine4d8f1b2012-06-21 18:03:05 -0700253 }
254
Scott Main5a1123e2012-09-26 12:51:28 -0700255 // If the selected page has a description, then it's a class or article homepage
256 if ($selListItem.find('a[description]').length) {
257 // this means we're on a class landing page
Scott Maine4d8f1b2012-06-21 18:03:05 -0700258 startClass = true;
259 }
260 } else {
261 // jump to the next topic in this section (if it exists)
262 $nextLink = $selListItem.next('li').find('a:eq(0)');
263 if (!$nextLink.length) {
Scott Main5a1123e2012-09-26 12:51:28 -0700264 isCrossingBoundary = true;
265 // no more topics in this section, jump to the first topic in the next section
266 $nextLink = $selListItem.parents('li:eq(0)').next('li.nav-section').find('a:eq(0)');
267 if (!$nextLink.length) { // Go up another layer to look for next page (lesson > class > course)
268 $nextLink = $selListItem.parents('li:eq(1)').next('li.nav-section').find('a:eq(0)');
Scott Maine4d8f1b2012-06-21 18:03:05 -0700269 }
270 }
271 }
Scott Main5a1123e2012-09-26 12:51:28 -0700272
273 if (startClass) {
274 $('.start-class-link').attr('href', $nextLink.attr('href')).removeClass("hide");
275
276 // if there's no training bar (below the start button),
277 // then we need to add a bottom border to button
278 if (!$("#tb").length) {
279 $('.start-class-link').css({'border-bottom':'1px solid #DADADA'});
Scott Maine4d8f1b2012-06-21 18:03:05 -0700280 }
Scott Main5a1123e2012-09-26 12:51:28 -0700281 } else if (isCrossingBoundary && !$('body.design').length) { // Design always crosses boundaries
282 $('.content-footer.next-class').show();
283 $('.next-page-link').attr('href','')
284 .removeClass("hide").addClass("disabled")
285 .click(function() { return false; });
286
287 $('.next-class-link').attr('href',$nextLink.attr('href'))
288 .removeClass("hide").append($nextLink.html());
289 $('.next-class-link').find('.new').empty();
290 } else {
291 $('.next-page-link').attr('href', $nextLink.attr('href')).removeClass("hide");
292 }
293
294 if (!startClass && $prevLink.length) {
295 var prevHref = $prevLink.attr('href');
296 if (prevHref == SITE_ROOT + 'index.html') {
297 // Don't show Previous when it leads to the homepage
298 } else {
299 $('.prev-page-link').attr('href', $prevLink.attr('href')).removeClass("hide");
300 }
301 }
302
303 // If this is a training 'article', there should be no prev/next nav
304 // ... if the grandparent is the "nav" ... and it has no child list items...
305 if (training && $selListItem.parents('ul').eq(1).is('[id="nav"]') &&
306 !$selListItem.find('li').length) {
307 $('.next-page-link,.prev-page-link').attr('href','').addClass("disabled")
308 .click(function() { return false; });
Scott Maine4d8f1b2012-06-21 18:03:05 -0700309 }
310
311 }
Scott Main5a1123e2012-09-26 12:51:28 -0700312
313
314
315 // Set up the course landing pages for Training with class names and descriptions
316 if ($('body.trainingcourse').length) {
317 var $classLinks = $selListItem.find('ul li a').not('#nav .nav-section .nav-section ul a');
318 var $classDescriptions = $classLinks.attr('description');
319
320 var $olClasses = $('<ol class="class-list"></ol>');
321 var $liClass;
322 var $imgIcon;
323 var $h2Title;
324 var $pSummary;
325 var $olLessons;
326 var $liLesson;
327 $classLinks.each(function(index) {
328 $liClass = $('<li></li>');
329 $h2Title = $('<a class="title" href="'+$(this).attr('href')+'"><h2>' + $(this).html()+'</h2><span></span></a>');
330 $pSummary = $('<p class="description">' + $(this).attr('description') + '</p>');
331
332 $olLessons = $('<ol class="lesson-list"></ol>');
333
334 $lessons = $(this).closest('li').find('ul li a');
335
336 if ($lessons.length) {
337 $imgIcon = $('<img src="'+toRoot+'assets/images/resource-tutorial.png" alt=""/>');
338 $lessons.each(function(index) {
339 $olLessons.append('<li><a href="'+$(this).attr('href')+'">' + $(this).html()+'</a></li>');
340 });
341 } else {
342 $imgIcon = $('<img src="'+toRoot+'assets/images/resource-article.png" alt=""/>');
343 $pSummary.addClass('article');
344 }
345
346 $liClass.append($h2Title).append($imgIcon).append($pSummary).append($olLessons);
347 $olClasses.append($liClass);
348 });
349 $('.jd-descr').append($olClasses);
350 }
351
Scott Maine4d8f1b2012-06-21 18:03:05 -0700352
353
354
355 // Set up expand/collapse behavior
356 $('#nav li.nav-section .nav-section-header').click(function() {
357 var section = $(this).closest('li.nav-section');
358 if (section.hasClass('expanded')) {
359 /* hide me */
360 // if (section.hasClass('selected') || section.find('li').hasClass('selected')) {
361 // /* but not if myself or my descendents are selected */
362 // return;
363 // }
364 section.children('ul').slideUp(250, function() {
365 section.closest('li').removeClass('expanded');
366 resizeNav();
367 });
368 } else {
369 /* show me */
370 // first hide all other siblings
371 var $others = $('li.nav-section.expanded', $(this).closest('ul'));
372 $others.removeClass('expanded').children('ul').slideUp(250);
373
374 // now expand me
375 section.closest('li').addClass('expanded');
376 section.children('ul').slideDown(250, function() {
377 resizeNav();
378 });
379 }
380 });
381
382 $(".scroll-pane").scroll(function(event) {
383 event.preventDefault();
384 return false;
385 });
386
387 /* Resize nav height when window height changes */
388 $(window).resize(function() {
389 if ($('#side-nav').length == 0) return;
390 var stylesheet = $('link[rel="stylesheet"][class="fullscreen"]');
391 setNavBarLeftPos(); // do this even if sidenav isn't fixed because it could become fixed
392 // make sidenav behave when resizing the window and side-scolling is a concern
393 if (navBarIsFixed) {
394 if ((stylesheet.attr("disabled") == "disabled") || stylesheet.length == 0) {
395 updateSideNavPosition();
396 } else {
397 updateSidenavFullscreenWidth();
398 }
399 }
400 resizeNav();
401 });
402
403
404 // Set up fixed navbar
405 var prevScrollLeft = 0; // used to compare current position to previous position of horiz scroll
406 $(window).scroll(function(event) {
407 if ($('#side-nav').length == 0) return;
408 if (event.target.nodeName == "DIV") {
409 // Dump scroll event if the target is a DIV, because that means the event is coming
410 // from a scrollable div and so there's no need to make adjustments to our layout
411 return;
412 }
413 var scrollTop = $(window).scrollTop();
414 var headerHeight = $('#header').outerHeight();
415 var subheaderHeight = $('#nav-x').outerHeight();
416 var searchResultHeight = $('#searchResults').is(":visible") ?
417 $('#searchResults').outerHeight() : 0;
418 var totalHeaderHeight = headerHeight + subheaderHeight + searchResultHeight;
Scott Mainb8d06a52012-12-19 18:38:24 -0800419 // we set the navbar fixed when the scroll position is beyond the height of the site header...
Scott Maine4d8f1b2012-06-21 18:03:05 -0700420 var navBarShouldBeFixed = scrollTop > totalHeaderHeight;
Scott Mainb8d06a52012-12-19 18:38:24 -0800421 // ... except if the document content is shorter than the sidenav height.
422 // (this is necessary to avoid crazy behavior on OSX Lion due to overscroll bouncing)
423 if ($("#doc-col").height() < $("#side-nav").height()) {
424 navBarShouldBeFixed = false;
425 }
Scott Maine4d8f1b2012-06-21 18:03:05 -0700426
427 var scrollLeft = $(window).scrollLeft();
428 // When the sidenav is fixed and user scrolls horizontally, reposition the sidenav to match
429 if (navBarIsFixed && (scrollLeft != prevScrollLeft)) {
430 updateSideNavPosition();
431 prevScrollLeft = scrollLeft;
432 }
433
434 // Don't continue if the header is sufficently far away
435 // (to avoid intensive resizing that slows scrolling)
436 if (navBarIsFixed && navBarShouldBeFixed) {
437 return;
438 }
439
440 if (navBarIsFixed != navBarShouldBeFixed) {
441 if (navBarShouldBeFixed) {
442 // make it fixed
443 var width = $('#devdoc-nav').width();
444 $('#devdoc-nav')
445 .addClass('fixed')
446 .css({'width':width+'px'})
447 .prependTo('#body-content');
448 // add neato "back to top" button
449 $('#devdoc-nav a.totop').css({'display':'block','width':$("#nav").innerWidth()+'px'});
450
451 // update the sidenaav position for side scrolling
452 updateSideNavPosition();
453 } else {
454 // make it static again
455 $('#devdoc-nav')
456 .removeClass('fixed')
457 .css({'width':'auto','margin':''})
458 .prependTo('#side-nav');
459 $('#devdoc-nav a.totop').hide();
460 }
461 navBarIsFixed = navBarShouldBeFixed;
462 }
463
464 resizeNav(250); // pass true in order to delay the scrollbar re-initialization for performance
465 });
466
467
468 var navBarLeftPos;
469 if ($('#devdoc-nav').length) {
470 setNavBarLeftPos();
471 }
472
473
474 // Stop expand/collapse behavior when clicking on nav section links (since we're navigating away
475 // from the page)
476 $('.nav-section-header').find('a:eq(0)').click(function(evt) {
477 window.location.href = $(this).attr('href');
478 return false;
479 });
480
481 // Set up play-on-hover <video> tags.
482 $('video.play-on-hover').bind('click', function(){
483 $(this).get(0).load(); // in case the video isn't seekable
484 $(this).get(0).play();
485 });
486
487 // Set up tooltips
488 var TOOLTIP_MARGIN = 10;
Scott Maindb3678b2012-10-23 14:13:41 -0700489 $('acronym,.tooltip-link').each(function() {
Scott Maine4d8f1b2012-06-21 18:03:05 -0700490 var $target = $(this);
491 var $tooltip = $('<div>')
492 .addClass('tooltip-box')
Scott Maindb3678b2012-10-23 14:13:41 -0700493 .append($target.attr('title'))
Scott Maine4d8f1b2012-06-21 18:03:05 -0700494 .hide()
495 .appendTo('body');
496 $target.removeAttr('title');
497
498 $target.hover(function() {
499 // in
500 var targetRect = $target.offset();
501 targetRect.width = $target.width();
502 targetRect.height = $target.height();
503
504 $tooltip.css({
505 left: targetRect.left,
506 top: targetRect.top + targetRect.height + TOOLTIP_MARGIN
507 });
508 $tooltip.addClass('below');
509 $tooltip.show();
510 }, function() {
511 // out
512 $tooltip.hide();
513 });
514 });
515
516 // Set up <h2> deeplinks
517 $('h2').click(function() {
518 var id = $(this).attr('id');
519 if (id) {
520 document.location.hash = id;
521 }
522 });
523
524 //Loads the +1 button
525 var po = document.createElement('script'); po.type = 'text/javascript'; po.async = true;
526 po.src = 'https://apis.google.com/js/plusone.js';
527 var s = document.getElementsByTagName('script')[0]; s.parentNode.insertBefore(po, s);
528
529
530 // Revise the sidenav widths to make room for the scrollbar
531 // which avoids the visible width from changing each time the bar appears
532 var $sidenav = $("#side-nav");
533 var sidenav_width = parseInt($sidenav.innerWidth());
534
535 $("#devdoc-nav #nav").css("width", sidenav_width - 4 + "px"); // 4px is scrollbar width
536
537
538 $(".scroll-pane").removeAttr("tabindex"); // get rid of tabindex added by jscroller
539
540 if ($(".scroll-pane").length > 1) {
541 // Check if there's a user preference for the panel heights
542 var cookieHeight = readCookie("reference_height");
543 if (cookieHeight) {
544 restoreHeight(cookieHeight);
545 }
546 }
547
548 resizeNav();
549
Scott Main015d6162013-01-29 09:01:52 -0800550 /* init the language selector based on user cookie for lang */
551 loadLangPref();
552 changeNavLang(getLangPref());
553
554 /* setup event handlers to ensure the overflow menu is visible while picking lang */
555 $("#language select")
556 .mousedown(function() {
557 $("div.morehover").addClass("hover"); })
558 .blur(function() {
559 $("div.morehover").removeClass("hover"); });
560
561 /* some global variable setup */
562 resizePackagesNav = $("#resize-packages-nav");
563 classesNav = $("#classes-nav");
564 devdocNav = $("#devdoc-nav");
565
566 var cookiePath = "";
567 if (location.href.indexOf("/reference/") != -1) {
568 cookiePath = "reference_";
569 } else if (location.href.indexOf("/guide/") != -1) {
570 cookiePath = "guide_";
571 } else if (location.href.indexOf("/tools/") != -1) {
572 cookiePath = "tools_";
573 } else if (location.href.indexOf("/training/") != -1) {
574 cookiePath = "training_";
575 } else if (location.href.indexOf("/design/") != -1) {
576 cookiePath = "design_";
577 } else if (location.href.indexOf("/distribute/") != -1) {
578 cookiePath = "distribute_";
579 }
Scott Maine4d8f1b2012-06-21 18:03:05 -0700580
581});
Scott Main7e447ed2013-02-19 17:22:37 -0800582// END of the onload event
Scott Maine4d8f1b2012-06-21 18:03:05 -0700583
584
Scott Mainf6145542013-04-01 16:38:11 -0700585function highlightSidenav() {
586 // select current page in sidenav and header, and set up prev/next links if they exist
587 var $selNavLink = $('#nav').find('a[href="' + mPagePath + '"]');
588 var $selListItem;
589 if ($selNavLink.length) {
590
591 // Find this page's <li> in sidenav and set selected
592 $selListItem = $selNavLink.closest('li');
593 $selListItem.addClass('selected');
594
595 // Traverse up the tree and expand all parent nav-sections
596 $selNavLink.parents('li.nav-section').each(function() {
597 $(this).addClass('expanded');
598 $(this).children('ul').show();
599 });
600 }
601}
602
Scott Maine4d8f1b2012-06-21 18:03:05 -0700603
604function toggleFullscreen(enable) {
605 var delay = 20;
606 var enabled = true;
607 var stylesheet = $('link[rel="stylesheet"][class="fullscreen"]');
608 if (enable) {
609 // Currently NOT USING fullscreen; enable fullscreen
610 stylesheet.removeAttr('disabled');
611 $('#nav-swap .fullscreen').removeClass('disabled');
612 $('#devdoc-nav').css({left:''});
613 setTimeout(updateSidenavFullscreenWidth,delay); // need to wait a moment for css to switch
614 enabled = true;
615 } else {
616 // Currently USING fullscreen; disable fullscreen
617 stylesheet.attr('disabled', 'disabled');
618 $('#nav-swap .fullscreen').addClass('disabled');
619 setTimeout(updateSidenavFixedWidth,delay); // need to wait a moment for css to switch
620 enabled = false;
621 }
622 writeCookie("fullscreen", enabled, null, null);
623 setNavBarLeftPos();
624 resizeNav(delay);
625 updateSideNavPosition();
626 setTimeout(initSidenavHeightResize,delay);
627}
628
629
630function setNavBarLeftPos() {
631 navBarLeftPos = $('#body-content').offset().left;
632}
633
634
635function updateSideNavPosition() {
636 var newLeft = $(window).scrollLeft() - navBarLeftPos;
637 $('#devdoc-nav').css({left: -newLeft});
638 $('#devdoc-nav .totop').css({left: -(newLeft - parseInt($('#side-nav').css('margin-left')))});
639}
640
641
642
643
644
645
646
647
648// TODO: use $(document).ready instead
649function addLoadEvent(newfun) {
650 var current = window.onload;
651 if (typeof window.onload != 'function') {
652 window.onload = newfun;
653 } else {
654 window.onload = function() {
655 current();
656 newfun();
657 }
658 }
659}
660
661var agent = navigator['userAgent'].toLowerCase();
662// If a mobile phone, set flag and do mobile setup
663if ((agent.indexOf("mobile") != -1) || // android, iphone, ipod
664 (agent.indexOf("blackberry") != -1) ||
665 (agent.indexOf("webos") != -1) ||
666 (agent.indexOf("mini") != -1)) { // opera mini browsers
667 isMobile = true;
668}
669
670
Scott Maine4d8f1b2012-06-21 18:03:05 -0700671addLoadEvent( function() {
672 $("pre:not(.no-pretty-print)").addClass("prettyprint");
673 prettyPrint();
674} );
675
Scott Maine4d8f1b2012-06-21 18:03:05 -0700676
677
678
679/* ######### RESIZE THE SIDENAV HEIGHT ########## */
680
681function resizeNav(delay) {
682 var $nav = $("#devdoc-nav");
683 var $window = $(window);
684 var navHeight;
685
686 // Get the height of entire window and the total header height.
687 // Then figure out based on scroll position whether the header is visible
688 var windowHeight = $window.height();
689 var scrollTop = $window.scrollTop();
690 var headerHeight = $('#header').outerHeight();
691 var subheaderHeight = $('#nav-x').outerHeight();
692 var headerVisible = (scrollTop < (headerHeight + subheaderHeight));
693
694 // get the height of space between nav and top of window.
695 // Could be either margin or top position, depending on whether the nav is fixed.
696 var topMargin = (parseInt($nav.css('margin-top')) || parseInt($nav.css('top'))) + 1;
697 // add 1 for the #side-nav bottom margin
698
699 // Depending on whether the header is visible, set the side nav's height.
700 if (headerVisible) {
701 // The sidenav height grows as the header goes off screen
702 navHeight = windowHeight - (headerHeight + subheaderHeight - scrollTop) - topMargin;
703 } else {
704 // Once header is off screen, the nav height is almost full window height
705 navHeight = windowHeight - topMargin;
706 }
707
708
709
710 $scrollPanes = $(".scroll-pane");
711 if ($scrollPanes.length > 1) {
712 // subtract the height of the api level widget and nav swapper from the available nav height
713 navHeight -= ($('#api-nav-header').outerHeight(true) + $('#nav-swap').outerHeight(true));
714
715 $("#swapper").css({height:navHeight + "px"});
716 if ($("#nav-tree").is(":visible")) {
717 $("#nav-tree").css({height:navHeight});
718 }
719
720 var classesHeight = navHeight - parseInt($("#resize-packages-nav").css("height")) - 10 + "px";
721 //subtract 10px to account for drag bar
722
723 // if the window becomes small enough to make the class panel height 0,
724 // then the package panel should begin to shrink
725 if (parseInt(classesHeight) <= 0) {
726 $("#resize-packages-nav").css({height:navHeight - 10}); //subtract 10px for drag bar
727 $("#packages-nav").css({height:navHeight - 10});
728 }
729
730 $("#classes-nav").css({'height':classesHeight, 'margin-top':'10px'});
731 $("#classes-nav .jspContainer").css({height:classesHeight});
732
733
734 } else {
735 $nav.height(navHeight);
736 }
737
738 if (delay) {
739 updateFromResize = true;
740 delayedReInitScrollbars(delay);
741 } else {
742 reInitScrollbars();
743 }
744
745}
746
747var updateScrollbars = false;
748var updateFromResize = false;
749
750/* Re-initialize the scrollbars to account for changed nav size.
751 * This method postpones the actual update by a 1/4 second in order to optimize the
752 * scroll performance while the header is still visible, because re-initializing the
753 * scroll panes is an intensive process.
754 */
755function delayedReInitScrollbars(delay) {
756 // If we're scheduled for an update, but have received another resize request
757 // before the scheduled resize has occured, just ignore the new request
758 // (and wait for the scheduled one).
759 if (updateScrollbars && updateFromResize) {
760 updateFromResize = false;
761 return;
762 }
763
764 // We're scheduled for an update and the update request came from this method's setTimeout
765 if (updateScrollbars && !updateFromResize) {
766 reInitScrollbars();
767 updateScrollbars = false;
768 } else {
769 updateScrollbars = true;
770 updateFromResize = false;
771 setTimeout('delayedReInitScrollbars()',delay);
772 }
773}
774
775/* Re-initialize the scrollbars to account for changed nav size. */
776function reInitScrollbars() {
777 var pane = $(".scroll-pane").each(function(){
778 var api = $(this).data('jsp');
779 if (!api) { setTimeout(reInitScrollbars,300); return;}
780 api.reinitialise( {verticalGutter:0} );
781 });
782 $(".scroll-pane").removeAttr("tabindex"); // get rid of tabindex added by jscroller
783}
784
785
786/* Resize the height of the nav panels in the reference,
787 * and save the new size to a cookie */
788function saveNavPanels() {
789 var basePath = getBaseUri(location.pathname);
790 var section = basePath.substring(1,basePath.indexOf("/",1));
791 writeCookie("height", resizePackagesNav.css("height"), section, null);
792}
793
794
795
796function restoreHeight(packageHeight) {
797 $("#resize-packages-nav").height(packageHeight);
798 $("#packages-nav").height(packageHeight);
799 // var classesHeight = navHeight - packageHeight;
800 // $("#classes-nav").css({height:classesHeight});
801 // $("#classes-nav .jspContainer").css({height:classesHeight});
802}
803
804
805
806/* ######### END RESIZE THE SIDENAV HEIGHT ########## */
807
808
809
810
811
812/** Scroll the jScrollPane to make the currently selected item visible
813 This is called when the page finished loading. */
814function scrollIntoView(nav) {
815 var $nav = $("#"+nav);
816 var element = $nav.jScrollPane({/* ...settings... */});
817 var api = element.data('jsp');
818
819 if ($nav.is(':visible')) {
820 var $selected = $(".selected", $nav);
821 if ($selected.length == 0) return;
822
823 var selectedOffset = $selected.position().top;
824 if (selectedOffset + 90 > $nav.height()) { // add 90 so that we scroll up even
825 // if the current item is close to the bottom
826 api.scrollTo(0, selectedOffset - ($nav.height() / 4), false); // scroll the item into view
827 // to be 1/4 of the way from the top
828 }
829 }
830}
831
832
833
834
835
836
837/* Show popup dialogs */
838function showDialog(id) {
839 $dialog = $("#"+id);
840 $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>');
841 $dialog.wrapInner('<div/>');
842 $dialog.removeClass("hide");
843}
844
845
846
847
848
849/* ######### COOKIES! ########## */
850
851function readCookie(cookie) {
852 var myCookie = cookie_namespace+"_"+cookie+"=";
853 if (document.cookie) {
854 var index = document.cookie.indexOf(myCookie);
855 if (index != -1) {
856 var valStart = index + myCookie.length;
857 var valEnd = document.cookie.indexOf(";", valStart);
858 if (valEnd == -1) {
859 valEnd = document.cookie.length;
860 }
861 var val = document.cookie.substring(valStart, valEnd);
862 return val;
863 }
864 }
865 return 0;
866}
867
868function writeCookie(cookie, val, section, expiration) {
869 if (val==undefined) return;
870 section = section == null ? "_" : "_"+section+"_";
871 if (expiration == null) {
872 var date = new Date();
873 date.setTime(date.getTime()+(10*365*24*60*60*1000)); // default expiration is one week
874 expiration = date.toGMTString();
875 }
876 var cookieValue = cookie_namespace + section + cookie + "=" + val
877 + "; expires=" + expiration+"; path=/";
878 document.cookie = cookieValue;
879}
880
881/* ######### END COOKIES! ########## */
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
Scott Maind7026f72013-06-17 15:08:49 -0700901/* MISC LIBRARY FUNCTIONS */
Scott Maine4d8f1b2012-06-21 18:03:05 -0700902
903
904
905
906
907function toggle(obj, slide) {
908 var ul = $("ul:first", obj);
909 var li = ul.parent();
910 if (li.hasClass("closed")) {
911 if (slide) {
912 ul.slideDown("fast");
913 } else {
914 ul.show();
915 }
916 li.removeClass("closed");
917 li.addClass("open");
918 $(".toggle-img", li).attr("title", "hide pages");
919 } else {
920 ul.slideUp("fast");
921 li.removeClass("open");
922 li.addClass("closed");
923 $(".toggle-img", li).attr("title", "show pages");
924 }
925}
926
927
Scott Maine4d8f1b2012-06-21 18:03:05 -0700928function buildToggleLists() {
929 $(".toggle-list").each(
930 function(i) {
931 $("div:first", this).append("<a class='toggle-img' href='#' title='show pages' onClick='toggle(this.parentNode.parentNode, true); return false;'></a>");
932 $(this).addClass("closed");
933 });
934}
935
936
937
Scott Maind7026f72013-06-17 15:08:49 -0700938function hideNestedItems(list, toggle) {
939 $list = $(list);
940 // hide nested lists
941 if($list.hasClass('showing')) {
942 $("li ol", $list).hide('fast');
943 $list.removeClass('showing');
944 // show nested lists
945 } else {
946 $("li ol", $list).show('fast');
947 $list.addClass('showing');
948 }
949 $(".more,.less",$(toggle)).toggle();
950}
Scott Maine4d8f1b2012-06-21 18:03:05 -0700951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979/* REFERENCE NAV SWAP */
980
981
982function getNavPref() {
983 var v = readCookie('reference_nav');
984 if (v != NAV_PREF_TREE) {
985 v = NAV_PREF_PANELS;
986 }
987 return v;
988}
989
990function chooseDefaultNav() {
991 nav_pref = getNavPref();
992 if (nav_pref == NAV_PREF_TREE) {
993 $("#nav-panels").toggle();
994 $("#panel-link").toggle();
995 $("#nav-tree").toggle();
996 $("#tree-link").toggle();
997 }
998}
999
1000function swapNav() {
1001 if (nav_pref == NAV_PREF_TREE) {
1002 nav_pref = NAV_PREF_PANELS;
1003 } else {
1004 nav_pref = NAV_PREF_TREE;
1005 init_default_navtree(toRoot);
1006 }
1007 var date = new Date();
1008 date.setTime(date.getTime()+(10*365*24*60*60*1000)); // keep this for 10 years
1009 writeCookie("nav", nav_pref, "reference", date.toGMTString());
1010
1011 $("#nav-panels").toggle();
1012 $("#panel-link").toggle();
1013 $("#nav-tree").toggle();
1014 $("#tree-link").toggle();
1015
1016 resizeNav();
1017
1018 // Gross nasty hack to make tree view show up upon first swap by setting height manually
1019 $("#nav-tree .jspContainer:visible")
1020 .css({'height':$("#nav-tree .jspContainer .jspPane").height() +'px'});
1021 // Another nasty hack to make the scrollbar appear now that we have height
1022 resizeNav();
1023
1024 if ($("#nav-tree").is(':visible')) {
1025 scrollIntoView("nav-tree");
1026 } else {
1027 scrollIntoView("packages-nav");
1028 scrollIntoView("classes-nav");
1029 }
1030}
1031
1032
1033
Scott Mainf5089842012-08-14 16:31:07 -07001034/* ############################################ */
Scott Maine4d8f1b2012-06-21 18:03:05 -07001035/* ########## LOCALIZATION ############ */
Scott Mainf5089842012-08-14 16:31:07 -07001036/* ############################################ */
Scott Maine4d8f1b2012-06-21 18:03:05 -07001037
1038function getBaseUri(uri) {
1039 var intlUrl = (uri.substring(0,6) == "/intl/");
1040 if (intlUrl) {
1041 base = uri.substring(uri.indexOf('intl/')+5,uri.length);
1042 base = base.substring(base.indexOf('/')+1, base.length);
1043 //alert("intl, returning base url: /" + base);
1044 return ("/" + base);
1045 } else {
1046 //alert("not intl, returning uri as found.");
1047 return uri;
1048 }
1049}
1050
1051function requestAppendHL(uri) {
1052//append "?hl=<lang> to an outgoing request (such as to blog)
1053 var lang = getLangPref();
1054 if (lang) {
1055 var q = 'hl=' + lang;
1056 uri += '?' + q;
1057 window.location = uri;
1058 return false;
1059 } else {
1060 return true;
1061 }
1062}
1063
1064
Scott Maine4d8f1b2012-06-21 18:03:05 -07001065function changeNavLang(lang) {
Scott Main6eb95f12012-10-02 17:12:23 -07001066 var $links = $("#devdoc-nav,#header,#nav-x,.training-nav-top,.content-footer").find("a["+lang+"-lang]");
1067 $links.each(function(i){ // for each link with a translation
1068 var $link = $(this);
1069 if (lang != "en") { // No need to worry about English, because a language change invokes new request
1070 // put the desired language from the attribute as the text
1071 $link.text($link.attr(lang+"-lang"))
Scott Maine4d8f1b2012-06-21 18:03:05 -07001072 }
Scott Main6eb95f12012-10-02 17:12:23 -07001073 });
Scott Maine4d8f1b2012-06-21 18:03:05 -07001074}
1075
Scott Main015d6162013-01-29 09:01:52 -08001076function changeLangPref(lang, submit) {
Scott Maine4d8f1b2012-06-21 18:03:05 -07001077 var date = new Date();
1078 expires = date.toGMTString(date.setTime(date.getTime()+(10*365*24*60*60*1000)));
1079 // keep this for 50 years
1080 //alert("expires: " + expires)
1081 writeCookie("pref_lang", lang, null, expires);
Scott Main015d6162013-01-29 09:01:52 -08001082
1083 // ####### TODO: Remove this condition once we're stable on devsite #######
1084 // This condition is only needed if we still need to support legacy GAE server
1085 if (devsite) {
1086 // Switch language when on Devsite server
1087 if (submit) {
1088 $("#setlang").submit();
1089 }
1090 } else {
1091 // Switch language when on legacy GAE server
Scott Main015d6162013-01-29 09:01:52 -08001092 if (submit) {
1093 window.location = getBaseUri(location.pathname);
1094 }
Scott Maine4d8f1b2012-06-21 18:03:05 -07001095 }
1096}
1097
1098function loadLangPref() {
1099 var lang = readCookie("pref_lang");
1100 if (lang != 0) {
1101 $("#language").find("option[value='"+lang+"']").attr("selected",true);
1102 }
1103}
1104
1105function getLangPref() {
1106 var lang = $("#language").find(":selected").attr("value");
1107 if (!lang) {
1108 lang = readCookie("pref_lang");
1109 }
1110 return (lang != 0) ? lang : 'en';
1111}
1112
1113/* ########## END LOCALIZATION ############ */
1114
1115
1116
1117
1118
1119
1120/* Used to hide and reveal supplemental content, such as long code samples.
1121 See the companion CSS in android-developer-docs.css */
1122function toggleContent(obj) {
1123 var div = $(obj.parentNode.parentNode);
1124 var toggleMe = $(".toggle-content-toggleme",div);
1125 if (div.hasClass("closed")) { // if it's closed, open it
1126 toggleMe.slideDown();
1127 $(".toggle-content-text", obj).toggle();
1128 div.removeClass("closed").addClass("open");
1129 $(".toggle-content-img", div).attr("title", "hide").attr("src", toRoot
1130 + "assets/images/triangle-opened.png");
1131 } else { // if it's open, close it
1132 toggleMe.slideUp('fast', function() { // Wait until the animation is done before closing arrow
1133 $(".toggle-content-text", obj).toggle();
1134 div.removeClass("open").addClass("closed");
1135 $(".toggle-content-img", div).attr("title", "show").attr("src", toRoot
1136 + "assets/images/triangle-closed.png");
1137 });
1138 }
1139 return false;
1140}
Scott Mainf5089842012-08-14 16:31:07 -07001141
1142
Scott Maindb3678b2012-10-23 14:13:41 -07001143/* New version of expandable content */
1144function toggleExpandable(link,id) {
1145 if($(id).is(':visible')) {
1146 $(id).slideUp();
1147 $(link).removeClass('expanded');
1148 } else {
1149 $(id).slideDown();
1150 $(link).addClass('expanded');
1151 }
1152}
1153
1154function hideExpandable(ids) {
1155 $(ids).slideUp();
Scott Main55d99832012-11-12 23:03:59 -08001156 $(ids).prev('h4').find('a.expandable').removeClass('expanded');
Scott Maindb3678b2012-10-23 14:13:41 -07001157}
1158
Scott Mainf5089842012-08-14 16:31:07 -07001159
1160
1161
1162
Robert Lyd2dd6e52012-11-29 21:28:48 -08001163/*
Scott Mainf5089842012-08-14 16:31:07 -07001164 * Slideshow 1.0
1165 * Used on /index.html and /develop/index.html for carousel
1166 *
1167 * Sample usage:
1168 * HTML -
1169 * <div class="slideshow-container">
1170 * <a href="" class="slideshow-prev">Prev</a>
1171 * <a href="" class="slideshow-next">Next</a>
1172 * <ul>
1173 * <li class="item"><img src="images/marquee1.jpg"></li>
1174 * <li class="item"><img src="images/marquee2.jpg"></li>
1175 * <li class="item"><img src="images/marquee3.jpg"></li>
1176 * <li class="item"><img src="images/marquee4.jpg"></li>
1177 * </ul>
1178 * </div>
1179 *
1180 * <script type="text/javascript">
1181 * $('.slideshow-container').dacSlideshow({
1182 * auto: true,
1183 * btnPrev: '.slideshow-prev',
1184 * btnNext: '.slideshow-next'
1185 * });
1186 * </script>
1187 *
1188 * Options:
1189 * btnPrev: optional identifier for previous button
1190 * btnNext: optional identifier for next button
Scott Maineb410352013-01-14 19:03:40 -08001191 * btnPause: optional identifier for pause button
Scott Mainf5089842012-08-14 16:31:07 -07001192 * auto: whether or not to auto-proceed
1193 * speed: animation speed
1194 * autoTime: time between auto-rotation
1195 * easing: easing function for transition
1196 * start: item to select by default
1197 * scroll: direction to scroll in
1198 * pagination: whether or not to include dotted pagination
1199 *
1200 */
1201
1202 (function($) {
1203 $.fn.dacSlideshow = function(o) {
1204
1205 //Options - see above
1206 o = $.extend({
1207 btnPrev: null,
1208 btnNext: null,
Scott Maineb410352013-01-14 19:03:40 -08001209 btnPause: null,
Scott Mainf5089842012-08-14 16:31:07 -07001210 auto: true,
1211 speed: 500,
1212 autoTime: 12000,
1213 easing: null,
1214 start: 0,
1215 scroll: 1,
1216 pagination: true
1217
1218 }, o || {});
1219
1220 //Set up a carousel for each
1221 return this.each(function() {
1222
1223 var running = false;
1224 var animCss = o.vertical ? "top" : "left";
1225 var sizeCss = o.vertical ? "height" : "width";
1226 var div = $(this);
1227 var ul = $("ul", div);
1228 var tLi = $("li", ul);
1229 var tl = tLi.size();
1230 var timer = null;
1231
1232 var li = $("li", ul);
1233 var itemLength = li.size();
1234 var curr = o.start;
1235
1236 li.css({float: o.vertical ? "none" : "left"});
1237 ul.css({margin: "0", padding: "0", position: "relative", "list-style-type": "none", "z-index": "1"});
1238 div.css({position: "relative", "z-index": "2", left: "0px"});
1239
1240 var liSize = o.vertical ? height(li) : width(li);
1241 var ulSize = liSize * itemLength;
1242 var divSize = liSize;
1243
1244 li.css({width: li.width(), height: li.height()});
1245 ul.css(sizeCss, ulSize+"px").css(animCss, -(curr*liSize));
1246
1247 div.css(sizeCss, divSize+"px");
1248
1249 //Pagination
1250 if (o.pagination) {
1251 var pagination = $("<div class='pagination'></div>");
1252 var pag_ul = $("<ul></ul>");
1253 if (tl > 1) {
1254 for (var i=0;i<tl;i++) {
1255 var li = $("<li>"+i+"</li>");
1256 pag_ul.append(li);
1257 if (i==o.start) li.addClass('active');
1258 li.click(function() {
1259 go(parseInt($(this).text()));
1260 })
1261 }
1262 pagination.append(pag_ul);
1263 div.append(pagination);
1264 }
1265 }
1266
1267 //Previous button
1268 if(o.btnPrev)
1269 $(o.btnPrev).click(function(e) {
1270 e.preventDefault();
1271 return go(curr-o.scroll);
1272 });
1273
1274 //Next button
1275 if(o.btnNext)
1276 $(o.btnNext).click(function(e) {
1277 e.preventDefault();
1278 return go(curr+o.scroll);
1279 });
Scott Maineb410352013-01-14 19:03:40 -08001280
1281 //Pause button
1282 if(o.btnPause)
1283 $(o.btnPause).click(function(e) {
1284 e.preventDefault();
1285 if ($(this).hasClass('paused')) {
1286 startRotateTimer();
1287 } else {
1288 pauseRotateTimer();
1289 }
1290 });
Scott Mainf5089842012-08-14 16:31:07 -07001291
1292 //Auto rotation
1293 if(o.auto) startRotateTimer();
1294
1295 function startRotateTimer() {
1296 clearInterval(timer);
1297 timer = setInterval(function() {
1298 if (curr == tl-1) {
1299 go(0);
1300 } else {
1301 go(curr+o.scroll);
1302 }
1303 }, o.autoTime);
Scott Maineb410352013-01-14 19:03:40 -08001304 $(o.btnPause).removeClass('paused');
1305 }
1306
1307 function pauseRotateTimer() {
1308 clearInterval(timer);
1309 $(o.btnPause).addClass('paused');
Scott Mainf5089842012-08-14 16:31:07 -07001310 }
1311
1312 //Go to an item
1313 function go(to) {
1314 if(!running) {
1315
1316 if(to<0) {
1317 to = itemLength-1;
1318 } else if (to>itemLength-1) {
1319 to = 0;
1320 }
1321 curr = to;
1322
1323 running = true;
1324
1325 ul.animate(
1326 animCss == "left" ? { left: -(curr*liSize) } : { top: -(curr*liSize) } , o.speed, o.easing,
1327 function() {
1328 running = false;
1329 }
1330 );
1331
1332 $(o.btnPrev + "," + o.btnNext).removeClass("disabled");
1333 $( (curr-o.scroll<0 && o.btnPrev)
1334 ||
1335 (curr+o.scroll > itemLength && o.btnNext)
1336 ||
1337 []
1338 ).addClass("disabled");
1339
1340
1341 var nav_items = $('li', pagination);
1342 nav_items.removeClass('active');
1343 nav_items.eq(to).addClass('active');
1344
1345
1346 }
1347 if(o.auto) startRotateTimer();
1348 return false;
1349 };
1350 });
1351 };
1352
1353 function css(el, prop) {
1354 return parseInt($.css(el[0], prop)) || 0;
1355 };
1356 function width(el) {
1357 return el[0].offsetWidth + css(el, 'marginLeft') + css(el, 'marginRight');
1358 };
1359 function height(el) {
1360 return el[0].offsetHeight + css(el, 'marginTop') + css(el, 'marginBottom');
1361 };
1362
1363 })(jQuery);
1364
1365
Robert Lyd2dd6e52012-11-29 21:28:48 -08001366/*
Scott Mainf5089842012-08-14 16:31:07 -07001367 * dacSlideshow 1.0
1368 * Used on develop/index.html for side-sliding tabs
1369 *
1370 * Sample usage:
1371 * HTML -
1372 * <div class="slideshow-container">
1373 * <a href="" class="slideshow-prev">Prev</a>
1374 * <a href="" class="slideshow-next">Next</a>
1375 * <ul>
1376 * <li class="item"><img src="images/marquee1.jpg"></li>
1377 * <li class="item"><img src="images/marquee2.jpg"></li>
1378 * <li class="item"><img src="images/marquee3.jpg"></li>
1379 * <li class="item"><img src="images/marquee4.jpg"></li>
1380 * </ul>
1381 * </div>
1382 *
1383 * <script type="text/javascript">
1384 * $('.slideshow-container').dacSlideshow({
1385 * auto: true,
1386 * btnPrev: '.slideshow-prev',
1387 * btnNext: '.slideshow-next'
1388 * });
1389 * </script>
1390 *
1391 * Options:
1392 * btnPrev: optional identifier for previous button
1393 * btnNext: optional identifier for next button
1394 * auto: whether or not to auto-proceed
1395 * speed: animation speed
1396 * autoTime: time between auto-rotation
1397 * easing: easing function for transition
1398 * start: item to select by default
1399 * scroll: direction to scroll in
1400 * pagination: whether or not to include dotted pagination
1401 *
1402 */
1403 (function($) {
1404 $.fn.dacTabbedList = function(o) {
1405
1406 //Options - see above
1407 o = $.extend({
1408 speed : 250,
1409 easing: null,
1410 nav_id: null,
1411 frame_id: null
1412 }, o || {});
1413
1414 //Set up a carousel for each
1415 return this.each(function() {
1416
1417 var curr = 0;
1418 var running = false;
1419 var animCss = "margin-left";
1420 var sizeCss = "width";
1421 var div = $(this);
1422
1423 var nav = $(o.nav_id, div);
1424 var nav_li = $("li", nav);
1425 var nav_size = nav_li.size();
1426 var frame = div.find(o.frame_id);
1427 var content_width = $(frame).find('ul').width();
1428 //Buttons
1429 $(nav_li).click(function(e) {
1430 go($(nav_li).index($(this)));
1431 })
1432
1433 //Go to an item
1434 function go(to) {
1435 if(!running) {
1436 curr = to;
1437 running = true;
1438
1439 frame.animate({ 'margin-left' : -(curr*content_width) }, o.speed, o.easing,
1440 function() {
1441 running = false;
1442 }
1443 );
1444
1445
1446 nav_li.removeClass('active');
1447 nav_li.eq(to).addClass('active');
1448
1449
1450 }
1451 return false;
1452 };
1453 });
1454 };
1455
1456 function css(el, prop) {
1457 return parseInt($.css(el[0], prop)) || 0;
1458 };
1459 function width(el) {
1460 return el[0].offsetWidth + css(el, 'marginLeft') + css(el, 'marginRight');
1461 };
1462 function height(el) {
1463 return el[0].offsetHeight + css(el, 'marginTop') + css(el, 'marginBottom');
1464 };
1465
1466 })(jQuery);
1467
1468
1469
1470
1471
1472/* ######################################################## */
1473/* ################ SEARCH SUGGESTIONS ################## */
1474/* ######################################################## */
1475
1476
Scott Main7e447ed2013-02-19 17:22:37 -08001477
Scott Main0e76e7e2013-03-12 10:24:07 -07001478var gSelectedIndex = -1; // the index position of currently highlighted suggestion
1479var gSelectedColumn = -1; // which column of suggestion lists is currently focused
1480
Scott Mainf5089842012-08-14 16:31:07 -07001481var gMatches = new Array();
1482var gLastText = "";
Scott Mainf5089842012-08-14 16:31:07 -07001483var gInitialized = false;
Scott Main7e447ed2013-02-19 17:22:37 -08001484var ROW_COUNT_FRAMEWORK = 20; // max number of results in list
1485var gListLength = 0;
1486
1487
1488var gGoogleMatches = new Array();
1489var ROW_COUNT_GOOGLE = 15; // max number of results in list
1490var gGoogleListLength = 0;
Scott Mainf5089842012-08-14 16:31:07 -07001491
Scott Main0e76e7e2013-03-12 10:24:07 -07001492var gDocsMatches = new Array();
1493var ROW_COUNT_DOCS = 100; // max number of results in list
1494var gDocsListLength = 0;
1495
Scott Mainde295272013-03-25 15:48:35 -07001496function onSuggestionClick(link) {
1497 // When user clicks a suggested document, track it
1498 _gaq.push(['_trackEvent', 'Suggestion Click', 'clicked: ' + $(link).text(),
1499 'from: ' + $("#search_autocomplete").val()]);
1500}
1501
Scott Mainf5089842012-08-14 16:31:07 -07001502function set_item_selected($li, selected)
1503{
1504 if (selected) {
1505 $li.attr('class','jd-autocomplete jd-selected');
1506 } else {
1507 $li.attr('class','jd-autocomplete');
1508 }
1509}
1510
1511function set_item_values(toroot, $li, match)
1512{
1513 var $link = $('a',$li);
1514 $link.html(match.__hilabel || match.label);
1515 $link.attr('href',toroot + match.link);
1516}
1517
Scott Main0e76e7e2013-03-12 10:24:07 -07001518function new_suggestion($list) {
Scott Main7e447ed2013-02-19 17:22:37 -08001519 var $li = $("<li class='jd-autocomplete'></li>");
1520 $list.append($li);
1521
1522 $li.mousedown(function() {
1523 window.location = this.firstChild.getAttribute("href");
1524 });
1525 $li.mouseover(function() {
Scott Main0e76e7e2013-03-12 10:24:07 -07001526 $('.search_filtered_wrapper li').removeClass('jd-selected');
Scott Main7e447ed2013-02-19 17:22:37 -08001527 $(this).addClass('jd-selected');
Scott Main0e76e7e2013-03-12 10:24:07 -07001528 gSelectedColumn = $(".search_filtered:visible").index($(this).closest('.search_filtered'));
1529 gSelectedIndex = $("li", $(".search_filtered:visible")[gSelectedColumn]).index(this);
Scott Main7e447ed2013-02-19 17:22:37 -08001530 });
Scott Mainde295272013-03-25 15:48:35 -07001531 $li.append("<a onclick='onSuggestionClick(this)'></a>");
Scott Main7e447ed2013-02-19 17:22:37 -08001532 $li.attr('class','show-item');
1533 return $li;
1534}
1535
Scott Mainf5089842012-08-14 16:31:07 -07001536function sync_selection_table(toroot)
1537{
Scott Mainf5089842012-08-14 16:31:07 -07001538 var $li; //list item jquery object
1539 var i; //list item iterator
Scott Main7e447ed2013-02-19 17:22:37 -08001540
Scott Main0e76e7e2013-03-12 10:24:07 -07001541 // if there are NO results at all, hide all columns
1542 if (!(gMatches.length > 0) && !(gGoogleMatches.length > 0) && !(gDocsMatches.length > 0)) {
1543 $('.suggest-card').hide(300);
1544 return;
1545 }
1546
1547 // if there are api results
Scott Main7e447ed2013-02-19 17:22:37 -08001548 if ((gMatches.length > 0) || (gGoogleMatches.length > 0)) {
Scott Main0e76e7e2013-03-12 10:24:07 -07001549 // reveal suggestion list
1550 $('.suggest-card.dummy').show();
1551 $('.suggest-card.reference').show();
1552 var listIndex = 0; // list index position
Scott Main7e447ed2013-02-19 17:22:37 -08001553
Scott Main0e76e7e2013-03-12 10:24:07 -07001554 // reset the lists
1555 $(".search_filtered_wrapper.reference li").remove();
Scott Main7e447ed2013-02-19 17:22:37 -08001556
Scott Main0e76e7e2013-03-12 10:24:07 -07001557 // ########### ANDROID RESULTS #############
1558 if (gMatches.length > 0) {
Scott Main7e447ed2013-02-19 17:22:37 -08001559
Scott Main0e76e7e2013-03-12 10:24:07 -07001560 // determine android results to show
1561 gListLength = gMatches.length < ROW_COUNT_FRAMEWORK ?
1562 gMatches.length : ROW_COUNT_FRAMEWORK;
1563 for (i=0; i<gListLength; i++) {
1564 var $li = new_suggestion($(".suggest-card.reference ul"));
1565 set_item_values(toroot, $li, gMatches[i]);
1566 set_item_selected($li, i == gSelectedIndex);
1567 }
1568 }
Scott Main7e447ed2013-02-19 17:22:37 -08001569
Scott Main0e76e7e2013-03-12 10:24:07 -07001570 // ########### GOOGLE RESULTS #############
1571 if (gGoogleMatches.length > 0) {
1572 // show header for list
1573 $(".suggest-card.reference ul").append("<li class='header'>in Google Services:</li>");
Scott Main7e447ed2013-02-19 17:22:37 -08001574
Scott Main0e76e7e2013-03-12 10:24:07 -07001575 // determine google results to show
1576 gGoogleListLength = gGoogleMatches.length < ROW_COUNT_GOOGLE ? gGoogleMatches.length : ROW_COUNT_GOOGLE;
1577 for (i=0; i<gGoogleListLength; i++) {
1578 var $li = new_suggestion($(".suggest-card.reference ul"));
1579 set_item_values(toroot, $li, gGoogleMatches[i]);
1580 set_item_selected($li, i == gSelectedIndex);
1581 }
1582 }
Scott Mainf5089842012-08-14 16:31:07 -07001583 } else {
Scott Main0e76e7e2013-03-12 10:24:07 -07001584 $('.suggest-card.reference').hide();
1585 $('.suggest-card.dummy').hide();
1586 }
1587
1588 // ########### JD DOC RESULTS #############
1589 if (gDocsMatches.length > 0) {
1590 // reset the lists
1591 $(".search_filtered_wrapper.docs li").remove();
1592
1593 // determine google results to show
1594 gDocsListLength = gDocsMatches.length < ROW_COUNT_DOCS ? gDocsMatches.length : ROW_COUNT_DOCS;
1595 for (i=0; i<gDocsListLength; i++) {
1596 var sugg = gDocsMatches[i];
1597 var $li;
1598 if (sugg.type == "design") {
1599 $li = new_suggestion($(".suggest-card.design ul"));
1600 } else
1601 if (sugg.type == "distribute") {
1602 $li = new_suggestion($(".suggest-card.distribute ul"));
1603 } else
1604 if (sugg.type == "training") {
1605 $li = new_suggestion($(".suggest-card.develop .child-card.training"));
1606 } else
1607 if (sugg.type == "guide"||"google") {
1608 $li = new_suggestion($(".suggest-card.develop .child-card.guides"));
1609 } else {
1610 continue;
1611 }
1612
1613 set_item_values(toroot, $li, sugg);
1614 set_item_selected($li, i == gSelectedIndex);
1615 }
1616
1617 // add heading and show or hide card
1618 if ($(".suggest-card.design li").length > 0) {
1619 $(".suggest-card.design ul").prepend("<li class='header'>Design:</li>");
1620 $(".suggest-card.design").show(300);
1621 } else {
1622 $('.suggest-card.design').hide(300);
1623 }
1624 if ($(".suggest-card.distribute li").length > 0) {
1625 $(".suggest-card.distribute ul").prepend("<li class='header'>Distribute:</li>");
1626 $(".suggest-card.distribute").show(300);
1627 } else {
1628 $('.suggest-card.distribute').hide(300);
1629 }
1630 if ($(".child-card.guides li").length > 0) {
1631 $(".child-card.guides").prepend("<li class='header'>Guides:</li>");
1632 $(".child-card.guides li").appendTo(".suggest-card.develop ul");
1633 }
1634 if ($(".child-card.training li").length > 0) {
1635 $(".child-card.training").prepend("<li class='header'>Training:</li>");
1636 $(".child-card.training li").appendTo(".suggest-card.develop ul");
1637 }
1638
1639 if ($(".suggest-card.develop li").length > 0) {
1640 $(".suggest-card.develop").show(300);
1641 } else {
1642 $('.suggest-card.develop').hide(300);
1643 }
1644
1645 } else {
1646 $('.search_filtered_wrapper.docs .suggest-card:not(.dummy)').hide(300);
Scott Mainf5089842012-08-14 16:31:07 -07001647 }
1648}
1649
Scott Main0e76e7e2013-03-12 10:24:07 -07001650/** Called by the search input's onkeydown and onkeyup events.
1651 * Handles navigation with keyboard arrows, Enter key to invoke search,
1652 * otherwise invokes search suggestions on key-up event.
1653 * @param e The JS event
1654 * @param kd True if the event is key-down
1655 * @param toroot A string for the site's root path
1656 * @returns True if the event should bubble up
1657 */
Scott Mainf5089842012-08-14 16:31:07 -07001658function search_changed(e, kd, toroot)
1659{
1660 var search = document.getElementById("search_autocomplete");
1661 var text = search.value.replace(/(^ +)|( +$)/g, '');
Scott Main0e76e7e2013-03-12 10:24:07 -07001662 // get the ul hosting the currently selected item
1663 gSelectedColumn = gSelectedColumn >= 0 ? gSelectedColumn : 0;
1664 var $columns = $(".search_filtered_wrapper").find(".search_filtered:visible");
1665 var $selectedUl = $columns[gSelectedColumn];
1666
Scott Mainf5089842012-08-14 16:31:07 -07001667 // show/hide the close button
1668 if (text != '') {
1669 $(".search .close").removeClass("hide");
1670 } else {
1671 $(".search .close").addClass("hide");
1672 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001673 // 27 = esc
1674 if (e.keyCode == 27) {
1675 // close all search results
1676 if (kd) $('.search .close').trigger('click');
1677 return true;
1678 }
Scott Mainf5089842012-08-14 16:31:07 -07001679 // 13 = enter
Scott Main0e76e7e2013-03-12 10:24:07 -07001680 else if (e.keyCode == 13) {
1681 if (gSelectedIndex < 0) {
1682 $('.suggest-card').hide();
Scott Main7e447ed2013-02-19 17:22:37 -08001683 if ($("#searchResults").is(":hidden") && (search.value != "")) {
1684 // if results aren't showing (and text not empty), return true to allow search to execute
Scott Mainf5089842012-08-14 16:31:07 -07001685 return true;
1686 } else {
1687 // otherwise, results are already showing, so allow ajax to auto refresh the results
1688 // and ignore this Enter press to avoid the reload.
1689 return false;
1690 }
1691 } else if (kd && gSelectedIndex >= 0) {
Scott Main0e76e7e2013-03-12 10:24:07 -07001692 // click the link corresponding to selected item
1693 $("a",$("li",$selectedUl)[gSelectedIndex]).get()[0].click();
Scott Mainf5089842012-08-14 16:31:07 -07001694 return false;
1695 }
1696 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001697 // Stop here if Google results are showing
1698 else if ($("#searchResults").is(":visible")) {
1699 return true;
1700 }
1701 // 38 UP ARROW
Scott Mainf5089842012-08-14 16:31:07 -07001702 else if (kd && (e.keyCode == 38)) {
Scott Main0e76e7e2013-03-12 10:24:07 -07001703 // if the next item is a header, skip it
1704 if ($($("li", $selectedUl)[gSelectedIndex-1]).hasClass("header")) {
Scott Mainf5089842012-08-14 16:31:07 -07001705 gSelectedIndex--;
Scott Main7e447ed2013-02-19 17:22:37 -08001706 }
1707 if (gSelectedIndex >= 0) {
Scott Main0e76e7e2013-03-12 10:24:07 -07001708 $('li', $selectedUl).removeClass('jd-selected');
Scott Main7e447ed2013-02-19 17:22:37 -08001709 gSelectedIndex--;
Scott Main0e76e7e2013-03-12 10:24:07 -07001710 $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
1711 // If user reaches top, reset selected column
1712 if (gSelectedIndex < 0) {
1713 gSelectedColumn = -1;
1714 }
Scott Mainf5089842012-08-14 16:31:07 -07001715 }
1716 return false;
1717 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001718 // 40 DOWN ARROW
Scott Mainf5089842012-08-14 16:31:07 -07001719 else if (kd && (e.keyCode == 40)) {
Scott Main0e76e7e2013-03-12 10:24:07 -07001720 // if the next item is a header, skip it
1721 if ($($("li", $selectedUl)[gSelectedIndex+1]).hasClass("header")) {
Scott Mainf5089842012-08-14 16:31:07 -07001722 gSelectedIndex++;
Scott Main7e447ed2013-02-19 17:22:37 -08001723 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001724 if ((gSelectedIndex < $("li", $selectedUl).length-1) ||
1725 ($($("li", $selectedUl)[gSelectedIndex+1]).hasClass("header"))) {
1726 $('li', $selectedUl).removeClass('jd-selected');
Scott Main7e447ed2013-02-19 17:22:37 -08001727 gSelectedIndex++;
Scott Main0e76e7e2013-03-12 10:24:07 -07001728 $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
Scott Mainf5089842012-08-14 16:31:07 -07001729 }
1730 return false;
1731 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001732 // Consider left/right arrow navigation
1733 // NOTE: Order of suggest columns are reverse order (index position 0 is on right)
1734 else if (kd && $columns.length > 1 && gSelectedColumn >= 0) {
1735 // 37 LEFT ARROW
1736 // go left only if current column is not left-most column (last column)
1737 if (e.keyCode == 37 && gSelectedColumn < $columns.length - 1) {
1738 $('li', $selectedUl).removeClass('jd-selected');
1739 gSelectedColumn++;
1740 $selectedUl = $columns[gSelectedColumn];
1741 // keep or reset the selected item to last item as appropriate
1742 gSelectedIndex = gSelectedIndex >
1743 $("li", $selectedUl).length-1 ?
1744 $("li", $selectedUl).length-1 : gSelectedIndex;
1745 // if the corresponding item is a header, move down
1746 if ($($("li", $selectedUl)[gSelectedIndex]).hasClass("header")) {
1747 gSelectedIndex++;
1748 }
1749 // set item selected
1750 $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
1751 return false;
1752 }
1753 // 39 RIGHT ARROW
1754 // go right only if current column is not the right-most column (first column)
1755 else if (e.keyCode == 39 && gSelectedColumn > 0) {
1756 $('li', $selectedUl).removeClass('jd-selected');
1757 gSelectedColumn--;
1758 $selectedUl = $columns[gSelectedColumn];
1759 // keep or reset the selected item to last item as appropriate
1760 gSelectedIndex = gSelectedIndex >
1761 $("li", $selectedUl).length-1 ?
1762 $("li", $selectedUl).length-1 : gSelectedIndex;
1763 // if the corresponding item is a header, move down
1764 if ($($("li", $selectedUl)[gSelectedIndex]).hasClass("header")) {
1765 gSelectedIndex++;
1766 }
1767 // set item selected
1768 $('li:nth-child('+(gSelectedIndex+1)+')', $selectedUl).addClass('jd-selected');
1769 return false;
1770 }
1771 }
1772
Scott Main7e447ed2013-02-19 17:22:37 -08001773 // if key-up event and not arrow down/up,
1774 // read the search query and add suggestsions to gMatches
Scott Main0e76e7e2013-03-12 10:24:07 -07001775 else if (!kd && (e.keyCode != 40)
1776 && (e.keyCode != 38)
1777 && (e.keyCode != 37)
1778 && (e.keyCode != 39)) {
1779 gSelectedIndex = -1;
Scott Mainf5089842012-08-14 16:31:07 -07001780 gMatches = new Array();
1781 matchedCount = 0;
Scott Main7e447ed2013-02-19 17:22:37 -08001782 gGoogleMatches = new Array();
1783 matchedCountGoogle = 0;
Scott Main0e76e7e2013-03-12 10:24:07 -07001784 gDocsMatches = new Array();
1785 matchedCountDocs = 0;
Scott Main7e447ed2013-02-19 17:22:37 -08001786
1787 // Search for Android matches
Scott Mainf5089842012-08-14 16:31:07 -07001788 for (var i=0; i<DATA.length; i++) {
1789 var s = DATA[i];
1790 if (text.length != 0 &&
1791 s.label.toLowerCase().indexOf(text.toLowerCase()) != -1) {
1792 gMatches[matchedCount] = s;
1793 matchedCount++;
1794 }
1795 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001796 rank_autocomplete_api_results(text, gMatches);
Scott Mainf5089842012-08-14 16:31:07 -07001797 for (var i=0; i<gMatches.length; i++) {
1798 var s = gMatches[i];
Scott Main7e447ed2013-02-19 17:22:37 -08001799 }
1800
1801
1802 // Search for Google matches
1803 for (var i=0; i<GOOGLE_DATA.length; i++) {
1804 var s = GOOGLE_DATA[i];
1805 if (text.length != 0 &&
1806 s.label.toLowerCase().indexOf(text.toLowerCase()) != -1) {
1807 gGoogleMatches[matchedCountGoogle] = s;
1808 matchedCountGoogle++;
Scott Mainf5089842012-08-14 16:31:07 -07001809 }
1810 }
Scott Main0e76e7e2013-03-12 10:24:07 -07001811 rank_autocomplete_api_results(text, gGoogleMatches);
Scott Main7e447ed2013-02-19 17:22:37 -08001812 for (var i=0; i<gGoogleMatches.length; i++) {
1813 var s = gGoogleMatches[i];
1814 }
1815
Scott Mainf5089842012-08-14 16:31:07 -07001816 highlight_autocomplete_result_labels(text);
Scott Main0e76e7e2013-03-12 10:24:07 -07001817
1818
1819
1820 // Search for JD docs
1821 if (text.length >= 3) {
1822 for (var i=0; i<JD_DATA.length; i++) {
1823 // Regex to match only the beginning of a word
1824 var textRegex = new RegExp("\\b" + text.toLowerCase(), "g");
1825 // current search comparison, with counters for tag and title,
1826 // used later to improve ranking
1827 var s = JD_DATA[i];
1828 s.matched_tag = 0;
1829 s.matched_title = 0;
1830 var matched = false;
1831
1832 // Check if query matches any tags; work backwards toward 1 to assist ranking
1833 for (var j = s.tags.length - 1; j >= 0; j--) {
1834 // it matches a tag
1835 if (s.tags[j].toLowerCase().match(textRegex)) {
1836 matched = true;
1837 s.matched_tag = j + 1; // add 1 to index position
1838 }
1839 }
1840 // Don't consider doc title for lessons (only for class landing pages)
1841 // ...it is not a training lesson (or is but has matched a tag)
1842 if (!(s.type == "training" && s.link.indexOf("index.html") == -1) || matched) {
1843 // it matches the doc title
1844 if (s.label.toLowerCase().match(textRegex)) {
1845 matched = true;
1846 s.matched_title = 1;
1847 }
1848 }
1849 if (matched) {
1850 gDocsMatches[matchedCountDocs] = s;
1851 matchedCountDocs++;
1852 }
1853 }
1854 rank_autocomplete_doc_results(text, gDocsMatches);
1855 }
1856
1857 // draw the suggestions
Scott Mainf5089842012-08-14 16:31:07 -07001858 sync_selection_table(toroot);
1859 return true; // allow the event to bubble up to the search api
1860 }
1861}
1862
Scott Main0e76e7e2013-03-12 10:24:07 -07001863/* Order the jd doc result list based on match quality */
1864function rank_autocomplete_doc_results(query, matches) {
1865 query = query || '';
1866 if (!matches || !matches.length)
1867 return;
1868
1869 var _resultScoreFn = function(match) {
1870 var score = 1.0;
1871
1872 // if the query matched a tag
1873 if (match.matched_tag > 0) {
1874 // multiply score by factor relative to position in tags list (max of 3)
1875 score *= 3 / match.matched_tag;
1876
1877 // if it also matched the title
1878 if (match.matched_title > 0) {
1879 score *= 2;
1880 }
1881 } else if (match.matched_title > 0) {
1882 score *= 3;
1883 }
1884
1885 return score;
1886 };
1887
1888 for (var i=0; i<matches.length; i++) {
1889 matches[i].__resultScore = _resultScoreFn(matches[i]);
1890 }
1891
1892 matches.sort(function(a,b){
1893 var n = b.__resultScore - a.__resultScore;
1894 if (n == 0) // lexicographical sort if scores are the same
1895 n = (a.label < b.label) ? -1 : 1;
1896 return n;
1897 });
1898}
1899
Scott Main7e447ed2013-02-19 17:22:37 -08001900/* Order the result list based on match quality */
Scott Main0e76e7e2013-03-12 10:24:07 -07001901function rank_autocomplete_api_results(query, matches) {
Scott Mainf5089842012-08-14 16:31:07 -07001902 query = query || '';
Scott Main7e447ed2013-02-19 17:22:37 -08001903 if (!matches || !matches.length)
Scott Mainf5089842012-08-14 16:31:07 -07001904 return;
1905
1906 // helper function that gets the last occurence index of the given regex
1907 // in the given string, or -1 if not found
1908 var _lastSearch = function(s, re) {
1909 if (s == '')
1910 return -1;
1911 var l = -1;
1912 var tmp;
1913 while ((tmp = s.search(re)) >= 0) {
1914 if (l < 0) l = 0;
1915 l += tmp;
1916 s = s.substr(tmp + 1);
1917 }
1918 return l;
1919 };
1920
1921 // helper function that counts the occurrences of a given character in
1922 // a given string
1923 var _countChar = function(s, c) {
1924 var n = 0;
1925 for (var i=0; i<s.length; i++)
1926 if (s.charAt(i) == c) ++n;
1927 return n;
1928 };
1929
1930 var queryLower = query.toLowerCase();
1931 var queryAlnum = (queryLower.match(/\w+/) || [''])[0];
1932 var partPrefixAlnumRE = new RegExp('\\b' + queryAlnum);
1933 var partExactAlnumRE = new RegExp('\\b' + queryAlnum + '\\b');
1934
1935 var _resultScoreFn = function(result) {
1936 // scores are calculated based on exact and prefix matches,
1937 // and then number of path separators (dots) from the last
1938 // match (i.e. favoring classes and deep package names)
1939 var score = 1.0;
1940 var labelLower = result.label.toLowerCase();
1941 var t;
1942 t = _lastSearch(labelLower, partExactAlnumRE);
1943 if (t >= 0) {
1944 // exact part match
1945 var partsAfter = _countChar(labelLower.substr(t + 1), '.');
1946 score *= 200 / (partsAfter + 1);
1947 } else {
1948 t = _lastSearch(labelLower, partPrefixAlnumRE);
1949 if (t >= 0) {
1950 // part prefix match
1951 var partsAfter = _countChar(labelLower.substr(t + 1), '.');
1952 score *= 20 / (partsAfter + 1);
1953 }
1954 }
1955
1956 return score;
1957 };
1958
Scott Main7e447ed2013-02-19 17:22:37 -08001959 for (var i=0; i<matches.length; i++) {
Scott Main0e76e7e2013-03-12 10:24:07 -07001960 // if the API is deprecated, default score is 0; otherwise, perform scoring
1961 if (matches[i].deprecated == "true") {
1962 matches[i].__resultScore = 0;
1963 } else {
1964 matches[i].__resultScore = _resultScoreFn(matches[i]);
1965 }
Scott Mainf5089842012-08-14 16:31:07 -07001966 }
1967
Scott Main7e447ed2013-02-19 17:22:37 -08001968 matches.sort(function(a,b){
Scott Mainf5089842012-08-14 16:31:07 -07001969 var n = b.__resultScore - a.__resultScore;
1970 if (n == 0) // lexicographical sort if scores are the same
1971 n = (a.label < b.label) ? -1 : 1;
1972 return n;
1973 });
1974}
1975
Scott Main7e447ed2013-02-19 17:22:37 -08001976/* Add emphasis to part of string that matches query */
Scott Mainf5089842012-08-14 16:31:07 -07001977function highlight_autocomplete_result_labels(query) {
1978 query = query || '';
Scott Main7e447ed2013-02-19 17:22:37 -08001979 if ((!gMatches || !gMatches.length) && (!gGoogleMatches || !gGoogleMatches.length))
Scott Mainf5089842012-08-14 16:31:07 -07001980 return;
1981
1982 var queryLower = query.toLowerCase();
1983 var queryAlnumDot = (queryLower.match(/[\w\.]+/) || [''])[0];
1984 var queryRE = new RegExp(
1985 '(' + queryAlnumDot.replace(/\./g, '\\.') + ')', 'ig');
1986 for (var i=0; i<gMatches.length; i++) {
1987 gMatches[i].__hilabel = gMatches[i].label.replace(
1988 queryRE, '<b>$1</b>');
1989 }
Scott Main7e447ed2013-02-19 17:22:37 -08001990 for (var i=0; i<gGoogleMatches.length; i++) {
1991 gGoogleMatches[i].__hilabel = gGoogleMatches[i].label.replace(
1992 queryRE, '<b>$1</b>');
1993 }
Scott Mainf5089842012-08-14 16:31:07 -07001994}
1995
1996function search_focus_changed(obj, focused)
1997{
1998 if (!focused) {
1999 if(obj.value == ""){
2000 $(".search .close").addClass("hide");
2001 }
Scott Main0e76e7e2013-03-12 10:24:07 -07002002 $(".suggest-card").hide();
Scott Mainf5089842012-08-14 16:31:07 -07002003 }
2004}
2005
2006function submit_search() {
2007 var query = document.getElementById('search_autocomplete').value;
2008 location.hash = 'q=' + query;
2009 loadSearchResults();
2010 $("#searchResults").slideDown('slow');
2011 return false;
2012}
2013
2014
2015function hideResults() {
2016 $("#searchResults").slideUp();
2017 $(".search .close").addClass("hide");
2018 location.hash = '';
2019
2020 $("#search_autocomplete").val("").blur();
2021
2022 // reset the ajax search callback to nothing, so results don't appear unless ENTER
2023 searchControl.setSearchStartingCallback(this, function(control, searcher, query) {});
Scott Main0e76e7e2013-03-12 10:24:07 -07002024
2025 // forcefully regain key-up event control (previously jacked by search api)
2026 $("#search_autocomplete").keyup(function(event) {
2027 return search_changed(event, false, toRoot);
2028 });
2029
Scott Mainf5089842012-08-14 16:31:07 -07002030 return false;
2031}
2032
2033
2034
2035/* ########################################################## */
2036/* ################ CUSTOM SEARCH ENGINE ################## */
2037/* ########################################################## */
2038
Scott Mainf5089842012-08-14 16:31:07 -07002039var searchControl;
Scott Main0e76e7e2013-03-12 10:24:07 -07002040google.load('search', '1', {"callback" : function() {
2041 searchControl = new google.search.SearchControl();
2042 } });
Scott Mainf5089842012-08-14 16:31:07 -07002043
2044function loadSearchResults() {
2045 document.getElementById("search_autocomplete").style.color = "#000";
2046
Scott Mainf5089842012-08-14 16:31:07 -07002047 searchControl = new google.search.SearchControl();
2048
2049 // use our existing search form and use tabs when multiple searchers are used
2050 drawOptions = new google.search.DrawOptions();
2051 drawOptions.setDrawMode(google.search.SearchControl.DRAW_MODE_TABBED);
2052 drawOptions.setInput(document.getElementById("search_autocomplete"));
2053
2054 // configure search result options
2055 searchOptions = new google.search.SearcherOptions();
2056 searchOptions.setExpandMode(GSearchControl.EXPAND_MODE_OPEN);
2057
2058 // configure each of the searchers, for each tab
2059 devSiteSearcher = new google.search.WebSearch();
2060 devSiteSearcher.setUserDefinedLabel("All");
2061 devSiteSearcher.setSiteRestriction("001482626316274216503:zu90b7s047u");
2062
2063 designSearcher = new google.search.WebSearch();
2064 designSearcher.setUserDefinedLabel("Design");
2065 designSearcher.setSiteRestriction("http://developer.android.com/design/");
2066
2067 trainingSearcher = new google.search.WebSearch();
2068 trainingSearcher.setUserDefinedLabel("Training");
2069 trainingSearcher.setSiteRestriction("http://developer.android.com/training/");
2070
2071 guidesSearcher = new google.search.WebSearch();
2072 guidesSearcher.setUserDefinedLabel("Guides");
2073 guidesSearcher.setSiteRestriction("http://developer.android.com/guide/");
2074
2075 referenceSearcher = new google.search.WebSearch();
2076 referenceSearcher.setUserDefinedLabel("Reference");
2077 referenceSearcher.setSiteRestriction("http://developer.android.com/reference/");
2078
Scott Maindf08ada2012-12-03 08:54:37 -08002079 googleSearcher = new google.search.WebSearch();
2080 googleSearcher.setUserDefinedLabel("Google Services");
2081 googleSearcher.setSiteRestriction("http://developer.android.com/google/");
2082
Scott Mainf5089842012-08-14 16:31:07 -07002083 blogSearcher = new google.search.WebSearch();
2084 blogSearcher.setUserDefinedLabel("Blog");
2085 blogSearcher.setSiteRestriction("http://android-developers.blogspot.com");
2086
2087 // add each searcher to the search control
2088 searchControl.addSearcher(devSiteSearcher, searchOptions);
2089 searchControl.addSearcher(designSearcher, searchOptions);
2090 searchControl.addSearcher(trainingSearcher, searchOptions);
2091 searchControl.addSearcher(guidesSearcher, searchOptions);
2092 searchControl.addSearcher(referenceSearcher, searchOptions);
Scott Maindf08ada2012-12-03 08:54:37 -08002093 searchControl.addSearcher(googleSearcher, searchOptions);
Scott Mainf5089842012-08-14 16:31:07 -07002094 searchControl.addSearcher(blogSearcher, searchOptions);
2095
2096 // configure result options
2097 searchControl.setResultSetSize(google.search.Search.LARGE_RESULTSET);
2098 searchControl.setLinkTarget(google.search.Search.LINK_TARGET_SELF);
2099 searchControl.setTimeoutInterval(google.search.SearchControl.TIMEOUT_SHORT);
2100 searchControl.setNoResultsString(google.search.SearchControl.NO_RESULTS_DEFAULT_STRING);
2101
2102 // upon ajax search, refresh the url and search title
2103 searchControl.setSearchStartingCallback(this, function(control, searcher, query) {
2104 updateResultTitle(query);
2105 var query = document.getElementById('search_autocomplete').value;
2106 location.hash = 'q=' + query;
2107 });
2108
Scott Mainde295272013-03-25 15:48:35 -07002109 // once search results load, set up click listeners
2110 searchControl.setSearchCompleteCallback(this, function(control, searcher, query) {
2111 addResultClickListeners();
2112 });
2113
Scott Mainf5089842012-08-14 16:31:07 -07002114 // draw the search results box
2115 searchControl.draw(document.getElementById("leftSearchControl"), drawOptions);
2116
2117 // get query and execute the search
2118 searchControl.execute(decodeURI(getQuery(location.hash)));
2119
2120 document.getElementById("search_autocomplete").focus();
2121 addTabListeners();
2122}
2123// End of loadSearchResults
2124
2125
2126google.setOnLoadCallback(function(){
2127 if (location.hash.indexOf("q=") == -1) {
2128 // if there's no query in the url, don't search and make sure results are hidden
2129 $('#searchResults').hide();
2130 return;
2131 } else {
2132 // first time loading search results for this page
2133 $('#searchResults').slideDown('slow');
2134 $(".search .close").removeClass("hide");
2135 loadSearchResults();
2136 }
2137}, true);
2138
2139// when an event on the browser history occurs (back, forward, load) requery hash and do search
2140$(window).hashchange( function(){
2141 // Exit if the hash isn't a search query or there's an error in the query
2142 if ((location.hash.indexOf("q=") == -1) || (query == "undefined")) {
2143 // If the results pane is open, close it.
2144 if (!$("#searchResults").is(":hidden")) {
2145 hideResults();
2146 }
2147 return;
2148 }
2149
2150 // Otherwise, we have a search to do
2151 var query = decodeURI(getQuery(location.hash));
2152 searchControl.execute(query);
2153 $('#searchResults').slideDown('slow');
2154 $("#search_autocomplete").focus();
2155 $(".search .close").removeClass("hide");
2156
2157 updateResultTitle(query);
2158});
2159
2160function updateResultTitle(query) {
2161 $("#searchTitle").html("Results for <em>" + escapeHTML(query) + "</em>");
2162}
2163
2164// forcefully regain key-up event control (previously jacked by search api)
2165$("#search_autocomplete").keyup(function(event) {
2166 return search_changed(event, false, toRoot);
2167});
2168
2169// add event listeners to each tab so we can track the browser history
2170function addTabListeners() {
2171 var tabHeaders = $(".gsc-tabHeader");
2172 for (var i = 0; i < tabHeaders.length; i++) {
2173 $(tabHeaders[i]).attr("id",i).click(function() {
2174 /*
2175 // make a copy of the page numbers for the search left pane
2176 setTimeout(function() {
2177 // remove any residual page numbers
2178 $('#searchResults .gsc-tabsArea .gsc-cursor-box.gs-bidi-start-align').remove();
2179 // move the page numbers to the left position; make a clone,
2180 // because the element is drawn to the DOM only once
2181 // and because we're going to remove it (previous line),
2182 // we need it to be available to move again as the user navigates
2183 $('#searchResults .gsc-webResult .gsc-cursor-box.gs-bidi-start-align:visible')
2184 .clone().appendTo('#searchResults .gsc-tabsArea');
2185 }, 200);
2186 */
2187 });
2188 }
2189 setTimeout(function(){$(tabHeaders[0]).click()},200);
2190}
2191
Scott Mainde295272013-03-25 15:48:35 -07002192// add analytics tracking events to each result link
2193function addResultClickListeners() {
2194 $("#searchResults a.gs-title").each(function(index, link) {
2195 // When user clicks enter for Google search results, track it
2196 $(link).click(function() {
2197 _gaq.push(['_trackEvent', 'Google Click', 'clicked: ' + $(this).text(),
2198 'from: ' + $("#search_autocomplete").val()]);
2199 });
2200 });
2201}
2202
Scott Mainf5089842012-08-14 16:31:07 -07002203
2204function getQuery(hash) {
2205 var queryParts = hash.split('=');
2206 return queryParts[1];
2207}
2208
2209/* returns the given string with all HTML brackets converted to entities
2210 TODO: move this to the site's JS library */
2211function escapeHTML(string) {
2212 return string.replace(/</g,"&lt;")
2213 .replace(/>/g,"&gt;");
2214}
2215
2216
2217
2218
2219
2220
2221
2222/* ######################################################## */
2223/* ################# JAVADOC REFERENCE ################### */
2224/* ######################################################## */
2225
Scott Main65511c02012-09-07 15:51:32 -07002226/* Initialize some droiddoc stuff, but only if we're in the reference */
Robert Ly67d75f12012-12-03 12:53:42 -08002227if (location.pathname.indexOf("/reference")) {
2228 if(!location.pathname.indexOf("/reference-gms/packages.html")
2229 && !location.pathname.indexOf("/reference-gcm/packages.html")
2230 && !location.pathname.indexOf("/reference/com/google") == 0) {
2231 $(document).ready(function() {
2232 // init available apis based on user pref
2233 changeApiLevel();
2234 initSidenavHeightResize()
2235 });
2236 }
Scott Main65511c02012-09-07 15:51:32 -07002237}
Scott Mainf5089842012-08-14 16:31:07 -07002238
2239var API_LEVEL_COOKIE = "api_level";
2240var minLevel = 1;
2241var maxLevel = 1;
2242
2243/******* SIDENAV DIMENSIONS ************/
2244
2245 function initSidenavHeightResize() {
2246 // Change the drag bar size to nicely fit the scrollbar positions
2247 var $dragBar = $(".ui-resizable-s");
2248 $dragBar.css({'width': $dragBar.parent().width() - 5 + "px"});
2249
2250 $( "#resize-packages-nav" ).resizable({
2251 containment: "#nav-panels",
2252 handles: "s",
2253 alsoResize: "#packages-nav",
2254 resize: function(event, ui) { resizeNav(); }, /* resize the nav while dragging */
2255 stop: function(event, ui) { saveNavPanels(); } /* once stopped, save the sizes to cookie */
2256 });
2257
2258 }
2259
2260function updateSidenavFixedWidth() {
2261 if (!navBarIsFixed) return;
2262 $('#devdoc-nav').css({
2263 'width' : $('#side-nav').css('width'),
2264 'margin' : $('#side-nav').css('margin')
2265 });
2266 $('#devdoc-nav a.totop').css({'display':'block','width':$("#nav").innerWidth()+'px'});
2267
2268 initSidenavHeightResize();
2269}
2270
2271function updateSidenavFullscreenWidth() {
2272 if (!navBarIsFixed) return;
2273 $('#devdoc-nav').css({
2274 'width' : $('#side-nav').css('width'),
2275 'margin' : $('#side-nav').css('margin')
2276 });
2277 $('#devdoc-nav .totop').css({'left': 'inherit'});
2278
2279 initSidenavHeightResize();
2280}
2281
2282function buildApiLevelSelector() {
2283 maxLevel = SINCE_DATA.length;
2284 var userApiLevel = parseInt(readCookie(API_LEVEL_COOKIE));
2285 userApiLevel = userApiLevel == 0 ? maxLevel : userApiLevel; // If there's no cookie (zero), use the max by default
2286
2287 minLevel = parseInt($("#doc-api-level").attr("class"));
2288 // Handle provisional api levels; the provisional level will always be the highest possible level
2289 // Provisional api levels will also have a length; other stuff that's just missing a level won't,
2290 // so leave those kinds of entities at the default level of 1 (for example, the R.styleable class)
2291 if (isNaN(minLevel) && minLevel.length) {
2292 minLevel = maxLevel;
2293 }
2294 var select = $("#apiLevelSelector").html("").change(changeApiLevel);
2295 for (var i = maxLevel-1; i >= 0; i--) {
2296 var option = $("<option />").attr("value",""+SINCE_DATA[i]).append(""+SINCE_DATA[i]);
2297 // if (SINCE_DATA[i] < minLevel) option.addClass("absent"); // always false for strings (codenames)
2298 select.append(option);
2299 }
2300
2301 // get the DOM element and use setAttribute cuz IE6 fails when using jquery .attr('selected',true)
2302 var selectedLevelItem = $("#apiLevelSelector option[value='"+userApiLevel+"']").get(0);
2303 selectedLevelItem.setAttribute('selected',true);
2304}
2305
2306function changeApiLevel() {
2307 maxLevel = SINCE_DATA.length;
2308 var selectedLevel = maxLevel;
2309
2310 selectedLevel = parseInt($("#apiLevelSelector option:selected").val());
2311 toggleVisisbleApis(selectedLevel, "body");
2312
2313 var date = new Date();
2314 date.setTime(date.getTime()+(10*365*24*60*60*1000)); // keep this for 10 years
2315 var expiration = date.toGMTString();
2316 writeCookie(API_LEVEL_COOKIE, selectedLevel, null, expiration);
2317
2318 if (selectedLevel < minLevel) {
2319 var thing = ($("#jd-header").html().indexOf("package") != -1) ? "package" : "class";
Scott Main8f24ca82012-11-16 10:34:22 -08002320 $("#naMessage").show().html("<div><p><strong>This " + thing
2321 + " requires API level " + minLevel + " or higher.</strong></p>"
2322 + "<p>This document is hidden because your selected API level for the documentation is "
2323 + selectedLevel + ". You can change the documentation API level with the selector "
2324 + "above the left navigation.</p>"
2325 + "<p>For more information about specifying the API level your app requires, "
2326 + "read <a href='" + toRoot + "training/basics/supporting-devices/platforms.html'"
2327 + ">Supporting Different Platform Versions</a>.</p>"
2328 + "<input type='button' value='OK, make this page visible' "
2329 + "title='Change the API level to " + minLevel + "' "
2330 + "onclick='$(\"#apiLevelSelector\").val(\"" + minLevel + "\");changeApiLevel();' />"
2331 + "</div>");
Scott Mainf5089842012-08-14 16:31:07 -07002332 } else {
2333 $("#naMessage").hide();
2334 }
2335}
2336
2337function toggleVisisbleApis(selectedLevel, context) {
2338 var apis = $(".api",context);
2339 apis.each(function(i) {
2340 var obj = $(this);
2341 var className = obj.attr("class");
2342 var apiLevelIndex = className.lastIndexOf("-")+1;
2343 var apiLevelEndIndex = className.indexOf(" ", apiLevelIndex);
2344 apiLevelEndIndex = apiLevelEndIndex != -1 ? apiLevelEndIndex : className.length;
2345 var apiLevel = className.substring(apiLevelIndex, apiLevelEndIndex);
2346 if (apiLevel.length == 0) { // for odd cases when the since data is actually missing, just bail
2347 return;
2348 }
2349 apiLevel = parseInt(apiLevel);
2350
2351 // Handle provisional api levels; if this item's level is the provisional one, set it to the max
2352 var selectedLevelNum = parseInt(selectedLevel)
2353 var apiLevelNum = parseInt(apiLevel);
2354 if (isNaN(apiLevelNum)) {
2355 apiLevelNum = maxLevel;
2356 }
2357
2358 // Grey things out that aren't available and give a tooltip title
2359 if (apiLevelNum > selectedLevelNum) {
2360 obj.addClass("absent").attr("title","Requires API Level \""
2361 + apiLevel + "\" or higher");
2362 }
2363 else obj.removeClass("absent").removeAttr("title");
2364 });
2365}
2366
2367
2368
2369
2370/* ################# SIDENAV TREE VIEW ################### */
2371
2372function new_node(me, mom, text, link, children_data, api_level)
2373{
2374 var node = new Object();
2375 node.children = Array();
2376 node.children_data = children_data;
2377 node.depth = mom.depth + 1;
2378
2379 node.li = document.createElement("li");
2380 mom.get_children_ul().appendChild(node.li);
2381
2382 node.label_div = document.createElement("div");
2383 node.label_div.className = "label";
2384 if (api_level != null) {
2385 $(node.label_div).addClass("api");
2386 $(node.label_div).addClass("api-level-"+api_level);
2387 }
2388 node.li.appendChild(node.label_div);
2389
2390 if (children_data != null) {
2391 node.expand_toggle = document.createElement("a");
2392 node.expand_toggle.href = "javascript:void(0)";
2393 node.expand_toggle.onclick = function() {
2394 if (node.expanded) {
2395 $(node.get_children_ul()).slideUp("fast");
2396 node.plus_img.src = me.toroot + "assets/images/triangle-closed-small.png";
2397 node.expanded = false;
2398 } else {
2399 expand_node(me, node);
2400 }
2401 };
2402 node.label_div.appendChild(node.expand_toggle);
2403
2404 node.plus_img = document.createElement("img");
2405 node.plus_img.src = me.toroot + "assets/images/triangle-closed-small.png";
2406 node.plus_img.className = "plus";
2407 node.plus_img.width = "8";
2408 node.plus_img.border = "0";
2409 node.expand_toggle.appendChild(node.plus_img);
2410
2411 node.expanded = false;
2412 }
2413
2414 var a = document.createElement("a");
2415 node.label_div.appendChild(a);
2416 node.label = document.createTextNode(text);
2417 a.appendChild(node.label);
2418 if (link) {
2419 a.href = me.toroot + link;
2420 } else {
2421 if (children_data != null) {
2422 a.className = "nolink";
2423 a.href = "javascript:void(0)";
2424 a.onclick = node.expand_toggle.onclick;
2425 // This next line shouldn't be necessary. I'll buy a beer for the first
2426 // person who figures out how to remove this line and have the link
2427 // toggle shut on the first try. --joeo@android.com
2428 node.expanded = false;
2429 }
2430 }
2431
2432
2433 node.children_ul = null;
2434 node.get_children_ul = function() {
2435 if (!node.children_ul) {
2436 node.children_ul = document.createElement("ul");
2437 node.children_ul.className = "children_ul";
2438 node.children_ul.style.display = "none";
2439 node.li.appendChild(node.children_ul);
2440 }
2441 return node.children_ul;
2442 };
2443
2444 return node;
2445}
2446
Robert Lyd2dd6e52012-11-29 21:28:48 -08002447
2448
2449
Scott Mainf5089842012-08-14 16:31:07 -07002450function expand_node(me, node)
2451{
2452 if (node.children_data && !node.expanded) {
2453 if (node.children_visited) {
2454 $(node.get_children_ul()).slideDown("fast");
2455 } else {
2456 get_node(me, node);
2457 if ($(node.label_div).hasClass("absent")) {
2458 $(node.get_children_ul()).addClass("absent");
2459 }
2460 $(node.get_children_ul()).slideDown("fast");
2461 }
2462 node.plus_img.src = me.toroot + "assets/images/triangle-opened-small.png";
2463 node.expanded = true;
2464
2465 // perform api level toggling because new nodes are new to the DOM
2466 var selectedLevel = $("#apiLevelSelector option:selected").val();
2467 toggleVisisbleApis(selectedLevel, "#side-nav");
2468 }
2469}
2470
2471function get_node(me, mom)
2472{
2473 mom.children_visited = true;
2474 for (var i in mom.children_data) {
2475 var node_data = mom.children_data[i];
2476 mom.children[i] = new_node(me, mom, node_data[0], node_data[1],
2477 node_data[2], node_data[3]);
2478 }
2479}
2480
2481function this_page_relative(toroot)
2482{
2483 var full = document.location.pathname;
2484 var file = "";
2485 if (toroot.substr(0, 1) == "/") {
2486 if (full.substr(0, toroot.length) == toroot) {
2487 return full.substr(toroot.length);
2488 } else {
2489 // the file isn't under toroot. Fail.
2490 return null;
2491 }
2492 } else {
2493 if (toroot != "./") {
2494 toroot = "./" + toroot;
2495 }
2496 do {
2497 if (toroot.substr(toroot.length-3, 3) == "../" || toroot == "./") {
2498 var pos = full.lastIndexOf("/");
2499 file = full.substr(pos) + file;
2500 full = full.substr(0, pos);
2501 toroot = toroot.substr(0, toroot.length-3);
2502 }
2503 } while (toroot != "" && toroot != "/");
2504 return file.substr(1);
2505 }
2506}
2507
2508function find_page(url, data)
2509{
2510 var nodes = data;
2511 var result = null;
2512 for (var i in nodes) {
2513 var d = nodes[i];
2514 if (d[1] == url) {
2515 return new Array(i);
2516 }
2517 else if (d[2] != null) {
2518 result = find_page(url, d[2]);
2519 if (result != null) {
2520 return (new Array(i).concat(result));
2521 }
2522 }
2523 }
2524 return null;
2525}
2526
Scott Mainf5089842012-08-14 16:31:07 -07002527function init_default_navtree(toroot) {
Scott Main25e73002013-03-27 15:24:06 -07002528 // load json file for navtree data
2529 $.getScript(toRoot + 'navtree_data.js', function(data, textStatus, jqxhr) {
2530 // when the file is loaded, initialize the tree
2531 if(jqxhr.status === 200) {
2532 init_navtree("tree-list", toroot, NAVTREE_DATA);
2533 }
2534 });
Scott Mainf5089842012-08-14 16:31:07 -07002535
2536 // perform api level toggling because because the whole tree is new to the DOM
2537 var selectedLevel = $("#apiLevelSelector option:selected").val();
2538 toggleVisisbleApis(selectedLevel, "#side-nav");
2539}
2540
2541function init_navtree(navtree_id, toroot, root_nodes)
2542{
2543 var me = new Object();
2544 me.toroot = toroot;
2545 me.node = new Object();
2546
2547 me.node.li = document.getElementById(navtree_id);
2548 me.node.children_data = root_nodes;
2549 me.node.children = new Array();
2550 me.node.children_ul = document.createElement("ul");
2551 me.node.get_children_ul = function() { return me.node.children_ul; };
2552 //me.node.children_ul.className = "children_ul";
2553 me.node.li.appendChild(me.node.children_ul);
2554 me.node.depth = 0;
2555
2556 get_node(me, me.node);
2557
2558 me.this_page = this_page_relative(toroot);
2559 me.breadcrumbs = find_page(me.this_page, root_nodes);
2560 if (me.breadcrumbs != null && me.breadcrumbs.length != 0) {
2561 var mom = me.node;
2562 for (var i in me.breadcrumbs) {
2563 var j = me.breadcrumbs[i];
2564 mom = mom.children[j];
2565 expand_node(me, mom);
2566 }
2567 mom.label_div.className = mom.label_div.className + " selected";
2568 addLoadEvent(function() {
2569 scrollIntoView("nav-tree");
2570 });
2571 }
2572}
2573
Robert Lyd2dd6e52012-11-29 21:28:48 -08002574/* TODO: eliminate redundancy with non-google functions */
2575function init_google_navtree(navtree_id, toroot, root_nodes)
2576{
2577 var me = new Object();
2578 me.toroot = toroot;
2579 me.node = new Object();
2580
2581 me.node.li = document.getElementById(navtree_id);
2582 me.node.children_data = root_nodes;
2583 me.node.children = new Array();
2584 me.node.children_ul = document.createElement("ul");
2585 me.node.get_children_ul = function() { return me.node.children_ul; };
2586 //me.node.children_ul.className = "children_ul";
2587 me.node.li.appendChild(me.node.children_ul);
2588 me.node.depth = 0;
2589
2590 get_google_node(me, me.node);
Robert Lyd2dd6e52012-11-29 21:28:48 -08002591}
2592
2593function new_google_node(me, mom, text, link, children_data, api_level)
2594{
2595 var node = new Object();
2596 var child;
2597 node.children = Array();
2598 node.children_data = children_data;
2599 node.depth = mom.depth + 1;
2600 node.get_children_ul = function() {
2601 if (!node.children_ul) {
Scott Mainac71b2b2012-11-30 14:40:58 -08002602 node.children_ul = document.createElement("ul");
2603 node.children_ul.className = "tree-list-children";
Robert Lyd2dd6e52012-11-29 21:28:48 -08002604 node.li.appendChild(node.children_ul);
2605 }
2606 return node.children_ul;
2607 };
2608 node.li = document.createElement("li");
2609
2610 mom.get_children_ul().appendChild(node.li);
2611
2612
2613 if(link) {
2614 child = document.createElement("a");
2615
2616 }
2617 else {
2618 child = document.createElement("span");
Scott Mainac71b2b2012-11-30 14:40:58 -08002619 child.className = "tree-list-subtitle";
Robert Lyd2dd6e52012-11-29 21:28:48 -08002620
2621 }
2622 if (children_data != null) {
2623 node.li.className="nav-section";
2624 node.label_div = document.createElement("div");
2625 node.label_div.className = "nav-section-header-ref";
2626 node.li.appendChild(node.label_div);
2627 get_google_node(me, node);
2628 node.label_div.appendChild(child);
2629 }
2630 else {
2631 node.li.appendChild(child);
2632 }
2633 if(link) {
2634 child.href = me.toroot + link;
2635 }
2636 node.label = document.createTextNode(text);
2637 child.appendChild(node.label);
2638
2639 node.children_ul = null;
2640
2641 return node;
2642}
2643
2644function get_google_node(me, mom)
2645{
2646 mom.children_visited = true;
2647 var linkText;
2648 for (var i in mom.children_data) {
2649 var node_data = mom.children_data[i];
2650 linkText = node_data[0];
2651
2652 if(linkText.match("^"+"com.google.android")=="com.google.android"){
2653 linkText = linkText.substr(19, linkText.length);
2654 }
2655 mom.children[i] = new_google_node(me, mom, linkText, node_data[1],
2656 node_data[2], node_data[3]);
2657 }
2658}
2659function showGoogleRefTree() {
2660 init_default_google_navtree(toRoot);
2661 init_default_gcm_navtree(toRoot);
Robert Lyd2dd6e52012-11-29 21:28:48 -08002662}
2663
2664function init_default_google_navtree(toroot) {
Scott Mainf6145542013-04-01 16:38:11 -07002665 // load json file for navtree data
2666 $.getScript(toRoot + 'gms_navtree_data.js', function(data, textStatus, jqxhr) {
2667 // when the file is loaded, initialize the tree
2668 if(jqxhr.status === 200) {
2669 init_google_navtree("gms-tree-list", toroot, GMS_NAVTREE_DATA);
2670 highlightSidenav();
2671 resizeNav();
2672 }
2673 });
Robert Lyd2dd6e52012-11-29 21:28:48 -08002674}
2675
2676function init_default_gcm_navtree(toroot) {
Scott Mainf6145542013-04-01 16:38:11 -07002677 // load json file for navtree data
2678 $.getScript(toRoot + 'gcm_navtree_data.js', function(data, textStatus, jqxhr) {
2679 // when the file is loaded, initialize the tree
2680 if(jqxhr.status === 200) {
2681 init_google_navtree("gcm-tree-list", toroot, GCM_NAVTREE_DATA);
2682 highlightSidenav();
2683 resizeNav();
2684 }
2685 });
Robert Lyd2dd6e52012-11-29 21:28:48 -08002686}
2687
Scott Mainf5089842012-08-14 16:31:07 -07002688/* TOGGLE INHERITED MEMBERS */
2689
2690/* Toggle an inherited class (arrow toggle)
2691 * @param linkObj The link that was clicked.
2692 * @param expand 'true' to ensure it's expanded. 'false' to ensure it's closed.
2693 * 'null' to simply toggle.
2694 */
2695function toggleInherited(linkObj, expand) {
2696 var base = linkObj.getAttribute("id");
2697 var list = document.getElementById(base + "-list");
2698 var summary = document.getElementById(base + "-summary");
2699 var trigger = document.getElementById(base + "-trigger");
2700 var a = $(linkObj);
2701 if ( (expand == null && a.hasClass("closed")) || expand ) {
2702 list.style.display = "none";
2703 summary.style.display = "block";
2704 trigger.src = toRoot + "assets/images/triangle-opened.png";
2705 a.removeClass("closed");
2706 a.addClass("opened");
2707 } else if ( (expand == null && a.hasClass("opened")) || (expand == false) ) {
2708 list.style.display = "block";
2709 summary.style.display = "none";
2710 trigger.src = toRoot + "assets/images/triangle-closed.png";
2711 a.removeClass("opened");
2712 a.addClass("closed");
2713 }
2714 return false;
2715}
2716
2717/* Toggle all inherited classes in a single table (e.g. all inherited methods)
2718 * @param linkObj The link that was clicked.
2719 * @param expand 'true' to ensure it's expanded. 'false' to ensure it's closed.
2720 * 'null' to simply toggle.
2721 */
2722function toggleAllInherited(linkObj, expand) {
2723 var a = $(linkObj);
2724 var table = $(a.parent().parent().parent()); // ugly way to get table/tbody
2725 var expandos = $(".jd-expando-trigger", table);
2726 if ( (expand == null && a.text() == "[Expand]") || expand ) {
2727 expandos.each(function(i) {
2728 toggleInherited(this, true);
2729 });
2730 a.text("[Collapse]");
2731 } else if ( (expand == null && a.text() == "[Collapse]") || (expand == false) ) {
2732 expandos.each(function(i) {
2733 toggleInherited(this, false);
2734 });
2735 a.text("[Expand]");
2736 }
2737 return false;
2738}
2739
2740/* Toggle all inherited members in the class (link in the class title)
2741 */
2742function toggleAllClassInherited() {
2743 var a = $("#toggleAllClassInherited"); // get toggle link from class title
2744 var toggles = $(".toggle-all", $("#body-content"));
2745 if (a.text() == "[Expand All]") {
2746 toggles.each(function(i) {
2747 toggleAllInherited(this, true);
2748 });
2749 a.text("[Collapse All]");
2750 } else {
2751 toggles.each(function(i) {
2752 toggleAllInherited(this, false);
2753 });
2754 a.text("[Expand All]");
2755 }
2756 return false;
2757}
2758
2759/* Expand all inherited members in the class. Used when initiating page search */
2760function ensureAllInheritedExpanded() {
2761 var toggles = $(".toggle-all", $("#body-content"));
2762 toggles.each(function(i) {
2763 toggleAllInherited(this, true);
2764 });
2765 $("#toggleAllClassInherited").text("[Collapse All]");
2766}
2767
2768
2769/* HANDLE KEY EVENTS
2770 * - Listen for Ctrl+F (Cmd on Mac) and expand all inherited members (to aid page search)
2771 */
2772var agent = navigator['userAgent'].toLowerCase();
2773var mac = agent.indexOf("macintosh") != -1;
2774
2775$(document).keydown( function(e) {
2776var control = mac ? e.metaKey && !e.ctrlKey : e.ctrlKey; // get ctrl key
2777 if (control && e.which == 70) { // 70 is "F"
2778 ensureAllInheritedExpanded();
2779 }
2780});