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:
Jürg Lehni 2014-01-04 21:57:29 +01:00
parent 9dea3f3b74
commit 4302682c1c
2 changed files with 32 additions and 60 deletions

View file

@ -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;
}, },

View file

@ -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];