Allow datetime inputs to be optional

Summary:
Fixes T3279. For ApplicationSearch (and in some other cases) I'd like users to be able to provide an optional date. This isn't currently possible.

Add a checkbox which disables or enables the input.

Test Plan: Used UIExample to enter dates. Used Calendar to enter dates.

Reviewers: chad, btrahan

Reviewed By: chad

CC: aran

Maniphest Tasks: T3279

Differential Revision: https://secure.phabricator.com/D6082
This commit is contained in:
epriestley
2013-05-30 16:19:43 -07:00
parent 6637acb3e4
commit 01a6d580ac
6 changed files with 203 additions and 87 deletions

View File

@@ -800,7 +800,7 @@ celerity_register_resource_map(array(
),
'aphront-form-view-css' =>
array(
'uri' => '/res/656ca1a3/rsrc/css/aphront/form-view.css',
'uri' => '/res/4fe4c174/rsrc/css/aphront/form-view.css',
'type' => 'css',
'requires' =>
array(
@@ -1641,7 +1641,7 @@ celerity_register_resource_map(array(
),
'javelin-behavior-fancy-datepicker' =>
array(
'uri' => '/res/f5c13e07/rsrc/js/core/behavior-fancy-datepicker.js',
'uri' => '/res/dcd7c2ca/rsrc/js/core/behavior-fancy-datepicker.js',
'type' => 'js',
'requires' =>
array(
@@ -3657,7 +3657,7 @@ celerity_register_resource_map(array(
),
'phui-form-css' =>
array(
'uri' => '/res/ba4d7995/rsrc/css/phui/phui-form.css',
'uri' => '/res/eb478e83/rsrc/css/phui/phui-form.css',
'type' => 'css',
'requires' =>
array(
@@ -3973,7 +3973,7 @@ celerity_register_resource_map(array(
), array(
'packages' =>
array(
'b7428f7c' =>
'1b14560c' =>
array(
'name' => 'core.pkg.css',
'symbols' =>
@@ -4022,7 +4022,7 @@ celerity_register_resource_map(array(
41 => 'phabricator-property-list-view-css',
42 => 'phabricator-tag-view-css',
),
'uri' => '/res/pkg/b7428f7c/core.pkg.css',
'uri' => '/res/pkg/1b14560c/core.pkg.css',
'type' => 'css',
),
'98f60e3f' =>
@@ -4216,16 +4216,16 @@ celerity_register_resource_map(array(
'reverse' =>
array(
'aphront-attached-file-view-css' => '6b1fccc6',
'aphront-dialog-view-css' => 'b7428f7c',
'aphront-error-view-css' => 'b7428f7c',
'aphront-form-view-css' => 'b7428f7c',
'aphront-list-filter-view-css' => 'b7428f7c',
'aphront-pager-view-css' => 'b7428f7c',
'aphront-panel-view-css' => 'b7428f7c',
'aphront-table-view-css' => 'b7428f7c',
'aphront-tokenizer-control-css' => 'b7428f7c',
'aphront-tooltip-css' => 'b7428f7c',
'aphront-typeahead-control-css' => 'b7428f7c',
'aphront-dialog-view-css' => '1b14560c',
'aphront-error-view-css' => '1b14560c',
'aphront-form-view-css' => '1b14560c',
'aphront-list-filter-view-css' => '1b14560c',
'aphront-pager-view-css' => '1b14560c',
'aphront-panel-view-css' => '1b14560c',
'aphront-table-view-css' => '1b14560c',
'aphront-tokenizer-control-css' => '1b14560c',
'aphront-tooltip-css' => '1b14560c',
'aphront-typeahead-control-css' => '1b14560c',
'differential-changeset-view-css' => 'dd27a69b',
'differential-core-view-css' => 'dd27a69b',
'differential-inline-comment-editor' => '9488bb69',
@@ -4239,7 +4239,7 @@ celerity_register_resource_map(array(
'differential-table-of-contents-css' => 'dd27a69b',
'diffusion-commit-view-css' => 'c8ce2d88',
'diffusion-icons-css' => 'c8ce2d88',
'global-drag-and-drop-css' => 'b7428f7c',
'global-drag-and-drop-css' => '1b14560c',
'inline-comment-summary-css' => 'dd27a69b',
'javelin-aphlict' => '98f60e3f',
'javelin-behavior' => 'c1359b5d',
@@ -4313,56 +4313,56 @@ celerity_register_resource_map(array(
'javelin-util' => 'c1359b5d',
'javelin-vector' => 'c1359b5d',
'javelin-workflow' => 'c1359b5d',
'lightbox-attachment-css' => 'b7428f7c',
'lightbox-attachment-css' => '1b14560c',
'maniphest-task-summary-css' => '6b1fccc6',
'maniphest-transaction-detail-css' => '6b1fccc6',
'phabricator-action-list-view-css' => 'b7428f7c',
'phabricator-application-launch-view-css' => 'b7428f7c',
'phabricator-action-list-view-css' => '1b14560c',
'phabricator-application-launch-view-css' => '1b14560c',
'phabricator-busy' => '98f60e3f',
'phabricator-content-source-view-css' => 'dd27a69b',
'phabricator-core-buttons-css' => 'b7428f7c',
'phabricator-core-css' => 'b7428f7c',
'phabricator-crumbs-view-css' => 'b7428f7c',
'phabricator-directory-css' => 'b7428f7c',
'phabricator-core-buttons-css' => '1b14560c',
'phabricator-core-css' => '1b14560c',
'phabricator-crumbs-view-css' => '1b14560c',
'phabricator-directory-css' => '1b14560c',
'phabricator-drag-and-drop-file-upload' => '9488bb69',
'phabricator-dropdown-menu' => '98f60e3f',
'phabricator-file-upload' => '98f60e3f',
'phabricator-filetree-view-css' => 'b7428f7c',
'phabricator-flag-css' => 'b7428f7c',
'phabricator-form-view-css' => 'b7428f7c',
'phabricator-header-view-css' => 'b7428f7c',
'phabricator-filetree-view-css' => '1b14560c',
'phabricator-flag-css' => '1b14560c',
'phabricator-form-view-css' => '1b14560c',
'phabricator-header-view-css' => '1b14560c',
'phabricator-hovercard' => '98f60e3f',
'phabricator-jump-nav' => 'b7428f7c',
'phabricator-jump-nav' => '1b14560c',
'phabricator-keyboard-shortcut' => '98f60e3f',
'phabricator-keyboard-shortcut-manager' => '98f60e3f',
'phabricator-main-menu-view' => 'b7428f7c',
'phabricator-main-menu-view' => '1b14560c',
'phabricator-menu-item' => '98f60e3f',
'phabricator-nav-view-css' => 'b7428f7c',
'phabricator-nav-view-css' => '1b14560c',
'phabricator-notification' => '98f60e3f',
'phabricator-notification-css' => 'b7428f7c',
'phabricator-notification-menu-css' => 'b7428f7c',
'phabricator-object-item-list-view-css' => 'b7428f7c',
'phabricator-notification-css' => '1b14560c',
'phabricator-notification-menu-css' => '1b14560c',
'phabricator-object-item-list-view-css' => '1b14560c',
'phabricator-object-selector-css' => 'dd27a69b',
'phabricator-phtize' => '98f60e3f',
'phabricator-prefab' => '98f60e3f',
'phabricator-project-tag-css' => '6b1fccc6',
'phabricator-property-list-view-css' => 'b7428f7c',
'phabricator-remarkup-css' => 'b7428f7c',
'phabricator-property-list-view-css' => '1b14560c',
'phabricator-remarkup-css' => '1b14560c',
'phabricator-shaped-request' => '9488bb69',
'phabricator-side-menu-view-css' => 'b7428f7c',
'phabricator-standard-page-view' => 'b7428f7c',
'phabricator-tag-view-css' => 'b7428f7c',
'phabricator-side-menu-view-css' => '1b14560c',
'phabricator-standard-page-view' => '1b14560c',
'phabricator-tag-view-css' => '1b14560c',
'phabricator-textareautils' => '98f60e3f',
'phabricator-tooltip' => '98f60e3f',
'phabricator-transaction-view-css' => 'b7428f7c',
'phabricator-zindex-css' => 'b7428f7c',
'phui-form-css' => 'b7428f7c',
'phui-icon-view-css' => 'b7428f7c',
'spacing-css' => 'b7428f7c',
'sprite-apps-large-css' => 'b7428f7c',
'sprite-gradient-css' => 'b7428f7c',
'sprite-icons-css' => 'b7428f7c',
'sprite-menu-css' => 'b7428f7c',
'syntax-highlighting-css' => 'b7428f7c',
'phabricator-transaction-view-css' => '1b14560c',
'phabricator-zindex-css' => '1b14560c',
'phui-form-css' => '1b14560c',
'phui-icon-view-css' => '1b14560c',
'spacing-css' => '1b14560c',
'sprite-apps-large-css' => '1b14560c',
'sprite-gradient-css' => '1b14560c',
'sprite-icons-css' => '1b14560c',
'sprite-menu-css' => '1b14560c',
'syntax-highlighting-css' => '1b14560c',
),
));

View File

@@ -19,20 +19,31 @@ final class PhabricatorFormExample extends PhabricatorUIExample {
->setName('start')
->setLabel('Start')
->setInitialTime(AphrontFormDateControl::TIME_START_OF_BUSINESS);
$start_value = $start_time->readValueFromRequest($request);
$end_time = id(new AphrontFormDateControl())
->setUser($user)
->setName('end')
->setLabel('End')
->setInitialTime(AphrontFormDateControl::TIME_END_OF_BUSINESS);
$end_value = $end_time->readValueFromRequest($request);
$null_time = id(new AphrontFormDateControl())
->setUser($user)
->setName('nulltime')
->setLabel('Nullable')
->setAllowNull(true);
if ($request->isFormPost()) {
$start_value = $start_time->readValueFromRequest($request);
$end_value = $end_time->readValueFromRequest($request);
$null_value = $null_time->readValueFromRequest($request);
}
$form = id(new AphrontFormView())
->setUser($user)
->setFlexible(true)
->appendChild($start_time)
->appendChild($end_time)
->appendChild($null_time)
->appendChild(
id(new AphrontFormSubmitControl())
->setValue('Submit'));

View File

@@ -3,11 +3,18 @@
final class AphrontFormDateControl extends AphrontFormControl {
private $initialTime;
private $zone;
private $valueDay;
private $valueMonth;
private $valueYear;
private $valueTime;
private $allowNull;
public function setAllowNull($allow_null) {
$this->allowNull = $allow_null;
return $this;
}
const TIME_START_OF_DAY = 'start-of-day';
const TIME_END_OF_DAY = 'end-of-day';
@@ -20,19 +27,17 @@ final class AphrontFormDateControl extends AphrontFormControl {
}
public function readValueFromRequest(AphrontRequest $request) {
$user = $this->user;
if (!$this->user) {
throw new Exception(
pht("Call setUser() before readValueFromRequest()!"));
}
$user_zone = $user->getTimezoneIdentifier();
$zone = new DateTimeZone($user_zone);
$day = $request->getInt($this->getDayInputName());
$month = $request->getInt($this->getMonthInputName());
$year = $request->getInt($this->getYearInputName());
$time = $request->getStr($this->getTimeInputName());
$enabled = $request->getBool($this->getCheckboxInputName());
if ($this->allowNull && !$enabled) {
$this->setError(null);
$this->setValue(null);
return;
}
$err = $this->getError();
@@ -45,6 +50,8 @@ final class AphrontFormDateControl extends AphrontFormControl {
// Assume invalid.
$err = 'Invalid';
$zone = $this->getTimezone();
try {
$date = new DateTime("{$year}-{$month}-{$day} {$time}", $zone);
$value = $date->format('U');
@@ -59,32 +66,7 @@ final class AphrontFormDateControl extends AphrontFormControl {
$this->setValue(null);
}
} else {
// TODO: We could eventually allow these to be customized per install or
// per user or both, but let's wait and see.
switch ($this->initialTime) {
case self::TIME_START_OF_DAY:
default:
$time = '12:00 AM';
break;
case self::TIME_START_OF_BUSINESS:
$time = '9:00 AM';
break;
case self::TIME_END_OF_BUSINESS:
$time = '5:00 PM';
break;
case self::TIME_END_OF_DAY:
$time = '11:59 PM';
break;
}
$today = $this->formatTime(time(), 'Y-m-d');
try {
$date = new DateTime("{$today} {$time}", $zone);
$value = $date->format('U');
} catch (Exception $ex) {
$value = null;
}
$value = $this->getInitialValue();
if ($value) {
$this->setValue($value);
} else {
@@ -176,7 +158,20 @@ final class AphrontFormDateControl extends AphrontFormControl {
return $this->getName().'_t';
}
private function getCheckboxInputName() {
return $this->getName().'_e';
}
protected function renderInput() {
$disabled = null;
if ($this->getValue() === null) {
$this->setValue($this->getInitialValue());
if ($this->allowNull) {
$disabled = 'disabled';
}
}
$min_year = $this->getMinYear();
$max_year = $this->getMaxYear();
@@ -198,6 +193,20 @@ final class AphrontFormDateControl extends AphrontFormControl {
12 => pht('Dec'),
);
$checkbox = null;
if ($this->allowNull) {
$checkbox = javelin_tag(
'input',
array(
'type' => 'checkbox',
'name' => $this->getCheckboxInputName(),
'sigil' => 'calendar-enable',
'class' => 'aphront-form-date-enabled-input',
'value' => 1,
'checked' => ($disabled === null ? 'checked' : null),
));
}
$years = range($this->getMinYear(), $this->getMaxYear());
$years = array_fuse($years);
@@ -207,6 +216,7 @@ final class AphrontFormDateControl extends AphrontFormControl {
array(
'name' => $this->getDayInputName(),
'sigil' => 'day-input',
'disabled' => $disabled,
));
$months_sel = AphrontFormSelectControl::renderSelectTag(
@@ -215,6 +225,7 @@ final class AphrontFormDateControl extends AphrontFormControl {
array(
'name' => $this->getMonthInputName(),
'sigil' => 'month-input',
'disabled' => $disabled,
));
$years_sel = AphrontFormSelectControl::renderSelectTag(
@@ -223,6 +234,7 @@ final class AphrontFormDateControl extends AphrontFormControl {
array(
'name' => $this->getYearInputName(),
'sigil' => 'year-input',
'disabled' => $disabled,
));
$cal_icon = javelin_tag(
@@ -234,7 +246,7 @@ final class AphrontFormDateControl extends AphrontFormControl {
),
'');
$time_sel = phutil_tag(
$time_sel = javelin_tag(
'input',
array(
'name' => $this->getTimeInputName(),
@@ -242,6 +254,7 @@ final class AphrontFormDateControl extends AphrontFormControl {
'value' => $this->getTimeInputValue(),
'type' => 'text',
'class' => 'aphront-form-date-time-input',
'disabled' => $disabled,
),
'');
@@ -252,8 +265,12 @@ final class AphrontFormDateControl extends AphrontFormControl {
array(
'class' => 'aphront-form-date-container',
'sigil' => 'phabricator-date-control',
'meta' => array(
'disabled' => (bool)$disabled,
),
),
array(
$checkbox,
$days_sel,
$months_sel,
$years_sel,
@@ -262,4 +279,51 @@ final class AphrontFormDateControl extends AphrontFormControl {
));
}
private function getTimezone() {
if ($this->zone) {
return $this->zone;
}
$user = $this->getUser();
if (!$this->getUser()) {
throw new Exception("Call setUser() before getTimezone()!");
}
$user_zone = $user->getTimezoneIdentifier();
$this->zone = new DateTimeZone($user_zone);
return $this->zone;
}
private function getInitialValue() {
$zone = $this->getTimezone();
// TODO: We could eventually allow these to be customized per install or
// per user or both, but let's wait and see.
switch ($this->initialTime) {
case self::TIME_START_OF_DAY:
default:
$time = '12:00 AM';
break;
case self::TIME_START_OF_BUSINESS:
$time = '9:00 AM';
break;
case self::TIME_END_OF_BUSINESS:
$time = '5:00 PM';
break;
case self::TIME_END_OF_DAY:
$time = '11:59 PM';
break;
}
$today = $this->formatTime(time(), 'Y-m-d');
try {
$date = new DateTime("{$today} {$time}", $zone);
$value = $date->format('U');
} catch (Exception $ex) {
$value = null;
}
return $value;
}
}