Co-authored-by: kolaente <k@knt.li> Reviewed-on: https://kolaente.dev/vikunja/frontend/pulls/380 Co-authored-by: konrad <konrad@kola-entertainments.de> Co-committed-by: konrad <konrad@kola-entertainments.de>
378 lines
8.8 KiB
Vue
378 lines
8.8 KiB
Vue
<template>
|
|
<card class="is-fullwidth has-overflow" :title="`Shared with these ${shareType}s`" :padding="false">
|
|
<div class="p-4" v-if="userIsAdmin">
|
|
<div class="field has-addons">
|
|
<p
|
|
class="control is-expanded"
|
|
v-bind:class="{ 'is-loading': searchService.loading }"
|
|
>
|
|
<multiselect
|
|
:loading="searchService.loading"
|
|
placeholder="Type to search..."
|
|
@search="find"
|
|
:search-results="found"
|
|
:label="searchLabel"
|
|
v-model="sharable"
|
|
/>
|
|
</p>
|
|
<p class="control">
|
|
<x-button @click="add()"> Share </x-button>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<table class="table is-striped is-hoverable is-fullwidth">
|
|
<tbody>
|
|
<tr :key="s.id" v-for="s in sharables">
|
|
<template v-if="shareType === 'user'">
|
|
<td>{{ s.getDisplayName() }}</td>
|
|
<td>
|
|
<template v-if="s.id === userInfo.id">
|
|
<b class="is-success">You</b>
|
|
</template>
|
|
</td>
|
|
</template>
|
|
<template v-if="shareType === 'team'">
|
|
<td>
|
|
<router-link
|
|
:to="{
|
|
name: 'teams.edit',
|
|
params: { id: s.id },
|
|
}"
|
|
>
|
|
{{ s.name }}
|
|
</router-link>
|
|
</td>
|
|
</template>
|
|
<td class="type">
|
|
<template v-if="s.right === rights.ADMIN">
|
|
<span class="icon is-small">
|
|
<icon icon="lock" />
|
|
</span>
|
|
Admin
|
|
</template>
|
|
<template v-else-if="s.right === rights.READ_WRITE">
|
|
<span class="icon is-small">
|
|
<icon icon="pen" />
|
|
</span>
|
|
Write
|
|
</template>
|
|
<template v-else>
|
|
<span class="icon is-small">
|
|
<icon icon="users" />
|
|
</span>
|
|
Read-only
|
|
</template>
|
|
</td>
|
|
<td class="actions" v-if="userIsAdmin">
|
|
<div class="select">
|
|
<select
|
|
@change="toggleType(s)"
|
|
class="button mr-2"
|
|
v-model="selectedRight[s.id]"
|
|
>
|
|
<option
|
|
:selected="s.right === rights.READ"
|
|
:value="rights.READ"
|
|
>
|
|
Read only
|
|
</option>
|
|
<option
|
|
:selected="s.right === rights.READ_WRITE"
|
|
:value="rights.READ_WRITE"
|
|
>
|
|
Read & write
|
|
</option>
|
|
<option
|
|
:selected="s.right === rights.ADMIN"
|
|
:value="rights.ADMIN"
|
|
>
|
|
Admin
|
|
</option>
|
|
</select>
|
|
</div>
|
|
<x-button
|
|
@click="
|
|
() => {
|
|
sharable = s
|
|
showDeleteModal = true
|
|
}
|
|
"
|
|
class="is-danger"
|
|
icon="trash-alt"
|
|
/>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
|
|
<modal
|
|
@close="showDeleteModal = false"
|
|
@submit="deleteSharable()"
|
|
v-if="showDeleteModal"
|
|
>
|
|
<span slot="header"
|
|
>Remove a {{ shareType }} from the {{ typeString }}</span
|
|
>
|
|
<p slot="text">
|
|
Are you sure you want to remove this {{ shareType }} from the
|
|
{{ typeString }}?<br />
|
|
<b>This CANNOT BE UNDONE!</b>
|
|
</p>
|
|
</modal>
|
|
</card>
|
|
</template>
|
|
|
|
<script>
|
|
import { mapState } from 'vuex'
|
|
|
|
import UserNamespaceService from '../../services/userNamespace'
|
|
import UserNamespaceModel from '../../models/userNamespace'
|
|
import UserListModel from '../../models/userList'
|
|
import UserListService from '../../services/userList'
|
|
import UserService from '../../services/user'
|
|
import UserModel from '../../models/user'
|
|
|
|
import TeamNamespaceService from '../../services/teamNamespace'
|
|
import TeamNamespaceModel from '../../models/teamNamespace'
|
|
import TeamListModel from '../../models/teamList'
|
|
import TeamListService from '../../services/teamList'
|
|
import TeamService from '../../services/team'
|
|
import TeamModel from '../../models/team'
|
|
|
|
import rights from '../../models/rights'
|
|
import Multiselect from '@/components/input/multiselect'
|
|
|
|
export default {
|
|
name: 'userTeamShare',
|
|
props: {
|
|
type: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
shareType: {
|
|
type: String,
|
|
default: '',
|
|
},
|
|
id: {
|
|
type: Number,
|
|
default: 0,
|
|
},
|
|
userIsAdmin: {
|
|
type: Boolean,
|
|
default: false,
|
|
},
|
|
},
|
|
data() {
|
|
return {
|
|
stuffService: Object, // This user service is either a userNamespaceService or a userListService, depending on the type we are using
|
|
stuffModel: Object,
|
|
searchService: Object,
|
|
sharable: Object,
|
|
|
|
found: [],
|
|
searchLabel: '',
|
|
rights: rights,
|
|
selectedRight: {},
|
|
|
|
typeString: '',
|
|
sharables: [], // This holds either teams or users who this namepace or list is shared with
|
|
showDeleteModal: false,
|
|
}
|
|
},
|
|
components: {
|
|
Multiselect,
|
|
},
|
|
computed: mapState({
|
|
userInfo: (state) => state.auth.info,
|
|
}),
|
|
created() {
|
|
if (this.shareType === 'user') {
|
|
this.searchService = new UserService()
|
|
this.sharable = new UserModel()
|
|
this.searchLabel = 'username'
|
|
|
|
if (this.type === 'list') {
|
|
this.typeString = `list`
|
|
this.stuffService = new UserListService()
|
|
this.stuffModel = new UserListModel({ listId: this.id })
|
|
} else if (this.type === 'namespace') {
|
|
this.typeString = `namespace`
|
|
this.stuffService = new UserNamespaceService()
|
|
this.stuffModel = new UserNamespaceModel({
|
|
namespaceId: this.id,
|
|
})
|
|
} else {
|
|
throw new Error('Unknown type: ' + this.type)
|
|
}
|
|
} else if (this.shareType === 'team') {
|
|
this.searchService = new TeamService()
|
|
this.sharable = new TeamModel()
|
|
this.searchLabel = 'name'
|
|
|
|
if (this.type === 'list') {
|
|
this.typeString = `list`
|
|
this.stuffService = new TeamListService()
|
|
this.stuffModel = new TeamListModel({ listId: this.id })
|
|
} else if (this.type === 'namespace') {
|
|
this.typeString = `namespace`
|
|
this.stuffService = new TeamNamespaceService()
|
|
this.stuffModel = new TeamNamespaceModel({
|
|
namespaceId: this.id,
|
|
})
|
|
} else {
|
|
throw new Error('Unknown type: ' + this.type)
|
|
}
|
|
} else {
|
|
throw new Error('Unkown share type')
|
|
}
|
|
|
|
this.load()
|
|
},
|
|
methods: {
|
|
load() {
|
|
this.stuffService
|
|
.getAll(this.stuffModel)
|
|
.then((r) => {
|
|
this.$set(this, 'sharables', r)
|
|
r.forEach((s) =>
|
|
this.$set(this.selectedRight, s.id, s.right)
|
|
)
|
|
})
|
|
.catch((e) => {
|
|
this.error(e, this)
|
|
})
|
|
},
|
|
deleteSharable() {
|
|
if (this.shareType === 'user') {
|
|
this.stuffModel.userId = this.sharable.username
|
|
} else if (this.shareType === 'team') {
|
|
this.stuffModel.teamId = this.sharable.id
|
|
}
|
|
this.stuffService
|
|
.delete(this.stuffModel)
|
|
.then(() => {
|
|
this.showDeleteModal = false
|
|
for (const i in this.sharables) {
|
|
if (
|
|
(this.sharables[i].id === this.stuffModel.userId &&
|
|
this.shareType === 'user') ||
|
|
(this.sharables[i].id === this.stuffModel.teamId &&
|
|
this.shareType === 'team')
|
|
) {
|
|
this.sharables.splice(i, 1)
|
|
}
|
|
}
|
|
this.success(
|
|
{
|
|
message:
|
|
'The ' +
|
|
this.shareType +
|
|
' was successfully deleted from the ' +
|
|
this.typeString +
|
|
'.',
|
|
},
|
|
this
|
|
)
|
|
})
|
|
.catch((e) => {
|
|
this.error(e, this)
|
|
})
|
|
},
|
|
add(admin) {
|
|
if (admin === null) {
|
|
admin = false
|
|
}
|
|
this.stuffModel.right = rights.READ
|
|
if (admin) {
|
|
this.stuffModel.right = rights.ADMIN
|
|
}
|
|
|
|
if (this.shareType === 'user') {
|
|
this.stuffModel.userId = this.sharable.username
|
|
} else if (this.shareType === 'team') {
|
|
this.stuffModel.teamId = this.sharable.id
|
|
}
|
|
|
|
this.stuffService
|
|
.create(this.stuffModel)
|
|
.then(() => {
|
|
this.success(
|
|
{
|
|
message:
|
|
'The ' +
|
|
this.shareType +
|
|
' was successfully added.',
|
|
},
|
|
this
|
|
)
|
|
this.load()
|
|
})
|
|
.catch((e) => {
|
|
this.error(e, this)
|
|
})
|
|
},
|
|
toggleType(sharable) {
|
|
if (
|
|
this.selectedRight[sharable.id] !== rights.ADMIN &&
|
|
this.selectedRight[sharable.id] !== rights.READ &&
|
|
this.selectedRight[sharable.id] !== rights.READ_WRITE
|
|
) {
|
|
this.selectedRight[sharable.id] = rights.READ
|
|
}
|
|
this.stuffModel.right = this.selectedRight[sharable.id]
|
|
|
|
if (this.shareType === 'user') {
|
|
this.stuffModel.userId = sharable.username
|
|
} else if (this.shareType === 'team') {
|
|
this.stuffModel.teamId = sharable.id
|
|
}
|
|
|
|
this.stuffService
|
|
.update(this.stuffModel)
|
|
.then((r) => {
|
|
for (const i in this.sharables) {
|
|
if (
|
|
(this.sharables[i].username ===
|
|
this.stuffModel.userId &&
|
|
this.shareType === 'user') ||
|
|
(this.sharables[i].id === this.stuffModel.teamId &&
|
|
this.shareType === 'team')
|
|
) {
|
|
this.$set(this.sharables[i], 'right', r.right)
|
|
}
|
|
}
|
|
this.success(
|
|
{
|
|
message:
|
|
'The ' +
|
|
this.shareType +
|
|
' right was successfully updated.',
|
|
},
|
|
this
|
|
)
|
|
})
|
|
.catch((e) => {
|
|
this.error(e, this)
|
|
})
|
|
},
|
|
find(query) {
|
|
if (query === '') {
|
|
this.$set(this, 'found', [])
|
|
return
|
|
}
|
|
|
|
this.searchService
|
|
.getAll({}, { s: query })
|
|
.then((response) => {
|
|
this.$set(this, 'found', response)
|
|
})
|
|
.catch((e) => {
|
|
this.error(e, this)
|
|
})
|
|
},
|
|
clearAll() {
|
|
this.$set(this, 'found', [])
|
|
},
|
|
},
|
|
}
|
|
</script>
|