Skip to content
Closed
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
<script lang="ts">
import * as Button from "$lib/ui/Button";
import { cn } from "$lib/utils";
import { toast } from "$lib/ui/Toast/toast";
import {
CheckmarkBadge02Icon,
Upload03Icon,
ViewIcon,
Copy01Icon,
} from "@hugeicons/core-free-icons";
import { HugeiconsIcon } from "@hugeicons/svelte";
import type { HTMLAttributes } from "svelte/elements";
Expand All @@ -15,8 +15,6 @@ interface userData {
interface IIdentityCard extends HTMLAttributes<HTMLElement> {
variant?: "eName" | "ePassport" | "eVault";
userId?: string;
viewBtn?: () => void;
shareBtn?: () => void;
userData?: userData;
totalStorage?: number;
usedStorage?: number;
Expand All @@ -25,8 +23,6 @@ interface IIdentityCard extends HTMLAttributes<HTMLElement> {
const {
variant = "eName",
userId,
viewBtn,
shareBtn,
userData,
totalStorage = 0,
usedStorage = 0,
Expand Down Expand Up @@ -68,24 +64,22 @@ const baseClasses = `relative ${variant === "eName" ? "bg-black-900" : variant =
className="text-secondary"
icon={CheckmarkBadge02Icon}
/>
<div class="flex gap-3 items-center">
{#if shareBtn}
<Button.Icon
icon={Upload03Icon}
iconColor={"white"}
strokeWidth={2}
onclick={shareBtn}
/>
{/if}
{#if viewBtn}
<Button.Icon
icon={ViewIcon}
iconColor={"white"}
strokeWidth={2}
onclick={viewBtn}
/>
{/if}
</div>
<Button.Icon
icon={Copy01Icon}
iconColor={"white"}
strokeWidth={2}
onclick={async () => {
if (userId) {
try {
await navigator.clipboard.writeText(userId);
toast.success("eName copied to clipboard");
} catch (error) {
console.error("Failed to copy:", error);
toast.error("Failed to copy eName");
}
}
}}
/>
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

⚠️ Potential issue | 🟡 Minor

Add an accessible label or tooltip for the copy button.

The copy-to-clipboard functionality works correctly with proper error handling and toast notifications. However, the button lacks an accessible label or tooltip, which may leave users uncertain about its purpose.

Consider adding an aria-label or title attribute to improve accessibility:

 <Button.Icon
     icon={Copy01Icon}
     iconColor={"white"}
     strokeWidth={2}
+    title="Copy eName to clipboard"
     onclick={async () => {

Optional: Consider adding brief visual feedback on successful copy.

For enhanced UX, you could temporarily change the icon or add a visual indicator when the copy succeeds, in addition to the toast notification.

📝 Committable suggestion

‼️ IMPORTANT
Carefully review the code before committing. Ensure that it accurately replaces the highlighted code, contains no missing lines, and has no issues with indentation. Thoroughly test & benchmark the code to ensure it meets the requirements.

Suggested change
<Button.Icon
icon={Copy01Icon}
iconColor={"white"}
strokeWidth={2}
onclick={async () => {
if (userId) {
try {
await navigator.clipboard.writeText(userId);
toast.success("eName copied to clipboard");
} catch (error) {
console.error("Failed to copy:", error);
toast.error("Failed to copy eName");
}
}
}}
/>
<Button.Icon
icon={Copy01Icon}
iconColor={"white"}
strokeWidth={2}
title="Copy eName to clipboard"
onclick={async () => {
if (userId) {
try {
await navigator.clipboard.writeText(userId);
toast.success("eName copied to clipboard");
} catch (error) {
console.error("Failed to copy:", error);
toast.error("Failed to copy eName");
}
}
}}
/>
🤖 Prompt for AI Agents
in infrastructure/eid-wallet/src/lib/fragments/IdentityCard/IdentityCard.svelte
around lines 70 to 85, the Copy button is missing an accessible label/tooltip;
add an aria-label (e.g., aria-label="Copy eName to clipboard") or a title prop
on the Button.Icon component so screen readers and hover users know its purpose,
and optionally add a short local state flash (e.g., set a transient "copied"
boolean for 1–2s to swap the icon or add a visual indicator) triggered on
successful copy while keeping the existing toast and error handling.

{:else if variant === "ePassport"}
<p
class="bg-white text-black flex items-center leading-0 justify-center rounded-full h-7 px-5 text-xs font-medium"
Expand All @@ -94,14 +88,6 @@ const baseClasses = `relative ${variant === "eName" ? "bg-black-900" : variant =
{userData.isFake ? "DEMO ID" : "VERIFIED ID"}
{/if}
</p>
{#if viewBtn}
<Button.Icon
icon={ViewIcon}
iconColor={"white"}
strokeWidth={2}
onclick={viewBtn}
/>
{/if}
{:else if variant === "eVault"}
<h3 class="text-black-300 text-3xl font-semibold mb-3 z-[1]">
{state.progressWidth} Used
Expand Down
18 changes: 10 additions & 8 deletions infrastructure/eid-wallet/src/lib/ui/Drawer/Drawer.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ interface IDrawerProps extends HTMLAttributes<HTMLDivElement> {
children?: Snippet;
handleSwipe?: (isOpen: boolean | undefined) => void;
dismissible?: boolean;
fullScreen?: boolean;
}
Comment thread
coderabbitai[bot] marked this conversation as resolved.
Outdated

let drawerElem: HTMLDivElement;
Expand All @@ -21,6 +22,7 @@ let {
children = undefined,
handleSwipe,
dismissible = true,
fullScreen = false,
...restProps
}: IDrawerProps = $props();

Expand Down Expand Up @@ -100,17 +102,17 @@ $effect(() => {

<style>
:global(.pane) {
width: 95% !important;
max-height: 600px !important;
min-height: 250px !important;
height: auto !important;
width: {fullScreen ? "100%" : "95%"} !important;
max-height: {fullScreen ? "100vh" : "600px"} !important;
min-height: {fullScreen ? "100vh" : "250px"} !important;
height: {fullScreen ? "100vh" : "auto"} !important;
position: fixed !important;
bottom: 30px !important;
bottom: {fullScreen ? "0" : "30px"} !important;
left: 50% !important;
transform: translateX(-50%) !important;
border-radius: 32px !important;
padding-block-start: 50px !important;
padding-block-end: 20px !important;
border-radius: {fullScreen ? "0" : "32px"} !important;
padding-block-start: {fullScreen ? "0" : "50px"} !important;
padding-block-end: {fullScreen ? "0" : "20px"} !important;
background-color: var(--color-white) !important;
overflow-y: auto !important; /* vertical scroll if needed */
overflow-x: hidden !important; /* prevent sideways scroll */
Expand Down
86 changes: 86 additions & 0 deletions infrastructure/eid-wallet/src/lib/ui/Toast/Toast.svelte
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
<script lang="ts">
import { toastStore } from "./toast";
import { cn } from "$lib/utils";
import { XIcon } from "@hugeicons/core-free-icons";
import { HugeiconsIcon } from "@hugeicons/svelte";

let toasts = $state<Array<{ id: string; message: string; type?: "success" | "error" | "info"; duration?: number }>>([]);

$effect(() => {
const unsubscribe = toastStore.subscribe((value) => {
toasts = value;
});
return unsubscribe;
});

function removeToast(id: string) {
toastStore.remove(id);
}

const getToastClasses = (type?: string) => {
switch (type) {
case "success":
return "bg-green-50 border-green-200 text-green-800";
case "error":
return "bg-red-50 border-red-200 text-red-800";
case "info":
default:
return "bg-blue-50 border-blue-200 text-blue-800";
}
};
</script>

<div
class="fixed top-4 left-1/2 -translate-x-1/2 z-[9999] flex flex-col gap-2 pointer-events-none w-full max-w-md px-4"
>
{#each toasts as toast (toast.id)}
<div
class={cn(
"pointer-events-auto flex items-center justify-between gap-3 rounded-lg border p-4 shadow-lg animate-in slide-in-from-top-2 fade-in",
getToastClasses(toast.type),
)}
role="alert"
>
<p class="text-sm font-medium flex-1">{toast.message}</p>
<button
onclick={() => removeToast(toast.id)}
class="flex-shrink-0 hover:opacity-70 transition-opacity"
aria-label="Close toast"
>
<HugeiconsIcon
icon={XIcon}
size={20}
strokeWidth={2}
className="text-current"
/>
</button>
</div>
{/each}
</div>

<style>
@keyframes slide-in-from-top-2 {
from {
transform: translateY(-100%);
opacity: 0;
}
to {
transform: translateY(0);
opacity: 1;
}
}

@keyframes fade-in {
from {
opacity: 0;
}
to {
opacity: 1;
}
}

.animate-in {
animation: slide-in-from-top-2 0.3s ease-out, fade-in 0.3s ease-out;
}
</style>
Comment on lines +69 to +95
Copy link
Copy Markdown
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

🛠️ Refactor suggestion | 🟠 Major

Add exit animations for smoother UX.

The component defines entrance animations (slide-in-from-top-2, fade-in) but no exit animations. When a toast is removed, it disappears abruptly, which creates a jarring user experience.

Consider using Svelte's transition directives for smoother enter/exit animations:

+import { fly, fade } from 'svelte/transition';
+
 {#each toasts as toast (toast.id)}
     <div
+        transition:fly={{ y: -20, duration: 300 }}
         class={cn(
-            "pointer-events-auto flex items-center justify-between gap-3 rounded-lg border p-4 shadow-lg animate-in slide-in-from-top-2 fade-in",
+            "pointer-events-auto flex items-center justify-between gap-3 rounded-lg border p-4 shadow-lg",
             getToastClasses(toast.type),
         )}
         role="alert"
     >

Then you can remove the custom CSS animations and use Svelte's built-in transition system.

Committable suggestion skipped: line range outside the PR's diff.

🤖 Prompt for AI Agents
in infrastructure/eid-wallet/src/lib/ui/Toast/Toast.svelte around lines 61-85,
the component only defines entrance CSS keyframe animations and lacks exit
animations, causing abrupt removals; replace or augment this with Svelte
transitions by importing a built-in transition (e.g., fly or slide/fade) and
apply a transition directive to the toast element (e.g., use:transitionName or
transition:fade with configured params) so the component animates both in and
out, and remove the custom CSS animations if the Svelte transitions cover both
directions to keep styles consistent.


53 changes: 53 additions & 0 deletions infrastructure/eid-wallet/src/lib/ui/Toast/toast.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
import { writable } from "svelte/store";

export interface Toast {
id: string;
message: string;
type?: "success" | "error" | "info";
duration?: number;
}

const createToastStore = () => {
const { subscribe, update } = writable<Toast[]>([]);

return {
subscribe,
add: (toast: Omit<Toast, "id">) => {
const id = crypto.randomUUID();
const newToast: Toast = {
id,
duration: 3000,
...toast,
};

update((toasts) => [...toasts, newToast]);

// Auto remove after duration
if (newToast.duration && newToast.duration > 0) {
setTimeout(() => {
remove(id);
}, newToast.duration);
}

return id;
},
remove: (id: string) => {
update((toasts) => toasts.filter((t) => t.id !== id));
},
clear: () => {
update(() => []);
},
};
};

export const toastStore = createToastStore();

export const toast = {
success: (message: string, duration?: number) =>
toastStore.add({ message, type: "success", duration }),
error: (message: string, duration?: number) =>
toastStore.add({ message, type: "error", duration }),
info: (message: string, duration?: number) =>
toastStore.add({ message, type: "info", duration }),
};

34 changes: 0 additions & 34 deletions infrastructure/eid-wallet/src/routes/(app)/main/+page.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -2,34 +2,25 @@
import { goto } from "$app/navigation";
import { Hero, IdentityCard } from "$lib/fragments";
import type { GlobalState } from "$lib/global";
import { Drawer } from "$lib/ui";
import * as Button from "$lib/ui/Button";
import {
CircleArrowDataTransferDiagonalFreeIcons,
QrCodeIcon,
Settings02Icon,
} from "@hugeicons/core-free-icons";
import { HugeiconsIcon } from "@hugeicons/svelte";
import { type Snippet, getContext, onMount } from "svelte";
import { onDestroy } from "svelte";
import { Shadow } from "svelte-loading-spinners";
import QrCode from "svelte-qrcode";

let userData: Record<string, unknown> | undefined = $state(undefined);
let greeting: string | undefined = $state(undefined);
let ename: string | undefined = $state(undefined);
let profileCreationStatus: "idle" | "loading" | "success" | "failed" =
$state("idle");

let shareQRdrawerOpen = $state(false);
let statusInterval: ReturnType<typeof setInterval> | undefined =
$state(undefined);

function shareQR() {
alert("QR Code shared!");
shareQRdrawerOpen = false;
}

async function retryProfileCreation() {
try {
await globalState.vaultController.retryProfileCreation();
Expand Down Expand Up @@ -137,14 +128,11 @@ onDestroy(() => {
<IdentityCard
variant="eName"
userId={ename ?? "Loading..."}
viewBtn={() => alert("View button clicked!")}
shareBtn={() => (shareQRdrawerOpen = true)}
/>
{/snippet}
{#snippet ePassport()}
<IdentityCard
variant="ePassport"
viewBtn={() => goto("/ePassport")}
userData={userData as Record<string, string>}
/>
{/snippet}
Expand All @@ -158,28 +146,6 @@ onDestroy(() => {
{@render Section("eVault", eVault)}
</main>

<Drawer
title="Scan QR Code"
bind:isPaneOpen={shareQRdrawerOpen}
class="flex flex-col gap-4 items-center justify-center"
>
<div
class="flex justify-center relative items-center overflow-hidden h-full rounded-3xl p-8 pt-0"
>
<QrCode size={320} value={ename ?? ""} />
</div>

<h4 class="text-center mt-2">Share your eName</h4>
<p class="text-black-700 text-center">
Anyone scanning this can see your eName
</p>
<div class="flex justify-center items-center mt-4">
<Button.Action variant="solid" callback={shareQR} class="w-full">
Share
</Button.Action>
</div>
</Drawer>

<Button.Nav href="/scan-qr">
<Button.Action
variant="solid"
Expand Down
28 changes: 8 additions & 20 deletions infrastructure/eid-wallet/src/routes/(app)/scan-qr/+page.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -99,15 +99,9 @@ $effect(() => {
});

async function handleAuthDrawerDecline() {
// If there's an error, "Okay" button closes modal and navigates to main
if ($authError) {
setCodeScannedDrawerOpen(false);
await goto("/main");
} else {
// Otherwise, "Decline" closes modal and restarts scanning
setCodeScannedDrawerOpen(false);
startScan();
}
// Cancel button always navigates to main dashboard
setCodeScannedDrawerOpen(false);
await goto("/main");
}

function handleAuthDrawerOpenChange(value: boolean) {
Expand All @@ -126,15 +120,9 @@ function handleLoggedInDrawerOpenChange(value: boolean) {
}

async function handleSigningDrawerDecline() {
// If there's an error, "Okay" button closes modal and navigates to main
if ($signingError) {
setSigningDrawerOpen(false);
await goto("/main");
} else {
// Otherwise, "Decline" closes modal and restarts scanning
setSigningDrawerOpen(false);
startScan();
}
// Cancel button always navigates to main dashboard
setSigningDrawerOpen(false);
await goto("/main");
}

function handleSigningDrawerOpenChange(value: boolean) {
Expand All @@ -148,9 +136,9 @@ function handleBlindVoteOptionChange(index: number) {
handleBlindVoteSelection(index);
}

function handleRevealDrawerCancel() {
async function handleRevealDrawerCancel() {
setRevealRequestOpen(false);
window.history.back();
await goto("/main");
}

function handleRevealDrawerOpenChange(value: boolean) {
Expand Down
Loading
Loading