2011-01-16 22:51:39 +01:00
|
|
|
<?php
|
|
|
|
|
2012-03-14 00:21:04 +01:00
|
|
|
final class AphrontDialogView extends AphrontView {
|
2011-01-16 22:51:39 +01:00
|
|
|
|
|
|
|
private $title;
|
|
|
|
private $submitButton;
|
|
|
|
private $cancelURI;
|
2011-05-28 20:36:00 +02:00
|
|
|
private $cancelText = 'Cancel';
|
2011-01-16 22:51:39 +01:00
|
|
|
private $submitURI;
|
2011-02-02 01:42:36 +01:00
|
|
|
private $user;
|
|
|
|
private $hidden = array();
|
2011-02-17 07:14:09 +01:00
|
|
|
private $class;
|
2011-02-17 23:32:01 +01:00
|
|
|
private $renderAsForm = true;
|
|
|
|
private $formID;
|
2011-02-02 01:42:36 +01:00
|
|
|
|
2011-06-10 00:28:29 +02:00
|
|
|
private $width = 'default';
|
|
|
|
const WIDTH_DEFAULT = 'default';
|
|
|
|
const WIDTH_FORM = 'form';
|
|
|
|
|
2011-02-02 01:42:36 +01:00
|
|
|
public function setUser(PhabricatorUser $user) {
|
|
|
|
$this->user = $user;
|
|
|
|
return $this;
|
|
|
|
}
|
2011-01-16 22:51:39 +01:00
|
|
|
|
|
|
|
public function setSubmitURI($uri) {
|
|
|
|
$this->submitURI = $uri;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setTitle($title) {
|
|
|
|
$this->title = $title;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function getTitle() {
|
|
|
|
return $this->title;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function addSubmitButton($text = 'Okay') {
|
|
|
|
$this->submitButton = $text;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2011-05-28 20:36:00 +02:00
|
|
|
public function addCancelButton($uri, $text = 'Cancel') {
|
2011-01-16 22:51:39 +01:00
|
|
|
$this->cancelURI = $uri;
|
2011-05-28 20:36:00 +02:00
|
|
|
$this->cancelText = $text;
|
2011-01-16 22:51:39 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2011-02-02 01:42:36 +01:00
|
|
|
public function addHiddenInput($key, $value) {
|
Improve behavior when user submits a no-op action in Differential
Summary:
See T730 and the slightly-less-pretty version of this in D1398.
When a user takes an action in Differential that has no effect (for instance,
accepting an already-accepted revision), prompt them:
Action Has No Effect
You can not accept this revision because it has already been accepted.
Do you want to post the feedback anyway, as a normal comment?
[Cancel] [Post as Comment]
If they have no comment text, the dialog only says "Cancel".
I think this is probably the best way to balance all the concerns here -- it
might occasionally be a little annoying, but that should be rare, and it should
never be confusing (the current workflow is extremely confusing).
This also fixes the issue where you can add all sorts of CCs who are already
part of the revision, either explicitly or via mentions.
Test Plan:
Posted some has-effect and has-no-effect comments, made different
choices in the dialog, everything seems to work OK?
Reviewers: vrana, btrahan, jungejason
Reviewed By: vrana
CC: aran, vrana
Maniphest Tasks: T730
Differential Revision: https://secure.phabricator.com/D1403
2012-01-14 20:39:22 +01:00
|
|
|
if (is_array($value)) {
|
|
|
|
foreach ($value as $hidden_key => $hidden_value) {
|
|
|
|
$this->hidden[] = array($key.'['.$hidden_key.']', $hidden_value);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$this->hidden[] = array($key, $value);
|
|
|
|
}
|
2011-02-02 01:42:36 +01:00
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2011-02-17 07:14:09 +01:00
|
|
|
public function setClass($class) {
|
|
|
|
$this->class = $class;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2011-02-17 23:32:01 +01:00
|
|
|
public function setRenderDialogAsDiv() {
|
|
|
|
// TODO: This API is awkward.
|
|
|
|
$this->renderAsForm = false;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function setFormID($id) {
|
|
|
|
$this->formID = $id;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2011-06-10 00:28:29 +02:00
|
|
|
public function setWidth($width) {
|
|
|
|
$this->width = $width;
|
|
|
|
return $this;
|
|
|
|
}
|
|
|
|
|
2011-01-16 22:51:39 +01:00
|
|
|
final public function render() {
|
2011-01-25 20:31:40 +01:00
|
|
|
require_celerity_resource('aphront-dialog-view-css');
|
2011-01-16 22:51:39 +01:00
|
|
|
|
|
|
|
$buttons = array();
|
|
|
|
if ($this->submitButton) {
|
2011-05-10 20:24:06 +02:00
|
|
|
$buttons[] = javelin_render_tag(
|
2011-02-05 02:53:14 +01:00
|
|
|
'button',
|
|
|
|
array(
|
|
|
|
'name' => '__submit__',
|
|
|
|
'sigil' => '__default__',
|
|
|
|
),
|
|
|
|
phutil_escape_html($this->submitButton));
|
2011-01-16 22:51:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
if ($this->cancelURI) {
|
2011-02-02 01:42:36 +01:00
|
|
|
$buttons[] = javelin_render_tag(
|
2011-01-16 22:51:39 +01:00
|
|
|
'a',
|
|
|
|
array(
|
|
|
|
'href' => $this->cancelURI,
|
|
|
|
'class' => 'button grey',
|
2011-02-02 01:42:36 +01:00
|
|
|
'name' => '__cancel__',
|
|
|
|
'sigil' => 'jx-workflow-button',
|
2011-01-16 22:51:39 +01:00
|
|
|
),
|
2011-05-28 20:36:00 +02:00
|
|
|
phutil_escape_html($this->cancelText));
|
2011-01-16 22:51:39 +01:00
|
|
|
}
|
2011-02-17 23:32:01 +01:00
|
|
|
$buttons = implode('', $buttons);
|
2011-01-16 22:51:39 +01:00
|
|
|
|
2011-02-02 01:42:36 +01:00
|
|
|
if (!$this->user) {
|
|
|
|
throw new Exception(
|
|
|
|
"You must call setUser() when rendering an AphrontDialogView.");
|
|
|
|
}
|
2011-02-17 23:32:01 +01:00
|
|
|
|
|
|
|
$more = $this->class;
|
|
|
|
|
2011-06-10 00:28:29 +02:00
|
|
|
switch ($this->width) {
|
|
|
|
case self::WIDTH_FORM:
|
|
|
|
$more .= ' aphront-dialog-view-width-'.$this->width;
|
|
|
|
break;
|
|
|
|
case self::WIDTH_DEFAULT:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
throw new Exception("Unknown dialog width '{$this->width}'!");
|
|
|
|
}
|
|
|
|
|
2011-02-17 23:32:01 +01:00
|
|
|
$attributes = array(
|
|
|
|
'class' => 'aphront-dialog-view '.$more,
|
|
|
|
'sigil' => 'jx-dialog',
|
|
|
|
);
|
|
|
|
|
|
|
|
$form_attributes = array(
|
|
|
|
'action' => $this->submitURI,
|
|
|
|
'method' => 'post',
|
|
|
|
'id' => $this->formID,
|
|
|
|
);
|
2011-02-02 01:42:36 +01:00
|
|
|
|
|
|
|
$hidden_inputs = array();
|
2011-02-17 23:32:01 +01:00
|
|
|
foreach ($this->hidden as $desc) {
|
|
|
|
list($key, $value) = $desc;
|
|
|
|
$hidden_inputs[] = javelin_render_tag(
|
2011-02-02 01:42:36 +01:00
|
|
|
'input',
|
|
|
|
array(
|
|
|
|
'type' => 'hidden',
|
|
|
|
'name' => $key,
|
|
|
|
'value' => $value,
|
2011-02-17 23:32:01 +01:00
|
|
|
'sigil' => 'aphront-dialog-application-input'
|
2011-02-02 01:42:36 +01:00
|
|
|
));
|
|
|
|
}
|
|
|
|
$hidden_inputs = implode("\n", $hidden_inputs);
|
2011-02-17 23:32:01 +01:00
|
|
|
$hidden_inputs =
|
|
|
|
'<input type="hidden" name="__dialog__" value="1" />'.
|
|
|
|
$hidden_inputs;
|
2011-02-02 01:42:36 +01:00
|
|
|
|
2011-02-17 07:14:09 +01:00
|
|
|
|
2011-02-17 23:32:01 +01:00
|
|
|
if (!$this->renderAsForm) {
|
|
|
|
$buttons = phabricator_render_form(
|
|
|
|
$this->user,
|
|
|
|
$form_attributes,
|
|
|
|
$hidden_inputs.$buttons);
|
|
|
|
}
|
|
|
|
|
|
|
|
$content =
|
2011-01-16 22:51:39 +01:00
|
|
|
'<div class="aphront-dialog-head">'.
|
|
|
|
phutil_escape_html($this->title).
|
|
|
|
'</div>'.
|
|
|
|
'<div class="aphront-dialog-body">'.
|
|
|
|
$this->renderChildren().
|
|
|
|
'</div>'.
|
|
|
|
'<div class="aphront-dialog-tail">'.
|
2011-02-17 23:32:01 +01:00
|
|
|
$buttons.
|
2011-01-16 22:51:39 +01:00
|
|
|
'<div style="clear: both;"></div>'.
|
2011-02-17 23:32:01 +01:00
|
|
|
'</div>';
|
|
|
|
|
|
|
|
if ($this->renderAsForm) {
|
|
|
|
return phabricator_render_form(
|
|
|
|
$this->user,
|
|
|
|
$form_attributes + $attributes,
|
|
|
|
$hidden_inputs.
|
|
|
|
$content);
|
|
|
|
} else {
|
|
|
|
return javelin_render_tag(
|
|
|
|
'div',
|
|
|
|
$attributes,
|
|
|
|
$content);
|
|
|
|
}
|
2011-01-16 22:51:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|