page list instead of hoc withloadmore
This commit is contained in:
parent
ce43c81ce8
commit
4a00d89494
9 changed files with 327 additions and 72 deletions
50
src/components/page_list/page_list.js
Normal file
50
src/components/page_list/page_list.js
Normal file
|
|
@ -0,0 +1,50 @@
|
|||
//import Checkbox from 'src/components/checkbox/checkbox.vue'
|
||||
import SelectableList from 'src/components/selectable_list/selectable_list.vue'
|
||||
|
||||
const PageList = {
|
||||
components: {
|
||||
SelectableList
|
||||
},
|
||||
props: {
|
||||
boxOnly: {
|
||||
type: Boolean,
|
||||
default: false
|
||||
},
|
||||
pageSize: {
|
||||
type: Number,
|
||||
default: 50
|
||||
},
|
||||
fetchPage: {
|
||||
type: Function,
|
||||
default: async () => []
|
||||
}
|
||||
},
|
||||
data () {
|
||||
return {
|
||||
pageIndex: 1,
|
||||
items: [],
|
||||
selected: [],
|
||||
canLoadMore: true
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
reset () {
|
||||
this.canLoadMore = true
|
||||
this.pageIndex = 1
|
||||
this.items = []
|
||||
this.loadMore() // load one page
|
||||
},
|
||||
loadMore () {
|
||||
this.fetchPage(this.$store, {
|
||||
page: this.pageIndex++,
|
||||
pageSize: this.pageSize
|
||||
}).then((items) => this.items = [...this.items, ...items])
|
||||
// fetch page, add to items
|
||||
//this.$forceUpdate()
|
||||
}
|
||||
},
|
||||
mounted () {
|
||||
this.reset()
|
||||
}
|
||||
}
|
||||
export default PageList
|
||||
58
src/components/page_list/page_list.vue
Normal file
58
src/components/page_list/page_list.vue
Normal file
|
|
@ -0,0 +1,58 @@
|
|||
<template>
|
||||
<div class="page-list">
|
||||
<SelectableList
|
||||
:box-only="true"
|
||||
:get-key="i => i"
|
||||
:items="items"
|
||||
>
|
||||
<template v-slot:item="slotProps">
|
||||
<slot name="item" v-bind="slotProps"/>
|
||||
</template>
|
||||
</SelectableList>
|
||||
<button
|
||||
v-if="canLoadMore"
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="loadMore"
|
||||
>
|
||||
{{ $t('page_list.load_more') }}
|
||||
</button>
|
||||
<p> prev first next </p>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script src="./page_list.js"></script>
|
||||
|
||||
<style lang="scss">
|
||||
.page-list {
|
||||
--__line-height: 1.5em;
|
||||
--__horizontal-gap: 0.75em;
|
||||
--__vertical-gap: 0.5em;
|
||||
|
||||
&-item-inner {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
|
||||
> * {
|
||||
min-width: 0;
|
||||
}
|
||||
}
|
||||
|
||||
&-header {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding: var(--__vertical-gap) var(--__horizontal-gap);
|
||||
border-bottom: 1px solid;
|
||||
border-bottom-color: var(--border);
|
||||
|
||||
&-actions {
|
||||
flex: 1;
|
||||
}
|
||||
}
|
||||
|
||||
&-checkbox-wrapper {
|
||||
padding-right: var(--__horizontal-gap);
|
||||
flex: none;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
|
@ -8,6 +8,9 @@ const AdminCard = {
|
|||
}
|
||||
},
|
||||
computed: {
|
||||
isLoaded () {
|
||||
return typeof(this.user) !== 'undefined'
|
||||
},
|
||||
user () {
|
||||
return this.$store.getters.findUser(this.userId)
|
||||
},
|
||||
|
|
|
|||
|
|
@ -1,4 +1,12 @@
|
|||
<template>
|
||||
<div
|
||||
v-if="!isLoaded"
|
||||
>
|
||||
loading user...
|
||||
</div>
|
||||
<div
|
||||
v-else
|
||||
>
|
||||
<BasicUserCard :user="user">
|
||||
<div class="admin-card-content-container">
|
||||
<!--<button
|
||||
|
|
@ -29,6 +37,7 @@
|
|||
</button>-->
|
||||
</div>
|
||||
</BasicUserCard>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<script src="./admin_card.js"></script>
|
||||
|
|
|
|||
|
|
@ -2,34 +2,11 @@
|
|||
import Checkbox from 'src/components/checkbox/checkbox.vue'
|
||||
import Select from 'src/components/select/select.vue'
|
||||
import BasicUserCard from 'src/components/basic_user_card/basic_user_card.vue'
|
||||
import withLoadMore from 'src/components/../hocs/with_load_more/with_load_more'
|
||||
import SelectableList from 'src/components/selectable_list/selectable_list.vue'
|
||||
import ProgressButton from 'src/components/progress_button/progress_button.vue'
|
||||
import AdminCard from 'src/components/settings_modal/admin_tabs/admin_card.vue'
|
||||
//import { ref } from 'vue'
|
||||
import PageList from 'src/components/page_list/page_list.vue'
|
||||
|
||||
const UserList = withLoadMore({
|
||||
fetch: (props, $store) => {
|
||||
console.log('fetch', props)
|
||||
return $store.dispatch('fetchAdminUsers')
|
||||
},
|
||||
select: (props, $store) => {
|
||||
console.log('select', props)
|
||||
const filterMethod = typeof props.filterMethod === 'function' ? props.filterMethod : () => true
|
||||
const users = $store.state.users.users.filter(user => user.id !== $store.state.users.currentUser.id && filterMethod(user))
|
||||
console.log('users', users)
|
||||
return users
|
||||
},
|
||||
destroy: () => {},
|
||||
childPropName: 'items'
|
||||
})(SelectableList)
|
||||
|
||||
const UserSortStrategy = Object.freeze({
|
||||
NONE: 0,
|
||||
NICKNAME: 1,
|
||||
DISPLAYNAME: 2,
|
||||
JOINED: 3
|
||||
})
|
||||
|
||||
const UsersTab = {
|
||||
provide () {
|
||||
|
|
@ -40,13 +17,17 @@ const UsersTab = {
|
|||
},
|
||||
data() {
|
||||
return {
|
||||
filterTerms: [],
|
||||
users: [],
|
||||
sortStrategy: UserSortStrategy.NONE,
|
||||
sortAscending: true,
|
||||
filters: {
|
||||
local: false,
|
||||
external: false,
|
||||
active: false,
|
||||
need_approval: false,
|
||||
unconfirmed: false,
|
||||
deactivated: false,
|
||||
is_admin: false,
|
||||
is_moderator: false
|
||||
},
|
||||
expandedUser: null,
|
||||
filterActive: 'active_only',
|
||||
filterLocal: 'local_only',
|
||||
loading: false
|
||||
}
|
||||
},
|
||||
|
|
@ -54,47 +35,39 @@ const UsersTab = {
|
|||
Checkbox,
|
||||
Select,
|
||||
BasicUserCard,
|
||||
UserList,
|
||||
PageList,
|
||||
ProgressButton,
|
||||
AdminCard,
|
||||
},
|
||||
computed: {
|
||||
knownDomains () {
|
||||
return this.$store.state.instance.knownDomains
|
||||
},
|
||||
user () {
|
||||
return this.$store.state.users.currentUser
|
||||
}
|
||||
},
|
||||
methods: {
|
||||
delete_selection () {
|
||||
console.log('delete selection')
|
||||
},
|
||||
delete_user () {},
|
||||
fetch_page (store, opts) {
|
||||
opts.query = ""
|
||||
console.log('current filters:', this.filters)
|
||||
opts.filters = this.filters
|
||||
opts.name = ""
|
||||
opts.email = ""
|
||||
const users = store.dispatch('fetchAdminUsers', opts)
|
||||
console.log('users', users)
|
||||
return users
|
||||
},
|
||||
reset () {
|
||||
this.$refs.userList.reset()
|
||||
},
|
||||
toggleLocal () {
|
||||
console.log('toggle local')
|
||||
this.filters.local = !this.filters.local
|
||||
this.reset()
|
||||
}
|
||||
},
|
||||
mounted() {
|
||||
console.log("mounted")
|
||||
/*const store = this.$store;
|
||||
const moduleTree = buildModuleTree(store._modules.root);
|
||||
console.log(JSON.stringify(moduleTree, null, 2));*/
|
||||
}
|
||||
}
|
||||
/*function buildModuleTree(module, path = []) {
|
||||
const fullPath = path.join('/') || 'root';
|
||||
|
||||
const moduleInfo = {
|
||||
path: fullPath,
|
||||
namespaced: module.namespaced,
|
||||
state: Object.keys(module.state),
|
||||
actions: module._rawModule.actions ? Object.keys(module._rawModule.actions) : [],
|
||||
mutations: module._rawModule.mutations ? Object.keys(module._rawModule.mutations) : [],
|
||||
getters: module._rawModule.getters ? Object.keys(module._rawModule.getters) : [],
|
||||
modules: []
|
||||
};
|
||||
|
||||
if (module._children) {
|
||||
for (const key in module._children) {
|
||||
const child = module._children[key];
|
||||
moduleInfo.modules.push(buildModuleTree(child, path.concat(key)));
|
||||
}
|
||||
}
|
||||
|
||||
return moduleInfo;
|
||||
}*/
|
||||
export default UsersTab
|
||||
|
|
|
|||
|
|
@ -1,14 +1,101 @@
|
|||
<template>
|
||||
<div :label="$t('admin_dash.tabs.instance')">
|
||||
<UserList
|
||||
<div :label="$t('admin_dash.tabs.users')">
|
||||
<Checkbox
|
||||
:model-value="filters[local]"
|
||||
@update:model-value="v => {filters.local = v; reset();}"
|
||||
>
|
||||
only local
|
||||
</Checkbox><br>
|
||||
<Checkbox
|
||||
:model-value="filters[external]"
|
||||
@update:model-value="v => {filters.external = v; reset();}"
|
||||
>
|
||||
only external
|
||||
</Checkbox><br>
|
||||
<Checkbox
|
||||
:model-value="filters[active]"
|
||||
@update:model-value="v => {filters.active = v; reset();}"
|
||||
>
|
||||
only active
|
||||
</Checkbox><br>
|
||||
<Checkbox
|
||||
:model-value="filters[need_approval]"
|
||||
@update:model-value="v => {filters.need_approval = v; reset();}"
|
||||
>
|
||||
only unconfirmed
|
||||
</Checkbox><br>
|
||||
<Checkbox
|
||||
:model-value="filters[deactivated]"
|
||||
@update:model-value="v => {filters.deactivated = v; reset();}"
|
||||
>
|
||||
only deactivated
|
||||
</Checkbox><br>
|
||||
<Checkbox
|
||||
:model-value="filters[is_admin]"
|
||||
@update:model-value="v => {filters.is_admin = v; reset();}"
|
||||
>
|
||||
only if admin
|
||||
</Checkbox><br>
|
||||
<Checkbox
|
||||
:model-value="filters[is_moderator]"
|
||||
@update:model-value="v => {filters.is_moderator = v; reset();}"
|
||||
>
|
||||
only if moderator
|
||||
</Checkbox><br>
|
||||
<button
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="delete_selection"
|
||||
>
|
||||
{{ $t('admin_dash.users.activate') }}
|
||||
</button>
|
||||
<button
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="activate_selection"
|
||||
>
|
||||
{{ $t('admin_dash.users.deactivate') }}
|
||||
</button>
|
||||
<button
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="deactivate_selection"
|
||||
>
|
||||
{{ $t('admin_dash.users.delete') }}
|
||||
</button>
|
||||
<PageList
|
||||
ref="userList"
|
||||
:refresh="true"
|
||||
:get-key="i => i"
|
||||
:boxOnly="true"
|
||||
:box-only="true"
|
||||
:page-size="50"
|
||||
:fetch-page="(store, opts) => this.fetch_page(store, opts)"
|
||||
>
|
||||
<template #item="{item}">
|
||||
<AdminCard :userId="item.id" />
|
||||
<AdminCard :user-id="item.id" />
|
||||
<button
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="delete_user"
|
||||
>
|
||||
{{ $t('admin_dash.users.activate') }}
|
||||
</button>
|
||||
<button
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="activate_user"
|
||||
>
|
||||
{{ $t('admin_dash.users.deactivate') }}
|
||||
</button>
|
||||
<button
|
||||
class="button button-default btn"
|
||||
type="button"
|
||||
@click="deactivate_user"
|
||||
>
|
||||
{{ $t('admin_dash.users.delete') }}
|
||||
</button>
|
||||
</template>
|
||||
</UserList>
|
||||
</PageList>
|
||||
</div>
|
||||
</template>
|
||||
<script src="./users_tab.js"></script>
|
||||
|
|
|
|||
|
|
@ -376,6 +376,9 @@
|
|||
"selectable_list": {
|
||||
"select_all": "Select all"
|
||||
},
|
||||
"page_list": {
|
||||
"load_more": "Load more"
|
||||
},
|
||||
"settings": {
|
||||
"add_language": "Add fallback language",
|
||||
"remove_language": "Remove",
|
||||
|
|
@ -1138,7 +1141,10 @@
|
|||
"local_only": "local only",
|
||||
"external_only": "external only"
|
||||
},
|
||||
"loading": "Loading..."
|
||||
"loading": "Loading...",
|
||||
"deactivate": "deactivate",
|
||||
"activate": "activate",
|
||||
"delete": "delete"
|
||||
},
|
||||
"limits": {
|
||||
"arbitrary_limits": "Arbitrary limits",
|
||||
|
|
|
|||
|
|
@ -60,8 +60,12 @@ const adminSettingsStorage = {
|
|||
}
|
||||
},
|
||||
actions: {
|
||||
fetchAdminUsers (store) {
|
||||
store.rootState.api.backendInteractor.adminListUsers().then((users) => users.forEach(user => store.dispatch('fetchUserIfMissing', user.id)))
|
||||
async fetchAdminUsers (store, opts) {
|
||||
const users = await store.rootState.api.backendInteractor.adminListUsers({opts})
|
||||
//await Promise.all(
|
||||
users.map(user => store.dispatch('fetchUserIfMissing', user.id))
|
||||
//)
|
||||
return users
|
||||
},
|
||||
loadFrontendsStuff ({ rootState, commit }) {
|
||||
rootState.api.backendInteractor.fetchAvailableFrontends()
|
||||
|
|
|
|||
|
|
@ -115,7 +115,30 @@ const PLEROMA_ADMIN_CONFIG_URL = '/api/pleroma/admin/config'
|
|||
const PLEROMA_ADMIN_DESCRIPTIONS_URL = '/api/pleroma/admin/config/descriptions'
|
||||
const PLEROMA_ADMIN_FRONTENDS_URL = '/api/pleroma/admin/frontends'
|
||||
const PLEROMA_ADMIN_FRONTENDS_INSTALL_URL = '/api/pleroma/admin/frontends/install'
|
||||
const PLEROMA_ADMIN_USERS_URL = '/api/v1/pleroma/admin/users'
|
||||
const PLEROMA_ADMIN_USERS_URL = ({page, pageSize, filters = {}, query = '', name = '', email = ''}) => {
|
||||
const {
|
||||
local = false,
|
||||
external = false,
|
||||
active = false,
|
||||
need_approval = false,
|
||||
unconfirmed = false,
|
||||
deactivated = false,
|
||||
is_admin = true,
|
||||
is_moderator = true,
|
||||
} = filters
|
||||
const filters_str = (local ? 'local,' : '')
|
||||
+ (external ? 'external,' : '')
|
||||
+ (active ? 'active,' : '')
|
||||
+ (need_approval ? 'need_approval,' : '')
|
||||
+ (unconfirmed ? 'unconfirmed,' : '')
|
||||
+ (deactivated ? 'deactivated,' : '')
|
||||
+ (is_admin ? 'is_admin,' : '')
|
||||
+ (is_moderator ? 'is_moderator,' : '')
|
||||
return `/api/v1/pleroma/admin/users?page=${page}&page_size=${pageSize}&filters=${filters_str}&query=${query}&name=${name}&email=${email}`
|
||||
}
|
||||
const PLEROMA_ADMIN_DELETE_USERS_URL = '/api/v1/pleroma/admin/users'
|
||||
const PLEROMA_ADMIN_ACTIVATE_USER_URL = '/api/v1/pleroma/admin/users/activate'
|
||||
const PLEROMA_ADMIN_DEACTIVATE_USER_URL = '/api/v1/pleroma/admin/users/deactivate'
|
||||
|
||||
const PLEROMA_EMOJI_RELOAD_URL = '/api/pleroma/admin/reload_emoji'
|
||||
const PLEROMA_EMOJI_IMPORT_FS_URL = '/api/pleroma/emoji/packs/import'
|
||||
|
|
@ -1441,10 +1464,49 @@ const dismissAnnouncement = ({ id, credentials }) => {
|
|||
})
|
||||
}
|
||||
|
||||
const adminListUsers = ({ credentials }) => {
|
||||
const adminListUsers = ({ opts, credentials }) => {
|
||||
// the reported list is hardly useful because standards are for dating i guess,
|
||||
// so make sure to fetchIfMissing right afterward using this call
|
||||
return promisedRequest({ url: PLEROMA_ADMIN_USERS_URL, credentials }).then((data) => data.users.map(parseUser))
|
||||
const url = PLEROMA_ADMIN_USERS_URL(opts)
|
||||
console.log('admin api: ', url)
|
||||
return promisedRequest({ url: url,
|
||||
credentials,
|
||||
method: 'GET'
|
||||
}).then((data) => data.users.map(parseUser))
|
||||
}
|
||||
|
||||
const adminDeleteUser = ({ nicknames, credentials }) => {
|
||||
return promisedRequest({
|
||||
url: PLEROMA_ADMIN_DELETE_USERS_URL,
|
||||
credentials,
|
||||
method: 'DELETE',
|
||||
headers: {
|
||||
'Content-Type': 'application/json'
|
||||
},
|
||||
body: JSON.stringify({'nicknames': nicknames})
|
||||
})
|
||||
}
|
||||
|
||||
const adminActivateUser = ({ nicknames, credentials }) => {
|
||||
return promisedRequest({ url: PLEROMA_ADMIN_ACTIVATE_USER_URL,
|
||||
credentials,
|
||||
method: 'PATCH',
|
||||
headers: {
|
||||
'Content-Type': 'application/json'
|
||||
},
|
||||
body: JSON.stringify({'nicknames': nicknames})
|
||||
})
|
||||
}
|
||||
|
||||
const adminDeactivateUser = ({ nicknames, credentials }) => {
|
||||
return promisedRequest({ url: PLEROMA_ADMIN_DEACTIVATE_USER_URL,
|
||||
credentials,
|
||||
method: 'PATCH',
|
||||
headers: {
|
||||
'Content-Type': 'application/json'
|
||||
},
|
||||
body: JSON.stringify({'nicknames': nicknames})
|
||||
})
|
||||
}
|
||||
|
||||
const announcementToPayload = ({ content, startsAt, endsAt, allDay }) => {
|
||||
|
|
@ -2120,6 +2182,9 @@ const apiService = {
|
|||
updateBookmarkFolder,
|
||||
deleteBookmarkFolder,
|
||||
adminListUsers,
|
||||
adminDeleteUser,
|
||||
adminActivateUser,
|
||||
adminDeactivateUser,
|
||||
}
|
||||
|
||||
export default apiService
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue