2016-01-26 14:02:23 -05:00
|
|
|
/*
|
|
|
|
* Paper.js - The Swiss Army Knife of Vector Graphics Scripting.
|
|
|
|
* http://paperjs.org/
|
|
|
|
*
|
2018-12-27 02:13:01 -05:00
|
|
|
* Copyright (c) 2011 - 2019, Juerg Lehni & Jonathan Puckey
|
2018-11-10 02:19:34 -05:00
|
|
|
* http://scratchdisk.com/ & https://puckey.studio/
|
2016-01-26 14:02:23 -05:00
|
|
|
*
|
|
|
|
* Distributed under the MIT license. See LICENSE file for details.
|
|
|
|
*
|
|
|
|
* All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
var fs = require('fs'),
|
|
|
|
path = require('path');
|
|
|
|
|
|
|
|
module.exports = function(paper) {
|
2016-02-15 17:59:31 -05:00
|
|
|
if (paper.PaperScript) {
|
2016-12-31 00:50:08 -05:00
|
|
|
var sourceMapSupport = 'require("source-map-support").install(paper.PaperScript.sourceMapSupport);\n',
|
2016-02-15 17:59:31 -05:00
|
|
|
sourceMaps = {};
|
2016-01-26 14:02:23 -05:00
|
|
|
|
2016-02-15 17:59:31 -05:00
|
|
|
paper.PaperScript.sourceMapSupport = {
|
|
|
|
retrieveSourceMap: function(source) {
|
|
|
|
var map = sourceMaps[source];
|
|
|
|
return map ? { url: source, map: map } : null;
|
|
|
|
}
|
|
|
|
};
|
2016-01-26 14:02:23 -05:00
|
|
|
|
2016-02-15 17:59:31 -05:00
|
|
|
// Register the .pjs extension for automatic compilation as PaperScript
|
|
|
|
require.extensions['.pjs'] = function(module, filename) {
|
|
|
|
// Requiring a PaperScript on Node.js returns an initialize method which
|
|
|
|
// needs to receive a Canvas object when called and returns the
|
|
|
|
// PaperScope.
|
|
|
|
module.exports = function(canvas) {
|
|
|
|
var source = fs.readFileSync(filename, 'utf8'),
|
2016-12-31 00:50:08 -05:00
|
|
|
code = sourceMapSupport + source,
|
2016-02-15 17:59:31 -05:00
|
|
|
compiled = paper.PaperScript.compile(code, {
|
|
|
|
url: filename,
|
|
|
|
source: source,
|
|
|
|
sourceMaps: true,
|
2016-12-31 00:50:08 -05:00
|
|
|
offset: -1 // remove sourceMapSupport...
|
2016-02-15 17:59:31 -05:00
|
|
|
}),
|
|
|
|
scope = new paper.PaperScope();
|
|
|
|
// Keep track of sourceMaps so retrieveSourceMap() can link them up
|
|
|
|
scope.setup(canvas);
|
|
|
|
scope.__filename = filename;
|
|
|
|
scope.__dirname = path.dirname(filename);
|
|
|
|
// Expose core methods and values
|
|
|
|
scope.require = require;
|
|
|
|
scope.console = console;
|
|
|
|
sourceMaps[filename] = compiled.map;
|
|
|
|
paper.PaperScript.execute(compiled, scope);
|
|
|
|
return scope;
|
|
|
|
};
|
2016-01-26 14:02:23 -05:00
|
|
|
};
|
2016-02-15 17:59:31 -05:00
|
|
|
}
|
2016-01-26 14:02:23 -05:00
|
|
|
|
2016-01-27 03:34:37 -05:00
|
|
|
paper.PaperScope.inject({
|
|
|
|
createCanvas: function(width, height, type) {
|
|
|
|
// Do not use CanvasProvider.getCanvas(), since we may be changing
|
2017-03-19 11:03:29 -04:00
|
|
|
// the underlying node-canvas when requesting PDF support, and don't
|
|
|
|
// want to release it after back into the pool.
|
2016-01-27 03:34:37 -05:00
|
|
|
var canvas = paper.document.createElement('canvas');
|
|
|
|
canvas.width = width;
|
|
|
|
canvas.height = height;
|
|
|
|
canvas.type = type;
|
|
|
|
return canvas;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
2016-09-18 15:38:28 -04:00
|
|
|
* @deprecated, use use {@link #createCanvas(width, height)} instead.
|
2016-01-27 03:34:37 -05:00
|
|
|
*/
|
|
|
|
Canvas: '#createCanvas'
|
|
|
|
});
|
|
|
|
|
2016-01-27 07:02:50 -05:00
|
|
|
// Override requestAnimationFrame() to avoid setInterval() timers.
|
2016-01-27 07:38:04 -05:00
|
|
|
// NOTE: In Node.js, we only support manual updating for now, but
|
|
|
|
// View#exportFrames() below offers a way to emulate animations by exporting
|
|
|
|
// them frame by frame at the given frame-rate.
|
2016-01-27 07:02:50 -05:00
|
|
|
paper.DomEvent.requestAnimationFrame = function(callback) {
|
|
|
|
};
|
|
|
|
|
2016-01-26 14:02:23 -05:00
|
|
|
// Node.js based image exporting code.
|
|
|
|
paper.CanvasView.inject({
|
2016-01-27 07:38:04 -05:00
|
|
|
// DOCS: CanvasView#exportFrames(options);
|
|
|
|
exportFrames: function(options) {
|
|
|
|
options = paper.Base.set({
|
2016-01-26 14:02:23 -05:00
|
|
|
fps: 30,
|
|
|
|
prefix: 'frame-',
|
2016-01-27 07:38:04 -05:00
|
|
|
amount: 1,
|
2017-10-04 16:39:19 -04:00
|
|
|
format: 'png' // Supported: 'png' or 'jpeg'
|
2016-01-27 07:38:04 -05:00
|
|
|
}, options);
|
|
|
|
if (!options.directory)
|
|
|
|
throw new Error('Missing options.directory');
|
2017-10-04 16:39:19 -04:00
|
|
|
if (options.format && !/^(jpeg|png)$/.test(options.format))
|
|
|
|
throw new Error('Unsupported format. Use "png" or "jpeg"');
|
2016-01-26 14:02:23 -05:00
|
|
|
var view = this,
|
|
|
|
count = 0,
|
2016-01-27 07:38:04 -05:00
|
|
|
frameDuration = 1 / options.fps,
|
2016-01-26 14:02:23 -05:00
|
|
|
startTime = Date.now(),
|
2016-01-27 07:38:04 -05:00
|
|
|
lastTime = startTime,
|
2016-06-13 23:20:58 -04:00
|
|
|
padding = options.padding || ((options.amount - 1) + '').length,
|
2016-01-27 07:38:04 -05:00
|
|
|
paddedStr = Array(padding + 1).join('0');
|
2016-01-26 14:02:23 -05:00
|
|
|
|
|
|
|
// Start exporting frames by exporting the first frame:
|
2016-01-27 07:38:04 -05:00
|
|
|
exportFrame(options);
|
2016-01-26 14:02:23 -05:00
|
|
|
|
2016-01-27 07:38:04 -05:00
|
|
|
function exportFrame() {
|
2016-01-27 07:27:11 -05:00
|
|
|
// Convert to a Base object, for #toString()
|
|
|
|
view.emit('frame', new paper.Base({
|
|
|
|
delta: frameDuration,
|
|
|
|
time: frameDuration * count,
|
|
|
|
count: count
|
|
|
|
}));
|
2017-10-04 16:39:19 -04:00
|
|
|
var file = path.join(options.directory,
|
|
|
|
options.prefix + (paddedStr + count).slice(-padding)
|
|
|
|
+ '.' + options.format);
|
2016-01-26 14:02:23 -05:00
|
|
|
var out = view.exportImage(file, function() {
|
2016-01-27 07:38:04 -05:00
|
|
|
// Once the file has been closed, export the next fame:
|
2016-01-26 14:02:23 -05:00
|
|
|
var then = Date.now();
|
2016-01-27 07:38:04 -05:00
|
|
|
if (options.onProgress) {
|
|
|
|
options.onProgress({
|
2016-01-26 14:02:23 -05:00
|
|
|
count: count,
|
2016-01-27 07:38:04 -05:00
|
|
|
amount: options.amount,
|
2016-01-27 16:23:56 -05:00
|
|
|
percentage: Math.round((count + 1) / options.amount
|
2016-01-26 14:02:23 -05:00
|
|
|
* 10000) / 100,
|
|
|
|
time: then - startTime,
|
|
|
|
delta: then - lastTime
|
|
|
|
});
|
|
|
|
}
|
|
|
|
lastTime = then;
|
2016-01-27 07:38:04 -05:00
|
|
|
if (++count < options.amount) {
|
|
|
|
exportFrame();
|
2016-01-26 14:02:23 -05:00
|
|
|
} else {
|
|
|
|
// Call onComplete handler when finished:
|
2016-01-27 07:38:04 -05:00
|
|
|
if (options.onComplete) {
|
|
|
|
options.onComplete();
|
2016-01-26 14:02:23 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
// DOCS: CanvasView#exportImage(path, callback);
|
|
|
|
exportImage: function(path, callback) {
|
2016-01-27 06:51:02 -05:00
|
|
|
this.update();
|
2016-01-26 14:02:23 -05:00
|
|
|
var out = fs.createWriteStream(path),
|
2017-10-05 06:09:07 -04:00
|
|
|
format = /\.jp(e?)g$/.test(path) ? 'jpeg' : 'png',
|
2017-10-04 16:39:19 -04:00
|
|
|
stream = this._element[format + 'Stream']();
|
2016-01-26 14:02:23 -05:00
|
|
|
stream.pipe(out);
|
|
|
|
if (callback) {
|
|
|
|
out.on('close', callback);
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|