mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-01 10:20:59 +01:00
5520dbfac6
Summary: Fixes T8036. In addition to making the mock edit work, this tightens quicksand code such that the correct page id is returned even if start() has not been called yet. It also tightens mock view where some functions should respect statics.enabled a bit more. Test Plan: clicked edit mock, mock crumb, edit mock, mock crum, edit mock, made edits and they worked! clicked edit mock, mock crumb, edit mock, mock crumb, edit mock, profile icon, hit browser back to edit mock, made edits and they worked! also observed mock view page not occasionally wigging out from image_onload race not having statics.enabled respect during the above Reviewers: epriestley Reviewed By: epriestley Subscribers: Korvin, epriestley Maniphest Tasks: T8036 Differential Revision: https://secure.phabricator.com/D12739
241 lines
6.6 KiB
JavaScript
241 lines
6.6 KiB
JavaScript
/**
|
|
* @provides javelin-behavior-pholio-mock-edit
|
|
* @requires javelin-behavior
|
|
* javelin-stratcom
|
|
* javelin-dom
|
|
* javelin-workflow
|
|
* javelin-quicksand
|
|
* phabricator-phtize
|
|
* phabricator-drag-and-drop-file-upload
|
|
* phabricator-draggable-list
|
|
*/
|
|
JX.behavior('pholio-mock-edit', function(config, statics) {
|
|
var pht = JX.phtize(config.pht);
|
|
var uploading = [];
|
|
|
|
|
|
/* -( Deleting Images )---------------------------------------------------- */
|
|
|
|
|
|
// When the user clicks the "X" on an image, we replace it with a "click to
|
|
// undo" element. If they click to undo, we put the original node back in the
|
|
// DOM.
|
|
var pholio_drop_remove = function(e) {
|
|
e.kill();
|
|
|
|
var node = e.getNode('pholio-drop-image');
|
|
var undo = render_undo();
|
|
|
|
JX.DOM.listen(undo, 'click', 'pholio-drop-undo', function(e) {
|
|
e.kill();
|
|
JX.DOM.replace(undo, node);
|
|
synchronize_order();
|
|
});
|
|
|
|
JX.DOM.replace(node, undo);
|
|
synchronize_order();
|
|
};
|
|
|
|
|
|
/* -( Reordering Images )-------------------------------------------------- */
|
|
|
|
|
|
// Reflect the display order in a hidden input.
|
|
var synchronize_order = function() {
|
|
var items = statics.draglist.findItems();
|
|
var order = [];
|
|
for (var ii = 0; ii < items.length; ii++) {
|
|
order.push(JX.Stratcom.getData(items[ii]).filePHID);
|
|
}
|
|
statics.nodes.order.value = order.join(',');
|
|
};
|
|
|
|
|
|
var build_draglist = function(node) {
|
|
var draglist = new JX.DraggableList('pholio-drop-image', node)
|
|
.setGhostNode(JX.$N('div', {className: 'drag-ghost'}))
|
|
.setFindItemsHandler(function() {
|
|
return JX.DOM.scry(node, 'div', 'pholio-drop-image');
|
|
});
|
|
|
|
// Only let the user drag images by the handle, not the whole entry.
|
|
draglist.listen('shouldBeginDrag', function(e) {
|
|
if (!e.getNode('pholio-drag-handle')) {
|
|
JX.Stratcom.context().prevent();
|
|
}
|
|
});
|
|
draglist.listen('didDrop', synchronize_order);
|
|
return draglist;
|
|
};
|
|
|
|
|
|
/* -( Build )-------------------------------------------------------------- */
|
|
|
|
|
|
var build_drop_upload = function(node) {
|
|
var drop = new JX.PhabricatorDragAndDropFileUpload(node)
|
|
.setURI(config.uploadURI);
|
|
|
|
drop.listen('didBeginDrag', function() {
|
|
JX.DOM.alterClass(node, 'pholio-drop-active', true);
|
|
});
|
|
|
|
drop.listen('didEndDrag', function() {
|
|
JX.DOM.alterClass(node, 'pholio-drop-active', false);
|
|
});
|
|
|
|
return drop;
|
|
};
|
|
|
|
var build_add_control = function(add_node) {
|
|
var drop = build_drop_upload(add_node);
|
|
|
|
drop.listen('willUpload', function(file) {
|
|
var node = render_uploading();
|
|
uploading.push({node: node, file: file});
|
|
statics.nodes.list.appendChild(node);
|
|
});
|
|
|
|
drop.listen('didUpload', function(file) {
|
|
var node;
|
|
for (var ii = 0; ii < uploading.length; ii++) {
|
|
if (uploading[ii].file === file) {
|
|
node = uploading[ii].node;
|
|
uploading.splice(ii, 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
JX.DOM.setContent(node, pht('uploaded'));
|
|
|
|
new JX.Workflow(config.renderURI, {filePHID: file.getPHID()})
|
|
.setHandler(function(response) {
|
|
var new_node = JX.$H(response.markup).getFragment().firstChild;
|
|
build_update_control(new_node);
|
|
|
|
JX.DOM.replace(node, new_node);
|
|
synchronize_order();
|
|
})
|
|
.start();
|
|
});
|
|
|
|
drop.start();
|
|
};
|
|
|
|
var build_list_controls = function(list_node) {
|
|
var nodes = JX.DOM.scry(list_node, 'div', 'pholio-drop-image');
|
|
for (var ii = 0; ii < nodes.length; ii++) {
|
|
build_update_control(nodes[ii]);
|
|
}
|
|
};
|
|
|
|
var build_update_control = function(node) {
|
|
var drop = build_drop_upload(node);
|
|
|
|
drop.listen('willUpload', function() {
|
|
JX.DOM.alterClass(node, 'pholio-replacing', true);
|
|
});
|
|
|
|
drop.listen('didUpload', function(file) {
|
|
var node_data = JX.Stratcom.getData(node);
|
|
|
|
var data = {
|
|
filePHID: file.getPHID(),
|
|
replacesPHID: node_data.replacesPHID || node_data.filePHID || null,
|
|
title: JX.DOM.find(node, 'input', 'image-title').value,
|
|
description: JX.DOM.find(node, 'textarea', 'image-description').value
|
|
};
|
|
|
|
new JX.Workflow(config.renderURI, data)
|
|
.setHandler(function(response) {
|
|
var new_node = JX.$H(response.markup).getFragment().firstChild;
|
|
build_update_control(new_node);
|
|
|
|
JX.DOM.replace(node, new_node);
|
|
JX.DOM.alterClass(node, 'pholio-replacing', false);
|
|
synchronize_order();
|
|
})
|
|
.start();
|
|
});
|
|
|
|
drop.start();
|
|
};
|
|
|
|
|
|
/* -( Rendering )---------------------------------------------------------- */
|
|
|
|
|
|
var render_uploading = function() {
|
|
return JX.$N(
|
|
'div',
|
|
{className: 'pholio-drop-uploading'},
|
|
pht('uploading'));
|
|
};
|
|
|
|
var render_undo = function() {
|
|
var link = JX.$N(
|
|
'a',
|
|
{href: '#', sigil: 'pholio-drop-undo'},
|
|
pht('undo'));
|
|
|
|
return JX.$N(
|
|
'div',
|
|
{className: 'pholio-drop-undo'},
|
|
[pht('removed'), ' ', link]);
|
|
};
|
|
|
|
|
|
/* -( Init )--------------------------------------------------------------- */
|
|
|
|
|
|
function update_statics(data, page_id, no_build) {
|
|
statics.nodes = {
|
|
list: JX.$(data.listID),
|
|
drop: JX.$(data.dropID),
|
|
order: JX.$(data.orderID)
|
|
};
|
|
|
|
if (!statics.mockEditCache[page_id]) {
|
|
statics.mockEditCache[page_id] = {};
|
|
}
|
|
statics.mockEditCache[page_id].config = config;
|
|
statics.mockEditCache[page_id].nodes = statics.nodes;
|
|
|
|
if (no_build !== true) {
|
|
build_add_control(statics.nodes.drop);
|
|
build_list_controls(statics.nodes.list);
|
|
statics.draglist = build_draglist(statics.nodes.list);
|
|
statics.mockEditCache[page_id].draglist = statics.draglist;
|
|
} else {
|
|
statics.draglist = statics.mockEditCache[page_id].draglist;
|
|
}
|
|
synchronize_order();
|
|
}
|
|
|
|
function install() {
|
|
statics.mockEditCache = {};
|
|
JX.Stratcom.listen('click', 'pholio-drop-remove', pholio_drop_remove);
|
|
JX.Stratcom.listen(
|
|
'quicksand-redraw',
|
|
null,
|
|
function (e) {
|
|
e.kill();
|
|
|
|
var data = e.getData();
|
|
if (!data.newResponse.mockEditConfig) {
|
|
return;
|
|
}
|
|
if (data.fromServer) {
|
|
// we ran update_statics(config) below already
|
|
} else {
|
|
var page_id = data.newResponseID;
|
|
var new_config = statics.mockEditCache[page_id].config;
|
|
update_statics(new_config, page_id, true);
|
|
}
|
|
});
|
|
return true;
|
|
}
|
|
|
|
statics.installed = statics.installed || install();
|
|
update_statics(config, JX.Quicksand.getCurrentPageID(), false);
|
|
});
|