mirror of
https://github.com/Drop-OSS/drop.git
synced 2025-11-13 08:12:40 +10:00
160 lines
5.8 KiB
Vue
160 lines
5.8 KiB
Vue
<template>
|
|
<div>
|
|
<div class="mx-auto max-w-2xl lg:mx-0">
|
|
<h2
|
|
class="mt-2 text-xl font-semibold tracking-tight text-zinc-100 sm:text-3xl"
|
|
>
|
|
{{ $t("users.admin.authentication.title") }}
|
|
</h2>
|
|
<p
|
|
class="mt-2 text-pretty text-sm font-medium text-zinc-400 sm:text-md/8"
|
|
>
|
|
{{ $t("users.admin.authentication.description") }}
|
|
</p>
|
|
</div>
|
|
<ul
|
|
role="list"
|
|
class="mt-8 grid grid-cols-1 gap-x-6 gap-y-8 lg:grid-cols-4 xl:gap-x-8"
|
|
>
|
|
<li
|
|
v-for="authMech in authenticationMechanisms"
|
|
:key="authMech.name"
|
|
class="group overflow-hidden rounded-xl border border-zinc-800 bg-zinc-900 shadow-sm transition-all duration-200 hover:shadow-lg hover:shadow-zinc-900/50 hover:scale-[1.02] hover:border-zinc-700"
|
|
>
|
|
<div class="flex items-center gap-x-4 border-b border-zinc-800 p-6">
|
|
<div
|
|
class="flex h-10 w-10 flex-none items-center justify-center rounded-lg bg-zinc-800/50 ring-1 ring-zinc-700/50 transition-all duration-200 group-hover:bg-zinc-800 group-hover:ring-zinc-600/50"
|
|
>
|
|
<component
|
|
:is="authMech.icon"
|
|
:alt="`${authMech.name} icon`"
|
|
class="h-6 w-6 text-zinc-100 transition-all duration-200 group-hover:scale-110"
|
|
/>
|
|
</div>
|
|
<div class="text-sm/6 font-medium text-zinc-100">
|
|
{{ authMech.name }}
|
|
</div>
|
|
<Menu v-if="authMech.route" as="div" class="relative ml-auto">
|
|
<MenuButton
|
|
class="-m-2.5 block p-2.5 text-zinc-400 hover:text-zinc-300 transition-colors duration-200"
|
|
>
|
|
<span class="sr-only">{{
|
|
$t("users.admin.authentication.srOpenOptions")
|
|
}}</span>
|
|
<EllipsisHorizontalIcon class="h-5 w-5" aria-hidden="true" />
|
|
</MenuButton>
|
|
<transition
|
|
enter-active-class="transition ease-out duration-100"
|
|
enter-from-class="transform opacity-0 scale-95"
|
|
enter-to-class="transform opacity-100 scale-100"
|
|
leave-active-class="transition ease-in duration-75"
|
|
leave-from-class="transform opacity-100 scale-100"
|
|
leave-to-class="transform opacity-0 scale-95"
|
|
>
|
|
<MenuItems
|
|
class="absolute right-0 z-10 mt-0.5 w-32 origin-top-right rounded-md bg-zinc-900 py-2 shadow-lg ring-1 ring-zinc-100/5 focus:outline-none"
|
|
>
|
|
<MenuItem v-slot="{ active }">
|
|
<NuxtLink
|
|
:href="authMech.route"
|
|
:class="[
|
|
active ? 'bg-zinc-800 outline-none' : '',
|
|
'block px-3 py-1 text-sm/6 text-zinc-100 transition-colors duration-200',
|
|
]"
|
|
>{{ $t("users.admin.authentication.configure")
|
|
}}<span class="sr-only">{{ authMech.name }}</span></NuxtLink
|
|
>
|
|
</MenuItem>
|
|
</MenuItems>
|
|
</transition>
|
|
</Menu>
|
|
</div>
|
|
<dl class="-my-3 divide-y divide-zinc-700 px-6 py-4 text-sm/6">
|
|
<div class="flex justify-between gap-x-4 py-3">
|
|
<dt class="text-zinc-400">
|
|
{{ $t("users.admin.authentication.enabledKey") }}
|
|
</dt>
|
|
<dd class="flex items-center">
|
|
<span
|
|
:class="[
|
|
'inline-flex items-center rounded-md px-2 py-1 text-xs font-medium ring-1 ring-inset',
|
|
authMech.enabled
|
|
? 'bg-green-400/10 text-green-400 ring-green-400/20'
|
|
: 'bg-red-400/10 text-red-400 ring-red-400/20',
|
|
]"
|
|
>
|
|
<CheckIcon v-if="authMech.enabled" class="w-4 h-4 mr-1" />
|
|
<XMarkIcon v-else class="w-4 h-4 mr-1" />
|
|
{{
|
|
authMech.enabled
|
|
? $t("users.admin.authentication.enabled")
|
|
: $t("users.admin.authentication.disabled")
|
|
}}
|
|
</span>
|
|
</dd>
|
|
</div>
|
|
<div v-if="authMech.settings">
|
|
<div
|
|
v-for="[key, value] in Object.entries(authMech.settings)"
|
|
:key="key"
|
|
class="flex flex-nowrap justify-between gap-x-4 py-2"
|
|
>
|
|
<dt class="text-zinc-400">{{ key }}</dt>
|
|
<dd class="text-zinc-300 truncate">
|
|
{{ value }}
|
|
</dd>
|
|
</div>
|
|
</div>
|
|
</dl>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { IconsSimpleAuthenticationLogo, IconsSSOLogo } from "#components";
|
|
import { Menu, MenuButton, MenuItem, MenuItems } from "@headlessui/vue";
|
|
import { EllipsisHorizontalIcon } from "@heroicons/vue/20/solid";
|
|
import { CheckIcon, XMarkIcon } from "@heroicons/vue/24/solid";
|
|
import type { AuthMec } from "~~/prisma/client/enums";
|
|
import type { Component } from "vue";
|
|
|
|
useHead({
|
|
title: "Authentication",
|
|
});
|
|
|
|
definePageMeta({
|
|
layout: "admin",
|
|
});
|
|
|
|
const { t } = useI18n();
|
|
|
|
const enabledMechanisms = await $dropFetch("/api/v1/admin/auth");
|
|
|
|
const authenticationMechanisms: Array<{
|
|
name: string;
|
|
mec: AuthMec;
|
|
icon: Component;
|
|
route?: string;
|
|
enabled: boolean;
|
|
settings?: { [key: string]: string | undefined } | undefined | boolean;
|
|
}> = [
|
|
{
|
|
name: t("users.admin.authentication.simple"),
|
|
mec: "Simple" as AuthMec,
|
|
icon: IconsSimpleAuthenticationLogo,
|
|
route: "/admin/users/auth/simple",
|
|
},
|
|
{
|
|
name: t("users.admin.authentication.oidc"),
|
|
mec: "OpenID" as AuthMec,
|
|
icon: IconsSSOLogo,
|
|
},
|
|
].map((e) => ({
|
|
...e,
|
|
enabled: !!enabledMechanisms[e.mec],
|
|
settings:
|
|
typeof enabledMechanisms[e.mec] === "object" && enabledMechanisms[e.mec],
|
|
}));
|
|
</script>
|