Simplified doc-site generation
[AGL/documentation.git] / theme / mkdocs_windmill / js / base.js
diff --git a/theme/mkdocs_windmill/js/base.js b/theme/mkdocs_windmill/js/base.js
new file mode 100644 (file)
index 0000000..51d5b5e
--- /dev/null
@@ -0,0 +1,568 @@
+/* global window, document, $, hljs, elasticlunr, base_url, is_top_frame */
+/* exported getParam, onIframeLoad */
+"use strict";
+
+// The full page consists of a main window with navigation and table of contents, and an inner
+// iframe containing the current article. Which article is shown is determined by the main
+// window's #hash portion of the URL. In fact, we use the simple rule: main window's URL of
+// "rootUrl#relPath" corresponds to iframe's URL of "rootUrl/relPath".
+//
+// The main frame and the contents of the index page actually live in a single generated html
+// file: the outer frame hides one half, and the inner hides the other. TODO: this should be
+// possible to greatly simplify after mkdocs-1.0 release.
+
+var mainWindow = is_top_frame ? window : (window.parent !== window ? window.parent : null);
+var iframeWindow = null;
+var rootUrl = qualifyUrl(base_url);
+var searchIndex = null;
+var showPageToc = true;
+var MutationObserver = window.MutationObserver || window.WebKitMutationObserver;
+
+var Keys = {
+  ENTER:  13,
+  ESCAPE: 27,
+  UP:     38,
+  DOWN:   40,
+};
+
+function startsWith(str, prefix) { return str.lastIndexOf(prefix, 0) === 0; }
+function endsWith(str, suffix) { return str.indexOf(suffix, str.length - suffix.length) !== -1; }
+
+/**
+ * Returns whether to use small-screen mode. Note that the same size is used in css @media block.
+ */
+function isSmallScreen() {
+  return window.matchMedia("(max-width: 600px)").matches;
+}
+
+/**
+ * Given a relative URL, returns the absolute one, relying on the browser to convert it.
+ */
+function qualifyUrl(url) {
+  var a = document.createElement('a');
+  a.href = url;
+  return a.href;
+}
+
+/**
+ * Turns an absolute path to relative, stripping out rootUrl + separator.
+ */
+function getRelPath(separator, absUrl) {
+  var prefix = rootUrl + (endsWith(rootUrl, separator) ? '' : separator);
+  return startsWith(absUrl, prefix) ? absUrl.slice(prefix.length) : null;
+}
+
+/**
+ * Turns a relative path to absolute, adding a prefix of rootUrl + separator.
+ */
+function getAbsUrl(separator, relPath) {
+  var sep = endsWith(rootUrl, separator) ? '' : separator;
+  return relPath === null ? null : rootUrl + sep + relPath;
+}
+
+/**
+ * Redirects the iframe to reflect the path represented by the main window's current URL.
+ * (In our design, nothing should change iframe's src except via updateIframe(), or back/forward
+ * history is likely to get messed up.)
+ */
+function updateIframe(enableForwardNav) {
+  // Grey out the "forward" button if we don't expect 'forward' to work.
+  $('#hist-fwd').toggleClass('greybtn', !enableForwardNav);
+
+  var targetRelPath = getRelPath('#', mainWindow.location.href) || '';
+  var targetIframeUrl = getAbsUrl('/', targetRelPath);
+  var loc = iframeWindow.location;
+  var currentIframeUrl = _safeGetLocationHref(loc);
+
+  console.log("updateIframe: %s -> %s (%s)", currentIframeUrl, targetIframeUrl,
+    currentIframeUrl === targetIframeUrl ? "same" : "replacing");
+
+  if (currentIframeUrl !== targetIframeUrl) {
+    loc.replace(targetIframeUrl);
+    onIframeBeforeLoad(targetIframeUrl);
+  }
+  document.body.scrollTop = 0;
+}
+
+/**
+ * Returns location.href, catching exception that's triggered if the iframe is on a different domain.
+ */
+function _safeGetLocationHref(location) {
+  try {
+    return location.href;
+  } catch (e) {
+    return null;
+  }
+}
+
+/**
+ * Returns the value of the given parameter in the URL's query portion.
+ */
+function getParam(key) {
+  var params = window.location.search.substring(1).split('&');
+  for (var i = 0; i < params.length; i++) {
+    var param = params[i].split('=');
+    if (param[0] === key) {
+      return decodeURIComponent(param[1].replace(/\+/g, '%20'));
+    }
+  }
+}
+
+/**
+ * Update the state of the button toggling table-of-contents. TOC has different behavior
+ * depending on screen size, so the button's behavior depends on that too.
+ */
+function updateTocButtonState() {
+  var shown;
+  if (isSmallScreen()) {
+    shown = $('.wm-toc-pane').hasClass('wm-toc-dropdown');
+  } else {
+    shown = !$('#main-content').hasClass('wm-toc-hidden');
+  }
+  $('#wm-toc-button').toggleClass('active', shown);
+}
+
+/**
+ * Update the height of the iframe container. On small screens, we adjust it to fit the iframe
+ * contents, so that the page scrolls as a whole rather than inside the iframe.
+ */
+function updateContentHeight() {
+  if (isSmallScreen()) {
+    $('.wm-content-pane').height(iframeWindow.document.body.offsetHeight + 20);
+    $('.wm-article').attr('scrolling', 'no');
+  } else {
+    $('.wm-content-pane').height('');
+    $('.wm-article').attr('scrolling', 'auto');
+  }
+}
+
+/**
+ * When TOC is a dropdown (on small screens), close it.
+ */
+function closeTempItems() {
+  $('.wm-toc-dropdown').removeClass('wm-toc-dropdown');
+  $('#mkdocs-search-query').closest('.wm-top-tool').removeClass('wm-top-tool-expanded');
+  updateTocButtonState();
+}
+
+/**
+ * Visit the given URL. This changes the hash of the top page to reflect the new URL's relative
+ * path, and points the iframe to the new URL.
+ */
+function visitUrl(url, event) {
+  var relPath = getRelPath('/', url);
+  if (relPath !== null) {
+    event.preventDefault();
+    var newUrl = getAbsUrl('#', relPath);
+    if (newUrl !== mainWindow.location.href) {
+      mainWindow.history.pushState(null, '', newUrl);
+      updateIframe(false);
+    }
+    closeTempItems();
+    iframeWindow.focus();
+  }
+}
+
+/**
+ * Adjusts link to point to a top page, converting URL from "base/path" to "base#path". It also
+ * sets a data-adjusted attribute on the link, to skip adjustments on future clicks.
+ */
+function adjustLink(linkEl) {
+  if (!linkEl.hasAttribute('data-wm-adjusted')) {
+    linkEl.setAttribute('data-wm-adjusted', 'done');
+    var relPath = getRelPath('/', linkEl.href);
+    if (relPath !== null) {
+      var newUrl = getAbsUrl('#', relPath);
+      linkEl.href = newUrl;
+    }
+  }
+}
+
+/**
+ * Given a URL, strips query and fragment, returning just the path.
+ */
+function cleanUrlPath(relUrl) {
+  return relUrl.replace(/[#?].*/, '');
+}
+
+/**
+ * Initialize the main window.
+ */
+function initMainWindow() {
+  // wm-toc-button either opens the table of contents in the side-pane, or (on smaller screens)
+  // shows the side-pane as a drop-down.
+  $('#wm-toc-button').on('click', function(e) {
+    if (isSmallScreen()) {
+      $('.wm-toc-pane').toggleClass('wm-toc-dropdown');
+      $('#wm-main-content').removeClass('wm-toc-hidden');
+    } else {
+      $('#main-content').toggleClass('wm-toc-hidden');
+      closeTempItems();
+    }
+    updateTocButtonState();
+  });
+
+  // Update the state of the wm-toc-button
+  updateTocButtonState();
+  $(window).on('resize', function() {
+    updateTocButtonState();
+    updateContentHeight();
+  });
+
+  // Connect up the Back and Forward buttons (if present).
+  $('#hist-back').on('click', function(e) { window.history.back(); });
+  $('#hist-fwd').on('click', function(e) { window.history.forward(); });
+
+  // When the side-pane is a dropdown, hide it on click-away.
+  $(window).on('blur', closeTempItems);
+
+  // When we click on an opener in the table of contents, open it.
+  $('.wm-toc-pane').on('click', '.wm-toc-opener', function(e) {
+    $(this).toggleClass('wm-toc-open');
+    $(this).next('.wm-toc-li-nested').collapse('toggle');
+  });
+  $('.wm-toc-pane').on('click', '.wm-page-toc-opener', function(e) {
+    // Ignore clicks while transitioning.
+    if ($(this).next('.wm-page-toc').hasClass('collapsing')) { return; }
+    showPageToc = !showPageToc;
+    $(this).toggleClass('wm-page-toc-open', showPageToc);
+    $(this).next('.wm-page-toc').collapse(showPageToc ? 'show' : 'hide');
+  });
+
+  // Once the article loads in the side-pane, close the dropdown.
+  $('.wm-article').on('load', function() {
+    document.title = iframeWindow.document.title;
+    updateContentHeight();
+
+    // We want to update content height whenever the height of the iframe's content changes.
+    // Using MutationObserver seems to be the best way to do that.
+    var observer = new MutationObserver(updateContentHeight);
+    observer.observe(iframeWindow.document.body, {
+      attributes: true,
+      childList: true,
+      characterData: true,
+      subtree: true
+    });
+
+    iframeWindow.focus();
+  });
+
+  // Initialize search functionality.
+  initSearch();
+
+  // Load the iframe now, and whenever we navigate the top frame.
+  setTimeout(function() { updateIframe(false); }, 0);
+  // For our usage, 'popstate' or 'hashchange' would work, but only 'hashchange' work on IE.
+  $(window).on('hashchange', function() { updateIframe(true); });
+}
+
+function onIframeBeforeLoad(url) {
+  $('.wm-current').removeClass('wm-current');
+  closeTempItems();
+
+  var tocLi = getTocLi(url);
+  tocLi.addClass('wm-current');
+  tocLi.parents('.wm-toc-li-nested')
+    // It's better to open parent items immediately without a transition.
+    .removeClass('collapsing').addClass('collapse in').height('')
+    .prev('.wm-toc-opener').addClass('wm-toc-open');
+}
+
+function getTocLi(url) {
+  var relPath = getAbsUrl('#', getRelPath('/', cleanUrlPath(url)));
+  var selector = '.wm-article-link[href="' + relPath + '"]';
+  return $(selector).closest('.wm-toc-li');
+}
+
+var _deferIframeLoad = false;
+
+// Sometimes iframe is loaded before main window's ready callback. In this case, we defer
+// onIframeLoad call until the main window has initialized.
+function ensureIframeLoaded() {
+  if (_deferIframeLoad) {
+    onIframeLoad();
+  }
+}
+
+function onIframeLoad() {
+  if (!iframeWindow) { _deferIframeLoad = true; return; }
+  var url = iframeWindow.location.href;
+  onIframeBeforeLoad(url);
+
+  if (iframeWindow.pageToc) {
+    var relPath = getAbsUrl('#', getRelPath('/', cleanUrlPath(url)));
+    renderPageToc(getTocLi(url), relPath, iframeWindow.pageToc);
+  }
+  iframeWindow.focus();
+}
+
+/**
+ * Hides a bootstrap collapsible element, and removes it from DOM once hidden.
+ */
+function collapseAndRemove(collapsibleElem) {
+  if (!collapsibleElem.hasClass('in')) {
+    // If the element is already hidden, just remove it immediately.
+    collapsibleElem.remove();
+  } else {
+    collapsibleElem.on('hidden.bs.collapse', function() {
+      collapsibleElem.remove();
+    })
+    .collapse('hide');
+  }
+}
+
+function renderPageToc(parentElem, pageUrl, pageToc) {
+  var ul = $('<ul class="wm-toctree">');
+  function addItem(tocItem) {
+    ul.append($('<li class="wm-toc-li">')
+      .append($('<a class="wm-article-link wm-page-toc-text">')
+        .attr('href', pageUrl + tocItem.url)
+        .attr('data-wm-adjusted', 'done')
+        .text(tocItem.title)));
+    if (tocItem.children) {
+      tocItem.children.forEach(addItem);
+    }
+  }
+  pageToc.forEach(addItem);
+
+  $('.wm-page-toc-opener').removeClass('wm-page-toc-opener wm-page-toc-open');
+  collapseAndRemove($('.wm-page-toc'));
+
+  parentElem.addClass('wm-page-toc-opener').toggleClass('wm-page-toc-open', showPageToc);
+  $('<li class="wm-page-toc wm-toc-li-nested collapse">').append(ul).insertAfter(parentElem)
+    .collapse(showPageToc ? 'show' : 'hide');
+}
+
+
+if (!mainWindow) {
+  // This is a page that ought to be in an iframe. Redirect to load the top page instead.
+  var topUrl = getAbsUrl('#', getRelPath('/', window.location.href));
+  if (topUrl) {
+    window.location.href = topUrl;
+  }
+
+} else {
+  // Adjust all links to point to the top page with the right hash fragment.
+  $(document).ready(function() {
+    $('a').each(function() { adjustLink(this); });
+  });
+
+  // For any dynamically-created links, adjust them on click.
+  $(document).on('click', 'a:not([data-wm-adjusted])', function(e) { adjustLink(this); });
+}
+
+if (is_top_frame) {
+  // Main window.
+  $(document).ready(function() {
+    iframeWindow = $('.wm-article')[0].contentWindow;
+    initMainWindow();
+    ensureIframeLoaded();
+  });
+
+} else {
+  // Article contents.
+  iframeWindow = window;
+  if (mainWindow) {
+    mainWindow.onIframeLoad();
+  }
+
+  // Other initialization of iframe contents.
+  hljs.initHighlightingOnLoad();
+  $(document).ready(function() {
+    $('table').addClass('table table-striped table-hover table-bordered table-condensed');
+  });
+}
+
+
+var searchIndexReady = false;
+
+/**
+ * Initialize search functionality.
+ */
+function initSearch() {
+  // Create elasticlunr index.
+  searchIndex = elasticlunr(function() {
+    this.setRef('location');
+    this.addField('title');
+    this.addField('text');
+  });
+
+  var searchBox = $('#mkdocs-search-query');
+  var searchResults = $('#mkdocs-search-results');
+
+  // Fetch the prebuilt index data, and add to the index.
+  $.getJSON(base_url + '/search/search_index.json')
+  .done(function(data) {
+    data.docs.forEach(function(doc) {
+      searchIndex.addDoc(doc);
+    });
+    searchIndexReady = true;
+    $(document).trigger('searchIndexReady');
+  });
+
+  function showSearchResults(optShow) {
+    var show = (optShow === false ? false : Boolean(searchBox.val()));
+    if (show) {
+      doSearch({
+        resultsElem: searchResults,
+        query: searchBox.val(),
+        snippetLen: 100,
+        limit: 10
+      });
+    }
+    searchResults.parent().toggleClass('open', show);
+    return show;
+  }
+
+  searchBox.on('click', function(e) {
+    if (!searchResults.parent().hasClass('open')) {
+      if (showSearchResults()) {
+        e.stopPropagation();
+      }
+    }
+  });
+
+  // Search automatically and show results on keyup event.
+  searchBox.on('keyup', function(e) {
+    var show = (e.which !== Keys.ESCAPE && e.which !== Keys.ENTER);
+    showSearchResults(show);
+  });
+
+  // Open the search box (and run the search) on up/down arrow keys.
+  searchBox.on('keydown', function(e) {
+    if (e.which === Keys.UP || e.which === Keys.DOWN) {
+      if (showSearchResults()) {
+        e.stopPropagation();
+        e.preventDefault();
+        setTimeout(function() {
+          searchResults.find('a').eq(e.which === Keys.UP ? -1 : 0).focus();
+        }, 0);
+      }
+    }
+  });
+
+  searchResults.on('keydown', function(e) {
+    if (e.which === Keys.UP || e.which === Keys.DOWN) {
+      if (searchResults.find('a').eq(e.which === Keys.UP ? 0 : -1)[0] === e.target) {
+        searchBox.focus();
+        e.stopPropagation();
+        e.preventDefault();
+      }
+    }
+  });
+
+  $(searchResults).on('click', '.search-all', function(e) {
+    e.stopPropagation();
+    e.preventDefault();
+    $('#wm-search-form').trigger('submit');
+  });
+
+  // Redirect to the search page on Enter or button-click (form submit).
+  $('#wm-search-form').on('submit', function(e) {
+    var url = this.action + '?' + $(this).serialize();
+    visitUrl(url, e);
+    searchResults.parent().removeClass('open');
+  });
+
+  $('#wm-search-show,#wm-search-go').on('click', function(e) {
+    if (isSmallScreen()) {
+      e.preventDefault();
+      var el = $('#mkdocs-search-query').closest('.wm-top-tool');
+      el.toggleClass('wm-top-tool-expanded');
+      if (el.hasClass('wm-top-tool-expanded')) {
+        setTimeout(function() {
+          $('#mkdocs-search-query').focus();
+          showSearchResults();
+        }, 0);
+        $('#mkdocs-search-query').focus();
+      }
+    }
+  });
+}
+
+function escapeRegex(s) {
+  return s.replace(/[-\/\\^$*+?.()|[\]{}]/g, '\\$&');
+}
+
+/**
+ * This helps construct useful snippets to show in search results, and highlight matches.
+ */
+function SnippetBuilder(query) {
+  var termsPattern = elasticlunr.tokenizer(query).map(escapeRegex).join("|");
+  this._termsRegex = termsPattern ? new RegExp(termsPattern, "gi") : null;
+}
+
+SnippetBuilder.prototype.getSnippet = function(text, len) {
+  if (!this._termsRegex) {
+    return text.slice(0, len);
+  }
+
+  // Find a position that includes something we searched for.
+  var pos = text.search(this._termsRegex);
+  if (pos < 0) { pos = 0; }
+
+  // Find a period before that position (a good starting point).
+  var start = text.lastIndexOf('.', pos) + 1;
+  if (pos - start > 30) {
+    // If too long to previous period, give it 30 characters, and find a space before that.
+    start = text.lastIndexOf(' ', pos - 30) + 1;
+  }
+  var rawSnippet = text.slice(start, start + len);
+  return rawSnippet.replace(this._termsRegex, '<b>$&</b>');
+};
+
+/**
+ * Search the elasticlunr index for the given query, and populate the dropdown with results.
+ */
+function doSearch(options) {
+  var resultsElem = options.resultsElem;
+  resultsElem.empty();
+
+  // If the index isn't ready, wait for it, and search again when ready.
+  if (!searchIndexReady) {
+    resultsElem.append($('<li class="disabled"><a class="search-link">SEARCHING...</a></li>'));
+    $(document).one('searchIndexReady', function() { doSearch(options); });
+    return;
+  }
+
+  var query = options.query;
+  var snippetLen = options.snippetLen;
+  var limit = options.limit;
+
+  if (query === '') { return; }
+
+  var results = searchIndex.search(query, {
+    fields: { title: {boost: 10}, text: { boost: 1 } },
+    expand: true,
+    bool: "AND"
+  });
+
+  var snippetBuilder = new SnippetBuilder(query);
+  if (results.length > 0){
+    var len = Math.min(results.length, limit || Infinity);
+    for (var i = 0; i < len; i++) {
+      var doc = searchIndex.documentStore.getDoc(results[i].ref);
+      var snippet = snippetBuilder.getSnippet(doc.text, snippetLen);
+      resultsElem.append(
+        $('<li>').append($('<a class="search-link">').attr('href', pathJoin(base_url, doc.location))
+          .append($('<div class="search-title">').text(doc.title))
+          .append($('<div class="search-text">').html(snippet)))
+      );
+    }
+    resultsElem.find('a').each(function() { adjustLink(this); });
+    if (limit) {
+      resultsElem.append($('<li role="separator" class="divider"></li>'));
+      resultsElem.append($(
+        '<li><a class="search-link search-all" href="' + base_url + '/search.html">' +
+        '<div class="search-title">SEE ALL RESULTS</div></a></li>'));
+    }
+  } else {
+    resultsElem.append($('<li class="disabled"><a class="search-link">NO RESULTS FOUND</a></li>'));
+  }
+}
+
+function pathJoin(prefix, suffix) {
+  var nPrefix = endsWith(prefix, "/") ? prefix.slice(0, -1) : prefix;
+  var nSuffix = startsWith(suffix, "/") ? suffix.slice(1) : suffix;
+  return nPrefix + "/" + nSuffix;
+}