|
@@ -5,806 +5,641 @@
|
|
* Core JS functions and initialization.
|
|
* Core JS functions and initialization.
|
|
**************************************************/
|
|
**************************************************/
|
|
|
|
|
|
-(function ($) {
|
|
|
|
-
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Responsive scrolling for URL hashes.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- // Dynamically get responsive navigation bar height for offsetting Scrollspy.
|
|
|
|
- function getNavBarHeight() {
|
|
|
|
- let $navbar = $('#navbar-main');
|
|
|
|
- let navbar_offset = $navbar.outerHeight();
|
|
|
|
- console.debug('Navbar height: ' + navbar_offset);
|
|
|
|
- return navbar_offset;
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- /**
|
|
|
|
- * Responsive hash scrolling.
|
|
|
|
- * Check for a URL hash as an anchor.
|
|
|
|
- * If it exists on current page, scroll to it responsively.
|
|
|
|
- * If `target` argument omitted (e.g. after event), assume it's the window's hash.
|
|
|
|
- */
|
|
|
|
- function scrollToAnchor(target) {
|
|
|
|
- // If `target` is undefined or HashChangeEvent object, set it to window's hash.
|
|
|
|
- // Decode the hash as browsers can encode non-ASCII characters (e.g. Chinese symbols).
|
|
|
|
- target = (typeof target === 'undefined' || typeof target === 'object') ? decodeURIComponent(window.location.hash) : target;
|
|
|
|
-
|
|
|
|
- // If target element exists, scroll to it taking into account fixed navigation bar offset.
|
|
|
|
- if ($(target).length) {
|
|
|
|
- // Escape special chars from IDs, such as colons found in Markdown footnote links.
|
|
|
|
- target = '#' + $.escapeSelector(target.substring(1)); // Previously, `target = target.replace(/:/g, '\\:');`
|
|
|
|
-
|
|
|
|
- let elementOffset = Math.ceil($(target).offset().top - getNavBarHeight()); // Round up to highlight right ID!
|
|
|
|
- $('body').addClass('scrolling');
|
|
|
|
- $('html, body').animate({
|
|
|
|
- scrollTop: elementOffset
|
|
|
|
- }, 600, function () {
|
|
|
|
- $('body').removeClass('scrolling');
|
|
|
|
- });
|
|
|
|
- } else {
|
|
|
|
- console.debug('Cannot scroll to target `#' + target + '`. ID not found!');
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+import {hugoEnvironment} from '@params';
|
|
|
|
|
|
- // Make Scrollspy responsive.
|
|
|
|
- function fixScrollspy() {
|
|
|
|
- let $body = $('body');
|
|
|
|
- let data = $body.data('bs.scrollspy');
|
|
|
|
- if (data) {
|
|
|
|
- data._config.offset = getNavBarHeight();
|
|
|
|
- $body.data('bs.scrollspy', data);
|
|
|
|
- $body.scrollspy('refresh');
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+import {fixMermaid} from './wowchemy-utils';
|
|
|
|
|
|
- function removeQueryParamsFromUrl() {
|
|
|
|
- if (window.history.replaceState) {
|
|
|
|
- let urlWithoutSearchParams = window.location.protocol + "//" + window.location.host + window.location.pathname + window.location.hash;
|
|
|
|
- window.history.replaceState({path: urlWithoutSearchParams}, '', urlWithoutSearchParams);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+import {
|
|
|
|
+ changeThemeModeClick,
|
|
|
|
+ initThemeVariation,
|
|
|
|
+ renderThemeVariation,
|
|
|
|
+ onMediaQueryListEvent,
|
|
|
|
+} from './wowchemy-theming';
|
|
|
|
|
|
- // Check for hash change event and fix responsive offset for hash links (e.g. Markdown footnotes).
|
|
|
|
- window.addEventListener("hashchange", scrollToAnchor);
|
|
|
|
|
|
+console.debug(`Environment: ${hugoEnvironment}`)
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Add smooth scrolling to all links inside the main navbar.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- $('#navbar-main li.nav-item a.nav-link, .js-scroll').on('click', function (event) {
|
|
|
|
- // Store requested URL hash.
|
|
|
|
- let hash = this.hash;
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * Responsive scrolling for URL hashes.
|
|
|
|
+ * --------------------------------------------------------------------------- */
|
|
|
|
|
|
- // If we are on a widget page and the navbar link is to a section on the same page.
|
|
|
|
- if (this.pathname === window.location.pathname && hash && $(hash).length && ($(".js-widget-page").length > 0)) {
|
|
|
|
- // Prevent default click behavior.
|
|
|
|
- event.preventDefault();
|
|
|
|
|
|
+// Dynamically get responsive navigation bar height for offsetting Scrollspy.
|
|
|
|
+function getNavBarHeight() {
|
|
|
|
+ let $navbar = $('#navbar-main');
|
|
|
|
+ let navbar_offset = $navbar.outerHeight();
|
|
|
|
+ console.debug('Navbar height: ' + navbar_offset);
|
|
|
|
+ return navbar_offset;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Responsive hash scrolling.
|
|
|
|
+ * Check for a URL hash as an anchor.
|
|
|
|
+ * If it exists on current page, scroll to it responsively.
|
|
|
|
+ * If `target` argument omitted (e.g. after event), assume it's the window's hash.
|
|
|
|
+ */
|
|
|
|
+function scrollToAnchor(target, duration = 600) {
|
|
|
|
+ // If `target` is undefined or HashChangeEvent object, set it to window's hash.
|
|
|
|
+ // Decode the hash as browsers can encode non-ASCII characters (e.g. Chinese symbols).
|
|
|
|
+ target = (typeof target === 'undefined' || typeof target === 'object') ? decodeURIComponent(window.location.hash) : target;
|
|
|
|
+
|
|
|
|
+ // If target element exists, scroll to it taking into account fixed navigation bar offset.
|
|
|
|
+ if ($(target).length) {
|
|
|
|
+ // Escape special chars from IDs, such as colons found in Markdown footnote links.
|
|
|
|
+ target = '#' + $.escapeSelector(target.substring(1)); // Previously, `target = target.replace(/:/g, '\\:');`
|
|
|
|
+
|
|
|
|
+ let elementOffset = Math.ceil($(target).offset().top - getNavBarHeight()); // Round up to highlight right ID!
|
|
|
|
+ $('body').addClass('scrolling');
|
|
|
|
+ $('html, body').animate({
|
|
|
|
+ scrollTop: elementOffset
|
|
|
|
+ }, duration, function () {
|
|
|
|
+ $('body').removeClass('scrolling');
|
|
|
|
+ });
|
|
|
|
+ } else {
|
|
|
|
+ console.debug('Cannot scroll to target `#' + target + '`. ID not found!');
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+// Make Scrollspy responsive.
|
|
|
|
+function fixScrollspy() {
|
|
|
|
+ let $body = $('body');
|
|
|
|
+ let data = $body.data('bs.scrollspy');
|
|
|
|
+ if (data) {
|
|
|
|
+ data._config.offset = getNavBarHeight();
|
|
|
|
+ $body.data('bs.scrollspy', data);
|
|
|
|
+ $body.scrollspy('refresh');
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
|
|
- // Use jQuery's animate() method for smooth page scrolling.
|
|
|
|
- // The numerical parameter specifies the time (ms) taken to scroll to the specified hash.
|
|
|
|
- let elementOffset = Math.ceil($(hash).offset().top - getNavBarHeight()); // Round up to highlight right ID!
|
|
|
|
|
|
+function removeQueryParamsFromUrl() {
|
|
|
|
+ if (window.history.replaceState) {
|
|
|
|
+ let urlWithoutSearchParams = window.location.protocol + "//" + window.location.host + window.location.pathname + window.location.hash;
|
|
|
|
+ window.history.replaceState({path: urlWithoutSearchParams}, '', urlWithoutSearchParams);
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
|
|
- // Uncomment to debug.
|
|
|
|
- // let scrollTop = $(window).scrollTop();
|
|
|
|
- // let scrollDelta = (elementOffset - scrollTop);
|
|
|
|
- // console.debug('Scroll Delta: ' + scrollDelta);
|
|
|
|
|
|
+// Check for hash change event and fix responsive offset for hash links (e.g. Markdown footnotes).
|
|
|
|
+window.addEventListener("hashchange", scrollToAnchor);
|
|
|
|
|
|
- $('html, body').animate({
|
|
|
|
- scrollTop: elementOffset
|
|
|
|
- }, 800);
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * Add smooth scrolling to all links inside the main navbar.
|
|
|
|
+ * --------------------------------------------------------------------------- */
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Hide mobile collapsable menu on clicking a link.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
|
|
+$('#navbar-main li.nav-item a.nav-link, .js-scroll').on('click', function (event) {
|
|
|
|
+ // Store requested URL hash.
|
|
|
|
+ let hash = this.hash;
|
|
|
|
|
|
- $(document).on('click', '.navbar-collapse.show', function (e) {
|
|
|
|
- //get the <a> element that was clicked, even if the <span> element that is inside the <a> element is e.target
|
|
|
|
- let targetElement = $(e.target).is('a') ? $(e.target) : $(e.target).parent();
|
|
|
|
|
|
+ // If we are on a widget page and the navbar link is to a section on the same page.
|
|
|
|
+ if (this.pathname === window.location.pathname && hash && $(hash).length && ($(".js-widget-page").length > 0)) {
|
|
|
|
+ // Prevent default click behavior.
|
|
|
|
+ event.preventDefault();
|
|
|
|
|
|
- if (targetElement.is('a') && targetElement.attr('class') != 'dropdown-toggle') {
|
|
|
|
- $(this).collapse('hide');
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
|
|
+ // Use jQuery's animate() method for smooth page scrolling.
|
|
|
|
+ // The numerical parameter specifies the time (ms) taken to scroll to the specified hash.
|
|
|
|
+ let elementOffset = Math.ceil($(hash).offset().top - getNavBarHeight()); // Round up to highlight right ID!
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Filter publications.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- // Active publication filters.
|
|
|
|
- let pubFilters = {};
|
|
|
|
-
|
|
|
|
- // Search term.
|
|
|
|
- let searchRegex;
|
|
|
|
-
|
|
|
|
- // Filter values (concatenated).
|
|
|
|
- let filterValues;
|
|
|
|
-
|
|
|
|
- // Publication container.
|
|
|
|
- let $grid_pubs = $('#container-publications');
|
|
|
|
-
|
|
|
|
- // Initialise Isotope publication layout if required.
|
|
|
|
- if ($grid_pubs.length)
|
|
|
|
- {
|
|
|
|
- $grid_pubs.isotope({
|
|
|
|
- itemSelector: '.isotope-item',
|
|
|
|
- percentPosition: true,
|
|
|
|
- masonry: {
|
|
|
|
- // Use Bootstrap compatible grid layout.
|
|
|
|
- columnWidth: '.grid-sizer'
|
|
|
|
- },
|
|
|
|
- filter: function () {
|
|
|
|
- let $this = $(this);
|
|
|
|
- let searchResults = searchRegex ? $this.text().match(searchRegex) : true;
|
|
|
|
- let filterResults = filterValues ? $this.is(filterValues) : true;
|
|
|
|
- return searchResults && filterResults;
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
|
|
+ // Uncomment to debug.
|
|
|
|
+ // let scrollTop = $(window).scrollTop();
|
|
|
|
+ // let scrollDelta = (elementOffset - scrollTop);
|
|
|
|
+ // console.debug('Scroll Delta: ' + scrollDelta);
|
|
|
|
|
|
- // Filter by search term.
|
|
|
|
- let $quickSearch = $('.filter-search').keyup(debounce(function () {
|
|
|
|
- searchRegex = new RegExp($quickSearch.val(), 'gi');
|
|
|
|
- $grid_pubs.isotope();
|
|
|
|
- }));
|
|
|
|
|
|
+ $('html, body').animate({
|
|
|
|
+ scrollTop: elementOffset
|
|
|
|
+ }, 800);
|
|
|
|
+ }
|
|
|
|
+});
|
|
|
|
|
|
- $('.pub-filters').on('change', function () {
|
|
|
|
- let $this = $(this);
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * Hide mobile collapsable menu on clicking a link.
|
|
|
|
+ * --------------------------------------------------------------------------- */
|
|
|
|
|
|
- // Get group key.
|
|
|
|
- let filterGroup = $this[0].getAttribute('data-filter-group');
|
|
|
|
|
|
+$(document).on('click', '.navbar-collapse.show', function (e) {
|
|
|
|
+ //get the <a> element that was clicked, even if the <span> element that is inside the <a> element is e.target
|
|
|
|
+ let targetElement = $(e.target).is('a') ? $(e.target) : $(e.target).parent();
|
|
|
|
|
|
- // Set filter for group.
|
|
|
|
- pubFilters[filterGroup] = this.value;
|
|
|
|
|
|
+ if (targetElement.is('a') && targetElement.attr('class') != 'dropdown-toggle') {
|
|
|
|
+ $(this).collapse('hide');
|
|
|
|
+ }
|
|
|
|
+});
|
|
|
|
|
|
- // Combine filters.
|
|
|
|
- filterValues = concatValues(pubFilters);
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * Filter publications.
|
|
|
|
+ * --------------------------------------------------------------------------- */
|
|
|
|
|
|
- // Activate filters.
|
|
|
|
- $grid_pubs.isotope();
|
|
|
|
|
|
+// Active publication filters.
|
|
|
|
+let pubFilters = {};
|
|
|
|
|
|
- // If filtering by publication type, update the URL hash to enable direct linking to results.
|
|
|
|
- if (filterGroup === "pubtype") {
|
|
|
|
- // Set hash URL to current filter.
|
|
|
|
- let url = $(this).val();
|
|
|
|
- if (url.substr(0, 9) === '.pubtype-') {
|
|
|
|
- window.location.hash = url.substr(9);
|
|
|
|
- } else {
|
|
|
|
- window.location.hash = '';
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
- }
|
|
|
|
|
|
+// Search term.
|
|
|
|
+let searchRegex;
|
|
|
|
|
|
- // Debounce input to prevent spamming filter requests.
|
|
|
|
- function debounce(fn, threshold) {
|
|
|
|
- let timeout;
|
|
|
|
- threshold = threshold || 100;
|
|
|
|
- return function debounced() {
|
|
|
|
- clearTimeout(timeout);
|
|
|
|
- let args = arguments;
|
|
|
|
- let _this = this;
|
|
|
|
-
|
|
|
|
- function delayed() {
|
|
|
|
- fn.apply(_this, args);
|
|
|
|
- }
|
|
|
|
|
|
+// Filter values (concatenated).
|
|
|
|
+let filterValues;
|
|
|
|
|
|
- timeout = setTimeout(delayed, threshold);
|
|
|
|
- };
|
|
|
|
- }
|
|
|
|
|
|
+// Publication container.
|
|
|
|
+let $grid_pubs = $('#container-publications');
|
|
|
|
|
|
- // Flatten object by concatenating values.
|
|
|
|
- function concatValues(obj) {
|
|
|
|
- let value = '';
|
|
|
|
- for (let prop in obj) {
|
|
|
|
- value += obj[prop];
|
|
|
|
|
|
+// Initialise Isotope publication layout if required.
|
|
|
|
+if ($grid_pubs.length) {
|
|
|
|
+ $grid_pubs.isotope({
|
|
|
|
+ itemSelector: '.isotope-item',
|
|
|
|
+ percentPosition: true,
|
|
|
|
+ masonry: {
|
|
|
|
+ // Use Bootstrap compatible grid layout.
|
|
|
|
+ columnWidth: '.grid-sizer'
|
|
|
|
+ },
|
|
|
|
+ filter: function () {
|
|
|
|
+ let $this = $(this);
|
|
|
|
+ let searchResults = searchRegex ? $this.text().match(searchRegex) : true;
|
|
|
|
+ let filterResults = filterValues ? $this.is(filterValues) : true;
|
|
|
|
+ return searchResults && filterResults;
|
|
}
|
|
}
|
|
- return value;
|
|
|
|
- }
|
|
|
|
|
|
+ });
|
|
|
|
|
|
- // Filter publications according to hash in URL.
|
|
|
|
- function filter_publications() {
|
|
|
|
- // Check for Isotope publication layout.
|
|
|
|
- if (!$grid_pubs.length)
|
|
|
|
- return
|
|
|
|
|
|
+ // Filter by search term.
|
|
|
|
+ let $quickSearch = $('.filter-search').keyup(debounce(function () {
|
|
|
|
+ searchRegex = new RegExp($quickSearch.val(), 'gi');
|
|
|
|
+ $grid_pubs.isotope();
|
|
|
|
+ }));
|
|
|
|
|
|
- let urlHash = window.location.hash.replace('#', '');
|
|
|
|
- let filterValue = '*';
|
|
|
|
|
|
+ $('.pub-filters').on('change', function () {
|
|
|
|
+ let $this = $(this);
|
|
|
|
|
|
- // Check if hash is numeric.
|
|
|
|
- if (urlHash != '' && !isNaN(urlHash)) {
|
|
|
|
- filterValue = '.pubtype-' + urlHash;
|
|
|
|
- }
|
|
|
|
|
|
+ // Get group key.
|
|
|
|
+ let filterGroup = $this[0].getAttribute('data-filter-group');
|
|
|
|
|
|
- // Set filter.
|
|
|
|
- let filterGroup = 'pubtype';
|
|
|
|
- pubFilters[filterGroup] = filterValue;
|
|
|
|
|
|
+ // Set filter for group.
|
|
|
|
+ pubFilters[filterGroup] = this.value;
|
|
|
|
+
|
|
|
|
+ // Combine filters.
|
|
filterValues = concatValues(pubFilters);
|
|
filterValues = concatValues(pubFilters);
|
|
|
|
|
|
// Activate filters.
|
|
// Activate filters.
|
|
$grid_pubs.isotope();
|
|
$grid_pubs.isotope();
|
|
|
|
|
|
- // Set selected option.
|
|
|
|
- $('.pubtype-select').val(filterValue);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Google Maps or OpenStreetMap via Leaflet.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- function initMap() {
|
|
|
|
- if ($('#map').length) {
|
|
|
|
- let map_provider = $('#map-provider').val();
|
|
|
|
- let lat = $('#map-lat').val();
|
|
|
|
- let lng = $('#map-lng').val();
|
|
|
|
- let zoom = parseInt($('#map-zoom').val());
|
|
|
|
- let address = $('#map-dir').val();
|
|
|
|
- let api_key = $('#map-api-key').val();
|
|
|
|
-
|
|
|
|
- if (map_provider == 1) {
|
|
|
|
- let map = new GMaps({
|
|
|
|
- div: '#map',
|
|
|
|
- lat: lat,
|
|
|
|
- lng: lng,
|
|
|
|
- zoom: zoom,
|
|
|
|
- zoomControl: true,
|
|
|
|
- zoomControlOpt: {
|
|
|
|
- style: 'SMALL',
|
|
|
|
- position: 'TOP_LEFT'
|
|
|
|
- },
|
|
|
|
- panControl: false,
|
|
|
|
- streetViewControl: false,
|
|
|
|
- mapTypeControl: false,
|
|
|
|
- overviewMapControl: false,
|
|
|
|
- scrollwheel: true,
|
|
|
|
- draggable: true
|
|
|
|
- });
|
|
|
|
-
|
|
|
|
- map.addMarker({
|
|
|
|
- lat: lat,
|
|
|
|
- lng: lng,
|
|
|
|
- click: function (e) {
|
|
|
|
- let url = 'https://www.google.com/maps/place/' + encodeURIComponent(address) + '/@' + lat + ',' + lng + '/';
|
|
|
|
- window.open(url, '_blank')
|
|
|
|
- },
|
|
|
|
- title: address
|
|
|
|
- })
|
|
|
|
|
|
+ // If filtering by publication type, update the URL hash to enable direct linking to results.
|
|
|
|
+ if (filterGroup === "pubtype") {
|
|
|
|
+ // Set hash URL to current filter.
|
|
|
|
+ let url = $(this).val();
|
|
|
|
+ if (url.substr(0, 9) === '.pubtype-') {
|
|
|
|
+ window.location.hash = url.substr(9);
|
|
} else {
|
|
} else {
|
|
- let map = new L.map('map').setView([lat, lng], zoom);
|
|
|
|
- if (map_provider == 3 && api_key.length) {
|
|
|
|
- L.tileLayer('https://api.mapbox.com/styles/v1/{id}/tiles/{z}/{x}/{y}?access_token={accessToken}', {
|
|
|
|
- attribution: 'Map data © <a href="http://openstreetmap.org">OpenStreetMap</a> contributors, <a href="http://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA</a>, Imagery © <a href="http://mapbox.com">Mapbox</a>',
|
|
|
|
- tileSize: 512,
|
|
|
|
- maxZoom: 18,
|
|
|
|
- zoomOffset: -1,
|
|
|
|
- id: 'mapbox/streets-v11',
|
|
|
|
- accessToken: api_key
|
|
|
|
- }).addTo(map);
|
|
|
|
- } else {
|
|
|
|
- L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
|
|
|
|
- maxZoom: 19,
|
|
|
|
- attribution: '© <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a>'
|
|
|
|
- }).addTo(map);
|
|
|
|
- }
|
|
|
|
- let marker = L.marker([lat, lng]).addTo(map);
|
|
|
|
- let url = lat + ',' + lng + '#map=' + zoom + '/' + lat + '/' + lng + '&layers=N';
|
|
|
|
- marker.bindPopup(address + '<p><a href="https://www.openstreetmap.org/directions?engine=osrm_car&route=' + url + '">Routing via OpenStreetMap</a></p>');
|
|
|
|
|
|
+ window.location.hash = '';
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- }
|
|
|
|
|
|
+ });
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+// Debounce input to prevent spamming filter requests.
|
|
|
|
+function debounce(fn, threshold) {
|
|
|
|
+ let timeout;
|
|
|
|
+ threshold = threshold || 100;
|
|
|
|
+ return function debounced() {
|
|
|
|
+ clearTimeout(timeout);
|
|
|
|
+ let args = arguments;
|
|
|
|
+ let _this = this;
|
|
|
|
+
|
|
|
|
+ function delayed() {
|
|
|
|
+ fn.apply(_this, args);
|
|
|
|
+ }
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * GitHub API.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
|
|
+ timeout = setTimeout(delayed, threshold);
|
|
|
|
+ };
|
|
|
|
+}
|
|
|
|
|
|
- function printLatestRelease(selector, repo) {
|
|
|
|
- $.getJSON('https://api.github.com/repos/' + repo + '/tags').done(function (json) {
|
|
|
|
- let release = json[0];
|
|
|
|
- $(selector).append(' ' + release.name);
|
|
|
|
- }).fail(function (jqxhr, textStatus, error) {
|
|
|
|
- let err = textStatus + ", " + error;
|
|
|
|
- console.log("Request Failed: " + err);
|
|
|
|
- });
|
|
|
|
|
|
+// Flatten object by concatenating values.
|
|
|
|
+function concatValues(obj) {
|
|
|
|
+ let value = '';
|
|
|
|
+ for (let prop in obj) {
|
|
|
|
+ value += obj[prop];
|
|
}
|
|
}
|
|
|
|
+ return value;
|
|
|
|
+}
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Toggle search dialog.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
|
|
+// Filter publications according to hash in URL.
|
|
|
|
+function filter_publications() {
|
|
|
|
+ // Check for Isotope publication layout.
|
|
|
|
+ if (!$grid_pubs.length)
|
|
|
|
+ return
|
|
|
|
|
|
- function toggleSearchDialog() {
|
|
|
|
- if ($('body').hasClass('searching')) {
|
|
|
|
- // Clear search query and hide search modal.
|
|
|
|
- $('[id=search-query]').blur();
|
|
|
|
- $('body').removeClass('searching compensate-for-scrollbar');
|
|
|
|
|
|
+ let urlHash = window.location.hash.replace('#', '');
|
|
|
|
+ let filterValue = '*';
|
|
|
|
|
|
- // Remove search query params from URL as user has finished searching.
|
|
|
|
- removeQueryParamsFromUrl();
|
|
|
|
|
|
+ // Check if hash is numeric.
|
|
|
|
+ if (urlHash != '' && !isNaN(urlHash)) {
|
|
|
|
+ filterValue = '.pubtype-' + urlHash;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // Set filter.
|
|
|
|
+ let filterGroup = 'pubtype';
|
|
|
|
+ pubFilters[filterGroup] = filterValue;
|
|
|
|
+ filterValues = concatValues(pubFilters);
|
|
|
|
+
|
|
|
|
+ // Activate filters.
|
|
|
|
+ $grid_pubs.isotope();
|
|
|
|
+
|
|
|
|
+ // Set selected option.
|
|
|
|
+ $('.pubtype-select').val(filterValue);
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+* Google Maps or OpenStreetMap via Leaflet.
|
|
|
|
+* --------------------------------------------------------------------------- */
|
|
|
|
+
|
|
|
|
+function initMap() {
|
|
|
|
+ if ($('#map').length) {
|
|
|
|
+ let map_provider = $('#map-provider').val();
|
|
|
|
+ let lat = $('#map-lat').val();
|
|
|
|
+ let lng = $('#map-lng').val();
|
|
|
|
+ let zoom = parseInt($('#map-zoom').val());
|
|
|
|
+ let address = $('#map-dir').val();
|
|
|
|
+ let api_key = $('#map-api-key').val();
|
|
|
|
+
|
|
|
|
+ if (map_provider == 1) {
|
|
|
|
+ let map = new GMaps({
|
|
|
|
+ div: '#map',
|
|
|
|
+ lat: lat,
|
|
|
|
+ lng: lng,
|
|
|
|
+ zoom: zoom,
|
|
|
|
+ zoomControl: true,
|
|
|
|
+ zoomControlOpt: {
|
|
|
|
+ style: 'SMALL',
|
|
|
|
+ position: 'TOP_LEFT'
|
|
|
|
+ },
|
|
|
|
+ panControl: false,
|
|
|
|
+ streetViewControl: false,
|
|
|
|
+ mapTypeControl: false,
|
|
|
|
+ overviewMapControl: false,
|
|
|
|
+ scrollwheel: true,
|
|
|
|
+ draggable: true
|
|
|
|
+ });
|
|
|
|
|
|
- // Prevent fixed positioned elements (e.g. navbar) moving due to scrollbars.
|
|
|
|
- $('#fancybox-style-noscroll').remove();
|
|
|
|
|
|
+ map.addMarker({
|
|
|
|
+ lat: lat,
|
|
|
|
+ lng: lng,
|
|
|
|
+ click: function (e) {
|
|
|
|
+ let url = 'https://www.google.com/maps/place/' + encodeURIComponent(address) + '/@' + lat + ',' + lng + '/';
|
|
|
|
+ window.open(url, '_blank')
|
|
|
|
+ },
|
|
|
|
+ title: address
|
|
|
|
+ })
|
|
} else {
|
|
} else {
|
|
- // Prevent fixed positioned elements (e.g. navbar) moving due to scrollbars.
|
|
|
|
- if (!$('#fancybox-style-noscroll').length && document.body.scrollHeight > window.innerHeight) {
|
|
|
|
- $('head').append(
|
|
|
|
- '<style id="fancybox-style-noscroll">.compensate-for-scrollbar{margin-right:' +
|
|
|
|
- (window.innerWidth - document.documentElement.clientWidth) +
|
|
|
|
- 'px;}</style>'
|
|
|
|
- );
|
|
|
|
- $('body').addClass('compensate-for-scrollbar');
|
|
|
|
|
|
+ let map = new L.map('map').setView([lat, lng], zoom);
|
|
|
|
+ if (map_provider == 3 && api_key.length) {
|
|
|
|
+ L.tileLayer('https://api.mapbox.com/styles/v1/{id}/tiles/{z}/{x}/{y}?access_token={accessToken}', {
|
|
|
|
+ attribution: 'Map data © <a href="http://openstreetmap.org">OpenStreetMap</a> contributors, <a href="http://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA</a>, Imagery © <a href="http://mapbox.com">Mapbox</a>',
|
|
|
|
+ tileSize: 512,
|
|
|
|
+ maxZoom: 18,
|
|
|
|
+ zoomOffset: -1,
|
|
|
|
+ id: 'mapbox/streets-v11',
|
|
|
|
+ accessToken: api_key
|
|
|
|
+ }).addTo(map);
|
|
|
|
+ } else {
|
|
|
|
+ L.tileLayer('https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png', {
|
|
|
|
+ maxZoom: 19,
|
|
|
|
+ attribution: '© <a href="http://www.openstreetmap.org/copyright">OpenStreetMap</a>'
|
|
|
|
+ }).addTo(map);
|
|
}
|
|
}
|
|
-
|
|
|
|
- // Show search modal.
|
|
|
|
- $('body').addClass('searching');
|
|
|
|
- $('.search-results').css({opacity: 0, visibility: 'visible'}).animate({opacity: 1}, 200);
|
|
|
|
- $('#search-query').focus();
|
|
|
|
|
|
+ let marker = L.marker([lat, lng]).addTo(map);
|
|
|
|
+ let url = lat + ',' + lng + '#map=' + zoom + '/' + lat + '/' + lng + '&layers=N';
|
|
|
|
+ marker.bindPopup(address + '<p><a href="https://www.openstreetmap.org/directions?engine=osrm_car&route=' + url + '">Routing via OpenStreetMap</a></p>');
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+}
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Change Theme Mode (0: Day, 1: Night, 2: Auto).
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- // TODO: import theme functions from load-theme.js to avoid duplication.
|
|
|
|
- function canChangeTheme() {
|
|
|
|
- // If var is set, then user is allowed to change the theme variation.
|
|
|
|
- return Boolean(window.wcDarkLightEnabled);
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- function getThemeMode() {
|
|
|
|
- return parseInt(localStorage.getItem('dark_mode') || 2);
|
|
|
|
- }
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * GitHub API.
|
|
|
|
+ * --------------------------------------------------------------------------- */
|
|
|
|
|
|
- function changeThemeModeClick(newMode) {
|
|
|
|
- console.info('Request to set theme.');
|
|
|
|
- if (!canChangeTheme()) {
|
|
|
|
- console.info('Cannot set theme - admin disabled theme selector.');
|
|
|
|
- return;
|
|
|
|
- }
|
|
|
|
- let isDarkTheme;
|
|
|
|
- switch (newMode) {
|
|
|
|
- case 0:
|
|
|
|
- localStorage.setItem('dark_mode', '1');
|
|
|
|
- isDarkTheme = true;
|
|
|
|
- console.info('User changed theme variation to Dark.');
|
|
|
|
- showActiveTheme(0);
|
|
|
|
- break;
|
|
|
|
- case 1:
|
|
|
|
- localStorage.setItem('dark_mode', '2');
|
|
|
|
- if (window.matchMedia('(prefers-color-scheme: dark)').matches) {
|
|
|
|
- // The visitor prefers dark themes and switching to the dark variation is allowed by admin.
|
|
|
|
- isDarkTheme = true;
|
|
|
|
- } else if (window.matchMedia('(prefers-color-scheme: light)').matches) {
|
|
|
|
- // The visitor prefers light themes and switching to the dark variation is allowed by admin.
|
|
|
|
- isDarkTheme = false;
|
|
|
|
- } else {
|
|
|
|
- isDarkTheme = isSiteThemeDark; // Use the site's default theme variation based on `light` in the theme file.
|
|
|
|
- }
|
|
|
|
- console.info('User changed theme variation to Auto.');
|
|
|
|
- showActiveTheme(1);
|
|
|
|
- break;
|
|
|
|
- default:
|
|
|
|
- localStorage.setItem('dark_mode', '0');
|
|
|
|
- isDarkTheme = false;
|
|
|
|
- console.info('User changed theme variation to Light.');
|
|
|
|
- showActiveTheme(2);
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- renderThemeVariation(isDarkTheme);
|
|
|
|
|
|
+function printLatestRelease(selector, repo) {
|
|
|
|
+ if (hugoEnvironment === 'production') {
|
|
|
|
+ $.getJSON('https://api.github.com/repos/' + repo + '/tags').done(function (json) {
|
|
|
|
+ let release = json[0];
|
|
|
|
+ $(selector).append(' ' + release.name);
|
|
|
|
+ }).fail(function (jqxhr, textStatus, error) {
|
|
|
|
+ let err = textStatus + ", " + error;
|
|
|
|
+ console.log("Request Failed: " + err);
|
|
|
|
+ });
|
|
}
|
|
}
|
|
-
|
|
|
|
- function showActiveTheme(mode){
|
|
|
|
- switch (mode) {
|
|
|
|
- case 0:
|
|
|
|
- // Dark.
|
|
|
|
- $('.js-set-theme-light').removeClass('dropdown-item-active');
|
|
|
|
- $('.js-set-theme-dark').addClass('dropdown-item-active');
|
|
|
|
- $('.js-set-theme-auto').removeClass('dropdown-item-active');
|
|
|
|
- break;
|
|
|
|
- case 1:
|
|
|
|
- // Auto.
|
|
|
|
- $('.js-set-theme-light').removeClass('dropdown-item-active');
|
|
|
|
- $('.js-set-theme-dark').removeClass('dropdown-item-active');
|
|
|
|
- $('.js-set-theme-auto').addClass('dropdown-item-active');
|
|
|
|
- break;
|
|
|
|
- default:
|
|
|
|
- // Light.
|
|
|
|
- $('.js-set-theme-light').addClass('dropdown-item-active');
|
|
|
|
- $('.js-set-theme-dark').removeClass('dropdown-item-active');
|
|
|
|
- $('.js-set-theme-auto').removeClass('dropdown-item-active');
|
|
|
|
- break;
|
|
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+* Toggle search dialog.
|
|
|
|
+* --------------------------------------------------------------------------- */
|
|
|
|
+
|
|
|
|
+function toggleSearchDialog() {
|
|
|
|
+ if ($('body').hasClass('searching')) {
|
|
|
|
+ // Clear search query and hide search modal.
|
|
|
|
+ $('[id=search-query]').blur();
|
|
|
|
+ $('body').removeClass('searching compensate-for-scrollbar');
|
|
|
|
+
|
|
|
|
+ // Remove search query params from URL as user has finished searching.
|
|
|
|
+ removeQueryParamsFromUrl();
|
|
|
|
+
|
|
|
|
+ // Prevent fixed positioned elements (e.g. navbar) moving due to scrollbars.
|
|
|
|
+ $('#fancybox-style-noscroll').remove();
|
|
|
|
+ } else {
|
|
|
|
+ // Prevent fixed positioned elements (e.g. navbar) moving due to scrollbars.
|
|
|
|
+ if (!$('#fancybox-style-noscroll').length && document.body.scrollHeight > window.innerHeight) {
|
|
|
|
+ $('head').append(
|
|
|
|
+ '<style id="fancybox-style-noscroll">.compensate-for-scrollbar{margin-right:' +
|
|
|
|
+ (window.innerWidth - document.documentElement.clientWidth) +
|
|
|
|
+ 'px;}</style>'
|
|
|
|
+ );
|
|
|
|
+ $('body').addClass('compensate-for-scrollbar');
|
|
}
|
|
}
|
|
- }
|
|
|
|
|
|
|
|
- function getThemeVariation() {
|
|
|
|
- if (!canChangeTheme()) {
|
|
|
|
- return isSiteThemeDark; // Use the site's default theme variation based on `light` in the theme file.
|
|
|
|
- }
|
|
|
|
- let currentThemeMode = getThemeMode();
|
|
|
|
- let isDarkTheme;
|
|
|
|
- switch (currentThemeMode) {
|
|
|
|
- case 0:
|
|
|
|
- isDarkTheme = false;
|
|
|
|
- break;
|
|
|
|
- case 1:
|
|
|
|
- isDarkTheme = true;
|
|
|
|
- break;
|
|
|
|
- default:
|
|
|
|
- if (window.matchMedia('(prefers-color-scheme: dark)').matches) {
|
|
|
|
- // The visitor prefers dark themes and switching to the dark variation is allowed by admin.
|
|
|
|
- isDarkTheme = true;
|
|
|
|
- } else if (window.matchMedia('(prefers-color-scheme: light)').matches) {
|
|
|
|
- // The visitor prefers light themes and switching to the dark variation is allowed by admin.
|
|
|
|
- isDarkTheme = false;
|
|
|
|
- } else {
|
|
|
|
- isDarkTheme = isSiteThemeDark; // Use the site's default theme variation based on `light` in the theme file.
|
|
|
|
- }
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- return isDarkTheme;
|
|
|
|
|
|
+ // Show search modal.
|
|
|
|
+ $('body').addClass('searching');
|
|
|
|
+ $('.search-results').css({opacity: 0, visibility: 'visible'}).animate({opacity: 1}, 200);
|
|
|
|
+ $('#search-query').focus();
|
|
}
|
|
}
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+* Normalize Bootstrap Carousel Slide Heights.
|
|
|
|
+* --------------------------------------------------------------------------- */
|
|
|
|
+
|
|
|
|
+function normalizeCarouselSlideHeights() {
|
|
|
|
+ $('.carousel').each(function () {
|
|
|
|
+ // Get carousel slides.
|
|
|
|
+ let items = $('.carousel-item', this);
|
|
|
|
+ // Reset all slide heights.
|
|
|
|
+ items.css('min-height', 0);
|
|
|
|
+ // Normalize all slide heights.
|
|
|
|
+ let maxHeight = Math.max.apply(null, items.map(function () {
|
|
|
|
+ return $(this).outerHeight()
|
|
|
|
+ }).get());
|
|
|
|
+ items.css('min-height', maxHeight + 'px');
|
|
|
|
+ })
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+* Fix Hugo's Goldmark output and Mermaid code blocks.
|
|
|
|
+* --------------------------------------------------------------------------- */
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * Fix Hugo's Goldmark output.
|
|
|
|
+ */
|
|
|
|
+function fixHugoOutput() {
|
|
|
|
+ // Fix Goldmark table of contents.
|
|
|
|
+ // - Must be performed prior to initializing ScrollSpy.
|
|
|
|
+ $('#TableOfContents').addClass('nav flex-column');
|
|
|
|
+ $('#TableOfContents li').addClass('nav-item');
|
|
|
|
+ $('#TableOfContents li a').addClass('nav-link');
|
|
|
|
+
|
|
|
|
+ // Fix Goldmark task lists (remove bullet points).
|
|
|
|
+ $("input[type='checkbox'][disabled]").parents('ul').addClass('task-list');
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+// Get an element's siblings.
|
|
|
|
+function getSiblings(elem) {
|
|
|
|
+ // Filter out itself.
|
|
|
|
+ return Array.prototype.filter.call(elem.parentNode.children, function (sibling) {
|
|
|
|
+ return sibling !== elem;
|
|
|
|
+ });
|
|
|
|
+}
|
|
|
|
|
|
- /**
|
|
|
|
- * Render theme variation (day or night).
|
|
|
|
- *
|
|
|
|
- * @param {boolean} isDarkTheme
|
|
|
|
- * @param {boolean} init
|
|
|
|
- * @returns {undefined}
|
|
|
|
- */
|
|
|
|
- function renderThemeVariation(isDarkTheme, init = false) {
|
|
|
|
- // Is code highlighting enabled in site config?
|
|
|
|
- const codeHlEnabled = $('link[title=hl-light]').length > 0;
|
|
|
|
- const codeHlLight = $('link[title=hl-light]')[0];
|
|
|
|
- const codeHlDark = $('link[title=hl-dark]')[0];
|
|
|
|
- const diagramEnabled = $('script[title=mermaid]').length > 0;
|
|
|
|
-
|
|
|
|
- // Check if re-render required.
|
|
|
|
- if (!init) {
|
|
|
|
- // If request to render light when light variation already rendered, return.
|
|
|
|
- // If request to render dark when dark variation already rendered, return.
|
|
|
|
- if ((isDarkTheme === false && !$('body').hasClass('dark')) || (isDarkTheme === true && $('body').hasClass('dark'))) {
|
|
|
|
- return;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * On document ready.
|
|
|
|
+ * --------------------------------------------------------------------------- */
|
|
|
|
|
|
- if (isDarkTheme === false) {
|
|
|
|
- if (!init) {
|
|
|
|
- // Only fade in the page when changing the theme variation.
|
|
|
|
- $('body').css({opacity: 0, visibility: 'visible'}).animate({opacity: 1}, 500);
|
|
|
|
- }
|
|
|
|
- $('body').removeClass('dark');
|
|
|
|
- if (codeHlEnabled) {
|
|
|
|
- codeHlLight.disabled = false;
|
|
|
|
- codeHlDark.disabled = true;
|
|
|
|
- }
|
|
|
|
- if (diagramEnabled) {
|
|
|
|
- if (init) {
|
|
|
|
- mermaid.initialize({theme: 'default', securityLevel: 'loose'});
|
|
|
|
- } else {
|
|
|
|
- // Have to reload to re-initialise Mermaid with the new theme and re-parse the Mermaid code blocks.
|
|
|
|
- location.reload();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- } else if (isDarkTheme === true) {
|
|
|
|
- if (!init) {
|
|
|
|
- // Only fade in the page when changing the theme variation.
|
|
|
|
- $('body').css({opacity: 0, visibility: 'visible'}).animate({opacity: 1}, 500);
|
|
|
|
- }
|
|
|
|
- $('body').addClass('dark');
|
|
|
|
- if (codeHlEnabled) {
|
|
|
|
- codeHlLight.disabled = true;
|
|
|
|
- codeHlDark.disabled = false;
|
|
|
|
- }
|
|
|
|
- if (diagramEnabled) {
|
|
|
|
- if (init) {
|
|
|
|
- mermaid.initialize({theme: 'dark', securityLevel: 'loose'});
|
|
|
|
- } else {
|
|
|
|
- // Have to reload to re-initialise Mermaid with the new theme and re-parse the Mermaid code blocks.
|
|
|
|
- location.reload();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+$(document).ready(function () {
|
|
|
|
+ fixHugoOutput();
|
|
|
|
+ fixMermaid();
|
|
|
|
|
|
- function initThemeVariation() {
|
|
|
|
- // If theme changer component present, set its icon according to the theme mode (day, night, or auto).
|
|
|
|
- if (canChangeTheme) {
|
|
|
|
- let themeMode = getThemeMode();
|
|
|
|
- switch (themeMode) {
|
|
|
|
- case 0:
|
|
|
|
- showActiveTheme(2);
|
|
|
|
- console.info('Initialize theme variation to Light.');
|
|
|
|
- break;
|
|
|
|
- case 1:
|
|
|
|
- showActiveTheme(0);
|
|
|
|
- console.info('Initialize theme variation to Dark.');
|
|
|
|
- break;
|
|
|
|
- default:
|
|
|
|
- showActiveTheme(1);
|
|
|
|
- console.info('Initialize theme variation to Auto.');
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- // Render the day or night theme.
|
|
|
|
- let isDarkTheme = getThemeVariation();
|
|
|
|
- renderThemeVariation(isDarkTheme, true);
|
|
|
|
|
|
+ // Initialise code highlighting if enabled for this page.
|
|
|
|
+ // Note: this block should be processed after the Mermaid code-->div conversion.
|
|
|
|
+ if (code_highlighting) {
|
|
|
|
+ hljs.initHighlighting();
|
|
}
|
|
}
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Normalize Bootstrap Carousel Slide Heights.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- function normalizeCarouselSlideHeights() {
|
|
|
|
- $('.carousel').each(function () {
|
|
|
|
- // Get carousel slides.
|
|
|
|
- let items = $('.carousel-item', this);
|
|
|
|
- // Reset all slide heights.
|
|
|
|
- items.css('min-height', 0);
|
|
|
|
- // Normalize all slide heights.
|
|
|
|
- let maxHeight = Math.max.apply(null, items.map(function () {
|
|
|
|
- return $(this).outerHeight()
|
|
|
|
- }).get());
|
|
|
|
- items.css('min-height', maxHeight + 'px');
|
|
|
|
- })
|
|
|
|
- }
|
|
|
|
|
|
+ // Render theme variation, including any HLJS and Mermaid themes.
|
|
|
|
+ let {isDarkTheme, themeMode} = initThemeVariation();
|
|
|
|
+ renderThemeVariation(isDarkTheme, themeMode, true);
|
|
|
|
+});
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * Fix Hugo's Goldmark output and Mermaid code blocks.
|
|
|
|
|
|
+/* ---------------------------------------------------------------------------
|
|
|
|
+ * On window loaded.
|
|
* --------------------------------------------------------------------------- */
|
|
* --------------------------------------------------------------------------- */
|
|
|
|
|
|
- /**
|
|
|
|
- * Fix Hugo's Goldmark output.
|
|
|
|
- */
|
|
|
|
- function fixHugoOutput() {
|
|
|
|
- // Fix Goldmark table of contents.
|
|
|
|
- // - Must be performed prior to initializing ScrollSpy.
|
|
|
|
- $('#TableOfContents').addClass('nav flex-column');
|
|
|
|
- $('#TableOfContents li').addClass('nav-item');
|
|
|
|
- $('#TableOfContents li a').addClass('nav-link');
|
|
|
|
-
|
|
|
|
- // Fix Goldmark task lists (remove bullet points).
|
|
|
|
- $("input[type='checkbox'][disabled]").parents('ul').addClass('task-list');
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- /**
|
|
|
|
- * Fix Mermaid.js clash with Highlight.js.
|
|
|
|
- * Refactor Mermaid code blocks as divs to prevent Highlight parsing them and enable Mermaid to parse them.
|
|
|
|
- */
|
|
|
|
- function fixMermaid() {
|
|
|
|
- let mermaids = [];
|
|
|
|
- [].push.apply(mermaids, document.getElementsByClassName('language-mermaid'));
|
|
|
|
- for (let i = 0; i < mermaids.length; i++) {
|
|
|
|
- $(mermaids[i]).unwrap('pre'); // Remove <pre> wrapper.
|
|
|
|
- $(mermaids[i]).replaceWith(function () {
|
|
|
|
- // Convert <code> block to <div> and add `mermaid` class so that Mermaid will parse it.
|
|
|
|
- return $("<div />").append($(this).contents()).addClass('mermaid');
|
|
|
|
- });
|
|
|
|
|
|
+$(window).on('load', function () {
|
|
|
|
+ // Init Isotope Layout Engine for instances of the Portfolio widget.
|
|
|
|
+ let isotopeInstances = document.querySelectorAll('.projects-container');
|
|
|
|
+ let isotopeInstancesCount = isotopeInstances.length;
|
|
|
|
+ let isotopeCounter = 0;
|
|
|
|
+ isotopeInstances.forEach(function (isotopeInstance, index) {
|
|
|
|
+ console.debug(`Loading Isotope instance ${index}`);
|
|
|
|
+
|
|
|
|
+ // Isotope instance
|
|
|
|
+ let iso;
|
|
|
|
+
|
|
|
|
+ // Get the layout for this Isotope instance
|
|
|
|
+ let isoSection = isotopeInstance.closest('section');
|
|
|
|
+ let layout = '';
|
|
|
|
+ if (isoSection.querySelector('.isotope').classList.contains('js-layout-row')) {
|
|
|
|
+ layout = 'fitRows';
|
|
|
|
+ } else {
|
|
|
|
+ layout = 'masonry';
|
|
}
|
|
}
|
|
- }
|
|
|
|
-
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * On document ready.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- $(document).ready(function () {
|
|
|
|
- fixHugoOutput();
|
|
|
|
- fixMermaid();
|
|
|
|
|
|
|
|
- // Initialise code highlighting if enabled for this page.
|
|
|
|
- // Note: this block should be processed after the Mermaid code-->div conversion.
|
|
|
|
- if (code_highlighting) {
|
|
|
|
- hljs.initHighlighting();
|
|
|
|
|
|
+ // Get default filter (if any) for this instance
|
|
|
|
+ let defaultFilter = isoSection.querySelector('.default-project-filter');
|
|
|
|
+ let filterText = '*';
|
|
|
|
+ if (defaultFilter !== null) {
|
|
|
|
+ filterText = defaultFilter.textContent;
|
|
}
|
|
}
|
|
|
|
+ console.debug(`Default Isotope filter: ${filterText}`);
|
|
|
|
+
|
|
|
|
+ // Init Isotope instance once its images have loaded.
|
|
|
|
+ imagesLoaded(isotopeInstance, function () {
|
|
|
|
+ iso = new Isotope(isotopeInstance, {
|
|
|
|
+ itemSelector: '.isotope-item',
|
|
|
|
+ layoutMode: layout,
|
|
|
|
+ masonry: {
|
|
|
|
+ gutter: 20
|
|
|
|
+ },
|
|
|
|
+ filter: filterText
|
|
|
|
+ });
|
|
|
|
|
|
- // Initialize theme variation.
|
|
|
|
- initThemeVariation();
|
|
|
|
-
|
|
|
|
- // Change theme mode.
|
|
|
|
- $('.js-set-theme-light').click(function (e) {
|
|
|
|
- e.preventDefault();
|
|
|
|
- changeThemeModeClick(2);
|
|
|
|
- });
|
|
|
|
- $('.js-set-theme-dark').click(function (e) {
|
|
|
|
- e.preventDefault();
|
|
|
|
- changeThemeModeClick(0);
|
|
|
|
- });
|
|
|
|
- $('.js-set-theme-auto').click(function (e) {
|
|
|
|
- e.preventDefault();
|
|
|
|
- changeThemeModeClick(1);
|
|
|
|
- });
|
|
|
|
|
|
+ // Filter Isotope items when a toolbar filter button is clicked.
|
|
|
|
+ let isoFilterButtons = isoSection.querySelectorAll('.project-filters a');
|
|
|
|
+ isoFilterButtons.forEach(button => button.addEventListener('click', (e) => {
|
|
|
|
+ e.preventDefault();
|
|
|
|
+ let selector = button.getAttribute('data-filter');
|
|
|
|
+
|
|
|
|
+ // Apply filter
|
|
|
|
+ console.debug(`Updating Isotope filter to ${selector}`);
|
|
|
|
+ iso.arrange({filter: selector});
|
|
|
|
+
|
|
|
|
+ // Update active toolbar filter button
|
|
|
|
+ button.classList.remove('active');
|
|
|
|
+ button.classList.add('active');
|
|
|
|
+ let buttonSiblings = getSiblings(button);
|
|
|
|
+ buttonSiblings.forEach(buttonSibling => {
|
|
|
|
+ buttonSibling.classList.remove('active');
|
|
|
|
+ buttonSibling.classList.remove('all');
|
|
|
|
+ });
|
|
|
|
+ }));
|
|
|
|
|
|
- // Live update of day/night mode on system preferences update (no refresh required).
|
|
|
|
- // Note: since we listen only for *dark* events, we won't detect other scheme changes such as light to no-preference.
|
|
|
|
- const darkModeMediaQuery = window.matchMedia('(prefers-color-scheme: dark)');
|
|
|
|
- darkModeMediaQuery.addListener((e) => {
|
|
|
|
- if (!canChangeTheme()) {
|
|
|
|
- // Changing theme variation is not allowed by admin.
|
|
|
|
- return;
|
|
|
|
- }
|
|
|
|
- const darkModeOn = e.matches;
|
|
|
|
- console.log(`OS dark mode preference changed to ${darkModeOn ? '🌒 on' : '☀️ off'}.`);
|
|
|
|
- let currentThemeVariation = parseInt(localStorage.getItem('dark_mode') || 2);
|
|
|
|
- let isDarkTheme;
|
|
|
|
- if (currentThemeVariation === 2) {
|
|
|
|
- if (window.matchMedia('(prefers-color-scheme: dark)').matches) {
|
|
|
|
- // The visitor prefers dark themes.
|
|
|
|
- isDarkTheme = true;
|
|
|
|
- } else if (window.matchMedia('(prefers-color-scheme: light)').matches) {
|
|
|
|
- // The visitor prefers light themes.
|
|
|
|
- isDarkTheme = false;
|
|
|
|
- } else {
|
|
|
|
- // The visitor does not have a day or night preference, so use the theme's default setting.
|
|
|
|
- isDarkTheme = isSiteThemeDark;
|
|
|
|
- }
|
|
|
|
- renderThemeVariation(isDarkTheme);
|
|
|
|
- }
|
|
|
|
|
|
+ // Check if all Isotope instances have loaded.
|
|
|
|
+ incrementIsotopeCounter();
|
|
});
|
|
});
|
|
});
|
|
});
|
|
|
|
|
|
- /* ---------------------------------------------------------------------------
|
|
|
|
- * On window loaded.
|
|
|
|
- * --------------------------------------------------------------------------- */
|
|
|
|
-
|
|
|
|
- $(window).on('load', function () {
|
|
|
|
- // Filter projects.
|
|
|
|
- $('.projects-container').each(function (index, container) {
|
|
|
|
- let $container = $(container);
|
|
|
|
- let $section = $container.closest('section');
|
|
|
|
- let layout;
|
|
|
|
- if ($section.find('.isotope').hasClass('js-layout-row')) {
|
|
|
|
- layout = 'fitRows';
|
|
|
|
- } else {
|
|
|
|
- layout = 'masonry';
|
|
|
|
|
|
+ // Hook to perform actions once all Isotope instances have loaded.
|
|
|
|
+ function incrementIsotopeCounter() {
|
|
|
|
+ isotopeCounter++;
|
|
|
|
+ if (isotopeCounter === isotopeInstancesCount) {
|
|
|
|
+ console.debug(`All Portfolio Isotope instances loaded.`);
|
|
|
|
+ // Once all Isotope instances and their images have loaded, scroll to hash (if set).
|
|
|
|
+ // Prevents scrolling to the wrong location due to the dynamic height of Isotope instances.
|
|
|
|
+ // Each Isotope instance height is affected by applying filters and loading images.
|
|
|
|
+ // Without this logic, the scroll location can appear correct, but actually a few pixels out and hence Scrollspy
|
|
|
|
+ // can highlight the wrong nav link.
|
|
|
|
+ if (window.location.hash) {
|
|
|
|
+ scrollToAnchor(decodeURIComponent(window.location.hash), 0);
|
|
}
|
|
}
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
|
|
- $container.imagesLoaded(function () {
|
|
|
|
- // Initialize Isotope after all images have loaded.
|
|
|
|
- $container.isotope({
|
|
|
|
- itemSelector: '.isotope-item',
|
|
|
|
- layoutMode: layout,
|
|
|
|
- masonry: {
|
|
|
|
- gutter: 20
|
|
|
|
- },
|
|
|
|
- filter: $section.find('.default-project-filter').text()
|
|
|
|
- });
|
|
|
|
-
|
|
|
|
- // Filter items when filter link is clicked.
|
|
|
|
- $section.find('.project-filters a').click(function () {
|
|
|
|
- let selector = $(this).attr('data-filter');
|
|
|
|
- $container.isotope({filter: selector});
|
|
|
|
- $(this).removeClass('active').addClass('active').siblings().removeClass('active all');
|
|
|
|
- return false;
|
|
|
|
- });
|
|
|
|
|
|
+ // Enable publication filter for publication index page.
|
|
|
|
+ if ($('.pub-filters-select')) {
|
|
|
|
+ filter_publications();
|
|
|
|
+ // Useful for changing hash manually (e.g. in development):
|
|
|
|
+ // window.addEventListener('hashchange', filter_publications, false);
|
|
|
|
+ }
|
|
|
|
|
|
- // If window hash is set, scroll to hash.
|
|
|
|
- // Placing this within `imagesLoaded` prevents scrolling to the wrong location due to dynamic image loading
|
|
|
|
- // affecting page layout and position of the target anchor ID.
|
|
|
|
- // Note: If there are multiple project widgets on a page, ideally only perform this once after images
|
|
|
|
- // from *all* project widgets have finished loading.
|
|
|
|
- if (window.location.hash) {
|
|
|
|
- scrollToAnchor();
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
|
|
+ // Load citation modal on 'Cite' click.
|
|
|
|
+ $('.js-cite-modal').click(function (e) {
|
|
|
|
+ e.preventDefault();
|
|
|
|
+ let filename = $(this).attr('data-filename');
|
|
|
|
+ let modal = $('#modal');
|
|
|
|
+ modal.find('.modal-body code').load(filename, function (response, status, xhr) {
|
|
|
|
+ if (status == 'error') {
|
|
|
|
+ let msg = "Error: ";
|
|
|
|
+ $('#modal-error').html(msg + xhr.status + " " + xhr.statusText);
|
|
|
|
+ } else {
|
|
|
|
+ $('.js-download-cite').attr('href', filename);
|
|
|
|
+ }
|
|
});
|
|
});
|
|
|
|
+ modal.modal('show');
|
|
|
|
+ });
|
|
|
|
|
|
- // Enable publication filter for publication index page.
|
|
|
|
- if ($('.pub-filters-select')) {
|
|
|
|
- filter_publications();
|
|
|
|
- // Useful for changing hash manually (e.g. in development):
|
|
|
|
- // window.addEventListener('hashchange', filter_publications, false);
|
|
|
|
|
|
+ // Copy citation text on 'Copy' click.
|
|
|
|
+ $('.js-copy-cite').click(function (e) {
|
|
|
|
+ e.preventDefault();
|
|
|
|
+ // Get selection.
|
|
|
|
+ let range = document.createRange();
|
|
|
|
+ let code_node = document.querySelector('#modal .modal-body');
|
|
|
|
+ range.selectNode(code_node);
|
|
|
|
+ window.getSelection().addRange(range);
|
|
|
|
+ try {
|
|
|
|
+ // Execute the copy command.
|
|
|
|
+ document.execCommand('copy');
|
|
|
|
+ } catch (e) {
|
|
|
|
+ console.log('Error: citation copy failed.');
|
|
}
|
|
}
|
|
|
|
+ // Remove selection.
|
|
|
|
+ window.getSelection().removeRange(range);
|
|
|
|
+ });
|
|
|
|
|
|
- // Scroll to top of page.
|
|
|
|
- $('.back-to-top').click(function (event) {
|
|
|
|
- event.preventDefault();
|
|
|
|
- $('html, body').animate({
|
|
|
|
- 'scrollTop': 0
|
|
|
|
- }, 800, function () {
|
|
|
|
- window.location.hash = "";
|
|
|
|
- });
|
|
|
|
- });
|
|
|
|
|
|
+ // Initialise Google Maps if necessary.
|
|
|
|
+ initMap();
|
|
|
|
|
|
- // Load citation modal on 'Cite' click.
|
|
|
|
- $('.js-cite-modal').click(function (e) {
|
|
|
|
- e.preventDefault();
|
|
|
|
- let filename = $(this).attr('data-filename');
|
|
|
|
- let modal = $('#modal');
|
|
|
|
- modal.find('.modal-body code').load(filename, function (response, status, xhr) {
|
|
|
|
- if (status == 'error') {
|
|
|
|
- let msg = "Error: ";
|
|
|
|
- $('#modal-error').html(msg + xhr.status + " " + xhr.statusText);
|
|
|
|
- } else {
|
|
|
|
- $('.js-download-cite').attr('href', filename);
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
- modal.modal('show');
|
|
|
|
- });
|
|
|
|
|
|
+ // Print latest version of GitHub projects.
|
|
|
|
+ let githubReleaseSelector = '.js-github-release';
|
|
|
|
+ if ($(githubReleaseSelector).length > 0) {
|
|
|
|
+ printLatestRelease(githubReleaseSelector, $(githubReleaseSelector).data('repo'));
|
|
|
|
+ }
|
|
|
|
|
|
- // Copy citation text on 'Copy' click.
|
|
|
|
- $('.js-copy-cite').click(function (e) {
|
|
|
|
- e.preventDefault();
|
|
|
|
- // Get selection.
|
|
|
|
- let range = document.createRange();
|
|
|
|
- let code_node = document.querySelector('#modal .modal-body');
|
|
|
|
- range.selectNode(code_node);
|
|
|
|
- window.getSelection().addRange(range);
|
|
|
|
- try {
|
|
|
|
- // Execute the copy command.
|
|
|
|
- document.execCommand('copy');
|
|
|
|
- } catch (e) {
|
|
|
|
- console.log('Error: citation copy failed.');
|
|
|
|
|
|
+ // Parse Wowchemy keyboard shortcuts.
|
|
|
|
+ document.addEventListener('keyup', (event) => {
|
|
|
|
+ if (event.code === "Escape") {
|
|
|
|
+ const body = document.body;
|
|
|
|
+ if (body.classList.contains('searching')) {
|
|
|
|
+ // Close search dialog.
|
|
|
|
+ toggleSearchDialog();
|
|
}
|
|
}
|
|
- // Remove selection.
|
|
|
|
- window.getSelection().removeRange(range);
|
|
|
|
- });
|
|
|
|
-
|
|
|
|
- // Initialise Google Maps if necessary.
|
|
|
|
- initMap();
|
|
|
|
-
|
|
|
|
- // Print latest version of GitHub projects.
|
|
|
|
- let githubReleaseSelector = '.js-github-release';
|
|
|
|
- if ($(githubReleaseSelector).length > 0)
|
|
|
|
- printLatestRelease(githubReleaseSelector, $(githubReleaseSelector).data('repo'));
|
|
|
|
|
|
+ }
|
|
|
|
+ // Use `key` to check for slash. Otherwise, with `code` we need to check for modifiers.
|
|
|
|
+ if (event.key === "/" ) {
|
|
|
|
+ let focusedElement = (
|
|
|
|
+ document.hasFocus() &&
|
|
|
|
+ document.activeElement !== document.body &&
|
|
|
|
+ document.activeElement !== document.documentElement &&
|
|
|
|
+ document.activeElement
|
|
|
|
+ ) || null;
|
|
|
|
+ let isInputFocused = focusedElement instanceof HTMLInputElement || focusedElement instanceof HTMLTextAreaElement;
|
|
|
|
+ if (search_config && !isInputFocused) {
|
|
|
|
+ // Open search dialog.
|
|
|
|
+ event.preventDefault();
|
|
|
|
+ toggleSearchDialog();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ });
|
|
|
|
|
|
|
|
+ // Search event handler
|
|
|
|
+ // Check that built-in search or Algolia enabled.
|
|
|
|
+ if (search_config) {
|
|
// On search icon click toggle search dialog.
|
|
// On search icon click toggle search dialog.
|
|
$('.js-search').click(function (e) {
|
|
$('.js-search').click(function (e) {
|
|
e.preventDefault();
|
|
e.preventDefault();
|
|
toggleSearchDialog();
|
|
toggleSearchDialog();
|
|
});
|
|
});
|
|
- $(document).on('keydown', function (e) {
|
|
|
|
- if (e.which == 27) {
|
|
|
|
- // `Esc` key pressed.
|
|
|
|
- if ($('body').hasClass('searching')) {
|
|
|
|
- toggleSearchDialog();
|
|
|
|
- }
|
|
|
|
- } else if (e.which == 191 && e.shiftKey == false && !$('input,textarea').is(':focus')) {
|
|
|
|
- // `/` key pressed outside of text input.
|
|
|
|
- e.preventDefault();
|
|
|
|
- toggleSearchDialog();
|
|
|
|
- }
|
|
|
|
- });
|
|
|
|
-
|
|
|
|
- // Init. author notes (tooltips).
|
|
|
|
- $('[data-toggle="tooltip"]').tooltip();
|
|
|
|
|
|
+ }
|
|
|
|
|
|
|
|
+ // Init. author notes (tooltips).
|
|
|
|
+ $('[data-toggle="tooltip"]').tooltip();
|
|
|
|
+
|
|
|
|
+ // Re-initialize Scrollspy with dynamic navbar height offset.
|
|
|
|
+ fixScrollspy();
|
|
|
|
+});
|
|
|
|
+
|
|
|
|
+// Theme chooser events.
|
|
|
|
+let linkLight = document.querySelector('.js-set-theme-light');
|
|
|
|
+let linkDark = document.querySelector('.js-set-theme-dark');
|
|
|
|
+let linkAuto = document.querySelector('.js-set-theme-auto');
|
|
|
|
+if (linkLight && linkDark && linkAuto) {
|
|
|
|
+ linkLight.addEventListener('click', event => {
|
|
|
|
+ event.preventDefault();
|
|
|
|
+ changeThemeModeClick(0);
|
|
});
|
|
});
|
|
-
|
|
|
|
- // Normalize Bootstrap carousel slide heights.
|
|
|
|
- $(window).on('load resize orientationchange', normalizeCarouselSlideHeights);
|
|
|
|
-
|
|
|
|
- // Automatic main menu dropdowns on mouse over.
|
|
|
|
- $('body').on('mouseenter mouseleave', '.dropdown', function (e) {
|
|
|
|
- var dropdown = $(e.target).closest('.dropdown');
|
|
|
|
- var menu = $('.dropdown-menu', dropdown);
|
|
|
|
- dropdown.addClass('show');
|
|
|
|
- menu.addClass('show');
|
|
|
|
- setTimeout(function () {
|
|
|
|
- dropdown[dropdown.is(':hover') ? 'addClass' : 'removeClass']('show');
|
|
|
|
- menu[dropdown.is(':hover') ? 'addClass' : 'removeClass']('show');
|
|
|
|
- }, 300);
|
|
|
|
-
|
|
|
|
- // Re-initialize Scrollspy with dynamic navbar height offset.
|
|
|
|
- fixScrollspy();
|
|
|
|
-
|
|
|
|
- if (window.location.hash) {
|
|
|
|
- // When accessing homepage from another page and `#top` hash is set, show top of page (no hash).
|
|
|
|
- if (window.location.hash == "#top") {
|
|
|
|
- window.location.hash = ""
|
|
|
|
- } else if (!$('.projects-container').length) {
|
|
|
|
- // If URL contains a hash and there are no dynamically loaded images on the page,
|
|
|
|
- // immediately scroll to target ID taking into account responsive offset.
|
|
|
|
- // Otherwise, wait for `imagesLoaded()` to complete before scrolling to hash to prevent scrolling to wrong
|
|
|
|
- // location.
|
|
|
|
- scrollToAnchor();
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- // Call `fixScrollspy` when window is resized.
|
|
|
|
- let resizeTimer;
|
|
|
|
- $(window).resize(function () {
|
|
|
|
- clearTimeout(resizeTimer);
|
|
|
|
- resizeTimer = setTimeout(fixScrollspy, 200);
|
|
|
|
- });
|
|
|
|
|
|
+ linkDark.addEventListener('click', event => {
|
|
|
|
+ event.preventDefault();
|
|
|
|
+ changeThemeModeClick(1);
|
|
});
|
|
});
|
|
-
|
|
|
|
-})(jQuery);
|
|
|
|
|
|
+ linkAuto.addEventListener('click', event => {
|
|
|
|
+ event.preventDefault();
|
|
|
|
+ changeThemeModeClick(2);
|
|
|
|
+ });
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+// Media Query events.
|
|
|
|
+// Live update of day/night mode on system preferences update (no refresh required).
|
|
|
|
+// Note: since we listen only for *dark* events, we won't detect other scheme changes such as light to no-preference.
|
|
|
|
+const darkModeMediaQuery = window.matchMedia('(prefers-color-scheme: dark)');
|
|
|
|
+darkModeMediaQuery.addEventListener("change", (event) => {
|
|
|
|
+ onMediaQueryListEvent(event);
|
|
|
|
+});
|
|
|
|
+
|
|
|
|
+// Normalize Bootstrap carousel slide heights for Slider widget instances.
|
|
|
|
+window.addEventListener('load', normalizeCarouselSlideHeights);
|
|
|
|
+window.addEventListener('resize', normalizeCarouselSlideHeights);
|
|
|
|
+window.addEventListener('orientationchange', normalizeCarouselSlideHeights);
|
|
|
|
+
|
|
|
|
+// Automatic main menu dropdowns on mouse over.
|
|
|
|
+$('body').on('mouseenter mouseleave', '.dropdown', function (e) {
|
|
|
|
+ var dropdown = $(e.target).closest('.dropdown');
|
|
|
|
+ var menu = $('.dropdown-menu', dropdown);
|
|
|
|
+ dropdown.addClass('show');
|
|
|
|
+ menu.addClass('show');
|
|
|
|
+ setTimeout(function () {
|
|
|
|
+ dropdown[dropdown.is(':hover') ? 'addClass' : 'removeClass']('show');
|
|
|
|
+ menu[dropdown.is(':hover') ? 'addClass' : 'removeClass']('show');
|
|
|
|
+ }, 300);
|
|
|
|
+});
|
|
|
|
+
|
|
|
|
+// Call `fixScrollspy` when window is resized.
|
|
|
|
+let resizeTimer;
|
|
|
|
+$(window).resize(function () {
|
|
|
|
+ clearTimeout(resizeTimer);
|
|
|
|
+ resizeTimer = setTimeout(fixScrollspy, 200);
|
|
|
|
+});
|