feat: introducing shared requests to admin dashboard (#3537)

Co-authored-by: jamesgeorge007 <jamesgeorge998001@gmail.com>
This commit is contained in:
Joel Jacob Stephen
2023-12-06 00:21:28 +05:30
committed by GitHub
parent 6fa722df7b
commit d9c75ed79e
11 changed files with 489 additions and 136 deletions

View File

@@ -0,0 +1,47 @@
<template>
<HoppButtonSecondary
v-tippy="{ theme: 'tooltip' }"
:title="title"
:icon="icon"
:color="icon === props.icon.default ? props.color : props.resetColor"
class="px-3"
@click="iconClickHandler"
/>
</template>
<script setup lang="ts">
import { FunctionalComponent, SVGAttributes } from 'vue';
import { refAutoReset } from '@vueuse/core';
const props = withDefaults(
defineProps<{
title: string;
color?: string;
// color to be displayed temporarily until reset
resetColor?: string;
icon: {
default: FunctionalComponent<SVGAttributes, {}>;
// icon to be displayed temporarily until reset
temporary: FunctionalComponent<SVGAttributes, {}>;
};
}>(),
{
color: 'white',
resetColor: 'emerald',
}
);
const emit = defineEmits<{
(e: 'click'): void;
}>();
const icon = refAutoReset<FunctionalComponent<SVGAttributes, {}>>(
props.icon.default,
1000
);
const iconClickHandler = () => {
icon.value = props.icon.temporary;
emit('click');
};
</script>

View File

@@ -0,0 +1,127 @@
<template>
<div class="rounded-md">
<div class="grid gap-6 mt-4">
<div
class="relative"
:class="
user.photoURL
? 'h-20 w-20'
: 'bg-primaryDark w-16 p-3 rounded-2xl mb-3'
"
>
<img
v-if="user.photoURL"
class="object-cover rounded-3xl mb-3"
:src="user.photoURL"
/>
<icon-lucide-user v-else class="text-4xl" />
<span
v-if="user.isAdmin"
class="absolute left-16 bottom-0 text-xs font-medium px-3 py-0.5 rounded-full bg-green-900 text-green-300"
>
{{ t('users.admin') }}
</span>
</div>
<template v-for="(info, key) in userInfo" :key="key">
<div v-if="info.condition">
<label class="text-secondaryDark" :for="key">{{ info.label }}</label>
<div
class="w-full p-3 mt-2 bg-divider border-gray-600 rounded-md focus:border-emerald-600 focus:ring focus:ring-opacity-40 focus:ring-emerald-500"
>
<span>{{ info.value }}</span>
</div>
</div>
</template>
</div>
<div class="flex justify-start mt-8">
<HoppButtonPrimary
:icon="user.isAdmin ? IconUserMinus : IconUserCheck"
:label="
user.isAdmin
? t('users.remove_admin_privilege')
: t('users.make_admin')
"
filled
outline
class="mr-4"
@click="
user.isAdmin
? emit('remove-admin', user.uid)
: emit('make-admin', user.uid)
"
/>
<HoppButtonSecondary
:icon="IconTrash"
:label="t('users.delete')"
filled
outline
class="mr-4 bg-red-600 text-white hover:text-gray-100"
@click="
user.isAdmin
? toast.error(t('state.remove_admin_to_delete_user'))
: emit('delete-user', user.uid)
"
/>
</div>
</div>
</template>
<script setup lang="ts">
import { format } from 'date-fns';
import { useI18n } from '~/composables/i18n';
import { useToast } from '~/composables/toast';
import IconTrash from '~icons/lucide/trash';
import IconUserCheck from '~icons/lucide/user-check';
import IconUserMinus from '~icons/lucide/user-minus';
import { UserInfoQuery } from '~/helpers/backend/graphql';
const t = useI18n();
const toast = useToast();
const props = defineProps<{
user: UserInfoQuery['infra']['userInfo'];
}>();
const emit = defineEmits<{
(event: 'delete-user', userID: string): void;
(event: 'make-admin', userID: string): void;
(event: 'remove-admin', userID: string): void;
}>();
// Get Proper Date Formats
const getCreatedDateAndTime = (date: string) =>
format(new Date(date), 'd-MM-yyyy hh:mm a');
// User Info
const { uid, displayName, email, createdOn } = props.user;
const userInfo = {
uid: {
condition: uid,
label: t('users.uid'),
value: uid,
},
displayName: {
condition: true,
label: t('users.name'),
value: displayName ?? t('users.unnamed'),
},
email: {
condition: email,
label: t('users.email'),
value: email,
},
createdOn: {
condition: createdOn,
label: t('users.created_on'),
value: getCreatedDateAndTime(createdOn),
},
};
</script>

View File

@@ -0,0 +1,191 @@
<template>
<div>
<div v-if="fetching" class="flex justify-center">
<HoppSmartSpinner />
</div>
<div v-else-if="error">{{ t('shared_requests.load_list_error') }}</div>
<div v-else-if="sharedRequests.length === 0" class="ml-3 mt-5 text-lg">
{{ t('users.no_shared_requests') }}
</div>
<div v-else class="mt-10">
<HoppSmartTable :list="sharedRequests">
<template #head>
<tr
class="text-secondary border-b border-dividerDark text-sm text-left bg-primaryLight"
>
<th class="px-6 py-2">{{ t('shared_requests.id') }}</th>
<th class="px-6 py-2 w-30">{{ t('shared_requests.url') }}</th>
<th class="px-6 py-2">{{ t('shared_requests.created_on') }}</th>
<!-- Empty Heading for the Action Button -->
<th class="px-6 py-2 text-center">Actions</th>
</tr>
</template>
<template #body="{ list: sharedRequests }">
<tr
v-for="request in sharedRequests"
:key="request.id"
class="text-secondaryDark hover:bg-divider hover:cursor-pointer rounded-xl"
>
<td class="flex py-4 px-7 max-w-50">
<span class="truncate">
{{ request.id }}
</span>
</td>
<td class="py-4 px-7 w-96">
{{ sharedRequestURL(request.request) }}
</td>
<td class="py-2 px-7">
{{ getCreatedDate(request.createdOn) }}
<div class="text-gray-400 text-tiny">
{{ getCreatedTime(request.createdOn) }}
</div>
</td>
<td class="flex justify-center">
<HoppButtonSecondary
v-tippy="{ theme: 'tooltip' }"
:title="t('shared_requests.open_request')"
:to="`${shortcodeBaseURL}/r/${request.id}`"
:blank="true"
:icon="IconExternalLink"
class="px-3 text-emerald-500 hover:text-accent"
/>
<UiAutoResetIcon
:title="t('shared_requests.copy')"
:icon="{ default: IconCopy, temporary: IconCheck }"
@click="copySharedRequest(request.id)"
/>
<HoppButtonSecondary
v-tippy="{ theme: 'tooltip' }"
:title="t('shared_requests.delete')"
:icon="IconTrash"
color="red"
class="px-3"
@click="deleteSharedRequest(request.id)"
/>
</td>
</tr>
</template>
</HoppSmartTable>
<!-- Pagination -->
<div
v-if="hasNextPage && sharedRequests.length >= sharedRequestsPerPage"
class="flex items-center w-28 px-3 py-2 mt-5 mx-auto font-semibold text-secondaryDark bg-divider hover:bg-dividerDark rounded-3xl cursor-pointer"
@click="fetchNextSharedRequests"
>
<span class="mr-2">{{ t('shared_requests.show_more') }}</span>
<icon-lucide-chevron-down />
</div>
</div>
</div>
<HoppSmartConfirmModal
:show="confirmDeletion"
:title="t('shared_requests.confirm_request_deletion')"
@hide-modal="confirmDeletion = false"
@resolve="deleteSharedRequestMutation(deleteSharedRequestID)"
/>
</template>
<script setup lang="ts">
import { format } from 'date-fns';
import { ref } from 'vue';
import { useMutation } from '@urql/vue';
import {
SharedRequestsDocument,
RevokeShortcodeByAdminDocument,
} from '../../helpers/backend/graphql';
import { usePagedQuery } from '~/composables/usePagedQuery';
import { useToast } from '~/composables/toast';
import { useI18n } from '~/composables/i18n';
import { copyToClipboard } from '~/helpers/utils/clipboard';
import IconTrash from '~icons/lucide/trash';
import IconCopy from '~icons/lucide/copy';
import IconCheck from '~icons/lucide/check';
import IconExternalLink from '~icons/lucide/external-link';
const t = useI18n();
const toast = useToast();
const props = defineProps<{
email: string;
}>();
// Get Desired Date Formats
const getCreatedDate = (date: string) => format(new Date(date), 'dd-MM-yyyy');
const getCreatedTime = (date: string) => format(new Date(date), 'hh:mm a');
//Fetch Shared Requests
const sharedRequestsPerPage = 30;
const {
fetching,
error,
goToNextPage: fetchNextSharedRequests,
refetch,
list: sharedRequests,
hasNextPage,
} = usePagedQuery(
SharedRequestsDocument,
(x) => x.infra.allShortcodes,
(x) => x.id,
sharedRequestsPerPage,
{ cursor: undefined, take: sharedRequestsPerPage, email: props.email }
);
// Return request endpoint from the request object
const sharedRequestURL = (request: string) => {
const parsedRequest = JSON.parse(request);
return parsedRequest.endpoint;
};
// Shortcode Base URL
const shortcodeBaseURL =
import.meta.env.VITE_SHORTCODE_BASE_URL ?? 'https://hopp.sh';
// Copy Shared Request to Clipboard
const copySharedRequest = (requestID: string) => {
copyToClipboard(`${shortcodeBaseURL}/r/${requestID}`);
toast.success(`${t('state.copied_to_clipboard')}`);
};
// Shared Request Deletion
const confirmDeletion = ref(false);
const deleteSharedRequestID = ref<string | null>(null);
const sharedRequestDeletion = useMutation(RevokeShortcodeByAdminDocument);
const deleteSharedRequest = (id: string) => {
confirmDeletion.value = true;
deleteSharedRequestID.value = id;
};
const deleteSharedRequestMutation = async (id: string | null) => {
if (!id) {
confirmDeletion.value = false;
toast.error(`${t('state.delete_request_failure')}`);
return;
}
const variables = { codeID: id };
await sharedRequestDeletion.executeMutation(variables).then((result) => {
if (result.error) {
toast.error(`${t('state.delete_request_failure')}`);
} else {
sharedRequests.value = sharedRequests.value.filter(
(request) => request.id !== id
);
refetch();
toast.success(`${t('state.delete_request_success')}`);
}
});
confirmDeletion.value = false;
deleteSharedRequestID.value = null;
};
</script>