mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-20 22:39:50 -05:00
Implement a better way of handling exceptions in PaperScript on Firefox and Chrome.
Use dynamically inserted script tag rather than compiled function since Firefox reports correct line numbers there.
This commit is contained in:
parent
9dea3f3b74
commit
4302682c1c
2 changed files with 32 additions and 60 deletions
|
@ -77,32 +77,16 @@ var Callback = {
|
||||||
if (!handlers)
|
if (!handlers)
|
||||||
return false;
|
return false;
|
||||||
var args = [].slice.call(arguments, 1),
|
var args = [].slice.call(arguments, 1),
|
||||||
PaperScript = paper.PaperScript,
|
|
||||||
handleException = PaperScript && PaperScript.handleException,
|
|
||||||
that = this;
|
that = this;
|
||||||
|
for (var i in handlers) {
|
||||||
function callHandlers() {
|
// When the handler function returns false, prevent the default
|
||||||
for (var i in handlers) {
|
// behaviour and stop propagation of the event by calling stop()
|
||||||
// When the handler function returns false, prevent the default
|
if (handlers[i].apply(that, args) === false
|
||||||
// behaviour and stop propagation of the event by calling stop()
|
&& event && event.stop) {
|
||||||
if (handlers[i].apply(that, args) === false
|
event.stop();
|
||||||
&& event && event.stop) {
|
break;
|
||||||
event.stop();
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// See PaperScript.handleException for an explanation of the following.
|
|
||||||
// Firefox is to blame for the necessity of this...
|
|
||||||
if (handleException) {
|
|
||||||
try {
|
|
||||||
callHandlers();
|
|
||||||
} catch (e) {
|
|
||||||
handleException(e);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
callHandlers();
|
|
||||||
}
|
|
||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
|
@ -248,15 +248,15 @@ var PaperScript = Base.exports.PaperScript = (function() {
|
||||||
// undefined arguments, so that their name exists, rather than
|
// undefined arguments, so that their name exists, rather than
|
||||||
// injecting a code line that defines them as variables.
|
// injecting a code line that defines them as variables.
|
||||||
// They are exported again at the end of the function.
|
// They are exported again at the end of the function.
|
||||||
handlers = ['onFrame', 'onResize'].concat(toolHandlers),
|
handlers = ['onFrame', 'onResize'].concat(toolHandlers);
|
||||||
res;
|
|
||||||
code = compile(code);
|
code = compile(code);
|
||||||
// compile a list of paramter names for all variables that need to
|
// compile a list of paramter names for all variables that need to
|
||||||
// appear as globals inside the script. At the same time, also collect
|
// appear as globals inside the script. At the same time, also collect
|
||||||
// their values, so we can pass them on as arguments in the function
|
// their values, so we can pass them on as arguments in the function
|
||||||
// call.
|
// call.
|
||||||
var params = ['_$_', '$_', 'view', 'tool'],
|
var params = ['_$_', '$_', 'view', 'tool'],
|
||||||
args = [_$_, $_ , view, tool];
|
args = [_$_, $_ , view, tool],
|
||||||
|
func;
|
||||||
// Look through all enumerable properties on the scope and expose these
|
// Look through all enumerable properties on the scope and expose these
|
||||||
// too as pseudo-globals.
|
// too as pseudo-globals.
|
||||||
for (var key in scope) {
|
for (var key in scope) {
|
||||||
|
@ -276,44 +276,32 @@ var PaperScript = Base.exports.PaperScript = (function() {
|
||||||
// We need an additional line that returns the handlers in one object.
|
// We need an additional line that returns the handlers in one object.
|
||||||
code += '\nreturn { ' + handlers + ' };';
|
code += '\nreturn { ' + handlers + ' };';
|
||||||
/*#*/ if (__options.environment == 'browser') {
|
/*#*/ if (__options.environment == 'browser') {
|
||||||
if (window.InstallTrigger) { // Firefox
|
if (window.InstallTrigger || window.chrome) { // Firefox and Chrome
|
||||||
// Add a semi-colon at the start so Firefox doesn't swallow empty
|
// On Firefox, all error numbers inside dynamically compiled code
|
||||||
// lines and shift error messages.
|
// are relative to the line where the eval / compilation happened.
|
||||||
code = ';' + code;
|
// To fix this issue, we're temporarily inserting a new script
|
||||||
// On Firefox, all error numbers inside evaled code are relative to
|
// tag. We also use this on Chrome to fix an issue with compiled
|
||||||
// the line where the eval happened. Totally silly, but that's how
|
// functions:
|
||||||
// it is. So we're calculating the base of lineNumbers, to remove it
|
// https://code.google.com/p/chromium/issues/detail?id=331655
|
||||||
// again from reported errors. Luckily, Firefox is the only browser
|
var script = document.createElement('script'),
|
||||||
// where we can define the lineNumber for exceptions.
|
head = document.head;
|
||||||
var handle = PaperScript.handleException;
|
// Do not add a new-line before the code on Chrome since the error
|
||||||
if (!handle) {
|
// messages are shifted by one line there...
|
||||||
handle = PaperScript.handleException = function(e) {
|
if (!window.chrome)
|
||||||
throw e.lineNumber >= lineNumber
|
code = '\n' + code;
|
||||||
? new Error(e.message, e.fileName,
|
script.appendChild(document.createTextNode(
|
||||||
e.lineNumber - lineNumber)
|
'paper._execute = function(' + params + ') {' + code + '\n}'
|
||||||
: e;
|
));
|
||||||
};
|
head.appendChild(script);
|
||||||
// We're using a crazy hack to detect wether the library is
|
func = paper._execute;
|
||||||
// minified or not: By generating a second error on the 2nd line
|
head.removeChild(script);
|
||||||
// and using the difference in line numbers to calculate the
|
|
||||||
// offset to the eval, it works in both casees.
|
|
||||||
var lineNumber = new Error().lineNumber;
|
|
||||||
lineNumber += (new Error().lineNumber - lineNumber) * 3;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
res = Function(params, code).apply(scope, args);
|
|
||||||
// NOTE: in order for the calculation of the above lineNumber
|
|
||||||
// offset to work, we cannot add any statements before the above
|
|
||||||
// line of code, nor can we put it into a separate function.
|
|
||||||
} catch (e) {
|
|
||||||
handle(e);
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
res = Function(params, code).apply(scope, args);
|
func = Function(params, code);
|
||||||
}
|
}
|
||||||
/*#*/ } else { // !__options.environment == 'browser'
|
/*#*/ } else { // !__options.environment == 'browser'
|
||||||
res = Function(params, code).apply(scope, args);
|
func = Function(params, code);
|
||||||
/*#*/ } // !__options.environment == 'browser'
|
/*#*/ } // !__options.environment == 'browser'
|
||||||
|
var res = func.apply(scope, args);
|
||||||
// Now install the 'global' tool and view handlers, and we're done!
|
// Now install the 'global' tool and view handlers, and we're done!
|
||||||
Base.each(toolHandlers, function(key) {
|
Base.each(toolHandlers, function(key) {
|
||||||
var value = res[key];
|
var value = res[key];
|
||||||
|
|
Loading…
Reference in a new issue