Export Board

Signed-off-by: David Loe <d.loewens24@gmail.com>
This commit is contained in:
David Loe
2021-05-10 15:03:35 +02:00
committed by Julius Härtl
parent d3cad6adf0
commit 88a9fe2161
15 changed files with 153 additions and 27 deletions

View File

@@ -9,6 +9,6 @@ module.exports = {
'jsdoc/check-param-names': ['off'], 'jsdoc/check-param-names': ['off'],
'jsdoc/no-undefined-types': ['off'], 'jsdoc/no-undefined-types': ['off'],
'jsdoc/require-property-description': ['off'], 'jsdoc/require-property-description': ['off'],
'import/no-named-as-default-member': ['off'] 'import/no-named-as-default-member': ['off'],
}, },
} }

View File

@@ -40,6 +40,7 @@ return [
['name' => 'board#deleteAcl', 'url' => '/boards/{boardId}/acl/{aclId}', 'verb' => 'DELETE'], ['name' => 'board#deleteAcl', 'url' => '/boards/{boardId}/acl/{aclId}', 'verb' => 'DELETE'],
['name' => 'board#clone', 'url' => '/boards/{boardId}/clone', 'verb' => 'POST'], ['name' => 'board#clone', 'url' => '/boards/{boardId}/clone', 'verb' => 'POST'],
['name' => 'board#transferOwner', 'url' => '/boards/{boardId}/transferOwner', 'verb' => 'PUT'], ['name' => 'board#transferOwner', 'url' => '/boards/{boardId}/transferOwner', 'verb' => 'PUT'],
['name' => 'board#export', 'url' => '/boards/{boardId}/export', 'verb' => 'GET'],
// stacks // stacks
['name' => 'stack#index', 'url' => '/stacks/{boardId}', 'verb' => 'GET'], ['name' => 'stack#index', 'url' => '/stacks/{boardId}', 'verb' => 'GET'],

View File

@@ -169,4 +169,13 @@ class BoardController extends ApiController {
return new DataResponse([], HTTP::STATUS_UNAUTHORIZED); return new DataResponse([], HTTP::STATUS_UNAUTHORIZED);
} }
/**
* @NoAdminRequired
* @param $boardId
*/
public function export($boardId) {
return $this->boardService->export($boardId);
}
} }

View File

@@ -63,7 +63,8 @@ use Psr\Container\NotFoundExceptionInterface;
class BoardService { class BoardService {
private BoardMapper $boardMapper; private BoardMapper $boardMapper;
private StackMapper $stackMapper; private StackMapper $stackMapper;
private LabelMapper $labelMapper; private LabelMapper $cardMapper;
private $labelMapper;
private AclMapper $aclMapper; private AclMapper $aclMapper;
private IConfig $config; private IConfig $config;
private IL10N $l10n; private IL10N $l10n;
@@ -85,6 +86,7 @@ class BoardService {
public function __construct( public function __construct(
BoardMapper $boardMapper, BoardMapper $boardMapper,
StackMapper $stackMapper, StackMapper $stackMapper,
CardMapper $cardMapper,
IConfig $config, IConfig $config,
IL10N $l10n, IL10N $l10n,
LabelMapper $labelMapper, LabelMapper $labelMapper,
@@ -92,7 +94,6 @@ class BoardService {
PermissionService $permissionService, PermissionService $permissionService,
NotificationHelper $notificationHelper, NotificationHelper $notificationHelper,
AssignmentMapper $assignedUsersMapper, AssignmentMapper $assignedUsersMapper,
CardMapper $cardMapper,
IUserManager $userManager, IUserManager $userManager,
IGroupManager $groupManager, IGroupManager $groupManager,
ActivityManager $activityManager, ActivityManager $activityManager,
@@ -105,6 +106,7 @@ class BoardService {
) { ) {
$this->boardMapper = $boardMapper; $this->boardMapper = $boardMapper;
$this->stackMapper = $stackMapper; $this->stackMapper = $stackMapper;
$this->cardMapper = $cardMapper;
$this->labelMapper = $labelMapper; $this->labelMapper = $labelMapper;
$this->config = $config; $this->config = $config;
$this->aclMapper = $aclMapper; $this->aclMapper = $aclMapper;
@@ -119,7 +121,6 @@ class BoardService {
$this->changeHelper = $changeHelper; $this->changeHelper = $changeHelper;
$this->userId = $userId; $this->userId = $userId;
$this->urlGenerator = $urlGenerator; $this->urlGenerator = $urlGenerator;
$this->cardMapper = $cardMapper;
$this->connection = $connection; $this->connection = $connection;
$this->boardServiceValidator = $boardServiceValidator; $this->boardServiceValidator = $boardServiceValidator;
} }
@@ -652,6 +653,27 @@ class BoardService {
} }
} }
/**
* @param $id
* @return Board
* @throws DoesNotExistException
* @throws \OCA\Deck\NoPermissionException
* @throws \OCP\AppFramework\Db\MultipleObjectsReturnedException
* @throws BadRequestException
*/
public function export($id) {
if (is_numeric($id) === false) {
throw new BadRequestException('board id must be a number');
}
$this->permissionService->checkPermission($this->boardMapper, $id, Acl::PERMISSION_READ);
$board = $this->boardMapper->find($id);
$this->enrichWithCards($board);
$this->enrichWithLabels($board);
return $board;
}
private function enrichWithStacks($board, $since = -1) { private function enrichWithStacks($board, $since = -1) {
$stacks = $this->stackMapper->findAll($board->getId(), null, null, $since); $stacks = $this->stackMapper->findAll($board->getId(), null, null, $since);
@@ -698,4 +720,23 @@ class BoardService {
$this->boardMapper->flushCache($boardId, $boardOwnerId); $this->boardMapper->flushCache($boardId, $boardOwnerId);
unset($this->boardsCache[$boardId]); unset($this->boardsCache[$boardId]);
} }
private function enrichWithCards($board, $since = -1) {
$stacks = $this->stackMapper->findAll($board->getId(), null, null, $since);
foreach ($stacks as $stack) {
$cards = $this->cardMapper->findAllByStack($stack->getId());
$fullCards = [];
foreach ($cards as $card) {
$fullCard = $this->cardMapper->find($card->getId());
array_push($fullCards, $fullCard);
}
$stack->setCards($fullCards);
}
if (\count($stacks) === 0) {
return;
}
$board->setStacks($stacks);
}
} }

View File

@@ -72,6 +72,13 @@
{{ t('deck', 'Archive board') }} {{ t('deck', 'Archive board') }}
</NcActionButton> </NcActionButton>
<NcActionButton v-if="!board.archived && board.acl.length === 0" :icon="board.settings['notify-due'] === 'off' ? 'icon-sound' : 'icon-sound-off'" @click="board.settings['notify-due'] === 'off' ? updateSetting('notify-due', 'all') : updateSetting('notify-due', 'off')" />
<NcActionButton v-if="canManage && !board.archived"
icon="icon-download"
:close-after-click="true"
@click="actionExport">
{{ t('deck', 'Export board') }}
</NcActionButton>
<NcActionButton v-if="!board.archived && board.acl.length === 0" :icon="board.settings['notify-due'] === 'off' ? 'icon-sound' : 'icon-sound-off'" @click="board.settings['notify-due'] === 'off' ? updateSetting('notify-due', 'all') : updateSetting('notify-due', 'off')"> <NcActionButton v-if="!board.archived && board.acl.length === 0" :icon="board.settings['notify-due'] === 'off' ? 'icon-sound' : 'icon-sound-off'" @click="board.settings['notify-due'] === 'off' ? updateSetting('notify-due', 'all') : updateSetting('notify-due', 'off')">
{{ board.settings['notify-due'] === 'off' ? t('deck', 'Turn on due date reminders') : t('deck', 'Turn off due date reminders') }} {{ board.settings['notify-due'] === 'off' ? t('deck', 'Turn on due date reminders') : t('deck', 'Turn off due date reminders') }}
</NcActionButton> </NcActionButton>
@@ -314,6 +321,9 @@ export default {
this.isDueSubmenuActive = false this.isDueSubmenuActive = false
this.updateDueSetting = null this.updateDueSetting = null
}, },
actionExport() {
this.boardApi.exportBoard(this.board)
},
}, },
} }
</script> </script>

View File

@@ -38,7 +38,7 @@ export class BoardApi {
* Updates a board. * Updates a board.
* *
* @param {Board} board the board object to update * @param {Board} board the board object to update
* @return {Promise} * @returns {Promise}
*/ */
updateBoard(board) { updateBoard(board) {
return axios.put(this.url(`/boards/${board.id}`), board) return axios.put(this.url(`/boards/${board.id}`), board)
@@ -63,7 +63,7 @@ export class BoardApi {
* @property {string} color * @property {string} color
* @param {BoardCreateObject} boardData The board data to send. * @param {BoardCreateObject} boardData The board data to send.
* color the hexadecimal color value formated /[0-9A-F]{6}/i * color the hexadecimal color value formated /[0-9A-F]{6}/i
* @return {Promise} * @returns {Promise}
*/ */
createBoard(boardData) { createBoard(boardData) {
return axios.post(this.url('/boards'), boardData) return axios.post(this.url('/boards'), boardData)
@@ -149,6 +149,71 @@ export class BoardApi {
} }
} }
exportBoard(board) {
return axios.get(this.url(`/boards/${board.id}/export`))
.then(
(response) => {
const fields = { title: t('deck', 'Card title'), description: t('deck', 'Description'), stackId: t('deck', 'List name'), labels: t('deck', 'Tags'), duedate: t('deck', 'Due date'), createdAt: t('deck', 'Created'), lastModified: t('deck', 'Modified') }
let row = ''
Object.keys(fields).forEach(field => {
row += '"' + fields[field] + '"' + '\t'
})
row = row.slice(0, -1)
let CSV = row + '\r\n'
response.data.stacks.forEach(stack => {
stack.cards.forEach(card => {
row = ''
Object.keys(fields).forEach(field => {
if (field === 'createdAt' || field === 'lastModified') {
const date = new Date(Number(card[field]) * 1000)
row += '"' + date.toLocaleDateString() + '"' + '\t'
} else if (field === 'stackId') {
row += '"' + stack.title + '"' + '\t'
} else if (field === 'labels') {
row += '"'
card[field].forEach(label => {
row += label.title + ', '
})
if (card[field].length > 0) {
row = row.slice(0, -1)
}
row += '"' + '\t'
} else {
row += '"' + card[field] + '"' + '\t'
}
})
row = row.slice(0, -1)
CSV += row + '\r\n'
})
})
let charCode = []
const byteArray = []
byteArray.push(255, 254)
for (let i = 0; i < CSV.length; ++i) {
charCode = CSV.charCodeAt(i)
byteArray.push(charCode & 0xff)
byteArray.push(charCode / 256 >>> 0)
}
const blob = new Blob([new Uint8Array(byteArray)], { type: 'text/csv;charset=UTF-16LE;' })
const blobUrl = URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = blobUrl // 'data:' + data
a.download = response.data.title + '.csv'
a.click()
a.remove()
return Promise.resolve()
},
(err) => {
return Promise.reject(err)
}
)
.catch((err) => {
return Promise.reject(err)
})
}
// Label API Calls // Label API Calls
deleteLabel(id) { deleteLabel(id) {
return axios.delete(this.url(`/labels/${id}`)) return axios.delete(this.url(`/labels/${id}`))

View File

@@ -78,7 +78,7 @@ export class StackApi {
/** /**
* @param {Stack} stack stack object to create * @param {Stack} stack stack object to create
* @return {Promise} * @returns {Promise}
*/ */
createStack(stack) { createStack(stack) {
return axios.post(this.url('/stacks'), stack) return axios.post(this.url('/stacks'), stack)

View File

@@ -378,7 +378,7 @@ export default new Vuex.Store({
* @param commit.commit * @param commit.commit
* @param commit * @param commit
* @param board The board to update. * @param board The board to update.
* @return {Promise<void>} * @returns {Promise<void>}
*/ */
async updateBoard({ commit }, board) { async updateBoard({ commit }, board) {
const storedBoard = await apiClient.updateBoard(board) const storedBoard = await apiClient.updateBoard(board)