1
0
Fork 0
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:
epriestley 2013-01-28 18:12:09 -08:00
parent fc4cb57357
commit 08bcb8c30a
7 changed files with 70 additions and 46 deletions

View file

@ -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()));

View file

@ -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) {

View file

@ -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) {

View file

@ -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 {

View file

@ -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) {

View file

@ -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.

View file

@ -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();