mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-19 14:10:14 -05:00
Merge branch 'refs/heads/paperscript-refactoring' into v8-optimizations
This commit is contained in:
commit
35890383d8
12 changed files with 220 additions and 175 deletions
|
@ -14,7 +14,7 @@
|
||||||
"test"
|
"test"
|
||||||
],
|
],
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"straps": "~1.4.1",
|
"straps": "~1.4.2",
|
||||||
"acorn": "git://github.com/paperjs/acorn#0.3.2",
|
"acorn": "git://github.com/paperjs/acorn#0.3.2",
|
||||||
"esprima": "~1.0.3",
|
"esprima": "~1.0.3",
|
||||||
"stats.js": "r11"
|
"stats.js": "r11"
|
||||||
|
|
|
@ -468,11 +468,7 @@ var Point = Base.extend(/** @lends Point# */{
|
||||||
* @bean
|
* @bean
|
||||||
*/
|
*/
|
||||||
getLength: function() {
|
getLength: function() {
|
||||||
// Supports a hidden parameter 'squared', which controls whether the
|
return Math.sqrt(this.x * this.x + this.y * this.y);
|
||||||
// squared length should be returned. Hide it so it produces a bean
|
|
||||||
// property called #length.
|
|
||||||
var length = this.x * this.x + this.y * this.y;
|
|
||||||
return arguments.length && arguments[0] ? length : Math.sqrt(length);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
setLength: function(length) {
|
setLength: function(length) {
|
||||||
|
|
|
@ -102,16 +102,15 @@ var PaperScope = Base.extend(/** @lends PaperScope# */{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The reference to the active tool.
|
* The reference to the active tool.
|
||||||
|
* @name PaperScope#tool
|
||||||
* @type Tool
|
* @type Tool
|
||||||
* @bean
|
|
||||||
*/
|
*/
|
||||||
getTool: function() {
|
|
||||||
// If no tool exists yet but one is requested, produce it now on the fly
|
/**
|
||||||
// so it can be used in PaperScript.
|
* The list of available tools.
|
||||||
if (!this._tool)
|
* @name PaperScope#tools
|
||||||
this._tool = new Tool();
|
* @type Tool[]
|
||||||
return this._tool;
|
*/
|
||||||
},
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A reference to the local scope. This is required, so `paper` will always
|
* A reference to the local scope. This is required, so `paper` will always
|
||||||
|
@ -125,16 +124,9 @@ var PaperScope = Base.extend(/** @lends PaperScope# */{
|
||||||
return this;
|
return this;
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
execute: function(code) {
|
||||||
* The list of available tools.
|
paper.PaperScript.execute(code, this);
|
||||||
* @name PaperScope#tools
|
|
||||||
* @type Tool[]
|
|
||||||
*/
|
|
||||||
|
|
||||||
evaluate: function(code) {
|
|
||||||
var res = paper.PaperScript.evaluate(code, this);
|
|
||||||
View.updateFocus();
|
View.updateFocus();
|
||||||
return res;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -166,10 +158,10 @@ var PaperScope = Base.extend(/** @lends PaperScope# */{
|
||||||
// Copy over all fields from this scope to the destination.
|
// Copy over all fields from this scope to the destination.
|
||||||
// Do not use Base.each, since we also want to enumerate over
|
// Do not use Base.each, since we also want to enumerate over
|
||||||
// fields on PaperScope.prototype, e.g. all classes
|
// fields on PaperScope.prototype, e.g. all classes
|
||||||
for (var key in this) {
|
for (var key in this)
|
||||||
if (!/^(version|_id)/.test(key))
|
// Exclude all 'hidden' fields
|
||||||
|
if (!/^_/.test(key))
|
||||||
scope[key] = this[key];
|
scope[key] = this[key];
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -228,14 +220,14 @@ var PaperScope = Base.extend(/** @lends PaperScope# */{
|
||||||
_id: 0,
|
_id: 0,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves a PaperScope object with the given id or associated with
|
* Retrieves a PaperScope object with the given id or associated
|
||||||
* the passed canvas element.
|
* with the passed canvas element.
|
||||||
*
|
*
|
||||||
* @param id
|
* @param id
|
||||||
*/
|
*/
|
||||||
get: function(id) {
|
get: function(id) {
|
||||||
// If a script tag is passed, get the id from it.
|
// If a script tag is passed, get the id from it.
|
||||||
if (typeof id === 'object')
|
if (id && id.getAttribute)
|
||||||
id = id.getAttribute('id');
|
id = id.getAttribute('id');
|
||||||
return this._scopes[id] || null;
|
return this._scopes[id] || null;
|
||||||
},
|
},
|
||||||
|
|
|
@ -14,19 +14,10 @@
|
||||||
* @name PaperScript
|
* @name PaperScript
|
||||||
* @namespace
|
* @namespace
|
||||||
*/
|
*/
|
||||||
// Note that due to the use of with(), PaperScript gets compiled outside the
|
var PaperScript = Base.exports.PaperScript = (function(root) {
|
||||||
// main paper scope, and is added to the PaperScope class. This allows for
|
// Locally turn of exports and define for inlined acorn / esprima.
|
||||||
// better minification and the future use of strict mode once it makes sense
|
// Just declaring the local vars is enough, as they will be undefined.
|
||||||
// in terms of performance.
|
var exports, define,
|
||||||
paper.PaperScope.prototype.PaperScript = (function(root) {
|
|
||||||
var Base = paper.Base,
|
|
||||||
PaperScope = paper.PaperScope,
|
|
||||||
// For local reference, for now only when setting lineNumberBase on
|
|
||||||
// Firefox.
|
|
||||||
PaperScript,
|
|
||||||
// Locally turn of exports and define for inlined acorn / esprima.
|
|
||||||
// Just declaring the local vars is enough, as they will be undefined.
|
|
||||||
exports, define,
|
|
||||||
// The scope into which the library is loaded.
|
// The scope into which the library is loaded.
|
||||||
scope = this;
|
scope = this;
|
||||||
/*#*/ if (__options.version == 'dev') {
|
/*#*/ if (__options.version == 'dev') {
|
||||||
|
@ -69,9 +60,9 @@ paper.PaperScope.prototype.PaperScript = (function(root) {
|
||||||
},
|
},
|
||||||
{}
|
{}
|
||||||
);
|
);
|
||||||
paper.Point.inject(fields);
|
Point.inject(fields);
|
||||||
paper.Size.inject(fields);
|
Size.inject(fields);
|
||||||
paper.Color.inject(fields);
|
Color.inject(fields);
|
||||||
|
|
||||||
// Use very short name for the binary operator (_$_) as well as the
|
// Use very short name for the binary operator (_$_) as well as the
|
||||||
// unary operator ($_), as operations will be replaced with then.
|
// unary operator ($_), as operations will be replaced with then.
|
||||||
|
@ -238,97 +229,117 @@ paper.PaperScope.prototype.PaperScript = (function(root) {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Evaluates parsed PaperScript code in the passed {@link PaperScope}
|
* Executes the parsed PaperScript code in a compiled function that receives
|
||||||
* object. It also installs handlers automatically for us.
|
* all properties of the passed {@link PaperScope} as arguments, to emulate
|
||||||
|
* a global scope with unaffected performance. It also installs global view
|
||||||
|
* and tool handlers automatically for you.
|
||||||
*
|
*
|
||||||
* @name PaperScript.evaluate
|
* @name PaperScript.execute
|
||||||
* @function
|
* @function
|
||||||
* @param {String} code The PaperScript code
|
* @param {String} code The PaperScript code
|
||||||
* @param {PaperScript} scope The scope in which the code is executed
|
* @param {PaperScript} scope The scope for which the code is executed
|
||||||
* @return {Object} the result of the code evaluation
|
|
||||||
*/
|
*/
|
||||||
function evaluate(code, scope) {
|
function execute(code, scope) {
|
||||||
// Set currently active scope.
|
// Set currently active scope.
|
||||||
paper = scope;
|
paper = scope;
|
||||||
var view = scope.project && scope.project.view,
|
var view = scope.getView(),
|
||||||
|
// Only create a tool object if something resembling a tool handler
|
||||||
|
// definition is contained in the code.
|
||||||
|
tool = /\s+on(?:Key|Mouse)(?:Up|Down|Move|Drag)/.test(code)
|
||||||
|
? new Tool()
|
||||||
|
: null,
|
||||||
|
toolHandlers = tool ? tool._events : [],
|
||||||
|
// Compile a list of all handlers that can be defined globally
|
||||||
|
// inside the PaperScript. These are passed on to the function as
|
||||||
|
// undefined arguments, so that their name exists, rather than
|
||||||
|
// injecting a code line that defines them as variables.
|
||||||
|
// They are exported again at the end of the function.
|
||||||
|
handlers = ['onFrame', 'onResize'].concat(toolHandlers),
|
||||||
res;
|
res;
|
||||||
// Define variables for potential handlers, so eval() calls below to
|
code = compile(code);
|
||||||
// fetch their values do not require try-catch around them.
|
// compile a list of paramter names for all variables that need to
|
||||||
// Use with() {} in order to make the scope the current 'global' scope
|
// appear as globals inside the script. At the same time, also collect
|
||||||
// instead of window.
|
// their values, so we can pass them on as arguments in the function
|
||||||
with (scope) {
|
// call.
|
||||||
// Within this, use a function scope, so local variables to not try
|
var params = ['_$_', '$_', 'view', 'tool'],
|
||||||
// and set themselves on the scope object.
|
args = [_$_, $_ , view, tool];
|
||||||
(function() {
|
// Look through all enumerable properties on the scope and expose these
|
||||||
var onActivate, onDeactivate, onEditOptions,
|
// too as pseudo-globals.
|
||||||
onMouseDown, onMouseUp, onMouseDrag, onMouseMove,
|
for (var key in scope) {
|
||||||
onKeyDown, onKeyUp, onFrame, onResize;
|
if (!/^_/.test(key)) {
|
||||||
code = compile(code);
|
params.push(key);
|
||||||
/*#*/ if (__options.environment == 'browser') {
|
args.push(scope[key]);
|
||||||
if (root.InstallTrigger) { // Firefox
|
}
|
||||||
// On Firefox, all error numbers inside evaled code are
|
}
|
||||||
// relative to the line where the eval happened. Totally
|
// Finally define the handler variable names as parameters and compose
|
||||||
// silly, but that's how it is. So we're calculating the
|
// the string describing the properties for the returned object at the
|
||||||
// base of lineNumbers, to remove it again from reported
|
// end of the code execution, so we can retrieve their values from the
|
||||||
// errors. Luckily, Firefox is the only browser where we can
|
// function call.
|
||||||
// define the lineNumber for exceptions.
|
handlers = Base.each(handlers, function(key) {
|
||||||
var handle = PaperScript.handleException;
|
params.push(key);
|
||||||
if (!handle) {
|
this.push(key + ': ' + key);
|
||||||
handle = PaperScript.handleException = function(e) {
|
}, []).join(', ');
|
||||||
throw e.lineNumber >= lineNumber
|
// We need an additional line that returns the handlers in one object.
|
||||||
? new Error(e.message, e.fileName,
|
code += '\nreturn { ' + handlers + ' };';
|
||||||
e.lineNumber - lineNumber)
|
/*#*/ if (__options.environment == 'browser') {
|
||||||
: e;
|
if (root.InstallTrigger) { // Firefox
|
||||||
}
|
// Add a semi-colon at the start so Firefox doesn't swallow empty
|
||||||
// We're using a crazy hack to detect wether the library
|
// lines and shift error messages.
|
||||||
// is minified or not: By generating a second error on
|
code = ';' + code;
|
||||||
// the 2nd line and using the difference in line numbers
|
// On Firefox, all error numbers inside evaled code are relative to
|
||||||
// to calculate the offset to the eval, it works in both
|
// the line where the eval happened. Totally silly, but that's how
|
||||||
// casees.
|
// it is. So we're calculating the base of lineNumbers, to remove it
|
||||||
var lineNumber = new Error().lineNumber;
|
// again from reported errors. Luckily, Firefox is the only browser
|
||||||
lineNumber += (new Error().lineNumber - lineNumber) * 3;
|
// where we can define the lineNumber for exceptions.
|
||||||
}
|
var handle = PaperScript.handleException;
|
||||||
try {
|
if (!handle) {
|
||||||
// Add a semi-colon at the start so Firefox doesn't
|
handle = PaperScript.handleException = function(e) {
|
||||||
// swallow empty lines and shift error messages.
|
throw e.lineNumber >= lineNumber
|
||||||
res = eval(';' + code);
|
? new Error(e.message, e.fileName,
|
||||||
} catch (e) {
|
e.lineNumber - lineNumber)
|
||||||
handle(e);
|
: e;
|
||||||
}
|
};
|
||||||
} else {
|
// We're using a crazy hack to detect wether the library is
|
||||||
res = eval(code);
|
// minified or not: By generating a second error on the 2nd line
|
||||||
}
|
// and using the difference in line numbers to calculate the
|
||||||
/*#*/ } else { // !__options.environment == 'browser'
|
// offset to the eval, it works in both casees.
|
||||||
res = eval(code);
|
var lineNumber = new Error().lineNumber;
|
||||||
/*#*/ } // !__options.environment == 'browser'
|
lineNumber += (new Error().lineNumber - lineNumber) * 3;
|
||||||
// Only look for tool handlers if something resembling their
|
}
|
||||||
// name is contained in the code.
|
try {
|
||||||
if (/on(?:Key|Mouse)(?:Up|Down|Move|Drag)/.test(code)) {
|
res = new Function(params, code).apply(scope, args);
|
||||||
Base.each(paper.Tool.prototype._events, function(key) {
|
// NOTE: in order for the calculation of the above lineNumber
|
||||||
var value = eval(key);
|
// offset to work, we cannot add any statements before the above
|
||||||
if (value) {
|
// line of code, nor can we put it into a separate function.
|
||||||
// Use the getTool accessor that handles auto tool
|
} catch (e) {
|
||||||
// creation for us:
|
handle(e);
|
||||||
scope.getTool()[key] = value;
|
}
|
||||||
}
|
} else {
|
||||||
});
|
res = new Function(params, code).apply(scope, args);
|
||||||
}
|
}
|
||||||
if (view) {
|
/*#*/ } else { // !__options.environment == 'browser'
|
||||||
view.setOnResize(onResize);
|
res = new Function(params, code).apply(scope, args);
|
||||||
// Fire resize event directly, so any user
|
/*#*/ } // !__options.environment == 'browser'
|
||||||
// defined resize handlers are called.
|
// Now install the 'global' tool and view handlers, and we're done!
|
||||||
view.fire('resize', {
|
Base.each(toolHandlers, function(key) {
|
||||||
size: view.size,
|
var value = res[key];
|
||||||
delta: new Point()
|
if (value)
|
||||||
});
|
tool[key] = value;
|
||||||
if (onFrame)
|
});
|
||||||
view.setOnFrame(onFrame);
|
if (view) {
|
||||||
// Automatically update view at the end.
|
if (res.onResize)
|
||||||
view.update();
|
view.setOnResize(res.onResize);
|
||||||
}
|
// Fire resize event directly, so any user
|
||||||
}).call(scope);
|
// defined resize handlers are called.
|
||||||
|
view.fire('resize', {
|
||||||
|
size: view.size,
|
||||||
|
delta: new Point()
|
||||||
|
});
|
||||||
|
if (res.onFrame)
|
||||||
|
view.setOnFrame(res.onFrame);
|
||||||
|
// Automatically update view at the end.
|
||||||
|
view.update();
|
||||||
}
|
}
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*#*/ if (__options.environment == 'browser') {
|
/*#*/ if (__options.environment == 'browser') {
|
||||||
|
@ -356,12 +367,12 @@ paper.PaperScope.prototype.PaperScript = (function(root) {
|
||||||
if (src) {
|
if (src) {
|
||||||
// If we're loading from a source, request that first and
|
// If we're loading from a source, request that first and
|
||||||
// then run later.
|
// then run later.
|
||||||
paper.Http.request('get', src, function(code) {
|
Http.request('get', src, function(code) {
|
||||||
evaluate(code, scope);
|
execute(code, scope);
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
// We can simply get the code form the script tag.
|
// We can simply get the code form the script tag.
|
||||||
evaluate(script.innerHTML, scope);
|
execute(script.innerHTML, scope);
|
||||||
}
|
}
|
||||||
// Mark script as loaded now.
|
// Mark script as loaded now.
|
||||||
script.setAttribute('data-paper-ignore', true);
|
script.setAttribute('data-paper-ignore', true);
|
||||||
|
@ -375,12 +386,12 @@ paper.PaperScope.prototype.PaperScript = (function(root) {
|
||||||
// Handle it asynchronously
|
// Handle it asynchronously
|
||||||
setTimeout(load);
|
setTimeout(load);
|
||||||
} else {
|
} else {
|
||||||
paper.DomEvent.add(window, { load: load });
|
DomEvent.add(window, { load: load });
|
||||||
}
|
}
|
||||||
|
|
||||||
return PaperScript = {
|
return {
|
||||||
compile: compile,
|
compile: compile,
|
||||||
evaluate: evaluate,
|
execute: execute,
|
||||||
load: load,
|
load: load,
|
||||||
lineNumberBase: 0
|
lineNumberBase: 0
|
||||||
};
|
};
|
||||||
|
@ -400,15 +411,15 @@ paper.PaperScope.prototype.PaperScript = (function(root) {
|
||||||
// Expose core methods and values
|
// Expose core methods and values
|
||||||
scope.require = require;
|
scope.require = require;
|
||||||
scope.console = console;
|
scope.console = console;
|
||||||
evaluate(source, scope);
|
execute(source, scope);
|
||||||
module.exports = scope;
|
module.exports = scope;
|
||||||
};
|
};
|
||||||
|
|
||||||
/*#*/ } // __options.environment == 'node'
|
/*#*/ } // __options.environment == 'node'
|
||||||
|
|
||||||
return PaperScript = {
|
return {
|
||||||
compile: compile,
|
compile: compile,
|
||||||
evaluate: evaluate
|
execute: execute
|
||||||
};
|
};
|
||||||
|
|
||||||
/*#*/ } // !__options.environment == 'browser'
|
/*#*/ } // !__options.environment == 'browser'
|
||||||
|
|
|
@ -10,16 +10,19 @@
|
||||||
* All rights reserved.
|
* All rights reserved.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/** @scope _global_ */ {
|
/**
|
||||||
|
* @name _global_
|
||||||
// DOCS: Find a way to put this description into _global_
|
* @namespace
|
||||||
|
*
|
||||||
/**
|
* When code is executed as PaperScript, the script's scope is populated with
|
||||||
* In a PaperScript context, the global scope is populated with all
|
* all fields of the currently active {@link PaperScope} object, which within
|
||||||
* fields of the currently active {@link PaperScope} object. In a JavaScript
|
* the script appear to be global.
|
||||||
* context, it only contains the {@link #paper} reference to the currently
|
*
|
||||||
* active {@link PaperScope} object, which also exposes all Paper classes.
|
* In a JavaScript context, only the {@link paper} variable is added to the
|
||||||
*/
|
* global scope, referencing the currently active {@link PaperScope} object,
|
||||||
|
* through which all properties and Paper.js classes can be accessed.
|
||||||
|
*/
|
||||||
|
/** @scope _global_ */{
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A reference to the currently active {@link PaperScope} object.
|
* A reference to the currently active {@link PaperScope} object.
|
||||||
|
@ -32,40 +35,58 @@
|
||||||
// DOCS: This does not work: @borrows PaperScope#version as _global_#version,
|
// DOCS: This does not work: @borrows PaperScope#version as _global_#version,
|
||||||
// so we're repeating documentation here form PaperScope:
|
// so we're repeating documentation here form PaperScope:
|
||||||
/**
|
/**
|
||||||
* {@grouptitle Global PaperScope Properties (for PaperScript)}
|
* {@grouptitle Global PaperScript Properties}
|
||||||
|
*
|
||||||
|
* The project for which the PaperScript is executed.
|
||||||
|
*
|
||||||
|
* Note that when working with mulitple projects, this does not necessarily
|
||||||
|
* reflect the currently active project. For this, use
|
||||||
|
* {@link PaperScope#project} instead.
|
||||||
*
|
*
|
||||||
* The currently active project.
|
|
||||||
* @name project
|
* @name project
|
||||||
* @type Project
|
* @type Project
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The list of all open projects within the current Paper.js context.
|
* The list of all open projects within the current Paper.js context.
|
||||||
|
*
|
||||||
* @name projects
|
* @name projects
|
||||||
* @type Project[]
|
* @type Project[]
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The reference to the active project's view.
|
* The reference to the project's view.
|
||||||
|
*
|
||||||
|
* Note that when working with mulitple projects, this does not necessarily
|
||||||
|
* reflect the view of the currently active project. For this, use
|
||||||
|
* {@link PaperScope#view} instead.
|
||||||
|
*
|
||||||
* @name view
|
* @name view
|
||||||
* @type View
|
* @type View
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The reference to the active tool.
|
* The reference to the tool object which is automatically created when global
|
||||||
|
* tool event handlers are defined.
|
||||||
|
*
|
||||||
|
* Note that when working with mulitple tools, this does not necessarily
|
||||||
|
* reflect the currently active tool. For this, use {@link PaperScope#tool}
|
||||||
|
* instead.
|
||||||
|
*
|
||||||
* @name tool
|
* @name tool
|
||||||
* @type Tool
|
* @type Tool
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The list of available tools.
|
* The list of available tools.
|
||||||
|
*
|
||||||
* @name tools
|
* @name tools
|
||||||
* @type Tool[]
|
* @type Tool[]
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@grouptitle View Event Handlers (for PaperScript)}
|
* {@grouptitle PaperScript View Event Handlers}
|
||||||
* A reference to the {@link View#onFrame} handler function.
|
* A global reference to the {@link View#onFrame} handler function.
|
||||||
*
|
*
|
||||||
* @name onFrame
|
* @name onFrame
|
||||||
* @property
|
* @property
|
||||||
|
@ -81,7 +102,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@grouptitle Mouse Event Handlers (for PaperScript)}
|
* {@grouptitle PaperScript Tool Event Handlers}
|
||||||
* A reference to the {@link Tool#onMouseDown} handler function.
|
* A reference to the {@link Tool#onMouseDown} handler function.
|
||||||
* @name onMouseDown
|
* @name onMouseDown
|
||||||
* @property
|
* @property
|
||||||
|
|
10
src/paper.js
10
src/paper.js
|
@ -138,12 +138,10 @@ var paper = new function(undefined) {
|
||||||
/*#*/ include('svg/SVGImport.js');
|
/*#*/ include('svg/SVGImport.js');
|
||||||
/*#*/ } // __options.svg
|
/*#*/ } // __options.svg
|
||||||
|
|
||||||
/*#*/ include('export.js');
|
|
||||||
return paper;
|
|
||||||
};
|
|
||||||
|
|
||||||
// include PaperScript separately outside the main paper scope, due to its use
|
|
||||||
// of with(). This also simplifies making its inclusion optional.
|
|
||||||
/*#*/ if (__options.paperscript) {
|
/*#*/ if (__options.paperscript) {
|
||||||
/*#*/ include('core/PaperScript.js');
|
/*#*/ include('core/PaperScript.js');
|
||||||
/*#*/ } // __options.paperscript
|
/*#*/ } // __options.paperscript
|
||||||
|
|
||||||
|
/*#*/ include('export.js');
|
||||||
|
return paper;
|
||||||
|
};
|
||||||
|
|
|
@ -151,6 +151,12 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
getStyle: function() {
|
||||||
|
// If this path is part of a CompoundPath, use the paren't style instead
|
||||||
|
var parent = this._parent;
|
||||||
|
return (parent instanceof CompoundPath ? parent : this)._style;
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The segments contained within the path.
|
* The segments contained within the path.
|
||||||
*
|
*
|
||||||
|
@ -1621,12 +1627,6 @@ var Path = PathItem.extend(/** @lends Path# */{
|
||||||
return this.getNearestLocation(point).getPoint();
|
return this.getNearestLocation(point).getPoint();
|
||||||
},
|
},
|
||||||
|
|
||||||
getStyle: function() {
|
|
||||||
// If this path is part of a CompoundPath, use the paren't style instead
|
|
||||||
var parent = this._parent;
|
|
||||||
return (parent && parent instanceof CompoundPath ? parent : this)._style;
|
|
||||||
},
|
|
||||||
|
|
||||||
// DOCS: toShape
|
// DOCS: toShape
|
||||||
|
|
||||||
toShape: function(insert) {
|
toShape: function(insert) {
|
||||||
|
|
|
@ -45,8 +45,10 @@ PathItem.inject(new function() {
|
||||||
// to compare both at the same time
|
// to compare both at the same time
|
||||||
? (loc1.getIndex() + loc1.getParameter())
|
? (loc1.getIndex() + loc1.getParameter())
|
||||||
- (loc2.getIndex() + loc2.getParameter())
|
- (loc2.getIndex() + loc2.getParameter())
|
||||||
// Sort by path id to group all locations on the same path.
|
// Sort by path index to group all locations on the same
|
||||||
: path1._id - path2._id;
|
// path in the sequnence that they are encountered within
|
||||||
|
// compound paths.
|
||||||
|
: path1._index - path2._index;
|
||||||
});
|
});
|
||||||
var others = collectOthers && [];
|
var others = collectOthers && [];
|
||||||
for (var i = intersections.length - 1; i >= 0; i--) {
|
for (var i = intersections.length - 1; i >= 0; i--) {
|
||||||
|
@ -222,6 +224,8 @@ PathItem.inject(new function() {
|
||||||
// at intersections.
|
// at intersections.
|
||||||
return /** @lends Path# */{
|
return /** @lends Path# */{
|
||||||
/**
|
/**
|
||||||
|
* {@grouptitle Boolean Path Operations}
|
||||||
|
*
|
||||||
* Merges the geometry of the specified path from this path's
|
* Merges the geometry of the specified path from this path's
|
||||||
* geometry and returns the result as a new path item.
|
* geometry and returns the result as a new path item.
|
||||||
*
|
*
|
||||||
|
|
|
@ -31,7 +31,12 @@ var PathItem = Item.extend(/** @lends PathItem# */{
|
||||||
* of {@link CurveLocation} objects. {@link CompoundPath} items are also
|
* of {@link CurveLocation} objects. {@link CompoundPath} items are also
|
||||||
* supported.
|
* supported.
|
||||||
*
|
*
|
||||||
* @param {PathItem} path the other item to find the intersections to.
|
* @name PathItem#getIntersections(path, sorted)
|
||||||
|
* @function
|
||||||
|
*
|
||||||
|
* @param {PathItem} path the other item to find the intersections with
|
||||||
|
* @param {Boolean} [sorted=true] controls wether to sort the results by
|
||||||
|
* offset
|
||||||
* @return {CurveLocation[]} the locations of all intersection between the
|
* @return {CurveLocation[]} the locations of all intersection between the
|
||||||
* paths
|
* paths
|
||||||
* @example {@paperscript}
|
* @example {@paperscript}
|
||||||
|
@ -59,7 +64,7 @@ var PathItem = Item.extend(/** @lends PathItem# */{
|
||||||
* }
|
* }
|
||||||
* }
|
* }
|
||||||
*/
|
*/
|
||||||
getIntersections: function(path) {
|
getIntersections: function(path, sorted) {
|
||||||
// First check the bounds of the two paths. If they don't intersect,
|
// First check the bounds of the two paths. If they don't intersect,
|
||||||
// we don't need to iterate through their curves.
|
// we don't need to iterate through their curves.
|
||||||
if (!this.getBounds().touches(path.getBounds()))
|
if (!this.getBounds().touches(path.getBounds()))
|
||||||
|
@ -81,6 +86,24 @@ var PathItem = Item.extend(/** @lends PathItem# */{
|
||||||
Curve.getIntersections(values1, values2[j], curve1, curves2[j],
|
Curve.getIntersections(values1, values2[j], curve1, curves2[j],
|
||||||
locations);
|
locations);
|
||||||
}
|
}
|
||||||
|
if (sorted || sorted === undefined) {
|
||||||
|
// Now sort the results into the right sequence.
|
||||||
|
// TODO: Share this code with PathItem.Boolean.js, potentially by
|
||||||
|
// using the new BinHeap class that's in preparation.
|
||||||
|
locations.sort(function(loc1, loc2) {
|
||||||
|
var path1 = loc1.getPath(),
|
||||||
|
path2 = loc2.getPath();
|
||||||
|
return path1 === path2
|
||||||
|
// We can add parameter (0 <= t <= 1) to index (integer)
|
||||||
|
// to compare both at the same time
|
||||||
|
? (loc1.getIndex() + loc1.getParameter())
|
||||||
|
- (loc2.getIndex() + loc2.getParameter())
|
||||||
|
// Sort by path index to group all locations on the same
|
||||||
|
// path in the sequnence that they are encountered
|
||||||
|
// within compound paths.
|
||||||
|
: path1._index - path2._index;
|
||||||
|
});
|
||||||
|
}
|
||||||
return locations;
|
return locations;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -45,7 +45,7 @@
|
||||||
var Tool = PaperScopeItem.extend(/** @lends Tool# */{
|
var Tool = PaperScopeItem.extend(/** @lends Tool# */{
|
||||||
_class: 'Tool',
|
_class: 'Tool',
|
||||||
_list: 'tools',
|
_list: 'tools',
|
||||||
_reference: '_tool', // PaperScope has accessor for #tool
|
_reference: 'tool',
|
||||||
_events: [ 'onActivate', 'onDeactivate', 'onEditOptions',
|
_events: [ 'onActivate', 'onDeactivate', 'onEditOptions',
|
||||||
'onMouseDown', 'onMouseUp', 'onMouseDrag', 'onMouseMove',
|
'onMouseDown', 'onMouseUp', 'onMouseDrag', 'onMouseMove',
|
||||||
'onKeyDown', 'onKeyUp' ],
|
'onKeyDown', 'onKeyUp' ],
|
||||||
|
|
|
@ -74,7 +74,7 @@ var Key = new function() {
|
||||||
type = down ? 'keydown' : 'keyup',
|
type = down ? 'keydown' : 'keyup',
|
||||||
view = View._focused,
|
view = View._focused,
|
||||||
scope = view && view.isVisible() && view._scope,
|
scope = view && view.isVisible() && view._scope,
|
||||||
tool = scope && scope._tool,
|
tool = scope && scope.tool,
|
||||||
name;
|
name;
|
||||||
keyMap[key] = down;
|
keyMap[key] = down;
|
||||||
// Detect modifiers and mark them as pressed / released
|
// Detect modifiers and mark them as pressed / released
|
||||||
|
|
|
@ -447,7 +447,7 @@ var View = Base.extend(Callback, /** @lends View# */{
|
||||||
|
|
||||||
viewToProject: function(/* point */) {
|
viewToProject: function(/* point */) {
|
||||||
return this._matrix._inverseTransform(Point.read(arguments));
|
return this._matrix._inverseTransform(Point.read(arguments));
|
||||||
},
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* {@grouptitle Event Handlers}
|
* {@grouptitle Event Handlers}
|
||||||
|
@ -664,7 +664,7 @@ var View = Base.extend(Callback, /** @lends View# */{
|
||||||
// Always first call the view's mouse handlers, as required by
|
// Always first call the view's mouse handlers, as required by
|
||||||
// CanvasView, and then handle the active tool, if any.
|
// CanvasView, and then handle the active tool, if any.
|
||||||
view._handleEvent('mousedown', point, event);
|
view._handleEvent('mousedown', point, event);
|
||||||
if (tool = view._scope._tool)
|
if (tool = view._scope.tool)
|
||||||
tool._handleEvent('mousedown', point, event);
|
tool._handleEvent('mousedown', point, event);
|
||||||
// In the end we always call update(), which only updates the view if
|
// In the end we always call update(), which only updates the view if
|
||||||
// anything has changed in the above calls.
|
// anything has changed in the above calls.
|
||||||
|
@ -673,7 +673,7 @@ var View = Base.extend(Callback, /** @lends View# */{
|
||||||
|
|
||||||
function handleMouseMove(view, point, event) {
|
function handleMouseMove(view, point, event) {
|
||||||
view._handleEvent('mousemove', point, event);
|
view._handleEvent('mousemove', point, event);
|
||||||
var tool = view._scope._tool;
|
var tool = view._scope.tool;
|
||||||
if (tool) {
|
if (tool) {
|
||||||
// If there's no onMouseDrag, fire onMouseMove while dragging.
|
// If there's no onMouseDrag, fire onMouseMove while dragging.
|
||||||
tool._handleEvent(dragging && tool.responds('mousedrag')
|
tool._handleEvent(dragging && tool.responds('mousedrag')
|
||||||
|
|
Loading…
Reference in a new issue