decktape/decktape.js
2016-06-05 22:38:34 +02:00

375 lines
13 KiB
JavaScript

require.paths.push(phantom.libraryPath + '/libs/');
var system = require('system');
// Node to PhantomJS bridging
var process = {
platform : { mac: 'darwin', windows: 'win32' }[system.os.name] || system.os.name,
env : system.env,
argv : system.args,
// To uncomment when system.stdout.isTTY is supported
//stdout : system.stdout,
exit : phantom.exit
};
// As opposed to PhantomJS, global variables declared in the main script are not
// accessible in modules loaded with require
if (system.platform === 'slimerjs')
require.globals.process = process;
var fs = require('fs'),
page = require('webpage').create(),
parser = require('nomnom'),
printer = require('printer').create(),
Promise = require('es6-promise').Promise;
var plugins = loadAvailablePlugins(phantom.libraryPath + '/plugins/');
parser.script('phantomjs decktape.js')
.options({
url: {
position : 1,
required : true,
help : 'URL of the slides deck'
},
filename: {
position : 2,
required : true,
help : 'Filename of the output PDF file'
},
size: {
abbr : 's',
metavar : '<size>',
type : 'string',
callback : parseSize,
transform : parseSize,
help : 'Size of the slides deck viewport: <width>x<height> (ex. 1280x720)'
},
pause: {
abbr : 'p',
metavar : '<ms>',
default : 1000,
help : 'Duration in milliseconds before each slide is exported'
},
loadPause: {
full : "load-pause",
metavar : '<ms>',
default : 0,
help : 'Duration in milliseconds between the page has loaded and starting to export slides'
},
screenshots: {
default : false,
flag : true,
help : 'Capture each slide as an image'
},
screenshotDirectory: {
full : 'screenshots-directory',
metavar : '<dir>',
default : 'screenshots',
help : 'Screenshots output directory'
},
screenshotSize: {
full : 'screenshots-size',
metavar : '<size>',
type : 'string',
list : true,
callback : parseSize,
transform : parseSize,
help : 'Screenshots resolution, can be repeated'
},
screenshotFormat: {
full : 'screenshots-format',
metavar : '<format>',
default : 'png',
choices : ['jpg', 'png'],
help : 'Screenshots image format, one of [jpg, png]'
},
slides: {
metavar : '<range>',
type : 'string',
callback : parseRange,
transform : parseRange,
help : 'Range of slides to be exported, a combination of slide indexes and ranges (e.g. \'1-3,5,8\')'
}
});
function parseSize(size) {
// TODO: support device viewport sizes and graphics display standard resolutions
// see http://viewportsizes.com/ and https://en.wikipedia.org/wiki/Graphics_display_resolution
var match = size.match(/^(\d+)x(\d+)$/);
if (!match)
return '<size> must follow the <width>x<height> notation, e.g., 1280x720';
else
return { width: match[1], height: match[2] };
}
function parseRange(range) {
var regex = /(\d)+(?:-(\d+))?/g;
if (!range.match(regex))
return '<range> must be a combination of slide indexes and ranges, e.g., \'1-3,5,8\'';
var slide;
var slides = {};
while ((slide = regex.exec(range)) !== null)
if (typeof slide[2] !== 'undefined')
for (var i = parseInt(slide[1]); i <= parseInt(slide[2]); i++)
slides[i] = true;
else
slides[parseInt(slide[1])] = true;
return slides;
}
parser.nocommand()
.help('Defaults to the automatic command.\n' +
'Iterates over the available plugins, picks the compatible one for presentation at the \n' +
'specified <url> and uses it to export and write the PDF into the specified <filename>.');
parser.command('automatic')
.help('Iterates over the available plugins, picks the compatible one for presentation at the \n' +
'specified <url> and uses it to export and write the PDF into the specified <filename>.');
Object.keys(plugins).forEach(function (id) {
var command = parser.command(id);
if (typeof plugins[id].options === 'object')
command.options(plugins[id].options);
if (typeof plugins[id].help === 'string')
command.help(plugins[id].help);
});
// TODO: should be deactivated as well when PhantomJS does not execute in a TTY context
if (system.os.name === 'windows')
parser.nocolors();
var options = parser.parse(system.args.slice(1));
page.onLoadStarted = function () {
console.log('Loading page ' + options.url + ' ...');
};
page.onResourceTimeout = function (request) {
console.log('+- Request timeout: ' + JSON.stringify(request));
};
page.onResourceError = function (resource) {
console.log('+- Unable to load resource from URL: ' + resource.url);
console.log('|_ Error code: ' + resource.errorCode);
console.log('|_ Description: ' + resource.errorString);
};
// PhantomJS emits this event for both pages and frames
page.onLoadFinished = function (status) {
console.log('Loading page finished with status: ' + status);
};
// Must be set before the page is opened
page.onConsoleMessage = function (msg) {
console.log(msg);
};
page.onError = function (msg, trace) {
console.log('+- ' + msg);
(trace || []).forEach(function (t) {
console.log('|_ ' + t.file + ': ' + t.line + (t.function ? ' (in function "' + t.function +'")' : ''));
});
};
openUrl(page, options.url)
.then(delay(options.loadPause))
.then(createPlugin)
.then(configurePlugin)
.then(configurePrinter)
.then(exportSlides)
.then(function (plugin) {
printer.end();
system.stdout.write('\nPrinted ' + plugin.exportedSlides + ' slides\n');
phantom.exit();
})
.catch(function (error) {
console.log(error);
phantom.exit(1);
});
function openUrl(page, url) {
return new Promise(function (resolve, reject) {
page.open(url, function (status) {
if (status !== 'success')
reject(Error('Unable to load the URL: ' + url));
else
resolve();
});
});
}
function createPlugin() {
var plugin;
if (!options.command || options.command === 'automatic') {
plugin = createActivePlugin();
if (!plugin) {
console.log('No supported DeckTape plugin detected, falling back to generic plugin');
plugin = plugins['generic'].create(page, options);
}
} else {
plugin = plugins[options.command].create(page, options);
if (!plugin.isActive()) {
throw Error('Unable to activate the ' + plugin.getName() + ' DeckTape plugin for the address: ' + options.url);
}
}
console.log(plugin.getName() + ' DeckTape plugin activated');
return plugin;
}
function loadAvailablePlugins(pluginPath) {
return fs.list(pluginPath).reduce(function (plugins, plugin) {
var matches = plugin.match(/^(.*)\.js$/);
if (matches && fs.isFile(pluginPath + plugin))
plugins[matches[1]] = require(pluginPath + matches[1]);
return plugins;
}, {});
}
function createActivePlugin() {
for (var id in plugins) {
if (id === 'generic')
continue;
var plugin = plugins[id].create(page, options);
if (plugin.isActive())
return plugin;
}
}
function configurePlugin(plugin) {
var config;
if (typeof plugin.configure === 'function')
config = Promise.resolve(plugin.configure()).then(value(plugin));
else
config = Promise.resolve(plugin);
return config.then(function (plugin) {
plugin.progressBarOverflow = 0;
plugin.currentSlide = 1;
plugin.exportedSlides = 0;
plugin.totalSlides = plugin.slideCount();
return plugin;
});
}
function configurePrinter(plugin) {
if (!options.size)
if (typeof plugin.size === 'function')
options.size = plugin.size();
else
// TODO: per-plugin default size
options.size = { width: 1280, height: 720 };
page.viewportSize = options.size;
printer.paperSize = {
width: options.size.width + 'px',
height: options.size.height + 'px',
margin: '0px'
};
printer.outputFileName = options.filename;
printer.begin();
return plugin;
}
// TODO: ideally defined in the plugin prototype
function hasNextSlide(plugin) {
if (typeof plugin.hasNextSlide === 'function')
return plugin.hasNextSlide();
else
return plugin.currentSlide < plugin.totalSlides;
}
// TODO: ideally defined in the plugin prototype
function nextSlide(plugin) {
plugin.currentSlide++;
return plugin.nextSlide();
}
// TODO: ideally defined in the plugin prototype
function printSlide(plugin) {
printer.printPage(page);
plugin.exportedSlides++;
}
function exportSlides(plugin) {
// TODO: support a more advanced "fragment to pause" mapping for special use cases like GIF animations
// TODO: support plugin optional promise to wait until a particular mutation instead of a pause
return Promise.resolve(plugin)
.then(delay(options.pause))
.then(exportSlide)
.then(hasNextSlide)
.then(function (hasNext) {
if (hasNext && (!options.slides || plugin.currentSlide < Math.max.apply(null, Object.keys(options.slides)))) {
nextSlide(plugin);
return exportSlides(plugin);
} else {
return plugin;
}
});
}
function exportSlide(plugin) {
// TODO: better logging when slide is skipped and move it to the main loop
system.stdout.write('\r' + progressBar(plugin));
if (options.slides && !options.slides[plugin.currentSlide])
return Promise.resolve(plugin);
var decktape = Promise.resolve(plugin).then(printSlide);
if (options.screenshots)
decktape = (options.screenshotSize || [options.size]).reduce(function (decktape, resolution) {
return decktape.then(function () { page.viewportSize = resolution })
// Delay page rendering to wait for the resize event to complete,
// e.g. for impress.js (may be needed to be configurable)
.then(delay(1000))
.then(function () {
page.render(options.screenshotDirectory + '/' + options.filename.replace('.pdf', '_' + plugin.currentSlide + '_' + resolution.width + 'x' + resolution.height + '.' + options.screenshotFormat), { onlyViewport: true });
})
}, decktape)
.then(function () { page.viewportSize = options.size })
.then(delay(1000));
return decktape.then(value(plugin));
}
function delay(time) {
return function (value) {
return new Promise(function (fulfill) {
setTimeout(fulfill, time, value);
});
}
}
function value(value) {
return function () {
return value;
}
}
// TODO: add progress bar, duration, ETA and file size
function progressBar(plugin) {
var cols = [];
var index = plugin.currentSlideIndex();
cols.push('Printing slide ');
cols.push(padding('#' + index, 8, ' ', false));
cols.push(' (');
cols.push(padding(plugin.currentSlide, plugin.totalSlides ? plugin.totalSlides.toString().length : 3, ' '));
cols.push('/');
cols.push(plugin.totalSlides || ' ?');
cols.push(') ...');
// erase overflowing slide fragments
cols.push(padding('', plugin.progressBarOverflow - Math.max(index.length + 1 - 8, 0), ' ', false));
plugin.progressBarOverflow = Math.max(index.length + 1 - 8, 0);
return cols.join('');
}
function padding(str, len, char, left) {
if (typeof str === 'number')
str = str.toString();
var l = len - str.length;
var p = [];
while (l-- > 0)
p.push(char);
return left === undefined || left ?
p.join('').concat(str) :
str.concat(p.join(''));
}