Files
traxxx-web/pages/admin/actors/+Page.vue

307 lines
5.6 KiB
Vue

<template>
<Admin class="page">
<div class="header">
<input
v-model="actorQuery"
type="search"
placeholder="Search actors"
class="input search"
@search="searchActors"
>
<div class="header-actions">
<button
class="button"
:disabled="selectedActors.size === 0"
@click="selectedActors = new Set()"
><Icon icon="cancel-square" />Deselect</button>
<button
class="button"
:disabled="selectedActors.size === 0"
@click="showMergeDialog = true"
><Icon icon="make-group" />Merge</button>
</div>
</div>
<table class="actors">
<thead class="actors-header">
<tr>
<th class="actor-id">ID</th>
<th class="actor-name">Entity</th>
<th class="actor-avatar">Avatar</th>
<th class="actor-name">Name</th>
</tr>
</thead>
<tbody class="actors-body">
<tr
v-for="(actor, index) in actors"
:key="`actor-${actor.id}`"
class="actor"
>
<td class="actor-id ellipsis">{{ actor.id }}</td>
<td
v-tooltip="actor.entity?.name || 'Global'"
class="actor-entity ellipsis"
>
<img
v-if="actor.entity"
:src="`/logos/${actor.entity.slug}/favicon_dark.png`"
class="actor-favicon"
>
<Icon
v-else
icon="device_hub"
class="actor-global"
/>
</td>
<td class="actor-avatar">
<img
v-if="actor.avatar"
:src="getPath(actor.avatar, 'lazy')"
loading="lazy"
class="avatar"
>
<img
v-if="actor.avatar"
:src="getPath(actor.avatar)"
loading="lazy"
class="avatar-zoom"
>
</td>
<td class="actor-name ellipsis">{{ actor.name }}</td>
<td class="actor-actions">
<div class="actions">
<Checkbox
:checked="selectedActors.has(actor.id)"
@change="(isChecked) => selectActors(actor, isChecked, index)"
/>
<Icon
v-tooltip="'Merge'"
icon="make-group"
class="actor-action action-merge"
@click="activeActor = actor; showMergeDialog = true;"
/>
<!--
<Icon
v-tooltip="'Delete'"
icon="bin"
class="actor-action action-delete"
/>
-->
</div>
</td>
</tr>
</tbody>
</table>
<Merge
v-if="showMergeDialog"
:actors="activeActor ? [activeActor] : actors.filter((actor) => selectedActors.has(actor.id))"
@close="showMergeDialog = false; activeActor = null;"
/>
</Admin>
</template>
<script setup>
import { ref, inject, onMounted } from 'vue';
import Admin from '#/components/admin/admin.vue';
import Checkbox from '#/components/form/checkbox.vue';
import Merge from '#/components/actors/merge.vue';
import getPath from '#/src/get-path.js';
import navigate from '#/src/navigate.js';
// import { get } from '#/src/api.js';
const { pageProps, urlParsed } = inject('pageContext');
const actors = ref(pageProps.actors);
const selectedActors = ref(new Set([]));
const activeActor = ref(null);
const actorQuery = ref(urlParsed.search.q || null);
const lastSelectedIndex = ref(null);
const holdingShift = ref(false);
const showMergeDialog = ref(false);
function selectActors(selectedActor, isChecked, index) {
const [start, end] = holdingShift.value
? [index, lastSelectedIndex.value].toSorted((indexA, indexB) => indexA - indexB)
: [index, index];
const actorIds = actors.value
.slice(start, end + 1)
.map((actor) => actor.id);
actorIds.forEach((actorId) => {
if (isChecked) {
selectedActors.value.add(actorId);
} else {
selectedActors.value.delete(actorId);
}
});
lastSelectedIndex.value = index;
}
async function searchActors() {
navigate('/admin/actors', { q: actorQuery.value }, { redirect: true });
}
onMounted(() => {
window.addEventListener('keydown', (event) => {
if (event.key === 'Shift') {
holdingShift.value = true;
}
});
window.addEventListener('keyup', (event) => {
if (event.key === 'Shift') {
holdingShift.value = false;
}
});
window.addEventListener('blur', () => {
holdingShift.value = false;
});
});
</script>
<style scoped>
.search {
max-width: 20rem;
flex-grow: 1;
}
.header {
display: flex;
gap: 1rem;
margin-bottom: 1rem;
}
.header-actions {
display: flex;
align-items: center;
gap: .5rem;
}
.actors-header tr {
text-align: left;
}
.actors, .actors th, .actors td {
border-collapse: collapse;
padding: .5rem;
border: 0;
}
.actor {
&:nth-child(2n) {
background: var(--glass-weak-50);
}
&:hover {
background: var(--glass-weak-40);
}
}
.actor-id {
width: 6rem;
font-family: monospace;
font-size: 1rem;
user-select: all;
}
.actor-entity {
width: 2rem;
}
.actor-favicon {
width: 1rem;
margin-left: .5rem;
}
.actor-global {
background: var(--primary);
fill: var(--text-light);
padding: .5rem;
border-radius: 1rem;
}
th.actor-actions {
display: flex;
align-items: center;
gap: .5rem;
.button:first-child {
margin-left: .5rem;
}
}
.actor-actions {
width: 1%;
white-space: nowrap;
}
.actions {
display: flex;
align-items: center;
}
.actor-action {
padding: .5rem .75rem;
fill: var(--glass);
&:hover {
cursor: pointer;
fill: var(--primary);
}
&:hover.action-delete {
fill: var(--error);
}
}
.actor-avatar {
width: 5rem;
position: relative;
line-height: 0;
&:hover .avatar-zoom,
&:active .avatar-zoom {
display: inline-block;
}
}
.avatar {
height: 3.5rem;
max-width: 100%;
}
.avatar-zoom {
display: none;
height: 50vh;
position: absolute;
top: 0;
left: 0;
z-index: 10;
box-shadow: 0 0 3px var(--shadow-weak-30);
pointer-events: none;
}
.check-container {
display: inline-flex;
padding: .5rem .75rem;
}
</style>