350 lines
10 KiB
JavaScript
350 lines
10 KiB
JavaScript
import { extend, queryAll } from "../utils/util.js";
|
|
|
|
/**
|
|
* Handles sorting and navigation of slide fragments.
|
|
* Fragments are elements within a slide that are
|
|
* revealed/animated incrementally.
|
|
*/
|
|
export default class Fragments {
|
|
constructor(Reveal) {
|
|
this.Reveal = Reveal;
|
|
}
|
|
|
|
/**
|
|
* Called when the reveal.js config is updated.
|
|
*/
|
|
configure(config, oldConfig) {
|
|
if (config.fragments === false) {
|
|
this.disable();
|
|
} else if (oldConfig.fragments === false) {
|
|
this.enable();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* If fragments are disabled in the deck, they should all be
|
|
* visible rather than stepped through.
|
|
*/
|
|
disable() {
|
|
queryAll(this.Reveal.getSlidesElement(), ".fragment").forEach((element) => {
|
|
element.classList.add("visible");
|
|
element.classList.remove("current-fragment");
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Reverse of #disable(). Only called if fragments have
|
|
* previously been disabled.
|
|
*/
|
|
enable() {
|
|
queryAll(this.Reveal.getSlidesElement(), ".fragment").forEach((element) => {
|
|
element.classList.remove("visible");
|
|
element.classList.remove("current-fragment");
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Returns an object describing the available fragment
|
|
* directions.
|
|
*
|
|
* @return {{prev: boolean, next: boolean}}
|
|
*/
|
|
availableRoutes() {
|
|
let currentSlide = this.Reveal.getCurrentSlide();
|
|
if (currentSlide && this.Reveal.getConfig().fragments) {
|
|
let fragments = currentSlide.querySelectorAll(".fragment:not(.disabled)");
|
|
let hiddenFragments = currentSlide.querySelectorAll(
|
|
".fragment:not(.disabled):not(.visible)"
|
|
);
|
|
|
|
return {
|
|
prev: fragments.length - hiddenFragments.length > 0,
|
|
next: !!hiddenFragments.length,
|
|
};
|
|
} else {
|
|
return { prev: false, next: false };
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Return a sorted fragments list, ordered by an increasing
|
|
* "data-fragment-index" attribute.
|
|
*
|
|
* Fragments will be revealed in the order that they are returned by
|
|
* this function, so you can use the index attributes to control the
|
|
* order of fragment appearance.
|
|
*
|
|
* To maintain a sensible default fragment order, fragments are presumed
|
|
* to be passed in document order. This function adds a "fragment-index"
|
|
* attribute to each node if such an attribute is not already present,
|
|
* and sets that attribute to an integer value which is the position of
|
|
* the fragment within the fragments list.
|
|
*
|
|
* @param {object[]|*} fragments
|
|
* @param {boolean} grouped If true the returned array will contain
|
|
* nested arrays for all fragments with the same index
|
|
* @return {object[]} sorted Sorted array of fragments
|
|
*/
|
|
sort(fragments, grouped = false) {
|
|
fragments = Array.from(fragments);
|
|
|
|
let ordered = [],
|
|
unordered = [],
|
|
sorted = [];
|
|
|
|
// Group ordered and unordered elements
|
|
fragments.forEach((fragment) => {
|
|
if (fragment.hasAttribute("data-fragment-index")) {
|
|
let index = parseInt(fragment.getAttribute("data-fragment-index"), 10);
|
|
|
|
if (!ordered[index]) {
|
|
ordered[index] = [];
|
|
}
|
|
|
|
ordered[index].push(fragment);
|
|
} else {
|
|
unordered.push([fragment]);
|
|
}
|
|
});
|
|
|
|
// Append fragments without explicit indices in their
|
|
// DOM order
|
|
ordered = ordered.concat(unordered);
|
|
|
|
// Manually count the index up per group to ensure there
|
|
// are no gaps
|
|
let index = 0;
|
|
|
|
// Push all fragments in their sorted order to an array,
|
|
// this flattens the groups
|
|
ordered.forEach((group) => {
|
|
group.forEach((fragment) => {
|
|
sorted.push(fragment);
|
|
fragment.setAttribute("data-fragment-index", index);
|
|
});
|
|
|
|
index++;
|
|
});
|
|
|
|
return grouped === true ? ordered : sorted;
|
|
}
|
|
|
|
/**
|
|
* Sorts and formats all of fragments in the
|
|
* presentation.
|
|
*/
|
|
sortAll() {
|
|
this.Reveal.getHorizontalSlides().forEach((horizontalSlide) => {
|
|
let verticalSlides = queryAll(horizontalSlide, "section");
|
|
verticalSlides.forEach((verticalSlide, y) => {
|
|
this.sort(verticalSlide.querySelectorAll(".fragment"));
|
|
}, this);
|
|
|
|
if (verticalSlides.length === 0)
|
|
this.sort(horizontalSlide.querySelectorAll(".fragment"));
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Refreshes the fragments on the current slide so that they
|
|
* have the appropriate classes (.visible + .current-fragment).
|
|
*
|
|
* @param {number} [index] The index of the current fragment
|
|
* @param {array} [fragments] Array containing all fragments
|
|
* in the current slide
|
|
*
|
|
* @return {{shown: array, hidden: array}}
|
|
*/
|
|
update(index, fragments) {
|
|
let changedFragments = {
|
|
shown: [],
|
|
hidden: [],
|
|
};
|
|
|
|
let currentSlide = this.Reveal.getCurrentSlide();
|
|
if (currentSlide && this.Reveal.getConfig().fragments) {
|
|
fragments =
|
|
fragments || this.sort(currentSlide.querySelectorAll(".fragment"));
|
|
|
|
if (fragments.length) {
|
|
let maxIndex = 0;
|
|
|
|
if (typeof index !== "number") {
|
|
let currentFragment = this.sort(
|
|
currentSlide.querySelectorAll(".fragment.visible")
|
|
).pop();
|
|
if (currentFragment) {
|
|
index = parseInt(
|
|
currentFragment.getAttribute("data-fragment-index") || 0,
|
|
10
|
|
);
|
|
}
|
|
}
|
|
|
|
Array.from(fragments).forEach((el, i) => {
|
|
if (el.hasAttribute("data-fragment-index")) {
|
|
i = parseInt(el.getAttribute("data-fragment-index"), 10);
|
|
}
|
|
|
|
maxIndex = Math.max(maxIndex, i);
|
|
|
|
// Visible fragments
|
|
if (i <= index) {
|
|
let wasVisible = el.classList.contains("visible");
|
|
el.classList.add("visible");
|
|
el.classList.remove("current-fragment");
|
|
|
|
if (i === index) {
|
|
// Announce the fragments one by one to the Screen Reader
|
|
this.Reveal.announceStatus(this.Reveal.getStatusText(el));
|
|
|
|
el.classList.add("current-fragment");
|
|
this.Reveal.slideContent.startEmbeddedContent(el);
|
|
}
|
|
|
|
if (!wasVisible) {
|
|
changedFragments.shown.push(el);
|
|
this.Reveal.dispatchEvent({
|
|
target: el,
|
|
type: "visible",
|
|
bubbles: false,
|
|
});
|
|
}
|
|
}
|
|
// Hidden fragments
|
|
else {
|
|
let wasVisible = el.classList.contains("visible");
|
|
el.classList.remove("visible");
|
|
el.classList.remove("current-fragment");
|
|
|
|
if (wasVisible) {
|
|
this.Reveal.slideContent.stopEmbeddedContent(el);
|
|
changedFragments.hidden.push(el);
|
|
this.Reveal.dispatchEvent({
|
|
target: el,
|
|
type: "hidden",
|
|
bubbles: false,
|
|
});
|
|
}
|
|
}
|
|
});
|
|
|
|
// Write the current fragment index to the slide <section>.
|
|
// This can be used by end users to apply styles based on
|
|
// the current fragment index.
|
|
index = typeof index === "number" ? index : -1;
|
|
index = Math.max(Math.min(index, maxIndex), -1);
|
|
currentSlide.setAttribute("data-fragment", index);
|
|
}
|
|
}
|
|
|
|
return changedFragments;
|
|
}
|
|
|
|
/**
|
|
* Formats the fragments on the given slide so that they have
|
|
* valid indices. Call this if fragments are changed in the DOM
|
|
* after reveal.js has already initialized.
|
|
*
|
|
* @param {HTMLElement} slide
|
|
* @return {Array} a list of the HTML fragments that were synced
|
|
*/
|
|
sync(slide = this.Reveal.getCurrentSlide()) {
|
|
return this.sort(slide.querySelectorAll(".fragment"));
|
|
}
|
|
|
|
/**
|
|
* Navigate to the specified slide fragment.
|
|
*
|
|
* @param {?number} index The index of the fragment that
|
|
* should be shown, -1 means all are invisible
|
|
* @param {number} offset Integer offset to apply to the
|
|
* fragment index
|
|
*
|
|
* @return {boolean} true if a change was made in any
|
|
* fragments visibility as part of this call
|
|
*/
|
|
goto(index, offset = 0) {
|
|
let currentSlide = this.Reveal.getCurrentSlide();
|
|
if (currentSlide && this.Reveal.getConfig().fragments) {
|
|
let fragments = this.sort(
|
|
currentSlide.querySelectorAll(".fragment:not(.disabled)")
|
|
);
|
|
if (fragments.length) {
|
|
// If no index is specified, find the current
|
|
if (typeof index !== "number") {
|
|
let lastVisibleFragment = this.sort(
|
|
currentSlide.querySelectorAll(".fragment:not(.disabled).visible")
|
|
).pop();
|
|
|
|
if (lastVisibleFragment) {
|
|
index = parseInt(
|
|
lastVisibleFragment.getAttribute("data-fragment-index") || 0,
|
|
10
|
|
);
|
|
} else {
|
|
index = -1;
|
|
}
|
|
}
|
|
|
|
// Apply the offset if there is one
|
|
index += offset;
|
|
|
|
let changedFragments = this.update(index, fragments);
|
|
|
|
if (changedFragments.hidden.length) {
|
|
this.Reveal.dispatchEvent({
|
|
type: "fragmenthidden",
|
|
data: {
|
|
fragment: changedFragments.hidden[0],
|
|
fragments: changedFragments.hidden,
|
|
},
|
|
});
|
|
}
|
|
|
|
if (changedFragments.shown.length) {
|
|
this.Reveal.dispatchEvent({
|
|
type: "fragmentshown",
|
|
data: {
|
|
fragment: changedFragments.shown[0],
|
|
fragments: changedFragments.shown,
|
|
},
|
|
});
|
|
}
|
|
|
|
this.Reveal.controls.update();
|
|
this.Reveal.progress.update();
|
|
|
|
if (this.Reveal.getConfig().fragmentInURL) {
|
|
this.Reveal.location.writeURL();
|
|
}
|
|
|
|
return !!(
|
|
changedFragments.shown.length || changedFragments.hidden.length
|
|
);
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* Navigate to the next slide fragment.
|
|
*
|
|
* @return {boolean} true if there was a next fragment,
|
|
* false otherwise
|
|
*/
|
|
next() {
|
|
return this.goto(null, 1);
|
|
}
|
|
|
|
/**
|
|
* Navigate to the previous slide fragment.
|
|
*
|
|
* @return {boolean} true if there was a previous fragment,
|
|
* false otherwise
|
|
*/
|
|
prev() {
|
|
return this.goto(null, -1);
|
|
}
|
|
}
|