mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-16 01:31:06 +01:00
Simplify form/lightbox stuff
Summary: I broke this a bit a few revisions ago by making `phabricator_render_csrf_magic()` return array instead of string without actually grepping for callsites. Instead of building a form in JS, build it in PHP and just change its action in JS. This is simpler and doesn't require us to expose CSRF magic in more places. This change triggered a rather subtle bug: if we rendered a normal form on the page (and thus installed the "disable forms when submitted" behavior), the download button would incorrectly disable after being clicked. This doesn't currently happen in Files because we don't put a form on the same page as the download button. Instead, make the "disable" behavior check if the form is downloading stuff, and not disable stuff if it is. Then mark the lightbox and Files form as both download forms. Test Plan: Used lightbox; used Files. Verified download buttons download the right stuff and behave correctly. Grepped for other download forms, but all the other places where we write "download" don't appear to actually be download links. Reviewers: vrana, btrahan Reviewed By: vrana CC: aran Maniphest Tasks: T2432 Differential Revision: https://secure.phabricator.com/D4685
This commit is contained in:
parent
fc4cb57357
commit
08bcb8c30a
7 changed files with 70 additions and 46 deletions
|
@ -72,6 +72,7 @@ final class PhabricatorFileInfoController extends PhabricatorFileController {
|
||||||
id(new PhabricatorActionView())
|
id(new PhabricatorActionView())
|
||||||
->setUser($user)
|
->setUser($user)
|
||||||
->setRenderAsForm(true)
|
->setRenderAsForm(true)
|
||||||
|
->setDownload(true)
|
||||||
->setName(pht('Download File'))
|
->setName(pht('Download File'))
|
||||||
->setIcon('download')
|
->setIcon('download')
|
||||||
->setHref($file->getViewURI()));
|
->setHref($file->getViewURI()));
|
||||||
|
|
|
@ -56,8 +56,8 @@ abstract class AphrontTagView extends AphrontView {
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
final public function getSigil() {
|
final public function getSigils() {
|
||||||
return $this->sigil;
|
return $this->sigils;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function addClass($class) {
|
public function addClass($class) {
|
||||||
|
|
|
@ -58,8 +58,6 @@ final class AphrontFormView extends AphrontView {
|
||||||
}
|
}
|
||||||
require_celerity_resource('aphront-form-view-css');
|
require_celerity_resource('aphront-form-view-css');
|
||||||
|
|
||||||
Javelin::initBehavior('aphront-form-disable-on-submit');
|
|
||||||
|
|
||||||
$layout = new AphrontFormLayoutView();
|
$layout = new AphrontFormLayoutView();
|
||||||
|
|
||||||
if (!$this->flexible) {
|
if (!$this->flexible) {
|
||||||
|
|
|
@ -8,6 +8,16 @@ final class PhabricatorActionView extends AphrontView {
|
||||||
private $disabled;
|
private $disabled;
|
||||||
private $workflow;
|
private $workflow;
|
||||||
private $renderAsForm;
|
private $renderAsForm;
|
||||||
|
private $download;
|
||||||
|
|
||||||
|
public function setDownload($download) {
|
||||||
|
$this->download = $download;
|
||||||
|
return $this;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getDownload() {
|
||||||
|
return $this->download;
|
||||||
|
}
|
||||||
|
|
||||||
public function setHref($href) {
|
public function setHref($href) {
|
||||||
$this->href = $href;
|
$this->href = $href;
|
||||||
|
@ -73,12 +83,20 @@ final class PhabricatorActionView extends AphrontView {
|
||||||
),
|
),
|
||||||
$this->name);
|
$this->name);
|
||||||
|
|
||||||
|
$sigils = array();
|
||||||
|
if ($this->workflow) {
|
||||||
|
$sigils[] = 'workflow';
|
||||||
|
}
|
||||||
|
if ($this->download) {
|
||||||
|
$sigils[] = 'download';
|
||||||
|
}
|
||||||
|
|
||||||
$item = phabricator_render_form(
|
$item = phabricator_render_form(
|
||||||
$this->user,
|
$this->user,
|
||||||
array(
|
array(
|
||||||
'action' => $this->href,
|
'action' => $this->href,
|
||||||
'method' => 'POST',
|
'method' => 'POST',
|
||||||
'sigil' => $this->workflow ? 'workflow' : null,
|
'sigil' => implode(' ', $sigils),
|
||||||
),
|
),
|
||||||
$item);
|
$item);
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -108,29 +108,47 @@ final class PhabricatorStandardPageView extends PhabricatorBarePageView {
|
||||||
|
|
||||||
Javelin::initBehavior('workflow', array());
|
Javelin::initBehavior('workflow', array());
|
||||||
|
|
||||||
$current_token = null;
|
|
||||||
$request = $this->getRequest();
|
$request = $this->getRequest();
|
||||||
|
$user = null;
|
||||||
if ($request) {
|
if ($request) {
|
||||||
$user = $request->getUser();
|
$user = $request->getUser();
|
||||||
if ($user) {
|
|
||||||
$current_token = $user->getCSRFToken();
|
|
||||||
$download_form = phabricator_render_form_magic($user);
|
|
||||||
$default_img_uri =
|
|
||||||
PhabricatorEnv::getCDNURI(
|
|
||||||
'/rsrc/image/icon/fatcow/document_black.png'
|
|
||||||
);
|
|
||||||
|
|
||||||
Javelin::initBehavior(
|
|
||||||
'lightbox-attachments',
|
|
||||||
array(
|
|
||||||
'defaultImageUri' => $default_img_uri,
|
|
||||||
'downloadForm' => $download_form,
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($user) {
|
||||||
|
$default_img_uri =
|
||||||
|
PhabricatorEnv::getCDNURI(
|
||||||
|
'/rsrc/image/icon/fatcow/document_black.png'
|
||||||
|
);
|
||||||
|
$download_form = phabricator_render_form(
|
||||||
|
$user,
|
||||||
|
array(
|
||||||
|
'action' => '#',
|
||||||
|
'method' => 'POST',
|
||||||
|
'class' => 'lightbox-download-form',
|
||||||
|
'sigil' => 'download',
|
||||||
|
),
|
||||||
|
phutil_tag(
|
||||||
|
'button',
|
||||||
|
array(),
|
||||||
|
pht('Download')));
|
||||||
|
|
||||||
|
Javelin::initBehavior(
|
||||||
|
'lightbox-attachments',
|
||||||
|
array(
|
||||||
|
'defaultImageUri' => $default_img_uri,
|
||||||
|
'downloadForm' => $download_form,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
Javelin::initBehavior('aphront-form-disable-on-submit');
|
||||||
Javelin::initBehavior('toggle-class', array());
|
Javelin::initBehavior('toggle-class', array());
|
||||||
Javelin::initBehavior('konami', array());
|
Javelin::initBehavior('konami', array());
|
||||||
|
|
||||||
|
$current_token = null;
|
||||||
|
if ($user) {
|
||||||
|
$current_token = $user->getCSRFToken();
|
||||||
|
}
|
||||||
|
|
||||||
Javelin::initBehavior(
|
Javelin::initBehavior(
|
||||||
'refresh-csrf',
|
'refresh-csrf',
|
||||||
array(
|
array(
|
||||||
|
@ -138,6 +156,7 @@ final class PhabricatorStandardPageView extends PhabricatorBarePageView {
|
||||||
'header' => AphrontRequest::getCSRFHeaderName(),
|
'header' => AphrontRequest::getCSRFHeaderName(),
|
||||||
'current' => $current_token,
|
'current' => $current_token,
|
||||||
));
|
));
|
||||||
|
|
||||||
Javelin::initBehavior('device');
|
Javelin::initBehavior('device');
|
||||||
|
|
||||||
if ($console) {
|
if ($console) {
|
||||||
|
|
|
@ -23,6 +23,12 @@ JX.behavior('aphront-form-disable-on-submit', function(config) {
|
||||||
|
|
||||||
root = e.getNode('tag:form');
|
root = e.getNode('tag:form');
|
||||||
|
|
||||||
|
// If the form is a "download" form, don't disable it on submit because
|
||||||
|
// we won't transition off the page.
|
||||||
|
if (JX.Stratcom.hasSigil(root, 'download')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Open the form to a new tab in Firefox explicitly (automatic in Chrome).
|
// Open the form to a new tab in Firefox explicitly (automatic in Chrome).
|
||||||
// We render some buttons as links so users may be confused that the links
|
// We render some buttons as links so users may be confused that the links
|
||||||
// don't open to new tabs with Ctrl+click as normal links.
|
// don't open to new tabs with Ctrl+click as normal links.
|
||||||
|
|
|
@ -15,7 +15,7 @@ JX.behavior('lightbox-attachments', function (config) {
|
||||||
var next = null;
|
var next = null;
|
||||||
var x_margin = 40;
|
var x_margin = 40;
|
||||||
var y_margin = 100;
|
var y_margin = 100;
|
||||||
var downloadForm = JX.$H(config.downloadForm);
|
var downloadForm = JX.$H(config.downloadForm).getFragment().firstChild;
|
||||||
|
|
||||||
function loadLightBox(e) {
|
function loadLightBox(e) {
|
||||||
if (!e.isNormalClick()) {
|
if (!e.isNormalClick()) {
|
||||||
|
@ -133,33 +133,11 @@ JX.behavior('lightbox-attachments', function (config) {
|
||||||
},
|
},
|
||||||
'Image '+current+' of '+total+'.'+extra_status
|
'Image '+current+' of '+total+'.'+extra_status
|
||||||
);
|
);
|
||||||
var form = JX.$N('form',
|
|
||||||
{
|
|
||||||
action : target_data.dUri,
|
|
||||||
method : 'POST',
|
|
||||||
className : 'lightbox-download-form'
|
|
||||||
},
|
|
||||||
downloadForm
|
|
||||||
);
|
|
||||||
JX.DOM.appendContent(form, JX.$N('button', {}, 'Download'));
|
|
||||||
JX.DOM.listen(form,
|
|
||||||
'click',
|
|
||||||
null,
|
|
||||||
function (e) {
|
|
||||||
e.prevent();
|
|
||||||
form.submit();
|
|
||||||
// Firefox and probably IE need this trick to work.
|
|
||||||
// Removing a form from the DOM while its submitting is
|
|
||||||
// tricky business.
|
|
||||||
setTimeout(JX.bind(null, closeLightBox, e), 0);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
var downloadSpan = JX.$N('span',
|
var downloadSpan = JX.$N('span',
|
||||||
{
|
{
|
||||||
className : 'lightbox-download'
|
className : 'lightbox-download'
|
||||||
},
|
});
|
||||||
form
|
|
||||||
);
|
|
||||||
var statusHTML = JX.$N('div',
|
var statusHTML = JX.$N('div',
|
||||||
{
|
{
|
||||||
className : 'lightbox-status'
|
className : 'lightbox-status'
|
||||||
|
@ -169,6 +147,10 @@ JX.behavior('lightbox-attachments', function (config) {
|
||||||
JX.DOM.appendContent(lightbox, statusHTML);
|
JX.DOM.appendContent(lightbox, statusHTML);
|
||||||
JX.DOM.alterClass(document.body, 'lightbox-attached', true);
|
JX.DOM.alterClass(document.body, 'lightbox-attached', true);
|
||||||
JX.Mask.show('jx-dark-mask');
|
JX.Mask.show('jx-dark-mask');
|
||||||
|
|
||||||
|
downloadForm.action = target_data.dUri;
|
||||||
|
downloadSpan.appendChild(downloadForm);
|
||||||
|
|
||||||
document.body.appendChild(lightbox);
|
document.body.appendChild(lightbox);
|
||||||
|
|
||||||
JX.Busy.start();
|
JX.Busy.start();
|
||||||
|
|
Loading…
Reference in a new issue