List edit UI overhaul
This commit is contained in:
parent
38bd59ceb0
commit
d074aefb4f
|
@ -85,8 +85,12 @@ export default {
|
||||||
isChats () {
|
isChats () {
|
||||||
return this.$route.name === 'chat' || this.$route.name === 'chats'
|
return this.$route.name === 'chat' || this.$route.name === 'chats'
|
||||||
},
|
},
|
||||||
|
isListEdit () {
|
||||||
|
return this.$route.name === 'lists-edit'
|
||||||
|
},
|
||||||
newPostButtonShown () {
|
newPostButtonShown () {
|
||||||
if (this.isChats) return false
|
if (this.isChats) return false
|
||||||
|
if (this.isListEdit) return false
|
||||||
return this.$store.getters.mergedConfig.alwaysShowNewPostButton || this.layoutType === 'mobile'
|
return this.$store.getters.mergedConfig.alwaysShowNewPostButton || this.layoutType === 'mobile'
|
||||||
},
|
},
|
||||||
showFeaturesPanel () { return this.$store.state.instance.showFeaturesPanel },
|
showFeaturesPanel () { return this.$store.state.instance.showFeaturesPanel },
|
||||||
|
|
|
@ -33,7 +33,7 @@
|
||||||
<div
|
<div
|
||||||
id="main-scroller"
|
id="main-scroller"
|
||||||
class="column main"
|
class="column main"
|
||||||
:class="{ '-full-height': isChats }"
|
:class="{ '-full-height': isChats || isListEdit }"
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
v-if="!currentUser"
|
v-if="!currentUser"
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import { mapState, mapGetters } from 'vuex'
|
import { mapState, mapGetters } from 'vuex'
|
||||||
import BasicUserCard from '../basic_user_card/basic_user_card.vue'
|
import BasicUserCard from '../basic_user_card/basic_user_card.vue'
|
||||||
import ListsUserSearch from '../lists_user_search/lists_user_search.vue'
|
import ListsUserSearch from '../lists_user_search/lists_user_search.vue'
|
||||||
|
import PanelLoading from 'src/components/panel_loading/panel_loading.vue'
|
||||||
import UserAvatar from '../user_avatar/user_avatar.vue'
|
import UserAvatar from '../user_avatar/user_avatar.vue'
|
||||||
|
import TabSwitcher from 'src/components/tab_switcher/tab_switcher.jsx'
|
||||||
import { library } from '@fortawesome/fontawesome-svg-core'
|
import { library } from '@fortawesome/fontawesome-svg-core'
|
||||||
import {
|
import {
|
||||||
faSearch,
|
faSearch,
|
||||||
|
@ -17,22 +19,32 @@ const ListsNew = {
|
||||||
components: {
|
components: {
|
||||||
BasicUserCard,
|
BasicUserCard,
|
||||||
UserAvatar,
|
UserAvatar,
|
||||||
ListsUserSearch
|
ListsUserSearch,
|
||||||
|
TabSwitcher,
|
||||||
|
PanelLoading
|
||||||
},
|
},
|
||||||
data () {
|
data () {
|
||||||
return {
|
return {
|
||||||
title: '',
|
title: '',
|
||||||
userIds: [],
|
titleDraft: '',
|
||||||
selectedUserIds: []
|
membersUserIds: [],
|
||||||
|
removedUserIds: new Set([]), // users we added for members, to undo
|
||||||
|
searchUserIds: [],
|
||||||
|
addedUserIds: new Set([]), // users we added from search, to undo
|
||||||
|
searchLoading: false,
|
||||||
|
reallyDelete: false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
created () {
|
created () {
|
||||||
this.$store.dispatch('fetchList', { listId: this.id })
|
this.$store.dispatch('fetchList', { listId: this.id })
|
||||||
.then(() => { this.title = this.findListTitle(this.id) })
|
.then(() => {
|
||||||
|
this.title = this.findListTitle(this.id)
|
||||||
|
this.titleDraft = this.title
|
||||||
|
})
|
||||||
this.$store.dispatch('fetchListAccounts', { listId: this.id })
|
this.$store.dispatch('fetchListAccounts', { listId: this.id })
|
||||||
.then(() => {
|
.then(() => {
|
||||||
this.selectedUserIds = this.findListAccounts(this.id)
|
this.membersUserIds = this.findListAccounts(this.id)
|
||||||
this.selectedUserIds.forEach(userId => {
|
this.membersUserIds.forEach(userId => {
|
||||||
this.$store.dispatch('fetchUserIfMissing', userId)
|
this.$store.dispatch('fetchUserIfMissing', userId)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -41,11 +53,12 @@ const ListsNew = {
|
||||||
id () {
|
id () {
|
||||||
return this.$route.params.id
|
return this.$route.params.id
|
||||||
},
|
},
|
||||||
users () {
|
membersUsers () {
|
||||||
return this.userIds.map(userId => this.findUser(userId))
|
return [...this.membersUserIds, ...this.addedUserIds]
|
||||||
|
.map(userId => this.findUser(userId)).filter(user => user)
|
||||||
},
|
},
|
||||||
selectedUsers () {
|
searchUsers () {
|
||||||
return this.selectedUserIds.map(userId => this.findUser(userId)).filter(user => user)
|
return this.searchUserIds.map(userId => this.findUser(userId)).filter(user => user)
|
||||||
},
|
},
|
||||||
...mapState({
|
...mapState({
|
||||||
currentUser: state => state.users.currentUser
|
currentUser: state => state.users.currentUser
|
||||||
|
@ -56,30 +69,51 @@ const ListsNew = {
|
||||||
onInput () {
|
onInput () {
|
||||||
this.search(this.query)
|
this.search(this.query)
|
||||||
},
|
},
|
||||||
selectUser (user) {
|
toggleRemoveMember (user) {
|
||||||
if (this.selectedUserIds.includes(user.id)) {
|
if (this.removedUserIds.has(user.id)) {
|
||||||
this.removeUser(user.id)
|
|
||||||
} else {
|
|
||||||
this.addUser(user)
|
this.addUser(user)
|
||||||
|
this.removedUserIds.delete(user.id)
|
||||||
|
} else {
|
||||||
|
this.removeUser(user.id)
|
||||||
|
this.removedUserIds.add(user.id)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
isSelected (user) {
|
toggleAddFromSearch (user) {
|
||||||
return this.selectedUserIds.includes(user.id)
|
if (this.addedUserIds.has(user.id)) {
|
||||||
|
this.removeUser(user.id)
|
||||||
|
this.addedUserIds.delete(user.id)
|
||||||
|
} else {
|
||||||
|
this.addUser(user)
|
||||||
|
this.addedUserIds.add(user.id)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
isRemoved (user) {
|
||||||
|
return this.removedUserIds.has(user.id)
|
||||||
|
},
|
||||||
|
isAdded (user) {
|
||||||
|
return this.addedUserIds.has(user.id)
|
||||||
},
|
},
|
||||||
addUser (user) {
|
addUser (user) {
|
||||||
this.selectedUserIds.push(user.id)
|
// this.selectedUserIds.push(user.id)
|
||||||
},
|
},
|
||||||
removeUser (userId) {
|
removeUser (userId) {
|
||||||
this.selectedUserIds = this.selectedUserIds.filter(id => id !== userId)
|
// this.selectedUserIds = this.selectedUserIds.filter(id => id !== userId)
|
||||||
},
|
},
|
||||||
onResults (results) {
|
onSearchLoading (results) {
|
||||||
this.userIds = results
|
this.searchLoading = true
|
||||||
},
|
},
|
||||||
updateList () {
|
onSearchLoadingDone (results) {
|
||||||
this.$store.dispatch('setList', { listId: this.id, title: this.title })
|
this.searchLoading = false
|
||||||
this.$store.dispatch('setListAccounts', { listId: this.id, accountIds: this.selectedUserIds })
|
},
|
||||||
|
onSearchResults (results) {
|
||||||
this.$router.push({ name: 'lists-timeline', params: { id: this.id } })
|
this.searchLoading = false
|
||||||
|
this.searchUserIds = results
|
||||||
|
},
|
||||||
|
updateListTitle () {
|
||||||
|
this.$store.dispatch('setList', { listId: this.id, title: this.titleDraft })
|
||||||
|
.then(() => {
|
||||||
|
this.title = this.findListTitle(this.id)
|
||||||
|
})
|
||||||
},
|
},
|
||||||
deleteList () {
|
deleteList () {
|
||||||
this.$store.dispatch('deleteList', { listId: this.id })
|
this.$store.dispatch('deleteList', { listId: this.id })
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
<template>
|
<template>
|
||||||
<div class="panel-default panel list-edit">
|
<div class="panel-default panel ListEdit">
|
||||||
<div
|
<div
|
||||||
ref="header"
|
ref="header"
|
||||||
class="panel-heading"
|
class="panel-heading list-edit-heading"
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
class="button-unstyled go-back-button"
|
class="button-unstyled go-back-button"
|
||||||
|
@ -13,54 +13,129 @@
|
||||||
icon="chevron-left"
|
icon="chevron-left"
|
||||||
/>
|
/>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
<div class="title">
|
||||||
<div class="input-wrap">
|
<i18n-t
|
||||||
<input
|
keypath="lists.editing_list"
|
||||||
ref="title"
|
|
||||||
v-model="title"
|
|
||||||
:placeholder="$t('lists.title')"
|
|
||||||
>
|
>
|
||||||
</div>
|
<template #listTitle>{{ title }}</template>
|
||||||
<div class="member-list">
|
</i18n-t>
|
||||||
<div
|
|
||||||
v-for="user in selectedUsers"
|
|
||||||
:key="user.id"
|
|
||||||
class="member"
|
|
||||||
>
|
|
||||||
<BasicUserCard
|
|
||||||
:user="user"
|
|
||||||
:class="isSelected(user) ? 'selected' : ''"
|
|
||||||
@click.capture.prevent="selectUser(user)"
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<ListsUserSearch @results="onResults" />
|
<div class="panel-body">
|
||||||
<div class="member-list">
|
<div class="input-wrap">
|
||||||
<div
|
<label for="list-edit-title">{{ $t('lists.title') }}</label>
|
||||||
v-for="user in users"
|
{{ ' ' }}
|
||||||
:key="user.id"
|
<input
|
||||||
class="member"
|
id="list-edit-title"
|
||||||
>
|
ref="title"
|
||||||
<BasicUserCard
|
v-model="titleDraft"
|
||||||
:user="user"
|
>
|
||||||
:class="isSelected(user) ? 'selected' : ''"
|
<button
|
||||||
@click.capture.prevent="selectUser(user)"
|
class="btn button-default follow-button"
|
||||||
/>
|
@click="updateListTitle"
|
||||||
|
>
|
||||||
|
{{ $t('lists.update_title') }}
|
||||||
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
<tab-switcher
|
||||||
|
class="list-member-management"
|
||||||
|
:scrollable-tabs="true"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
:label="$t('lists.manage_members')"
|
||||||
|
class="members-list"
|
||||||
|
>
|
||||||
|
<div class="users-list">
|
||||||
|
<div
|
||||||
|
v-for="user in membersUsers"
|
||||||
|
:key="user.id"
|
||||||
|
class="member"
|
||||||
|
>
|
||||||
|
<BasicUserCard
|
||||||
|
:user="user"
|
||||||
|
>
|
||||||
|
<button
|
||||||
|
class="btn button-default follow-button"
|
||||||
|
@click="toggleRemoveMember(user)"
|
||||||
|
>
|
||||||
|
{{ isRemoved(user) ? $t('general.undo') : $t('lists.remove_from_list') }}
|
||||||
|
</button>
|
||||||
|
</BasicUserCard>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="search-list"
|
||||||
|
:label="$t('lists.add_members')"
|
||||||
|
>
|
||||||
|
<ListsUserSearch
|
||||||
|
@results="onSearchResults"
|
||||||
|
@loading="onSearchLoading"
|
||||||
|
@loadingDone="onSearchLoadingDone"
|
||||||
|
/>
|
||||||
|
<div v-if="searchLoading" class="loading">
|
||||||
|
<PanelLoading />
|
||||||
|
</div>
|
||||||
|
<div v-else class="users-list">
|
||||||
|
<div
|
||||||
|
v-for="user in searchUsers"
|
||||||
|
:key="user.id"
|
||||||
|
class="member"
|
||||||
|
>
|
||||||
|
<BasicUserCard
|
||||||
|
:user="user"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
v-if="membersUserIds.includes(user.id)"
|
||||||
|
>
|
||||||
|
{{ $t('lists.is_in_list') }}
|
||||||
|
</span>
|
||||||
|
<button
|
||||||
|
v-if="!membersUserIds.includes(user.id)"
|
||||||
|
class="btn button-default follow-button"
|
||||||
|
@click="toggleAddFromSearch(user)"
|
||||||
|
>
|
||||||
|
{{ isAdded(user) ? $t('general.undo') : $t('lists.add_to_list') }}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
v-else
|
||||||
|
class="btn button-default follow-button"
|
||||||
|
@click="toggleRemoveMember(user)"
|
||||||
|
>
|
||||||
|
{{ isRemoved(user) ? $t('general.undo') : $t('lists.remove_from_list') }}
|
||||||
|
</button>
|
||||||
|
</BasicUserCard>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</tab-switcher>
|
||||||
|
</div>
|
||||||
|
<div class="panel-footer">
|
||||||
|
<span class="spacer" />
|
||||||
|
<button
|
||||||
|
class="btn button-default delete-button"
|
||||||
|
@click="reallyDelete = true"
|
||||||
|
v-if="!reallyDelete"
|
||||||
|
>
|
||||||
|
{{ $t('lists.delete') }}
|
||||||
|
</button>
|
||||||
|
<template v-else>
|
||||||
|
{{ $t('lists.really_delete') }}
|
||||||
|
<button
|
||||||
|
class="btn button-default delete-button"
|
||||||
|
@click="deleteList"
|
||||||
|
>
|
||||||
|
{{ $t('general.yes') }}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
class="btn button-default delete-button"
|
||||||
|
@click="reallyDelete = false"
|
||||||
|
>
|
||||||
|
{{ $t('general.no') }}
|
||||||
|
</button>
|
||||||
|
</template>
|
||||||
</div>
|
</div>
|
||||||
<button
|
|
||||||
:disabled="title && title.length === 0"
|
|
||||||
class="btn button-default"
|
|
||||||
@click="updateList"
|
|
||||||
>
|
|
||||||
{{ $t('lists.save') }}
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
class="btn button-default"
|
|
||||||
@click="deleteList"
|
|
||||||
>
|
|
||||||
{{ $t('lists.delete') }}
|
|
||||||
</button>
|
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
@ -69,28 +144,43 @@
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
@import '../../_variables.scss';
|
@import '../../_variables.scss';
|
||||||
|
|
||||||
.list-edit {
|
.ListEdit {
|
||||||
.input-wrap {
|
--panel-body-padding: 0.5em;
|
||||||
display: flex;
|
|
||||||
margin: 0.7em 0.5em 0.7em 0.5em;
|
|
||||||
|
|
||||||
input {
|
height: calc(100vh - var(--navbar-height));
|
||||||
width: 100%;
|
overflow: hidden;
|
||||||
}
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
|
.list-edit-heading {
|
||||||
|
grid-template-columns: auto minmax(50%, 1fr);
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel-body {
|
||||||
|
display: flex;
|
||||||
|
flex: 1;
|
||||||
|
flex-direction: column;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.list-member-management {
|
||||||
|
flex: 1 0 auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-icon {
|
.search-icon {
|
||||||
margin-right: 0.3em;
|
margin-right: 0.3em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.member-list {
|
.users-list {
|
||||||
padding-bottom: 0.7rem;
|
padding-bottom: 0.7rem;
|
||||||
|
overflow-y: auto;
|
||||||
}
|
}
|
||||||
|
|
||||||
.basic-user-card:hover,
|
& .search-list,
|
||||||
.basic-user-card.selected {
|
& .members-list {
|
||||||
cursor: pointer;
|
overflow: hidden;
|
||||||
background-color: var(--selectedPost, $fallback--lightBg);
|
flex-direction: column;
|
||||||
|
min-height: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.go-back-button {
|
.go-back-button {
|
||||||
|
@ -102,7 +192,15 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
margin: 0.5em;
|
margin: 0 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.panel-footer {
|
||||||
|
grid-template-columns: minmax(10%, 1fr);
|
||||||
|
|
||||||
|
.delete-button {
|
||||||
|
min-width: 9em;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -15,6 +15,7 @@ const ListsUserSearch = {
|
||||||
components: {
|
components: {
|
||||||
Checkbox
|
Checkbox
|
||||||
},
|
},
|
||||||
|
emits: ['loading', 'loadingDone', 'results'],
|
||||||
data () {
|
data () {
|
||||||
return {
|
return {
|
||||||
loading: false,
|
loading: false,
|
||||||
|
@ -33,12 +34,16 @@ const ListsUserSearch = {
|
||||||
}
|
}
|
||||||
|
|
||||||
this.loading = true
|
this.loading = true
|
||||||
|
this.$emit('loading')
|
||||||
this.userIds = []
|
this.userIds = []
|
||||||
this.$store.dispatch('search', { q: query, resolve: true, type: 'accounts', following: this.followingOnly })
|
this.$store.dispatch('search', { q: query, resolve: true, type: 'accounts', following: this.followingOnly })
|
||||||
.then(data => {
|
.then(data => {
|
||||||
this.loading = false
|
|
||||||
this.$emit('results', data.accounts.map(a => a.id))
|
this.$emit('results', data.accounts.map(a => a.id))
|
||||||
})
|
})
|
||||||
|
.finally(() => {
|
||||||
|
this.loading = false
|
||||||
|
this.$emit('loadingDone')
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
<template>
|
<template>
|
||||||
<div>
|
<div class="ListsUserSearch">
|
||||||
<div class="input-wrap">
|
<div class="input-wrap">
|
||||||
<div class="input-search">
|
<div class="input-search">
|
||||||
<FAIcon
|
<FAIcon
|
||||||
|
@ -29,17 +29,19 @@
|
||||||
<style lang="scss">
|
<style lang="scss">
|
||||||
@import '../../_variables.scss';
|
@import '../../_variables.scss';
|
||||||
|
|
||||||
.input-wrap {
|
.ListsUserSearch {
|
||||||
display: flex;
|
.input-wrap {
|
||||||
margin: 0.7em 0.5em 0.7em 0.5em;
|
display: flex;
|
||||||
|
margin: 0.7em 0.5em 0.7em 0.5em;
|
||||||
|
|
||||||
input {
|
input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.search-icon {
|
||||||
|
margin-right: 0.3em;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.search-icon {
|
|
||||||
margin-right: 0.3em;
|
|
||||||
}
|
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
|
@ -10,7 +10,8 @@ library.add(
|
||||||
|
|
||||||
const HIDDEN_FOR_PAGES = new Set([
|
const HIDDEN_FOR_PAGES = new Set([
|
||||||
'chats',
|
'chats',
|
||||||
'chat'
|
'chat',
|
||||||
|
'lists-edit'
|
||||||
])
|
])
|
||||||
|
|
||||||
const MobilePostStatusButton = {
|
const MobilePostStatusButton = {
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
overflow-x: auto;
|
overflow-x: auto;
|
||||||
padding-top: 5px;
|
padding-top: 5px;
|
||||||
flex-direction: row;
|
flex-direction: row;
|
||||||
|
flex: 0 0 auto;
|
||||||
|
|
||||||
&::after, &::before {
|
&::after, &::before {
|
||||||
content: '';
|
content: '';
|
||||||
|
|
|
@ -80,6 +80,9 @@
|
||||||
"confirm": "Confirm",
|
"confirm": "Confirm",
|
||||||
"verify": "Verify",
|
"verify": "Verify",
|
||||||
"close": "Close",
|
"close": "Close",
|
||||||
|
"undo": "Undo",
|
||||||
|
"yes": "Yes",
|
||||||
|
"no": "No",
|
||||||
"peek": "Peek",
|
"peek": "Peek",
|
||||||
"role": {
|
"role": {
|
||||||
"admin": "Admin",
|
"admin": "Admin",
|
||||||
|
@ -981,7 +984,15 @@
|
||||||
"save": "Save changes",
|
"save": "Save changes",
|
||||||
"delete": "Delete list",
|
"delete": "Delete list",
|
||||||
"following_only": "Limit to Following",
|
"following_only": "Limit to Following",
|
||||||
"manage_lists": "Manage lists"
|
"manage_lists": "Manage lists",
|
||||||
|
"manage_members": "Manage list members",
|
||||||
|
"add_members": "Search for more users",
|
||||||
|
"remove_from_list": "Remove from list",
|
||||||
|
"add_to_list": "Add to list",
|
||||||
|
"is_in_list": "Already in list",
|
||||||
|
"editing_list": "Editing list {listTitle}",
|
||||||
|
"update_title": "Save Title",
|
||||||
|
"really_delete": "Really delete list?"
|
||||||
},
|
},
|
||||||
"file_type": {
|
"file_type": {
|
||||||
"audio": "Audio",
|
"audio": "Audio",
|
||||||
|
|
Loading…
Reference in a new issue