Merge branch 'feature/update-status-count' into 'develop'
Update status count when an instance was selected Closes #112 See merge request pleroma/admin-fe!129
This commit is contained in:
commit
9c28eddec1
10 changed files with 328 additions and 12 deletions
|
@ -6,6 +6,15 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).
|
||||||
|
|
||||||
## Unreleased
|
## Unreleased
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- Statuses count changes when an instance is selected and shows the amount of statuses from an originating instance
|
||||||
|
- Add a dialog window with a confirmation when a remove button is clicked on the Settings page
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Send `true` and `false` as booleans if they are values of single selects on the Settings page
|
||||||
|
|
||||||
## [2.0.3] - 2020-04-29
|
## [2.0.3] - 2020-04-29
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
4
src/api/__mocks__/peers.js
Normal file
4
src/api/__mocks__/peers.js
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
export async function fetchPeers(authHost, token) {
|
||||||
|
const data = ['lain.com', 'heaven.com']
|
||||||
|
return Promise.resolve({ data })
|
||||||
|
}
|
|
@ -5,3 +5,66 @@ export async function changeStatusScope(id, sensitive, visibility, authHost, tok
|
||||||
export async function deleteStatus(id, authHost, token) {
|
export async function deleteStatus(id, authHost, token) {
|
||||||
return Promise.resolve()
|
return Promise.resolve()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function fetchStatusesByInstance({ instance, authHost, token, pageSize, page }) {
|
||||||
|
let data
|
||||||
|
if (pageSize === 1) {
|
||||||
|
data = page === 1 || page === 2
|
||||||
|
? [{
|
||||||
|
'account': {
|
||||||
|
'avatar': 'http://localhost:4000/images/avi.png',
|
||||||
|
'display_name': 'sky',
|
||||||
|
'url': 'http://localhost:4000/users/sky'
|
||||||
|
},
|
||||||
|
'content': 'A nice young couple contacted us from Brazil to decorate their newly acquired apartment.',
|
||||||
|
'created_at': '2020-01-31T18:20:01.000Z',
|
||||||
|
'id': '9rZIr0Jzao5Gjgfmro',
|
||||||
|
'sensitive': false,
|
||||||
|
'url': 'http://localhost:4000/objects/7af9abbd-fb6c-4318-aeb7-6636c138ac98',
|
||||||
|
'visibility': 'unlisted'
|
||||||
|
}]
|
||||||
|
: []
|
||||||
|
} else {
|
||||||
|
data = [
|
||||||
|
{
|
||||||
|
'account': {
|
||||||
|
'avatar': 'http://localhost:4000/images/avi.png',
|
||||||
|
'display_name': 'sky',
|
||||||
|
'url': 'http://localhost:4000/users/sky'
|
||||||
|
},
|
||||||
|
'content': 'A nice young couple contacted us from Brazil to decorate their newly acquired apartment.',
|
||||||
|
'created_at': '2020-01-31T18:20:01.000Z',
|
||||||
|
'id': '9rZIr0Jzao5Gjgfmro',
|
||||||
|
'sensitive': false,
|
||||||
|
'url': 'http://localhost:4000/objects/7af9abbd-fb6c-4318-aeb7-6636c138ac98',
|
||||||
|
'visibility': 'unlisted'
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'account': {
|
||||||
|
'avatar': 'http://localhost:4000/images/avi.png',
|
||||||
|
'display_name': 'sky',
|
||||||
|
'url': 'http://localhost:4000/users/sky'
|
||||||
|
},
|
||||||
|
'content': 'the happiest man ever',
|
||||||
|
'created_at': '2019-11-23T12:56:18.000Z',
|
||||||
|
'id': '9pFoVfWMU3A96Rzq3k',
|
||||||
|
'sensitive': false,
|
||||||
|
'url': 'http://localhost:4000/objects/449c90fe-c457-4c64-baf2-fe6d0a59ca25',
|
||||||
|
'visibility': 'unlisted'
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
return Promise.resolve({ data })
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchStatusesCount(instance, authHost, token) {
|
||||||
|
const data = instance === 'heaven.com'
|
||||||
|
? {
|
||||||
|
'status_visibility':
|
||||||
|
{ 'direct': 1, 'private': 2, 'public': 3, 'unlisted': 0 }
|
||||||
|
}
|
||||||
|
: {
|
||||||
|
'status_visibility':
|
||||||
|
{ 'direct': 4, 'private': 10, 'public': 4, 'unlisted': 10 }
|
||||||
|
}
|
||||||
|
return Promise.resolve({ data })
|
||||||
|
}
|
||||||
|
|
|
@ -30,10 +30,10 @@ export async function fetchStatuses({ godmode, localOnly, authHost, token, pageS
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fetchStatusesCount(authHost, token) {
|
export async function fetchStatusesCount(instance, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
baseURL: baseName(authHost),
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/stats`,
|
url: instance ? `/api/pleroma/admin/stats?instance=${instance}` : `/api/pleroma/admin/stats`,
|
||||||
method: 'get',
|
method: 'get',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
|
|
|
@ -58,6 +58,14 @@ const status = {
|
||||||
dispatch('FetchStatusesByInstance')
|
dispatch('FetchStatusesByInstance')
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
ClearState({ commit }) {
|
||||||
|
commit('CHANGE_SELECTED_INSTANCE', '')
|
||||||
|
commit('SET_STATUSES_BY_INSTANCE', [])
|
||||||
|
commit('CHANGE_LOCAL_CHECKBOX_VALUE', false)
|
||||||
|
commit('CHANGE_GODMODE_CHECKBOX_VALUE', false)
|
||||||
|
commit('SET_ALL_LOADED', false)
|
||||||
|
commit('CHANGE_PAGE', 1)
|
||||||
|
},
|
||||||
async DeleteStatus({ dispatch, getters }, { statusId, reportCurrentPage, userId, godmode, fetchStatusesByInstance }) {
|
async DeleteStatus({ dispatch, getters }, { statusId, reportCurrentPage, userId, godmode, fetchStatusesByInstance }) {
|
||||||
await deleteStatus(statusId, getters.authHost, getters.token)
|
await deleteStatus(statusId, getters.authHost, getters.token)
|
||||||
if (reportCurrentPage !== 0) { // called from Reports
|
if (reportCurrentPage !== 0) { // called from Reports
|
||||||
|
@ -68,14 +76,15 @@ const status = {
|
||||||
dispatch('FetchStatusesByInstance')
|
dispatch('FetchStatusesByInstance')
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
async FetchStatusesCount({ commit, getters }) {
|
async FetchStatusesCount({ commit, getters }, instance) {
|
||||||
commit('SET_LOADING', true)
|
commit('SET_LOADING', true)
|
||||||
const { data } = await fetchStatusesCount(getters.authHost, getters.token)
|
const { data } = await fetchStatusesCount(instance, getters.authHost, getters.token)
|
||||||
commit('SET_STATUS_VISIBILITY', data.status_visibility)
|
commit('SET_STATUS_VISIBILITY', data.status_visibility)
|
||||||
commit('SET_LOADING', false)
|
commit('SET_LOADING', false)
|
||||||
},
|
},
|
||||||
async FetchStatusesByInstance({ commit, getters, state, rootState }) {
|
async FetchStatusesByInstance({ commit, dispatch, getters, state, rootState }) {
|
||||||
commit('SET_LOADING', true)
|
commit('SET_LOADING', true)
|
||||||
|
dispatch('FetchStatusesCount', state.statusesByInstance.selectedInstance)
|
||||||
if (state.statusesByInstance.selectedInstance === '') {
|
if (state.statusesByInstance.selectedInstance === '') {
|
||||||
commit('SET_STATUSES_BY_INSTANCE', [])
|
commit('SET_STATUSES_BY_INSTANCE', [])
|
||||||
} else {
|
} else {
|
||||||
|
|
|
@ -8,10 +8,18 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="statuses-header-container">
|
<div class="statuses-header-container">
|
||||||
<el-button-group>
|
<el-button-group>
|
||||||
<el-button plain>{{ $t('statuses.direct') }}: {{ statusVisibility.direct }}</el-button>
|
<el-button plain class="direct-button">
|
||||||
<el-button plain>{{ $t('statuses.private') }}: {{ statusVisibility.private }}</el-button>
|
{{ $t('statuses.direct') }}: {{ normalizedCount(statusVisibility.direct) }}
|
||||||
<el-button plain>{{ $t('statuses.public') }}: {{ statusVisibility.public }}</el-button>
|
</el-button>
|
||||||
<el-button plain>{{ $t('statuses.unlisted') }}: {{ statusVisibility.unlisted }}</el-button>
|
<el-button plain class="private-button">
|
||||||
|
{{ $t('statuses.private') }}: {{ normalizedCount(statusVisibility.private) }}
|
||||||
|
</el-button>
|
||||||
|
<el-button plain class="public-button">
|
||||||
|
{{ $t('statuses.public') }}: {{ normalizedCount(statusVisibility.public) }}
|
||||||
|
</el-button>
|
||||||
|
<el-button plain class="unlisted-button">
|
||||||
|
{{ $t('statuses.unlisted') }}: {{ normalizedCount(statusVisibility.unlisted) }}
|
||||||
|
</el-button>
|
||||||
</el-button-group>
|
</el-button-group>
|
||||||
</div>
|
</div>
|
||||||
<div class="filter-container">
|
<div class="filter-container">
|
||||||
|
@ -61,6 +69,7 @@
|
||||||
import MultipleUsersMenu from '@/views/users/components/MultipleUsersMenu'
|
import MultipleUsersMenu from '@/views/users/components/MultipleUsersMenu'
|
||||||
import Status from '@/components/Status'
|
import Status from '@/components/Status'
|
||||||
import RebootButton from '@/components/RebootButton'
|
import RebootButton from '@/components/RebootButton'
|
||||||
|
import numeral from 'numeral'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'Statuses',
|
name: 'Statuses',
|
||||||
|
@ -142,7 +151,14 @@ export default {
|
||||||
this.$store.dispatch('FetchPeers')
|
this.$store.dispatch('FetchPeers')
|
||||||
this.$store.dispatch('FetchStatusesCount')
|
this.$store.dispatch('FetchStatusesCount')
|
||||||
},
|
},
|
||||||
|
destroyed() {
|
||||||
|
this.clearSelection()
|
||||||
|
this.$store.dispatch('ClearState')
|
||||||
|
},
|
||||||
methods: {
|
methods: {
|
||||||
|
clearSelection() {
|
||||||
|
this.selectedUsers = []
|
||||||
|
},
|
||||||
handleFilterChange() {
|
handleFilterChange() {
|
||||||
this.$store.dispatch('HandlePageChange', 1)
|
this.$store.dispatch('HandlePageChange', 1)
|
||||||
this.$store.dispatch('FetchStatusesByInstance')
|
this.$store.dispatch('FetchStatusesByInstance')
|
||||||
|
@ -152,14 +168,14 @@ export default {
|
||||||
|
|
||||||
this.$store.dispatch('FetchStatusesPageByInstance')
|
this.$store.dispatch('FetchStatusesPageByInstance')
|
||||||
},
|
},
|
||||||
clearSelection() {
|
|
||||||
this.selectedUsers = []
|
|
||||||
},
|
|
||||||
handleStatusSelection(user) {
|
handleStatusSelection(user) {
|
||||||
if (this.selectedUsers.find(selectedUser => user.id === selectedUser.id) !== undefined) {
|
if (this.selectedUsers.find(selectedUser => user.id === selectedUser.id) !== undefined) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
this.selectedUsers = [...this.selectedUsers, user]
|
this.selectedUsers = [...this.selectedUsers, user]
|
||||||
|
},
|
||||||
|
normalizedCount(count) {
|
||||||
|
return numeral(count).format('0a')
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -175,6 +191,13 @@ export default {
|
||||||
margin: 0 0 10px;
|
margin: 0 0 10px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
.statuses-header-container {
|
||||||
|
.el-button.is-plain:focus, .el-button.is-plain:hover {
|
||||||
|
border-color: #DCDFE6;
|
||||||
|
color: #606266;
|
||||||
|
cursor: default
|
||||||
|
}
|
||||||
|
}
|
||||||
.checkbox-container {
|
.checkbox-container {
|
||||||
margin-bottom: 15px;
|
margin-bottom: 15px;
|
||||||
}
|
}
|
||||||
|
@ -228,8 +251,26 @@ export default {
|
||||||
.statuses-header-container {
|
.statuses-header-container {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
|
.el-button-group {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
.el-button {
|
.el-button {
|
||||||
padding: 10px 6.5px;
|
padding: 10px 6.5px;
|
||||||
|
width: 50%;
|
||||||
|
}
|
||||||
|
.el-button-group>.el-button:first-child {
|
||||||
|
border-bottom-left-radius: 0;
|
||||||
|
}
|
||||||
|
.el-button-group>.el-button:not(:first-child):not(:last-child).private-button {
|
||||||
|
border-top-right-radius: 4px;
|
||||||
|
}
|
||||||
|
.el-button-group>.el-button:not(:first-child):not(:last-child).public-button {
|
||||||
|
border-bottom-left-radius: 4px;
|
||||||
|
border-top: white;
|
||||||
|
}
|
||||||
|
.el-button-group>.el-button:last-child {
|
||||||
|
border-top-right-radius: 0;
|
||||||
|
border-top: white;
|
||||||
}
|
}
|
||||||
.reboot-button {
|
.reboot-button {
|
||||||
margin: 10px 0 0 0;
|
margin: 10px 0 0 0;
|
||||||
|
|
103
test/views/statuses/index.test.js
Normal file
103
test/views/statuses/index.test.js
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
import Vuex from 'vuex'
|
||||||
|
import { mount, createLocalVue, config } from '@vue/test-utils'
|
||||||
|
import flushPromises from 'flush-promises'
|
||||||
|
import Element from 'element-ui'
|
||||||
|
import Statuses from '@/views/statuses/index'
|
||||||
|
import storeConfig from './store.conf'
|
||||||
|
import { cloneDeep } from 'lodash'
|
||||||
|
|
||||||
|
config.mocks["$t"] = () => {}
|
||||||
|
|
||||||
|
const localVue = createLocalVue()
|
||||||
|
localVue.use(Vuex)
|
||||||
|
localVue.use(Element)
|
||||||
|
|
||||||
|
jest.mock('@/api/app')
|
||||||
|
jest.mock('@/api/status')
|
||||||
|
jest.mock('@/api/peers')
|
||||||
|
jest.mock('@/api/nodeInfo')
|
||||||
|
|
||||||
|
describe('Statuses', () => {
|
||||||
|
let store
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
store = new Vuex.Store(cloneDeep(storeConfig))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('fetches peers and statuses count', async (done) => {
|
||||||
|
mount(Statuses, {
|
||||||
|
store,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
|
||||||
|
await flushPromises()
|
||||||
|
const statusVisibilityCount = store.state.status.statusVisibility
|
||||||
|
expect(statusVisibilityCount.direct).toEqual(4)
|
||||||
|
expect(statusVisibilityCount.private).toEqual(10)
|
||||||
|
expect(statusVisibilityCount.public).toEqual(4)
|
||||||
|
expect(statusVisibilityCount.unlisted).toEqual(10)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('fetches statuses from selected instance and updates the count', async (done) => {
|
||||||
|
const wrapper = mount(Statuses, {
|
||||||
|
store,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
store.dispatch('HandleFilterChange', 'heaven.com')
|
||||||
|
wrapper.vm.handleFilterChange()
|
||||||
|
await flushPromises()
|
||||||
|
const statusVisibilityCount = store.state.status.statusVisibility
|
||||||
|
|
||||||
|
expect(statusVisibilityCount.direct).toEqual(1)
|
||||||
|
expect(statusVisibilityCount.private).toEqual(2)
|
||||||
|
expect(statusVisibilityCount.public).toEqual(3)
|
||||||
|
expect(statusVisibilityCount.unlisted).toEqual(0)
|
||||||
|
expect(store.state.status.fetchedStatuses.length).toEqual(2)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('handles status select', async (done) => {
|
||||||
|
const wrapper = mount(Statuses, {
|
||||||
|
store: store,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
store.dispatch('HandleFilterChange', 'heaven.com')
|
||||||
|
wrapper.vm.handleFilterChange()
|
||||||
|
await flushPromises()
|
||||||
|
wrapper.find('.status-checkbox input').setChecked()
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
expect(wrapper.vm.selectedUsers.length).toEqual(1)
|
||||||
|
expect(wrapper.vm.selectedUsers[0].display_name).toBe('sky')
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('clear state after component was destroyed', async (done) => {
|
||||||
|
const wrapper = mount(Statuses, {
|
||||||
|
store: store,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
store.dispatch('HandleFilterChange', 'heaven.com')
|
||||||
|
wrapper.vm.handleFilterChange()
|
||||||
|
await flushPromises()
|
||||||
|
wrapper.find('.status-checkbox input').setChecked()
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
expect(wrapper.vm.selectedUsers.length).toEqual(1)
|
||||||
|
expect(store.state.status.statusesByInstance.selectedInstance).toBe('heaven.com')
|
||||||
|
expect(store.state.status.fetchedStatuses.length).toEqual(2)
|
||||||
|
wrapper.destroy()
|
||||||
|
|
||||||
|
expect(wrapper.vm.selectedUsers.length).toEqual(0)
|
||||||
|
expect(store.state.status.statusesByInstance.selectedInstance).toBe('')
|
||||||
|
expect(store.state.status.fetchedStatuses.length).toEqual(0)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
})
|
53
test/views/statuses/pagination.test.js
Normal file
53
test/views/statuses/pagination.test.js
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
import Vuex from 'vuex'
|
||||||
|
import { mount, createLocalVue, config } from '@vue/test-utils'
|
||||||
|
import flushPromises from 'flush-promises'
|
||||||
|
import Element from 'element-ui'
|
||||||
|
import Statuses from '@/views/statuses/index'
|
||||||
|
import storeConfig from './storeForPagination.conf'
|
||||||
|
import { cloneDeep } from 'lodash'
|
||||||
|
|
||||||
|
config.mocks["$t"] = () => {}
|
||||||
|
|
||||||
|
const localVue = createLocalVue()
|
||||||
|
localVue.use(Vuex)
|
||||||
|
localVue.use(Element)
|
||||||
|
|
||||||
|
jest.mock('@/api/app')
|
||||||
|
jest.mock('@/api/status')
|
||||||
|
jest.mock('@/api/peers')
|
||||||
|
jest.mock('@/api/nodeInfo')
|
||||||
|
|
||||||
|
describe('Statuses', () => {
|
||||||
|
let store
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
store = new Vuex.Store(cloneDeep(storeConfig))
|
||||||
|
})
|
||||||
|
|
||||||
|
it('pagination', async (done) => {
|
||||||
|
const wrapper = mount(Statuses, {
|
||||||
|
store,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
store.dispatch('HandleFilterChange', 'heaven.com')
|
||||||
|
wrapper.vm.handleFilterChange()
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
expect(store.state.status.statusesByInstance.allLoaded).toBe(false)
|
||||||
|
expect(store.state.status.statusesByInstance.page).toBe(1)
|
||||||
|
wrapper.find('.statuses-pagination button').trigger('click')
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
expect(store.state.status.statusesByInstance.allLoaded).toBe(false)
|
||||||
|
expect(store.state.status.statusesByInstance.page).toBe(2)
|
||||||
|
|
||||||
|
wrapper.find('.statuses-pagination button').trigger('click')
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
expect(store.state.status.statusesByInstance.allLoaded).toBe(true)
|
||||||
|
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
})
|
17
test/views/statuses/store.conf.js
Normal file
17
test/views/statuses/store.conf.js
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
import app from '@/store/modules/app'
|
||||||
|
import peers from '@/store/modules/peers'
|
||||||
|
import user from '@/store/modules/user'
|
||||||
|
import settings from '@/store/modules/settings'
|
||||||
|
import status from '@/store/modules/status'
|
||||||
|
import getters from '@/store/getters'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
modules: {
|
||||||
|
app,
|
||||||
|
peers,
|
||||||
|
settings,
|
||||||
|
status,
|
||||||
|
user: { ...user, state: { ...user.state, authHost: 'localhost:4000' }}
|
||||||
|
},
|
||||||
|
getters
|
||||||
|
}
|
17
test/views/statuses/storeForPagination.conf.js
Normal file
17
test/views/statuses/storeForPagination.conf.js
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
import app from '@/store/modules/app'
|
||||||
|
import peers from '@/store/modules/peers'
|
||||||
|
import user from '@/store/modules/user'
|
||||||
|
import settings from '@/store/modules/settings'
|
||||||
|
import status from '@/store/modules/status'
|
||||||
|
import getters from '@/store/getters'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
modules: {
|
||||||
|
app,
|
||||||
|
peers,
|
||||||
|
settings,
|
||||||
|
status: { ...status, state: { ...status.state, statusesByInstance: { ...status.state.statusesByInstance, pageSize: 1 }}},
|
||||||
|
user: { ...user, state: { ...user.state, authHost: 'localhost:4000' }}
|
||||||
|
},
|
||||||
|
getters
|
||||||
|
}
|
Loading…
Reference in a new issue