mirror of
https://we.phorge.it/source/phorge.git
synced 2024-12-20 20:40:56 +01:00
Implemented support for different export formats.
Summary: This adds support for different export formats to Excel via a drop-down on the Export page as per the discussion in D5443. Test Plan: Export some things from Maniphest. Do a simple implementation of ManiphestExcelFormat just for testing and make sure that it appears in the list after you run `arc liberate`. Reviewers: epriestley Reviewed By: epriestley CC: aran, Korvin Maniphest Tasks: T2575 Differential Revision: https://secure.phabricator.com/D5642 Conflicts: src/applications/maniphest/controller/ManiphestExportController.php
This commit is contained in:
parent
bdef566f2e
commit
8ba593b3f3
4 changed files with 214 additions and 120 deletions
|
@ -621,6 +621,8 @@ phutil_register_library_map(array(
|
||||||
'ManiphestDAO' => 'applications/maniphest/storage/ManiphestDAO.php',
|
'ManiphestDAO' => 'applications/maniphest/storage/ManiphestDAO.php',
|
||||||
'ManiphestDefaultTaskExtensions' => 'applications/maniphest/extensions/ManiphestDefaultTaskExtensions.php',
|
'ManiphestDefaultTaskExtensions' => 'applications/maniphest/extensions/ManiphestDefaultTaskExtensions.php',
|
||||||
'ManiphestEdgeEventListener' => 'applications/maniphest/event/ManiphestEdgeEventListener.php',
|
'ManiphestEdgeEventListener' => 'applications/maniphest/event/ManiphestEdgeEventListener.php',
|
||||||
|
'ManiphestExcelDefaultFormat' => 'applications/maniphest/export/ManiphestExcelDefaultFormat.php',
|
||||||
|
'ManiphestExcelFormat' => 'applications/maniphest/export/ManiphestExcelFormat.php',
|
||||||
'ManiphestExportController' => 'applications/maniphest/controller/ManiphestExportController.php',
|
'ManiphestExportController' => 'applications/maniphest/controller/ManiphestExportController.php',
|
||||||
'ManiphestHovercardEventListener' => 'applications/maniphest/event/ManiphestHovercardEventListener.php',
|
'ManiphestHovercardEventListener' => 'applications/maniphest/event/ManiphestHovercardEventListener.php',
|
||||||
'ManiphestPeopleMenuEventListener' => 'applications/maniphest/event/ManiphestPeopleMenuEventListener.php',
|
'ManiphestPeopleMenuEventListener' => 'applications/maniphest/event/ManiphestPeopleMenuEventListener.php',
|
||||||
|
@ -2302,6 +2304,7 @@ phutil_register_library_map(array(
|
||||||
'ManiphestDAO' => 'PhabricatorLiskDAO',
|
'ManiphestDAO' => 'PhabricatorLiskDAO',
|
||||||
'ManiphestDefaultTaskExtensions' => 'ManiphestTaskExtensions',
|
'ManiphestDefaultTaskExtensions' => 'ManiphestTaskExtensions',
|
||||||
'ManiphestEdgeEventListener' => 'PhutilEventListener',
|
'ManiphestEdgeEventListener' => 'PhutilEventListener',
|
||||||
|
'ManiphestExcelDefaultFormat' => 'ManiphestExcelFormat',
|
||||||
'ManiphestExportController' => 'ManiphestController',
|
'ManiphestExportController' => 'ManiphestController',
|
||||||
'ManiphestHovercardEventListener' => 'PhutilEventListener',
|
'ManiphestHovercardEventListener' => 'PhutilEventListener',
|
||||||
'ManiphestPeopleMenuEventListener' => 'PhutilEventListener',
|
'ManiphestPeopleMenuEventListener' => 'PhutilEventListener',
|
||||||
|
|
|
@ -62,6 +62,12 @@ final class ManiphestExportController extends ManiphestController {
|
||||||
return new Aphront404Response();
|
return new Aphront404Response();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$formats = ManiphestExcelFormat::loadAllFormats();
|
||||||
|
$export_formats = array();
|
||||||
|
foreach ($formats as $format_class => $format_object) {
|
||||||
|
$export_formats[$format_class] = $format_object->getName();
|
||||||
|
}
|
||||||
|
|
||||||
if (!$request->isDialogFormPost()) {
|
if (!$request->isDialogFormPost()) {
|
||||||
$dialog = new AphrontDialogView();
|
$dialog = new AphrontDialogView();
|
||||||
$dialog->setUser($user);
|
$dialog->setUser($user);
|
||||||
|
@ -70,12 +76,26 @@ final class ManiphestExportController extends ManiphestController {
|
||||||
$dialog->appendChild(phutil_tag('p', array(), pht(
|
$dialog->appendChild(phutil_tag('p', array(), pht(
|
||||||
'Do you want to export the query results to Excel?')));
|
'Do you want to export the query results to Excel?')));
|
||||||
|
|
||||||
|
$form = id(new AphrontFormLayoutView())
|
||||||
|
->appendChild(
|
||||||
|
id(new AphrontFormSelectControl())
|
||||||
|
->setLabel(pht('Format:'))
|
||||||
|
->setName("excel-format")
|
||||||
|
->setOptions($export_formats));
|
||||||
|
|
||||||
|
$dialog->appendChild($form);
|
||||||
|
|
||||||
$dialog->addCancelButton('/maniphest/');
|
$dialog->addCancelButton('/maniphest/');
|
||||||
$dialog->addSubmitButton(pht('Export to Excel'));
|
$dialog->addSubmitButton(pht('Export to Excel'));
|
||||||
return id(new AphrontDialogResponse())->setDialog($dialog);
|
return id(new AphrontDialogResponse())->setDialog($dialog);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$format = idx($formats, $request->getStr("excel-format"));
|
||||||
|
if ($format === null) {
|
||||||
|
throw new Exception('Excel format object not found.');
|
||||||
|
}
|
||||||
|
|
||||||
$query->setParameter('limit', null);
|
$query->setParameter('limit', null);
|
||||||
$query->setParameter('offset', null);
|
$query->setParameter('offset', null);
|
||||||
$query->setParameter('order', 'p');
|
$query->setParameter('order', 'p');
|
||||||
|
@ -92,114 +112,7 @@ final class ManiphestExportController extends ManiphestController {
|
||||||
$handles += $project_handles;
|
$handles += $project_handles;
|
||||||
|
|
||||||
$workbook = new PHPExcel();
|
$workbook = new PHPExcel();
|
||||||
$sheet = $workbook->setActiveSheetIndex(0);
|
$format->buildWorkbook($workbook, $tasks, $handles, $user);
|
||||||
$sheet->setTitle(pht('Tasks'));
|
|
||||||
|
|
||||||
$widths = array(
|
|
||||||
null,
|
|
||||||
15,
|
|
||||||
null,
|
|
||||||
10,
|
|
||||||
15,
|
|
||||||
15,
|
|
||||||
60,
|
|
||||||
30,
|
|
||||||
20,
|
|
||||||
100,
|
|
||||||
);
|
|
||||||
|
|
||||||
foreach ($widths as $col => $width) {
|
|
||||||
if ($width !== null) {
|
|
||||||
$sheet->getColumnDimension($this->col($col))->setWidth($width);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$status_map = ManiphestTaskStatus::getTaskStatusMap();
|
|
||||||
$pri_map = ManiphestTaskPriority::getTaskPriorityMap();
|
|
||||||
|
|
||||||
$date_format = null;
|
|
||||||
|
|
||||||
$rows = array();
|
|
||||||
$rows[] = array(
|
|
||||||
pht('ID'),
|
|
||||||
pht('Owner'),
|
|
||||||
pht('Status'),
|
|
||||||
pht('Priority'),
|
|
||||||
pht('Date Created'),
|
|
||||||
pht('Date Updated'),
|
|
||||||
pht('Title'),
|
|
||||||
pht('Projects'),
|
|
||||||
pht('URI'),
|
|
||||||
pht('Description'),
|
|
||||||
);
|
|
||||||
|
|
||||||
$is_date = array(
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
true,
|
|
||||||
true,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
false,
|
|
||||||
);
|
|
||||||
|
|
||||||
$header_format = array(
|
|
||||||
'font' => array(
|
|
||||||
'bold' => true,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
foreach ($tasks as $task) {
|
|
||||||
$task_owner = null;
|
|
||||||
if ($task->getOwnerPHID()) {
|
|
||||||
$task_owner = $handles[$task->getOwnerPHID()]->getName();
|
|
||||||
}
|
|
||||||
|
|
||||||
$projects = array();
|
|
||||||
foreach ($task->getProjectPHIDs() as $phid) {
|
|
||||||
$projects[] = $handles[$phid]->getName();
|
|
||||||
}
|
|
||||||
$projects = implode(', ', $projects);
|
|
||||||
|
|
||||||
$rows[] = array(
|
|
||||||
'T'.$task->getID(),
|
|
||||||
$task_owner,
|
|
||||||
idx($status_map, $task->getStatus(), '?'),
|
|
||||||
idx($pri_map, $task->getPriority(), '?'),
|
|
||||||
$this->computeExcelDate($task->getDateCreated()),
|
|
||||||
$this->computeExcelDate($task->getDateModified()),
|
|
||||||
$task->getTitle(),
|
|
||||||
$projects,
|
|
||||||
PhabricatorEnv::getProductionURI('/T'.$task->getID()),
|
|
||||||
phutil_utf8_shorten($task->getDescription(), 512),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
foreach ($rows as $row => $cols) {
|
|
||||||
foreach ($cols as $col => $spec) {
|
|
||||||
$cell_name = $this->col($col).($row + 1);
|
|
||||||
$cell = $sheet
|
|
||||||
->setCellValue($cell_name, $spec, $return_cell = true);
|
|
||||||
|
|
||||||
if ($row == 0) {
|
|
||||||
$sheet->getStyle($cell_name)->applyFromArray($header_format);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($is_date[$col]) {
|
|
||||||
$code = PHPExcel_Style_NumberFormat::FORMAT_DATE_YYYYMMDD2;
|
|
||||||
$sheet
|
|
||||||
->getStyle($cell_name)
|
|
||||||
->getNumberFormat()
|
|
||||||
->setFormatCode($code);
|
|
||||||
} else {
|
|
||||||
$cell->setDataType(PHPExcel_Cell_DataType::TYPE_STRING);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$writer = PHPExcel_IOFactory::createWriter($workbook, 'Excel2007');
|
$writer = PHPExcel_IOFactory::createWriter($workbook, 'Excel2007');
|
||||||
|
|
||||||
ob_start();
|
ob_start();
|
||||||
|
@ -210,19 +123,8 @@ final class ManiphestExportController extends ManiphestController {
|
||||||
|
|
||||||
return id(new AphrontFileResponse())
|
return id(new AphrontFileResponse())
|
||||||
->setMimeType($mime)
|
->setMimeType($mime)
|
||||||
->setDownload('maniphest_tasks_'.date('Ymd').'.xlsx')
|
->setDownload($format->getFileName().'.xlsx')
|
||||||
->setContent($data);
|
->setContent($data);
|
||||||
}
|
}
|
||||||
|
|
||||||
private function computeExcelDate($epoch) {
|
|
||||||
$seconds_per_day = (60 * 60 * 24);
|
|
||||||
$offset = ($seconds_per_day * 25569);
|
|
||||||
|
|
||||||
return ($epoch + $offset) / $seconds_per_day;
|
|
||||||
}
|
|
||||||
|
|
||||||
private function col($n) {
|
|
||||||
return chr(ord('A') + $n);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,142 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @group maniphest
|
||||||
|
*/
|
||||||
|
final class ManiphestExcelDefaultFormat extends ManiphestExcelFormat {
|
||||||
|
|
||||||
|
public function getName() {
|
||||||
|
return pht('Default');
|
||||||
|
}
|
||||||
|
|
||||||
|
public function getFileName() {
|
||||||
|
return 'maniphest_tasks_'.date('Ymd');
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @phutil-external-symbol class PHPExcel
|
||||||
|
* @phutil-external-symbol class PHPExcel_IOFactory
|
||||||
|
* @phutil-external-symbol class PHPExcel_Style_NumberFormat
|
||||||
|
* @phutil-external-symbol class PHPExcel_Cell_DataType
|
||||||
|
*/
|
||||||
|
public function buildWorkbook(
|
||||||
|
PHPExcel $workbook,
|
||||||
|
array $tasks,
|
||||||
|
array $handles,
|
||||||
|
PhabricatorUser $user) {
|
||||||
|
|
||||||
|
$sheet = $workbook->setActiveSheetIndex(0);
|
||||||
|
$sheet->setTitle(pht('Tasks'));
|
||||||
|
|
||||||
|
$widths = array(
|
||||||
|
null,
|
||||||
|
15,
|
||||||
|
null,
|
||||||
|
10,
|
||||||
|
15,
|
||||||
|
15,
|
||||||
|
60,
|
||||||
|
30,
|
||||||
|
20,
|
||||||
|
100,
|
||||||
|
);
|
||||||
|
|
||||||
|
foreach ($widths as $col => $width) {
|
||||||
|
if ($width !== null) {
|
||||||
|
$sheet->getColumnDimension($this->col($col))->setWidth($width);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$status_map = ManiphestTaskStatus::getTaskStatusMap();
|
||||||
|
$pri_map = ManiphestTaskPriority::getTaskPriorityMap();
|
||||||
|
|
||||||
|
$date_format = null;
|
||||||
|
|
||||||
|
$rows = array();
|
||||||
|
$rows[] = array(
|
||||||
|
pht('ID'),
|
||||||
|
pht('Owner'),
|
||||||
|
pht('Status'),
|
||||||
|
pht('Priority'),
|
||||||
|
pht('Date Created'),
|
||||||
|
pht('Date Updated'),
|
||||||
|
pht('Title'),
|
||||||
|
pht('Projects'),
|
||||||
|
pht('URI'),
|
||||||
|
pht('Description'),
|
||||||
|
);
|
||||||
|
|
||||||
|
$is_date = array(
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
true,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
false,
|
||||||
|
);
|
||||||
|
|
||||||
|
$header_format = array(
|
||||||
|
'font' => array(
|
||||||
|
'bold' => true,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
foreach ($tasks as $task) {
|
||||||
|
$task_owner = null;
|
||||||
|
if ($task->getOwnerPHID()) {
|
||||||
|
$task_owner = $handles[$task->getOwnerPHID()]->getName();
|
||||||
|
}
|
||||||
|
|
||||||
|
$projects = array();
|
||||||
|
foreach ($task->getProjectPHIDs() as $phid) {
|
||||||
|
$projects[] = $handles[$phid]->getName();
|
||||||
|
}
|
||||||
|
$projects = implode(', ', $projects);
|
||||||
|
|
||||||
|
$rows[] = array(
|
||||||
|
'T'.$task->getID(),
|
||||||
|
$task_owner,
|
||||||
|
idx($status_map, $task->getStatus(), '?'),
|
||||||
|
idx($pri_map, $task->getPriority(), '?'),
|
||||||
|
$this->computeExcelDate($task->getDateCreated()),
|
||||||
|
$this->computeExcelDate($task->getDateModified()),
|
||||||
|
$task->getTitle(),
|
||||||
|
$projects,
|
||||||
|
PhabricatorEnv::getProductionURI('/T'.$task->getID()),
|
||||||
|
phutil_utf8_shorten($task->getDescription(), 512),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($rows as $row => $cols) {
|
||||||
|
foreach ($cols as $col => $spec) {
|
||||||
|
$cell_name = $this->col($col).($row + 1);
|
||||||
|
$cell = $sheet
|
||||||
|
->setCellValue($cell_name, $spec, $return_cell = true);
|
||||||
|
|
||||||
|
if ($row == 0) {
|
||||||
|
$sheet->getStyle($cell_name)->applyFromArray($header_format);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($is_date[$col]) {
|
||||||
|
$code = PHPExcel_Style_NumberFormat::FORMAT_DATE_YYYYMMDD2;
|
||||||
|
$sheet
|
||||||
|
->getStyle($cell_name)
|
||||||
|
->getNumberFormat()
|
||||||
|
->setFormatCode($code);
|
||||||
|
} else {
|
||||||
|
$cell->setDataType(PHPExcel_Cell_DataType::TYPE_STRING);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
private function col($n) {
|
||||||
|
return chr(ord('A') + $n);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
47
src/applications/maniphest/export/ManiphestExcelFormat.php
Normal file
47
src/applications/maniphest/export/ManiphestExcelFormat.php
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @group maniphest
|
||||||
|
*/
|
||||||
|
abstract class ManiphestExcelFormat {
|
||||||
|
|
||||||
|
final public static function loadAllFormats() {
|
||||||
|
$classes = id(new PhutilSymbolLoader())
|
||||||
|
->setAncestorClass(__CLASS__)
|
||||||
|
->setConcreteOnly(true)
|
||||||
|
->selectAndLoadSymbols();
|
||||||
|
|
||||||
|
$objects = array();
|
||||||
|
foreach ($classes as $class) {
|
||||||
|
$objects[$class['name']] = newv($class['name'], array());
|
||||||
|
}
|
||||||
|
|
||||||
|
$objects = msort($objects, 'getOrder');
|
||||||
|
|
||||||
|
return $objects;
|
||||||
|
}
|
||||||
|
|
||||||
|
public abstract function getName();
|
||||||
|
public abstract function getFileName();
|
||||||
|
|
||||||
|
public function getOrder() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function computeExcelDate($epoch) {
|
||||||
|
$seconds_per_day = (60 * 60 * 24);
|
||||||
|
$offset = ($seconds_per_day * 25569);
|
||||||
|
|
||||||
|
return ($epoch + $offset) / $seconds_per_day;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @phutil-external-symbol class PHPExcel
|
||||||
|
*/
|
||||||
|
public abstract function buildWorkbook(
|
||||||
|
PHPExcel $workbook,
|
||||||
|
array $tasks,
|
||||||
|
array $handles,
|
||||||
|
PhabricatorUser $user);
|
||||||
|
|
||||||
|
}
|
Loading…
Reference in a new issue