feat(settings): aba de diagnóstico e cópia de user ID no menu do usuário

- Nova aba "Diagnóstico" em Settings com:
  - Identidade: user ID (copiável), nome, e-mail
  - Sessão: criada em / expira em
  - Aplicação: versão, NODE_ENV, build SHA (se definido)
  - Configuração do servidor: S3, e-mail e domínio público — apenas booleans, sem expor credenciais
  - Saúde: status e latência do banco de dados
  - Uso: contagem de lançamentos, anexos, anotações e itens no inbox
- Botão de cópia do user ID no dropdown do avatar (ao lado do e-mail)

Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
Felipe Coutinho
2026-04-11 18:06:40 +00:00
parent 9ecafdb15f
commit 5f7bfb98da
4 changed files with 368 additions and 6 deletions

View File

@@ -1,6 +1,8 @@
"use client";
import {
RiCheckLine,
RiFileCopyLine,
RiHistoryLine,
RiLogoutCircleLine,
RiMegaphoneLine,
@@ -15,6 +17,11 @@ import { version } from "@/package.json";
import { FeedbackDialogBody } from "@/shared/components/navigation/navbar/feedback-dialog";
import { Badge } from "@/shared/components/ui/badge";
import { Dialog, DialogTrigger } from "@/shared/components/ui/dialog";
import {
Tooltip,
TooltipContent,
TooltipTrigger,
} from "@/shared/components/ui/tooltip";
import {
DropdownMenu,
DropdownMenuContent,
@@ -50,6 +57,13 @@ export function NavbarUser({
const router = useRouter();
const [logoutLoading, setLogoutLoading] = useState(false);
const [feedbackOpen, setFeedbackOpen] = useState(false);
const [copied, setCopied] = useState(false);
function handleCopyId() {
navigator.clipboard.writeText(user.id);
setCopied(true);
setTimeout(() => setCopied(false), 2000);
}
const avatarSrc = pagadorAvatarUrl
? getAvatarSrc(pagadorAvatarUrl)
@@ -106,9 +120,30 @@ export function NavbarUser({
</div>
<div className="flex flex-col min-w-0">
<span className="text-sm font-medium truncate">{user.name}</span>
<span className="text-xs text-muted-foreground truncate">
{user.email}
</span>
<div className="flex items-center gap-1 min-w-0">
<span className="text-xs text-muted-foreground truncate">
{user.email}
</span>
<Tooltip>
<TooltipTrigger asChild>
<button
type="button"
onClick={handleCopyId}
className="shrink-0 text-muted-foreground/50 hover:text-muted-foreground transition-colors"
aria-label="Copiar ID do usuário"
>
{copied ? (
<RiCheckLine className="size-3 text-success" />
) : (
<RiFileCopyLine className="size-3" />
)}
</button>
</TooltipTrigger>
<TooltipContent side="bottom">
{copied ? "Copiado!" : "Copiar ID do usuário"}
</TooltipContent>
</Tooltip>
</div>
</div>
</DropdownMenuLabel>