@@ -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'],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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'],
|
||||||
|
|||||||
@@ -1,17 +1,17 @@
|
|||||||
const { defineConfig } = require('cypress')
|
const { defineConfig } = require('cypress')
|
||||||
|
|
||||||
module.exports = defineConfig({
|
module.exports = defineConfig({
|
||||||
projectId: '1s7wkc',
|
projectId: '1s7wkc',
|
||||||
viewportWidth: 1280,
|
viewportWidth: 1280,
|
||||||
viewportHeight: 720,
|
viewportHeight: 720,
|
||||||
e2e: {
|
e2e: {
|
||||||
// We've imported your old cypress plugins here.
|
// We've imported your old cypress plugins here.
|
||||||
// You may want to clean this up later by importing these.
|
// You may want to clean this up later by importing these.
|
||||||
setupNodeEvents(on, config) {
|
setupNodeEvents(on, config) {
|
||||||
return require('./cypress/plugins/index.js')(on, config)
|
return require('./cypress/plugins/index.js')(on, config)
|
||||||
},
|
},
|
||||||
baseUrl: 'http://nextcloud.local/index.php',
|
baseUrl: 'http://nextcloud.local/index.php',
|
||||||
experimentalSessionAndOrigin: true,
|
experimentalSessionAndOrigin: true,
|
||||||
specPattern: 'cypress/e2e/**/*.{js,jsx,ts,tsx}',
|
specPattern: 'cypress/e2e/**/*.{js,jsx,ts,tsx}',
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -213,7 +213,7 @@ export default {
|
|||||||
confirmClasses: 'error',
|
confirmClasses: 'error',
|
||||||
cancel: t('deck', 'Cancel'),
|
cancel: t('deck', 'Cancel'),
|
||||||
},
|
},
|
||||||
async (result) => {
|
async(result) => {
|
||||||
if (result) {
|
if (result) {
|
||||||
try {
|
try {
|
||||||
this.isLoading = true
|
this.isLoading = true
|
||||||
|
|||||||
@@ -225,7 +225,7 @@ export default {
|
|||||||
},
|
},
|
||||||
shareFromFiles() {
|
shareFromFiles() {
|
||||||
picker.pick()
|
picker.pick()
|
||||||
.then(async (path) => {
|
.then(async(path) => {
|
||||||
console.debug(`path ${path} selected for sharing`)
|
console.debug(`path ${path} selected for sharing`)
|
||||||
if (!path.startsWith('/')) {
|
if (!path.startsWith('/')) {
|
||||||
throw new Error(t('files', 'Invalid path selected'))
|
throw new Error(t('files', 'Invalid path selected'))
|
||||||
|
|||||||
@@ -259,7 +259,7 @@ export default {
|
|||||||
updateDescription() {
|
updateDescription() {
|
||||||
this.descriptionLastEdit = Date.now()
|
this.descriptionLastEdit = Date.now()
|
||||||
clearTimeout(this.descriptionSaveTimeout)
|
clearTimeout(this.descriptionSaveTimeout)
|
||||||
this.descriptionSaveTimeout = setTimeout(async () => {
|
this.descriptionSaveTimeout = setTimeout(async() => {
|
||||||
await this.saveDescription()
|
await this.saveDescription()
|
||||||
}, 2500)
|
}, 2500)
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ const createCancelToken = () => axios.CancelToken.source()
|
|||||||
function search({ query, cursor }) {
|
function search({ query, cursor }) {
|
||||||
const cancelToken = createCancelToken()
|
const cancelToken = createCancelToken()
|
||||||
|
|
||||||
const request = async () => axios.get(generateOcsUrl('apps/deck/api/v1.0/search'), {
|
const request = async() => axios.get(generateOcsUrl('apps/deck/api/v1.0/search'), {
|
||||||
cancelToken: cancelToken.token,
|
cancelToken: cancelToken.token,
|
||||||
params: {
|
params: {
|
||||||
term: query,
|
term: query,
|
||||||
|
|||||||
@@ -48,7 +48,7 @@ export default {
|
|||||||
bodyFormData.append('cardId', this.cardId)
|
bodyFormData.append('cardId', this.cardId)
|
||||||
bodyFormData.append('type', type)
|
bodyFormData.append('type', type)
|
||||||
bodyFormData.append('file', file)
|
bodyFormData.append('file', file)
|
||||||
await queue.add(async () => {
|
await queue.add(async() => {
|
||||||
try {
|
try {
|
||||||
await this.$store.dispatch('createAttachment', {
|
await this.$store.dispatch('createAttachment', {
|
||||||
cardId: this.cardId,
|
cardId: this.cardId,
|
||||||
|
|||||||
@@ -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}`))
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -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)
|
||||||
|
|||||||
@@ -46,7 +46,7 @@ export default {
|
|||||||
ComponentVM.$destroy()
|
ComponentVM.$destroy()
|
||||||
reject(new Error('Canceled'))
|
reject(new Error('Canceled'))
|
||||||
})
|
})
|
||||||
ComponentVM.$root.$on('select', async (id) => {
|
ComponentVM.$root.$on('select', async(id) => {
|
||||||
const result = await createShare({
|
const result = await createShare({
|
||||||
path: self.fileInfo.path + '/' + self.fileInfo.name,
|
path: self.fileInfo.path + '/' + self.fileInfo.name,
|
||||||
shareType: 12,
|
shareType: 12,
|
||||||
|
|||||||
Reference in New Issue
Block a user