This repository has been archived on 2022-05-02. You can view files and clone it, but cannot push or open issues or pull requests.
Ecosysteme/presentation/plugin/menu/plugin.js

1252 lines
37 KiB
JavaScript

/*
* Reveal.js menu plugin
* MIT licensed
* (c) Greg Denehy 2020
*/
const Plugin = () => {
const ieVersion = (function () {
let browser = /(msie) ([\w.]+)/.exec(
window.navigator.userAgent.toLowerCase()
);
if (browser && browser[1] === 'msie') {
return parseFloat(browser[2]);
}
return null;
})();
var deck;
var config;
var options;
var initialised = false;
function scriptPath() {
// obtain plugin path from the script element
var path;
const script = document.querySelector('script[src$="menu.js"]');
if (script) {
var sel = document.querySelector('script[src$="menu.js"]');
if (sel) {
path = sel.src.slice(0, -7);
}
} else {
path = import.meta.url.slice(0, import.meta.url.lastIndexOf('/') + 1);
}
return path;
}
function initOptions(config) {
options = config.menu || {};
options.path = options.path || scriptPath() || 'plugin/menu/';
if (!options.path.endsWith('/')) {
options.path += '/';
}
// Set defaults
if (options.side === undefined) options.side = 'left';
if (options.numbers === undefined) options.numbers = false;
if (typeof options.titleSelector !== 'string')
options.titleSelector = 'h1, h2, h3, h4, h5';
if (options.hideMissingTitles === undefined)
options.hideMissingTitles = false;
if (options.useTextContentForMissingTitles === undefined)
options.useTextContentForMissingTitles = false;
if (options.markers === undefined) options.markers = true;
if (typeof options.themesPath !== 'string')
options.themesPath = 'dist/theme/';
if (!options.themesPath.endsWith('/')) options.themesPath += '/';
if (!select('link#theme')) options.themes = false;
if (options.themes === true) {
options.themes = [
{ name: 'Black', theme: options.themesPath + 'black.css' },
{ name: 'White', theme: options.themesPath + 'white.css' },
{ name: 'League', theme: options.themesPath + 'league.css' },
{ name: 'Sky', theme: options.themesPath + 'sky.css' },
{ name: 'Beige', theme: options.themesPath + 'beige.css' },
{ name: 'Simple', theme: options.themesPath + 'simple.css' },
{ name: 'Serif', theme: options.themesPath + 'serif.css' },
{ name: 'Blood', theme: options.themesPath + 'blood.css' },
{ name: 'Night', theme: options.themesPath + 'night.css' },
{ name: 'Moon', theme: options.themesPath + 'moon.css' },
{ name: 'Solarized', theme: options.themesPath + 'solarized.css' }
];
} else if (!Array.isArray(options.themes)) {
options.themes = false;
}
if (options.transitions === undefined) options.transitions = false;
if (options.transitions === true) {
options.transitions = [
'None',
'Fade',
'Slide',
'Convex',
'Concave',
'Zoom'
];
} else if (
options.transitions !== false &&
(!Array.isArray(options.transitions) ||
!options.transitions.every(function (e) {
return typeof e === 'string';
}))
) {
console.error(
"reveal.js-menu error: transitions config value must be 'true' or an array of strings, eg ['None', 'Fade', 'Slide')"
);
options.transitions = false;
}
if (ieVersion && ieVersion <= 9) {
// transitions aren't support in IE9 anyway, so no point in showing them
options.transitions = false;
}
if (typeof options.openButton === 'undefined') options.openButton = true;
if (typeof options.openSlideNumber === 'undefined')
options.openSlideNumber = false;
if (typeof options.keyboard === 'undefined') options.keyboard = true;
if (typeof options.sticky === 'undefined') options.sticky = false;
if (typeof options.autoOpen === 'undefined') options.autoOpen = true;
if (typeof options.delayInit === 'undefined') options.delayInit = false;
if (typeof options.openOnInit === 'undefined') options.openOnInit = false;
}
var mouseSelectionEnabled = true;
function disableMouseSelection() {
mouseSelectionEnabled = false;
}
function reenableMouseSelection() {
// wait until the mouse has moved before re-enabling mouse selection
// to avoid selections on scroll
select('nav.slide-menu').addEventListener('mousemove', function fn(e) {
select('nav.slide-menu').removeEventListener('mousemove', fn);
//XXX this should select the item under the mouse
mouseSelectionEnabled = true;
});
}
//
// Keyboard handling
//
function getOffset(el) {
var _x = 0;
var _y = 0;
while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {
_x += el.offsetLeft - el.scrollLeft;
_y += el.offsetTop - el.scrollTop;
el = el.offsetParent;
}
return { top: _y, left: _x };
}
function visibleOffset(el) {
var offsetFromTop = getOffset(el).top - el.offsetParent.offsetTop;
if (offsetFromTop < 0) return -offsetFromTop;
var offsetFromBottom =
el.offsetParent.offsetHeight -
(el.offsetTop - el.offsetParent.scrollTop + el.offsetHeight);
if (offsetFromBottom < 0) return offsetFromBottom;
return 0;
}
function keepVisible(el) {
var offset = visibleOffset(el);
if (offset) {
disableMouseSelection();
el.scrollIntoView(offset > 0);
reenableMouseSelection();
}
}
function scrollItemToTop(el) {
disableMouseSelection();
el.offsetParent.scrollTop = el.offsetTop;
reenableMouseSelection();
}
function scrollItemToBottom(el) {
disableMouseSelection();
el.offsetParent.scrollTop =
el.offsetTop - el.offsetParent.offsetHeight + el.offsetHeight;
reenableMouseSelection();
}
function selectItem(el) {
el.classList.add('selected');
keepVisible(el);
if (options.sticky && options.autoOpen) openItem(el);
}
function onDocumentKeyDown(event) {
// opening menu is handled by registering key binding with Reveal below
if (isOpen()) {
event.stopImmediatePropagation();
switch (event.keyCode) {
// case 77:
// closeMenu();
// break;
// h, left - change panel
case 72:
case 37:
prevPanel();
break;
// l, right - change panel
case 76:
case 39:
nextPanel();
break;
// k, up
case 75:
case 38:
var currItem =
select('.active-menu-panel .slide-menu-items li.selected') ||
select('.active-menu-panel .slide-menu-items li.active');
if (currItem) {
selectAll('.active-menu-panel .slide-menu-items li').forEach(
function (item) {
item.classList.remove('selected');
}
);
var nextItem =
select(
'.active-menu-panel .slide-menu-items li[data-item="' +
(parseInt(currItem.getAttribute('data-item')) - 1) +
'"]'
) || currItem;
selectItem(nextItem);
} else {
var item = select(
'.active-menu-panel .slide-menu-items li.slide-menu-item'
);
if (item) selectItem(item);
}
break;
// j, down
case 74:
case 40:
var currItem =
select('.active-menu-panel .slide-menu-items li.selected') ||
select('.active-menu-panel .slide-menu-items li.active');
if (currItem) {
selectAll('.active-menu-panel .slide-menu-items li').forEach(
function (item) {
item.classList.remove('selected');
}
);
var nextItem =
select(
'.active-menu-panel .slide-menu-items li[data-item="' +
(parseInt(currItem.getAttribute('data-item')) + 1) +
'"]'
) || currItem;
selectItem(nextItem);
} else {
var item = select(
'.active-menu-panel .slide-menu-items li.slide-menu-item'
);
if (item) selectItem(item);
}
break;
// pageup, u
case 33:
case 85:
var itemsAbove = selectAll(
'.active-menu-panel .slide-menu-items li'
).filter(function (item) {
return visibleOffset(item) > 0;
});
var visibleItems = selectAll(
'.active-menu-panel .slide-menu-items li'
).filter(function (item) {
return visibleOffset(item) == 0;
});
var firstVisible =
itemsAbove.length > 0 &&
Math.abs(visibleOffset(itemsAbove[itemsAbove.length - 1])) <
itemsAbove[itemsAbove.length - 1].clientHeight
? itemsAbove[itemsAbove.length - 1]
: visibleItems[0];
if (firstVisible) {
if (
firstVisible.classList.contains('selected') &&
itemsAbove.length > 0
) {
// at top of viewport already, page scroll (if not at start)
// ...move selected item to bottom, and change selection to last fully visible item at top
scrollItemToBottom(firstVisible);
visibleItems = selectAll(
'.active-menu-panel .slide-menu-items li'
).filter(function (item) {
return visibleOffset(item) == 0;
});
if (visibleItems[0] == firstVisible) {
// prev item is still beyond the viewport (for custom panels)
firstVisible = itemsAbove[itemsAbove.length - 1];
} else {
firstVisible = visibleItems[0];
}
}
selectAll('.active-menu-panel .slide-menu-items li').forEach(
function (item) {
item.classList.remove('selected');
}
);
selectItem(firstVisible);
// ensure selected item is positioned at the top of the viewport
scrollItemToTop(firstVisible);
}
break;
// pagedown, d
case 34:
case 68:
var visibleItems = selectAll(
'.active-menu-panel .slide-menu-items li'
).filter(function (item) {
return visibleOffset(item) == 0;
});
var itemsBelow = selectAll(
'.active-menu-panel .slide-menu-items li'
).filter(function (item) {
return visibleOffset(item) < 0;
});
var lastVisible =
itemsBelow.length > 0 &&
Math.abs(visibleOffset(itemsBelow[0])) < itemsBelow[0].clientHeight
? itemsBelow[0]
: visibleItems[visibleItems.length - 1];
if (lastVisible) {
if (
lastVisible.classList.contains('selected') &&
itemsBelow.length > 0
) {
// at bottom of viewport already, page scroll (if not at end)
// ...move selected item to top, and change selection to last fully visible item at bottom
scrollItemToTop(lastVisible);
visibleItems = selectAll(
'.active-menu-panel .slide-menu-items li'
).filter(function (item) {
return visibleOffset(item) == 0;
});
if (visibleItems[visibleItems.length - 1] == lastVisible) {
// next item is still beyond the viewport (for custom panels)
lastVisible = itemsBelow[0];
} else {
lastVisible = visibleItems[visibleItems.length - 1];
}
}
selectAll('.active-menu-panel .slide-menu-items li').forEach(
function (item) {
item.classList.remove('selected');
}
);
selectItem(lastVisible);
// ensure selected item is positioned at the bottom of the viewport
scrollItemToBottom(lastVisible);
}
break;
// home
case 36:
selectAll('.active-menu-panel .slide-menu-items li').forEach(
function (item) {
item.classList.remove('selected');
}
);
var item = select(
'.active-menu-panel .slide-menu-items li:first-of-type'
);
if (item) {
item.classList.add('selected');
keepVisible(item);
}
break;
// end
case 35:
selectAll('.active-menu-panel .slide-menu-items li').forEach(
function (item) {
item.classList.remove('selected');
}
);
var item = select(
'.active-menu-panel .slide-menu-items:last-of-type li:last-of-type'
);
if (item) {
item.classList.add('selected');
keepVisible(item);
}
break;
// space, return
case 32:
case 13:
var currItem = select(
'.active-menu-panel .slide-menu-items li.selected'
);
if (currItem) {
openItem(currItem, true);
}
break;
// esc
case 27:
closeMenu(null, true);
break;
}
}
}
//
// Utilty functions
//
function openMenu(event) {
if (event) event.preventDefault();
if (!isOpen()) {
select('body').classList.add('slide-menu-active');
select('.reveal').classList.add(
'has-' + options.effect + '-' + options.side
);
select('.slide-menu').classList.add('active');
select('.slide-menu-overlay').classList.add('active');
// identify active theme
if (options.themes) {
selectAll('div[data-panel="Themes"] li').forEach(function (i) {
i.classList.remove('active');
});
selectAll(
'li[data-theme="' + select('link#theme').getAttribute('href') + '"]'
).forEach(function (i) {
i.classList.add('active');
});
}
// identify active transition
if (options.transitions) {
selectAll('div[data-panel="Transitions"] li').forEach(function (i) {
i.classList.remove('active');
});
selectAll('li[data-transition="' + config.transition + '"]').forEach(
function (i) {
i.classList.add('active');
}
);
}
// set item selections to match active items
var items = selectAll('.slide-menu-panel li.active');
items.forEach(function (i) {
i.classList.add('selected');
keepVisible(i);
});
}
}
function closeMenu(event, force) {
if (event) event.preventDefault();
if (!options.sticky || force) {
select('body').classList.remove('slide-menu-active');
select('.reveal').classList.remove(
'has-' + options.effect + '-' + options.side
);
select('.slide-menu').classList.remove('active');
select('.slide-menu-overlay').classList.remove('active');
selectAll('.slide-menu-panel li.selected').forEach(function (i) {
i.classList.remove('selected');
});
}
}
function toggleMenu(event) {
if (isOpen()) {
closeMenu(event, true);
} else {
openMenu(event);
}
}
function isOpen() {
return select('body').classList.contains('slide-menu-active');
}
function openPanel(event, ref) {
openMenu(event);
var panel = ref;
if (typeof ref !== 'string') {
panel = event.currentTarget.getAttribute('data-panel');
}
select('.slide-menu-toolbar > li.active-toolbar-button').classList.remove(
'active-toolbar-button'
);
select('li[data-panel="' + panel + '"]').classList.add(
'active-toolbar-button'
);
select('.slide-menu-panel.active-menu-panel').classList.remove(
'active-menu-panel'
);
select('div[data-panel="' + panel + '"]').classList.add(
'active-menu-panel'
);
}
function nextPanel() {
var next =
(parseInt(select('.active-toolbar-button').getAttribute('data-button')) +
1) %
buttons;
openPanel(
null,
select('.toolbar-panel-button[data-button="' + next + '"]').getAttribute(
'data-panel'
)
);
}
function prevPanel() {
var next =
parseInt(select('.active-toolbar-button').getAttribute('data-button')) -
1;
if (next < 0) {
next = buttons - 1;
}
openPanel(
null,
select('.toolbar-panel-button[data-button="' + next + '"]').getAttribute(
'data-panel'
)
);
}
function openItem(item, force) {
var h = parseInt(item.getAttribute('data-slide-h'));
var v = parseInt(item.getAttribute('data-slide-v'));
var theme = item.getAttribute('data-theme');
var highlightTheme = item.getAttribute('data-highlight-theme');
var transition = item.getAttribute('data-transition');
if (!isNaN(h) && !isNaN(v)) {
deck.slide(h, v);
}
if (theme) {
changeStylesheet('theme', theme);
}
if (highlightTheme) {
changeStylesheet('highlight-theme', highlightTheme);
}
if (transition) {
deck.configure({ transition: transition });
}
var link = select('a', item);
if (link) {
if (
force ||
!options.sticky ||
(options.autoOpen && link.href.startsWith('#')) ||
link.href.startsWith(
window.location.origin + window.location.pathname + '#'
)
) {
link.click();
}
}
closeMenu();
}
function clicked(event) {
if (event.target.nodeName !== 'A') {
event.preventDefault();
}
openItem(event.currentTarget);
}
function highlightCurrentSlide() {
var state = deck.getState();
selectAll('li.slide-menu-item, li.slide-menu-item-vertical').forEach(
function (item) {
item.classList.remove('past');
item.classList.remove('active');
item.classList.remove('future');
var h = parseInt(item.getAttribute('data-slide-h'));
var v = parseInt(item.getAttribute('data-slide-v'));
if (h < state.indexh || (h === state.indexh && v < state.indexv)) {
item.classList.add('past');
} else if (h === state.indexh && v === state.indexv) {
item.classList.add('active');
} else {
item.classList.add('future');
}
}
);
}
function matchRevealStyle() {
var revealStyle = window.getComputedStyle(select('.reveal'));
var element = select('.slide-menu');
element.style.fontFamily = revealStyle.fontFamily;
//XXX could adjust the complete menu style to match the theme, ie colors, etc
}
var buttons = 0;
function initMenu() {
if (!initialised) {
var parent = select('.reveal').parentElement;
var top = create('div', { class: 'slide-menu-wrapper' });
parent.appendChild(top);
var panels = create('nav', {
class: 'slide-menu slide-menu--' + options.side
});
if (typeof options.width === 'string') {
if (
['normal', 'wide', 'third', 'half', 'full'].indexOf(options.width) !=
-1
) {
panels.classList.add('slide-menu--' + options.width);
} else {
panels.classList.add('slide-menu--custom');
panels.style.width = options.width;
}
}
top.appendChild(panels);
matchRevealStyle();
var overlay = create('div', { class: 'slide-menu-overlay' });
top.appendChild(overlay);
overlay.onclick = function () {
closeMenu(null, true);
};
var toolbar = create('ol', { class: 'slide-menu-toolbar' });
select('.slide-menu').appendChild(toolbar);
function addToolbarButton(title, ref, icon, style, fn, active) {
var attrs = {
'data-button': '' + buttons++,
class:
'toolbar-panel-button' + (active ? ' active-toolbar-button' : '')
};
if (ref) {
attrs['data-panel'] = ref;
}
var button = create('li', attrs);
if (icon.startsWith('fa-')) {
button.appendChild(create('i', { class: style + ' ' + icon }));
} else {
button.innerHTML = icon + '</i>';
}
button.appendChild(create('br'), select('i', button));
button.appendChild(
create('span', { class: 'slide-menu-toolbar-label' }, title),
select('i', button)
);
button.onclick = fn;
toolbar.appendChild(button);
return button;
}
addToolbarButton('Slides', 'Slides', 'fa-images', 'fas', openPanel, true);
if (options.custom) {
options.custom.forEach(function (element, index, array) {
addToolbarButton(
element.title,
'Custom' + index,
element.icon,
null,
openPanel
);
});
}
if (options.themes) {
addToolbarButton('Themes', 'Themes', 'fa-adjust', 'fas', openPanel);
}
if (options.transitions) {
addToolbarButton(
'Transitions',
'Transitions',
'fa-sticky-note',
'fas',
openPanel
);
}
var button = create('li', {
id: 'close',
class: 'toolbar-panel-button'
});
button.appendChild(create('i', { class: 'fas fa-times' }));
button.appendChild(create('br'));
button.appendChild(
create('span', { class: 'slide-menu-toolbar-label' }, 'Close')
);
button.onclick = function () {
closeMenu(null, true);
};
toolbar.appendChild(button);
//
// Slide links
//
function generateItem(type, section, i, h, v) {
var link = '/#/' + h;
if (typeof v === 'number' && !isNaN(v)) link += '/' + v;
function text(selector, parent) {
if (selector === '') return null;
var el = parent ? select(selector, section) : select(selector);
if (el) return el.textContent;
return null;
}
var title =
section.getAttribute('data-menu-title') ||
text('.menu-title', section) ||
text(options.titleSelector, section);
if (!title && options.useTextContentForMissingTitles) {
// attempt to figure out a title based on the text in the slide
title = section.textContent.trim();
if (title) {
title =
title
.split('\n')
.map(function (t) {
return t.trim();
})
.join(' ')
.trim()
.replace(/^(.{16}[^\s]*).*/, '$1') // limit to 16 chars plus any consecutive non-whitespace chars (to avoid breaking words)
.replace(/&/g, '&amp;')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;')
.replace(/"/g, '&quot;')
.replace(/'/g, '&#039;') + '...';
}
}
if (!title) {
if (options.hideMissingTitles) return '';
type += ' no-title';
title = 'Slide ' + (i + 1);
}
var item = create('li', {
class: type,
'data-item': i,
'data-slide-h': h,
'data-slide-v': v === undefined ? 0 : v
});
if (options.markers) {
item.appendChild(
create('i', { class: 'fas fa-check-circle fa-fw past' })
);
item.appendChild(
create('i', {
class: 'fas fa-arrow-alt-circle-right fa-fw active'
})
);
item.appendChild(
create('i', { class: 'far fa-circle fa-fw future' })
);
}
if (options.numbers) {
// Number formatting taken from reveal.js
var value = [];
var format = 'h.v';
// Check if a custom number format is available
if (typeof options.numbers === 'string') {
format = options.numbers;
} else if (typeof config.slideNumber === 'string') {
// Take user defined number format for slides
format = config.slideNumber;
}
switch (format) {
case 'c':
value.push(i + 1);
break;
case 'c/t':
value.push(i + 1, '/', deck.getTotalSlides());
break;
case 'h/v':
value.push(h + 1);
if (typeof v === 'number' && !isNaN(v)) value.push('/', v + 1);
break;
default:
value.push(h + 1);
if (typeof v === 'number' && !isNaN(v)) value.push('.', v + 1);
}
item.appendChild(
create(
'span',
{ class: 'slide-menu-item-number' },
value.join('') + '. '
)
);
}
item.appendChild(
create('span', { class: 'slide-menu-item-title' }, title)
);
return item;
}
function createSlideMenu() {
if (
!document.querySelector(
'section[data-markdown]:not([data-markdown-parsed])'
)
) {
var panel = create('div', {
'data-panel': 'Slides',
class: 'slide-menu-panel active-menu-panel'
});
panel.appendChild(create('ul', { class: 'slide-menu-items' }));
panels.appendChild(panel);
var items = select(
'.slide-menu-panel[data-panel="Slides"] > .slide-menu-items'
);
var slideCount = 0;
selectAll('.slides > section').forEach(function (section, h) {
var subsections = selectAll('section', section);
if (subsections.length > 0) {
subsections.forEach(function (subsection, v) {
var type =
v === 0 ? 'slide-menu-item' : 'slide-menu-item-vertical';
var item = generateItem(type, subsection, slideCount, h, v);
if (item) {
items.appendChild(item);
}
slideCount++;
});
} else {
var item = generateItem(
'slide-menu-item',
section,
slideCount,
h
);
if (item) {
items.appendChild(item);
}
slideCount++;
}
});
selectAll('.slide-menu-item, .slide-menu-item-vertical').forEach(
function (i) {
i.onclick = clicked;
}
);
highlightCurrentSlide();
} else {
// wait for markdown to be loaded and parsed
setTimeout(createSlideMenu, 100);
}
}
createSlideMenu();
deck.addEventListener('slidechanged', highlightCurrentSlide);
//
// Custom menu panels
//
if (options.custom) {
function xhrSuccess() {
if (this.status >= 200 && this.status < 300) {
this.panel.innerHTML = this.responseText;
enableCustomLinks(this.panel);
} else {
showErrorMsg(this);
}
}
function xhrError() {
showErrorMsg(this);
}
function loadCustomPanelContent(panel, sURL) {
var oReq = new XMLHttpRequest();
oReq.panel = panel;
oReq.arguments = Array.prototype.slice.call(arguments, 2);
oReq.onload = xhrSuccess;
oReq.onerror = xhrError;
oReq.open('get', sURL, true);
oReq.send(null);
}
function enableCustomLinks(panel) {
selectAll('ul.slide-menu-items li.slide-menu-item', panel).forEach(
function (item, i) {
item.setAttribute('data-item', i + 1);
item.onclick = clicked;
item.addEventListener('mouseenter', handleMouseHighlight);
}
);
}
function showErrorMsg(response) {
var msg =
'<p>ERROR: The attempt to fetch ' +
response.responseURL +
' failed with HTTP status ' +
response.status +
' (' +
response.statusText +
').</p>' +
'<p>Remember that you need to serve the presentation HTML from a HTTP server.</p>';
response.panel.innerHTML = msg;
}
options.custom.forEach(function (element, index, array) {
var panel = create('div', {
'data-panel': 'Custom' + index,
class: 'slide-menu-panel slide-menu-custom-panel'
});
if (element.content) {
panel.innerHTML = element.content;
enableCustomLinks(panel);
} else if (element.src) {
loadCustomPanelContent(panel, element.src);
}
panels.appendChild(panel);
});
}
//
// Themes
//
if (options.themes) {
var panel = create('div', {
class: 'slide-menu-panel',
'data-panel': 'Themes'
});
panels.appendChild(panel);
var menu = create('ul', { class: 'slide-menu-items' });
panel.appendChild(menu);
options.themes.forEach(function (t, i) {
var attrs = {
class: 'slide-menu-item',
'data-item': '' + (i + 1)
};
if (t.theme) {
attrs['data-theme'] = t.theme;
}
if (t.highlightTheme) {
attrs['data-highlight-theme'] = t.highlightTheme;
}
var item = create('li', attrs, t.name);
menu.appendChild(item);
item.onclick = clicked;
});
}
//
// Transitions
//
if (options.transitions) {
var panel = create('div', {
class: 'slide-menu-panel',
'data-panel': 'Transitions'
});
panels.appendChild(panel);
var menu = create('ul', { class: 'slide-menu-items' });
panel.appendChild(menu);
options.transitions.forEach(function (name, i) {
var item = create(
'li',
{
class: 'slide-menu-item',
'data-transition': name.toLowerCase(),
'data-item': '' + (i + 1)
},
name
);
menu.appendChild(item);
item.onclick = clicked;
});
}
//
// Open menu options
//
if (options.openButton) {
// add menu button
var div = create('div', { class: 'slide-menu-button' });
var link = create('a', { href: '#' });
link.appendChild(create('i', { class: 'fas fa-bars' }));
div.appendChild(link);
select('.reveal').appendChild(div);
div.onclick = openMenu;
}
if (options.openSlideNumber) {
var slideNumber = select('div.slide-number');
slideNumber.onclick = openMenu;
}
//
// Handle mouse overs
//
selectAll('.slide-menu-panel .slide-menu-items li').forEach(function (
item
) {
item.addEventListener('mouseenter', handleMouseHighlight);
});
function handleMouseHighlight(event) {
if (mouseSelectionEnabled) {
selectAll('.active-menu-panel .slide-menu-items li.selected').forEach(
function (i) {
i.classList.remove('selected');
}
);
event.currentTarget.classList.add('selected');
}
}
}
if (options.keyboard) {
//XXX add keyboard option for custom key codes, etc.
document.addEventListener('keydown', onDocumentKeyDown, false);
// handle key presses within speaker notes
window.addEventListener('message', function (event) {
var data;
try {
data = JSON.parse(event.data);
} catch (e) {}
if (data && data.method === 'triggerKey') {
onDocumentKeyDown({
keyCode: data.args[0],
stopImmediatePropagation: function () {}
});
}
});
// Prevent reveal from processing keyboard events when the menu is open
if (
config.keyboardCondition &&
typeof config.keyboardCondition === 'function'
) {
// combine user defined keyboard condition with the menu's own condition
var userCondition = config.keyboardCondition;
config.keyboardCondition = function (event) {
return userCondition(event) && (!isOpen() || event.keyCode == 77);
};
} else {
config.keyboardCondition = function (event) {
return !isOpen() || event.keyCode == 77;
};
}
deck.addKeyBinding(
{ keyCode: 77, key: 'M', description: 'Toggle menu' },
toggleMenu
);
}
if (options.openOnInit) {
openMenu();
}
initialised = true;
}
/**
* Extend object a with the properties of object b.
* If there's a conflict, object b takes precedence.
*/
function extend(a, b) {
for (var i in b) {
a[i] = b[i];
}
}
/**
* Dispatches an event of the specified type from the
* reveal DOM element.
*/
function dispatchEvent(type, args) {
var event = document.createEvent('HTMLEvents', 1, 2);
event.initEvent(type, true, true);
extend(event, args);
document.querySelector('.reveal').dispatchEvent(event);
// If we're in an iframe, post each reveal.js event to the
// parent window. Used by the notes plugin
if (config.postMessageEvents && window.parent !== window.self) {
window.parent.postMessage(
JSON.stringify({
namespace: 'reveal',
eventName: type,
state: deck.getState()
}),
'*'
);
}
}
function select(selector, el) {
if (!el) {
el = document;
}
return el.querySelector(selector);
}
function selectAll(selector, el) {
if (!el) {
el = document;
}
return Array.prototype.slice.call(el.querySelectorAll(selector));
}
function create(tagName, attrs, content) {
var el = document.createElement(tagName);
if (attrs) {
Object.getOwnPropertyNames(attrs).forEach(function (n) {
el.setAttribute(n, attrs[n]);
});
}
if (content) el.innerHTML = content;
return el;
}
function changeStylesheet(id, href) {
// take note of the previous theme and remove it, then create a new stylesheet reference and insert it
// this is required to force a load event so we can change the menu style to match the new style
var stylesheet = select('link#' + id);
var parent = stylesheet.parentElement;
var sibling = stylesheet.nextElementSibling;
stylesheet.remove();
var newStylesheet = stylesheet.cloneNode();
newStylesheet.setAttribute('href', href);
newStylesheet.onload = function () {
matchRevealStyle();
};
parent.insertBefore(newStylesheet, sibling);
}
// modified from math plugin
function loadResource(url, type, callback) {
var head = document.querySelector('head');
var resource;
if (type === 'script') {
resource = document.createElement('script');
resource.type = 'text/javascript';
resource.src = url;
} else if (type === 'stylesheet') {
resource = document.createElement('link');
resource.rel = 'stylesheet';
resource.href = url;
}
// Wrapper for callback to make sure it only fires once
var finish = function () {
if (typeof callback === 'function') {
callback.call();
callback = null;
}
};
resource.onload = finish;
// IE
resource.onreadystatechange = function () {
if (this.readyState === 'loaded') {
finish();
}
};
// Normal browsers
head.appendChild(resource);
}
function loadPlugin() {
// does not support IE8 or below
var supported = !ieVersion || ieVersion >= 9;
// do not load the menu in the upcoming slide panel in the speaker notes
if (
deck.isSpeakerNotes() &&
window.location.search.endsWith('controls=false')
) {
supported = false;
}
if (supported) {
if (!options.delayInit) initMenu();
dispatchEvent('menu-ready');
}
}
return {
id: 'menu',
init: reveal => {
deck = reveal;
config = deck.getConfig();
initOptions(config);
loadResource(options.path + 'menu.css', 'stylesheet', function () {
if (options.loadIcons === undefined || options.loadIcons) {
loadResource(
options.path + 'font-awesome/css/all.css',
'stylesheet',
loadPlugin
);
} else {
loadPlugin();
}
});
},
toggle: toggleMenu,
openMenu: openMenu,
closeMenu: closeMenu,
openPanel: openPanel,
isOpen: isOpen,
initialiseMenu: initMenu,
isMenuInitialised: function () {
return initialised;
}
};
};
// polyfill
if (!String.prototype.startsWith) {
String.prototype.startsWith = function (searchString, position) {
return this.substr(position || 0, searchString.length) === searchString;
};
}
if (!String.prototype.endsWith) {
String.prototype.endsWith = function (search, this_len) {
if (this_len === undefined || this_len > this.length) {
this_len = this.length;
}
return this.substring(this_len - search.length, this_len) === search;
};
}
export default Plugin;