decktape/plugins/generic.js
2022-12-09 17:17:04 +01:00

96 lines
2.9 KiB
JavaScript

// The generic plugin emulates end-user interaction by pressing keyboard
// and detects changes to the DOM. The deck is considered over when no change
// is detected afterward.
import { pause } from '../libs/util.js';
export const options = {
key : {
default : 'ArrowRight',
metavar : '<key>',
help : 'Key pressed to navigate to next slide',
},
maxSlides : {
full : 'max-slides',
metavar : '<size>',
help : 'Maximum number of slides to export',
},
media: {
default : 'screen',
choices : ['screen', 'print'],
metavar : '<media>',
help : 'CSS media type to emulate, one of [print, screen]',
},
};
export const help =
`Emulates the end-user interaction by pressing the key with the specified --key option
and iterates over the presentation as long as:
- Any change to the DOM is detected by observing mutation events targeting the body element
and its subtree,
- Nor the number of slides exported has reached the specified --max-slides option.
The --key option must be one of the 'KeyboardEvent' keys and defaults to [${options.key.default}].`;
export const create = (page, opts) => new Generic(page, opts);
class Generic {
constructor(page, opts) {
this.page = page;
this.options = opts;
this.currentSlide = 1;
this.isNextSlideDetected = false;
this.key = this.options.key || options.key.default;
this.media = this.options.media || options.media.default;
}
getName() {
return 'Generic';
}
isActive() {
return true;
}
async configure() {
await this.page.emulateMediaType(this.media);
await this.page.exposeFunction('onMutation', _ => (this.isNextSlideDetected = true));
await this.page.evaluate(_ =>
new MutationObserver(_ => window.onMutation()).observe(document, {
attributes : true,
childList : true,
subtree : true,
})
);
}
slideCount() {
return undefined;
}
// A priori knowledge is impossible to achieve in a generic way. Thus the only way is to
// actually emulate end-user interaction by pressing the configured key and check whether
// the DOM has changed a posteriori.
async hasNextSlide() {
if (this.options.maxSlides && this.currentSlide >= this.options.maxSlides) {
return false;
}
await this.page.keyboard.press(this.key);
// TODO: use mutation event directly instead of relying on a timeout
// TODO: detect cycle to avoid infinite navigation for frameworks
// that support loopable presentations like impress.js and flowtime.js
await pause(1000);
return this.isNextSlideDetected;
}
nextSlide() {
this.currentSlide++;
this.isNextSlideDetected = false;
}
async currentSlideIndex() {
const hash = await this.page.evaluate(_ => window.location.hash);
const [, fragment] = hash.match(/^#\/?([^?]*)/) || [];
return fragment || this.currentSlide;
}
}