This repository has been archived on 2024-07-22. You can view files and clone it, but cannot push or open issues or pull requests.
Zaimki/routes/admin.vue

188 lines
7.6 KiB
Vue
Raw Normal View History

2020-10-29 15:41:40 -07:00
<template>
2020-12-30 15:03:30 -08:00
<NotFound v-if="!$isGranted('panel') && !$isGranted('users')"/>
2020-11-16 11:43:44 -08:00
<div v-else>
2020-10-29 15:41:40 -07:00
<h2>
<Icon v="user-cog"/>
<T>admin.header</T>
</h2>
2021-06-09 23:45:13 -07:00
<p>Stats counted: {{$datetime(stats.calculatedAt)}}</p>
2020-12-30 15:03:30 -08:00
<section v-if="$isGranted('users')">
2021-04-04 17:39:58 -07:00
<details class="border mb-3" @click="loadUsers">
2020-11-24 16:11:17 -08:00
<summary class="bg-light p-3">
<Icon v="users"/>
Users
2020-12-30 15:03:30 -08:00
({{stats.users.overall}} overall, {{stats.users.admins}} admins, {{visibleUsers.length}} visible)
2020-11-24 16:11:17 -08:00
</summary>
<div class="border-top">
2021-04-04 17:39:58 -07:00
<Loading :value="users" size="5rem">
<div class="input-group mt-4">
<input class="form-control" v-model="userFilter" :placeholder="$t('crud.filterLong')"/>
<button :class="['btn', adminsFilter ? 'btn-secondary' : 'btn-outline-secondary']"
@click="adminsFilter = !adminsFilter"
>
Only admins
</button>
<button :class="['btn', localeFilter ? 'btn-secondary' : 'btn-outline-secondary']"
@click="localeFilter = !localeFilter"
>
Only this version
</button>
</div>
<Table :data="visibleUsers" :columns="4">
<template v-slot:header>
<th class="text-nowrap">
<T>admin.user.user</T>
</th>
<th class="text-nowrap">
<T>admin.user.email</T>
</th>
<th class="text-nowrap">
<T>admin.user.roles</T>
</th>
<th class="text-nowrap">
<T>admin.user.profiles</T>
</th>
</template>
2020-10-29 15:41:40 -07:00
2021-04-04 17:39:58 -07:00
<template v-slot:row="s">
<td>
<Avatar :user="s.el" dsize="2rem"/>
{{s.el.username}}
</td>
<td>
<p>
<a :href="`mailto:${s.el.email}`" target="_blank" rel="noopener">
{{s.el.email}}
</a>
</p>
<ul v-if="s.el.socialConnections.length" class="list-inline">
<li v-for="conn in s.el.socialConnections" class="list-inline-item">
<Icon :v="socialProviders[conn].icon || conn" set="b"/>
</li>
</ul>
</td>
<td>
<Roles :user="s.el"/>
</td>
<td>
<ul class="list-unstyled">
<li v-for="locale in s.el.profiles" v-if="locales[locale]">
<LocaleLink :link="`/@${s.el.username}`" :locale="locale">
{{ locales[locale].name }}
</LocaleLink>
</li>
</ul>
</td>
</template>
</Table>
</Loading>
2020-11-24 16:11:17 -08:00
</div>
</details>
</section>
2021-04-10 04:04:16 -07:00
<ChartSet name="users" :data="stats.users.chart" init="cumulative"/>
2021-04-08 15:43:57 -07:00
2020-11-24 16:11:17 -08:00
<section v-for="(locale, k) in stats.locales" :key="k">
<details class="border mb-3" open>
<summary class="bg-light p-3">
<LocaleLink :locale="k" link="/">{{locale.name}}</LocaleLink>
</summary>
<div class="p-3 border-top d-flex justify-content-between flex-column flex-md-row">
<div class="flex-grow-1">
<h4 class="h5">
<Icon v="id-card"/>
Profiles
</h4>
{{locale.profiles}}
</div>
<div class="flex-grow-1">
<h4 class="h5">
<Icon v="tags"/>
Pronouns
</h4>
2020-11-27 11:30:21 -08:00
<ListExpandable :data="locale.pronouns"/>
</div>
<div class="flex-grow-1">
<h4 class="h5">
<Icon v="flag"/>
Flags
</h4>
<ListExpandable :data="locale.flags"/>
2020-11-24 16:11:17 -08:00
</div>
<div class="flex-grow-1">
<h4 class="h5">
<Icon v="atom-alt"/>
Dictionary
</h4>
<ul class="list-unstyled">
<li>
<strong>Approved</strong>: {{locale.nouns.approved}}
</li>
<li>
<strong>Awaiting</strong>: {{locale.nouns.awaiting}}
</li>
</ul>
</div>
</div>
2021-04-08 15:43:57 -07:00
<div class="p-3 border-top">
2021-04-10 04:04:16 -07:00
<ChartSet name="profiles" :data="locale.chart"/>
2021-04-08 15:43:57 -07:00
</div>
2020-11-24 16:11:17 -08:00
</details>
</section>
2020-10-29 15:41:40 -07:00
</div>
</template>
<script>
2020-12-30 15:03:30 -08:00
import {head, isGranted} from "../src/helpers";
2020-11-02 12:45:45 -08:00
import {socialProviders} from "../src/data";
2020-10-29 15:41:40 -07:00
export default {
2020-11-02 12:45:45 -08:00
data() {
2020-11-27 11:30:21 -08:00
return {
socialProviders,
2020-12-22 02:05:33 -08:00
userFilter: '',
2020-12-30 15:03:30 -08:00
localeFilter: true,
adminsFilter: false,
2021-04-04 17:39:58 -07:00
users: undefined,
2020-11-27 11:30:21 -08:00
}
2020-11-02 12:45:45 -08:00
},
2020-10-29 15:41:40 -07:00
async asyncData({ app, store }) {
2020-12-30 15:03:30 -08:00
let stats = { users: {}};
2020-10-29 15:41:40 -07:00
2020-12-30 15:03:30 -08:00
try {
stats = await app.$axios.$get(`/admin/stats`);
} catch {}
2020-11-24 16:11:17 -08:00
2020-10-29 15:41:40 -07:00
return {
2020-11-24 16:11:17 -08:00
stats,
2020-10-29 15:41:40 -07:00
};
},
2021-04-04 17:39:58 -07:00
methods: {
async loadUsers() {
if (this.users === undefined) {
this.users = await this.$axios.$get(`/admin/users`);
}
},
},
2020-12-22 02:05:33 -08:00
computed: {
visibleUsers() {
2021-04-04 17:39:58 -07:00
if (this.users === undefined) {
return [];
}
2020-12-30 15:03:30 -08:00
return Object.values(this.users).filter(u =>
u.username.toLowerCase().includes(this.userFilter.toLowerCase())
&& (!this.adminsFilter || u.roles !== '')
&& (!this.localeFilter || u.profiles.includes(this.config.locale))
);
2020-12-22 02:05:33 -08:00
},
},
2020-10-29 15:41:40 -07:00
head() {
return head({
title: this.$t('admin.header'),
});
},
}
</script>