mirror of
https://github.com/astefanutti/decktape.git
synced 2024-12-12 13:49:13 +03:00
375 lines
13 KiB
JavaScript
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(''));
|
|
}
|