2011-02-01 15:52:04 -08:00
|
|
|
/** @provides javelin-workflow-dev */
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @requires javelin-install javelin-vector javelin-dom
|
|
|
|
* @provides javelin-mask
|
|
|
|
* @javelin
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Show a transparent "mask" over the page; used by Workflow to draw visual
|
|
|
|
* attention to modal dialogs.
|
|
|
|
*/
|
|
|
|
JX.install('Mask', {
|
|
|
|
statics : {
|
|
|
|
_depth : 0,
|
|
|
|
_mask : null,
|
|
|
|
show : function() {
|
|
|
|
if (!JX.Mask._depth) {
|
|
|
|
JX.Mask._mask = JX.$N('div', {className: 'jx-mask'});
|
|
|
|
document.body.appendChild(JX.Mask._mask);
|
|
|
|
JX.$V.getDocument().setDim(JX.Mask._mask);
|
|
|
|
}
|
|
|
|
++JX.Mask._depth;
|
|
|
|
},
|
|
|
|
hide : function() {
|
|
|
|
--JX.Mask._depth;
|
|
|
|
if (!JX.Mask._depth) {
|
|
|
|
JX.DOM.remove(JX.Mask._mask);
|
|
|
|
JX.Mask._mask = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
/**
|
|
|
|
* @requires javelin-stratcom
|
|
|
|
* javelin-request
|
|
|
|
* javelin-dom
|
|
|
|
* javelin-vector
|
|
|
|
* javelin-install
|
|
|
|
* javelin-util
|
|
|
|
* javelin-mask
|
|
|
|
* @provides javelin-workflow
|
|
|
|
* @javelin
|
|
|
|
*/
|
|
|
|
|
|
|
|
JX.install('Workflow', {
|
|
|
|
construct : function(uri, data) {
|
|
|
|
if (__DEV__) {
|
|
|
|
if (!uri || uri == '#') {
|
|
|
|
throw new Error(
|
|
|
|
'new JX.Workflow(<?>, ...): '+
|
|
|
|
'bogus URI provided when creating workflow.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this.setURI(uri);
|
|
|
|
this.setData(data || {});
|
|
|
|
},
|
|
|
|
|
|
|
|
events : ['error', 'finally', 'submit'],
|
|
|
|
|
|
|
|
statics : {
|
|
|
|
_stack : [],
|
|
|
|
newFromForm : function(form, data) {
|
|
|
|
var inputs = [].concat(
|
|
|
|
JX.DOM.scry(form, 'input'),
|
|
|
|
JX.DOM.scry(form, 'button'),
|
|
|
|
JX.DOM.scry(form, 'textarea'));
|
|
|
|
|
|
|
|
for (var ii = 0; ii < inputs.length; ii++) {
|
|
|
|
if (inputs[ii].disabled) {
|
|
|
|
delete inputs[ii];
|
|
|
|
} else {
|
|
|
|
inputs[ii].disabled = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var workflow = new JX.Workflow(
|
|
|
|
form.getAttribute('action'),
|
|
|
|
JX.copy(data || {}, JX.DOM.serialize(form)));
|
|
|
|
workflow.setMethod(form.getAttribute('method'));
|
|
|
|
workflow.listen('finally', function() {
|
|
|
|
for (var ii = 0; ii < inputs.length; ii++) {
|
|
|
|
inputs[ii] && (inputs[ii].disabled = false);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return workflow;
|
|
|
|
},
|
|
|
|
newFromLink : function(link) {
|
|
|
|
var workflow = new JX.Workflow(link.href);
|
|
|
|
return workflow;
|
|
|
|
},
|
|
|
|
_push : function(workflow) {
|
|
|
|
JX.Mask.show();
|
|
|
|
JX.Workflow._stack.push(workflow);
|
|
|
|
},
|
|
|
|
_pop : function() {
|
|
|
|
var dialog = JX.Workflow._stack.pop();
|
|
|
|
(dialog.getCloseHandler() || JX.bag)();
|
|
|
|
dialog._destroy();
|
|
|
|
JX.Mask.hide();
|
|
|
|
},
|
|
|
|
disable : function() {
|
|
|
|
JX.Workflow._disabled = true;
|
|
|
|
},
|
|
|
|
_onbutton : function(event) {
|
|
|
|
|
|
|
|
if (JX.Stratcom.pass()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (JX.Workflow._disabled) {
|
|
|
|
return;
|
|
|
|
}
|
2011-02-01 16:42:36 -08:00
|
|
|
|
2011-02-01 15:52:04 -08:00
|
|
|
var t = event.getTarget();
|
|
|
|
if (t.name == '__cancel__' || t.name == '__close__') {
|
|
|
|
JX.Workflow._pop();
|
|
|
|
} else {
|
|
|
|
|
|
|
|
var form = event.getNode('jx-dialog');
|
|
|
|
var data = JX.DOM.serialize(form);
|
|
|
|
data[t.name] = true;
|
|
|
|
data.__wflow__ = true;
|
|
|
|
|
|
|
|
var active = JX.Workflow._stack[JX.Workflow._stack.length - 1];
|
|
|
|
var e = active.invoke('submit', {form: form, data: data});
|
|
|
|
if (!e.getStopped()) {
|
|
|
|
active._destroy();
|
|
|
|
active
|
|
|
|
.setURI(form.getAttribute('action') || active.getURI())
|
|
|
|
.setData(data)
|
|
|
|
.start();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
event.prevent();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
members : {
|
|
|
|
_root : null,
|
|
|
|
_pushed : false,
|
|
|
|
_onload : function(r) {
|
|
|
|
// It is permissible to send back a falsey redirect to force a page
|
|
|
|
// reload, so we need to take this branch if the key is present.
|
|
|
|
if (r && (typeof r.redirect != 'undefined')) {
|
|
|
|
JX.go(r.redirect, true);
|
|
|
|
} else if (r && r.dialog) {
|
|
|
|
this._push();
|
|
|
|
this._root = JX.$N(
|
|
|
|
'div',
|
|
|
|
{className: 'jx-client-dialog'},
|
|
|
|
JX.HTML(r.dialog));
|
|
|
|
JX.DOM.listen(
|
|
|
|
this._root,
|
|
|
|
'click',
|
2011-02-01 16:42:36 -08:00
|
|
|
[['jx-workflow-button'], ['tag:button']],
|
2011-02-01 15:52:04 -08:00
|
|
|
JX.Workflow._onbutton);
|
|
|
|
document.body.appendChild(this._root);
|
|
|
|
var d = JX.$V.getDim(this._root);
|
|
|
|
var v = JX.$V.getViewport();
|
|
|
|
var s = JX.$V.getScroll();
|
|
|
|
JX.$V((v.x - d.x) / 2, s.y + 100).setPos(this._root);
|
|
|
|
try {
|
2011-02-04 17:53:14 -08:00
|
|
|
try {
|
|
|
|
JX.DOM.focus(JX.DOM.find(this._root, 'button', '__default__'));
|
|
|
|
} catch (_ignored) {}
|
2011-02-01 15:52:04 -08:00
|
|
|
var inputs = JX.DOM.scry(this._root, 'input')
|
|
|
|
.concat(JX.DOM.scry(this._root, 'textarea'));
|
|
|
|
var miny = Number.POSITIVE_INFINITY;
|
|
|
|
var target = null;
|
|
|
|
for (var ii = 0; ii < inputs.length; ++ii) {
|
|
|
|
if (inputs[ii].type != 'hidden') {
|
|
|
|
// Find the topleft-most displayed element.
|
|
|
|
var p = JX.$V(inputs[ii]);
|
|
|
|
if (p.y < miny) {
|
|
|
|
miny = p.y;
|
|
|
|
target = inputs[ii];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
target && JX.DOM.focus(target);
|
|
|
|
} catch (_ignored) {}
|
|
|
|
} else if (this.getHandler()) {
|
|
|
|
this.getHandler()(r);
|
|
|
|
this._pop();
|
|
|
|
} else if (r) {
|
|
|
|
if (__DEV__) {
|
|
|
|
throw new Error('Response to workflow request went unhandled.');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_push : function() {
|
|
|
|
if (!this._pushed) {
|
|
|
|
this._pushed = true;
|
|
|
|
JX.Workflow._push(this);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_pop : function() {
|
|
|
|
if (this._pushed) {
|
|
|
|
this._pushed = false;
|
|
|
|
JX.Workflow._pop();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_destroy : function() {
|
|
|
|
if (this._root) {
|
|
|
|
JX.DOM.remove(this._root);
|
|
|
|
this._root = null;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
start : function() {
|
|
|
|
var uri = this.getURI();
|
|
|
|
var method = this.getMethod();
|
|
|
|
var r = new JX.Request(uri, JX.bind(this, this._onload));
|
|
|
|
r.setData(this.getData());
|
|
|
|
r.setDataSerializer(this.getDataSerializer());
|
|
|
|
if (method) {
|
|
|
|
r.setMethod(method);
|
|
|
|
}
|
|
|
|
r.listen('finally', JX.bind(this, this.invoke, 'finally'));
|
|
|
|
r.listen('error', JX.bind(this, function(error) {
|
|
|
|
var e = this.invoke('error', error);
|
|
|
|
if (e.getStopped()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO: Default error behavior? On Facebook Lite, we just shipped the
|
|
|
|
// user to "/error/". We could emit a blanket 'workflow-failed' type
|
|
|
|
// event instead.
|
|
|
|
}));
|
|
|
|
r.send();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
properties : {
|
|
|
|
handler : null,
|
|
|
|
closeHandler : null,
|
|
|
|
data : null,
|
|
|
|
dataSerializer : null,
|
|
|
|
method : null,
|
|
|
|
URI : null
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|