144 lines
5.6 KiB
Vue
144 lines
5.6 KiB
Vue
<template>
|
|
<layout>
|
|
<h1
|
|
v-if="context.auth.attemptedUsername && context.auth.showUsername"
|
|
class="text-3xl font-semibold text-center text-gray-700">
|
|
Welcome {{ context.auth?.attemptedUsername }}!
|
|
</h1>
|
|
<h1 v-else class="text-3xl font-semibold text-center text-gray-700">
|
|
Welcome!
|
|
</h1>
|
|
<p class="text-center mt-3">
|
|
Please select a second factor you would like to use:
|
|
</p>
|
|
<div class="flex flex-col items-center mt-10 text-gray-700">
|
|
<div
|
|
v-if="$data.selection.webauthn.show"
|
|
@click="useMethod($data.selection.webauthn.execId)"
|
|
class="flex flex-row items-center w-full p-5 bg-white shadow-lg transition ease-in-out duration-150 hover:scale-105 hover:cursor-pointer rounded-md">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="w-8 h-8"
|
|
fill="currentColor"
|
|
viewBox="0 0 16 16">
|
|
<path
|
|
d="M6 .5a.5.5 0 0 1 .5-.5h4a.5.5 0 0 1 .5.5v4H6v-4ZM7 1v1h1V1H7Zm2 0v1h1V1H9ZM5.5 5a.5.5 0 0 0-.5.5V15a1 1 0 0 0 1 1h5a1 1 0 0 0 1-1V5.5a.5.5 0 0 0-.5-.5h-6Z" />
|
|
</svg>
|
|
<div class="flex flex-col ml-5 leading-none">
|
|
<h2 class="mb-1.5 font-semibold">Hardware Security Key</h2>
|
|
<p class="text-sm">Authenticate using a WebAuthn capable device</p>
|
|
</div>
|
|
</div>
|
|
<div
|
|
v-if="$data.selection.otp.show"
|
|
@click="useMethod($data.selection.otp.execId)"
|
|
class="flex flex-row items-center w-full p-5 mt-4 bg-white shadow-lg transition ease-in-out duration-150 hover:scale-105 hover:cursor-pointer rounded-md">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="w-8 h-8"
|
|
fill="currentColor"
|
|
viewBox="0 0 16 16">
|
|
<path
|
|
d="M8.515 1.019A7 7 0 0 0 8 1V0a8 8 0 0 1 .589.022l-.074.997zm2.004.45a7.003 7.003 0 0 0-.985-.299l.219-.976c.383.086.76.2 1.126.342l-.36.933zm1.37.71a7.01 7.01 0 0 0-.439-.27l.493-.87a8.025 8.025 0 0 1 .979.654l-.615.789a6.996 6.996 0 0 0-.418-.302zm1.834 1.79a6.99 6.99 0 0 0-.653-.796l.724-.69c.27.285.52.59.747.91l-.818.576zm.744 1.352a7.08 7.08 0 0 0-.214-.468l.893-.45a7.976 7.976 0 0 1 .45 1.088l-.95.313a7.023 7.023 0 0 0-.179-.483zm.53 2.507a6.991 6.991 0 0 0-.1-1.025l.985-.17c.067.386.106.778.116 1.17l-1 .025zm-.131 1.538c.033-.17.06-.339.081-.51l.993.123a7.957 7.957 0 0 1-.23 1.155l-.964-.267c.046-.165.086-.332.12-.501zm-.952 2.379c.184-.29.346-.594.486-.908l.914.405c-.16.36-.345.706-.555 1.038l-.845-.535zm-.964 1.205c.122-.122.239-.248.35-.378l.758.653a8.073 8.073 0 0 1-.401.432l-.707-.707z" />
|
|
<path
|
|
d="M8 1a7 7 0 1 0 4.95 11.95l.707.707A8.001 8.001 0 1 1 8 0v1z" />
|
|
<path
|
|
d="M7.5 3a.5.5 0 0 1 .5.5v5.21l3.248 1.856a.5.5 0 0 1-.496.868l-3.5-2A.5.5 0 0 1 7 9V3.5a.5.5 0 0 1 .5-.5z" />
|
|
</svg>
|
|
<div class="flex flex-col ml-5 leading-none">
|
|
<h2 class="mb-1.5 font-semibold">Authenticator App</h2>
|
|
<p class="text-sm">
|
|
Authenticate using a one time code from your authenticator app
|
|
</p>
|
|
</div>
|
|
</div>
|
|
<div
|
|
v-if="$data.selection.recovery.show"
|
|
@click="useMethod($data.selection.recovery.execId)"
|
|
class="flex flex-row items-center w-full p-5 mt-4 bg-white shadow-lg transition ease-in-out duration-150 hover:scale-105 hover:cursor-pointer rounded-md">
|
|
<svg
|
|
xmlns="http://www.w3.org/2000/svg"
|
|
class="w-8 h-8"
|
|
fill="currentColor"
|
|
viewBox="0 0 16 16">
|
|
<path
|
|
fill-rule="evenodd"
|
|
d="M8 3a5 5 0 1 1-4.546 2.914.5.5 0 0 0-.908-.417A6 6 0 1 0 8 2v1z" />
|
|
<path
|
|
d="M8 4.466V.534a.25.25 0 0 0-.41-.192L5.23 2.308a.25.25 0 0 0 0 .384l2.36 1.966A.25.25 0 0 0 8 4.466z" />
|
|
</svg>
|
|
<div class="flex flex-col ml-5 leading-none">
|
|
<h2 class="mb-1.5 font-semibold">Recovery Code</h2>
|
|
<p class="text-sm">Authenticate using one of your recovery codes</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</layout>
|
|
</template>
|
|
<script lang="ts">
|
|
import { defineComponent } from "vue";
|
|
import Layout from "~/components/Layout.vue";
|
|
import type { KcContextBase } from "~/types/context";
|
|
import { formPost } from "~/functions/utils";
|
|
|
|
export default defineComponent({
|
|
name: "SelectAuthenticator",
|
|
components: {
|
|
Layout,
|
|
},
|
|
data() {
|
|
return {
|
|
context: (window as any).kcContext as KcContextBase.SelectAuthenticator,
|
|
selection: {
|
|
webauthn: {
|
|
show: false,
|
|
execId: "",
|
|
},
|
|
otp: {
|
|
show: false,
|
|
execId: "",
|
|
},
|
|
recovery: {
|
|
show: false,
|
|
execId: "",
|
|
},
|
|
},
|
|
};
|
|
},
|
|
mounted: function () {
|
|
this.context.auth.authenticationSelections.forEach(selection => {
|
|
if (
|
|
selection.authenticationExecution.authenticator ===
|
|
"webauthn-authenticator"
|
|
) {
|
|
this.selection.webauthn.execId = selection.authExecId;
|
|
this.selection.webauthn.show = true;
|
|
} else if (
|
|
selection.authenticationExecution.authenticator === "auth-otp-form"
|
|
) {
|
|
this.selection.otp.execId = selection.authExecId;
|
|
this.selection.otp.show = true;
|
|
} else if (
|
|
selection.authenticationExecution.authenticator ===
|
|
"auth-recovery-authn-code-form"
|
|
) {
|
|
this.selection.recovery.execId = selection.authExecId;
|
|
this.selection.recovery.show = true;
|
|
}
|
|
});
|
|
},
|
|
methods: {
|
|
useMethod(execId: string) {
|
|
formPost(this.context.url.loginAction, {
|
|
authenticationExecution: execId,
|
|
});
|
|
},
|
|
},
|
|
});
|
|
</script>
|
|
<style>
|
|
@tailwind base;
|
|
@tailwind components;
|
|
@tailwind utilities;
|
|
</style>
|