feat: Implement onboarding tour and admin features

- Activate onboarding tour for new users without groups.
- Remove WhatsApp connection steps from the tour.
- Add admin menu and button to hide WhatsApp connect button for all users.
- Disable Finance Home icon redirect to landing page.
- Adjust tour to teach menu open/close.
- Edited: src/hooks/useOnboardingTour.ts, src/components/onboarding/OnboardingTour.tsx, src/components/layout/NewModernLayout.tsx, src/components/layout/Header.tsx, src/App.tsx
This commit is contained in:
gpt-engineer-app[bot] 2025-07-02 04:56:21 +00:00
parent 9f634a7a6e
commit 33f2d389cd
7 changed files with 196 additions and 24 deletions

View File

@ -27,6 +27,7 @@ import Categorias from "./pages/Categorias";
import AdminFAQ from "./pages/AdminFAQ";
import Assinatura from "./pages/Assinatura";
import AvisosContas from "./pages/AvisosContas";
import Admin from "./pages/Admin";
const queryClient = new QueryClient();
@ -75,6 +76,7 @@ function App() {
<Route path="/complete-profile" element={<CompleteProfile />} />
<Route path="/categorias" element={<Categorias />} />
<Route path="/admin/faq" element={<AdminFAQ />} />
<Route path="/admin" element={<Admin />} />
<Route path="/assinatura" element={<Assinatura />} />
<Route path="/avisos-contas" element={<AvisosContas />} />
</Route>

View File

@ -0,0 +1,42 @@
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
import { Switch } from '@/components/ui/switch';
import { Label } from '@/components/ui/label';
import { Shield } from 'lucide-react';
import { useAdminSettings } from '@/hooks/useAdminSettings';
const AdminPanel = () => {
const { isAdmin, hideWhatsAppButton, loading, toggleWhatsAppButton } = useAdminSettings();
if (!isAdmin || loading) {
return null;
}
return (
<Card className="w-full max-w-2xl mx-auto">
<CardHeader>
<CardTitle className="flex items-center gap-2">
<Shield className="h-5 w-5 text-red-600" />
Painel Administrativo
</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<div className="flex items-center justify-between">
<Label htmlFor="hide-whatsapp" className="text-sm font-medium">
Esconder botão "Conectar WhatsApp" para todos os usuários
</Label>
<Switch
id="hide-whatsapp"
checked={hideWhatsAppButton}
onCheckedChange={toggleWhatsAppButton}
/>
</div>
<p className="text-xs text-gray-500">
Quando ativado, o botão "Conectar WhatsApp" ficará oculto para todos os usuários do sistema.
</p>
</CardContent>
</Card>
);
};
export default AdminPanel;

View File

@ -27,12 +27,14 @@ import {
MessageSquareText,
Users,
Settings,
Bell
Bell,
Shield
} from 'lucide-react';
import OnboardingTour from '@/components/onboarding/OnboardingTour';
import HelpIcon from '@/components/help/HelpIcon';
import UserProfileButton from '@/components/dashboard/UserProfileButton';
import { useOnboardingTour } from '@/hooks/useOnboardingTour';
import { useAdminSettings } from '@/hooks/useAdminSettings';
import {
Tooltip,
TooltipContent,
@ -46,10 +48,7 @@ interface NewModernLayoutProps {
const Logo = () => {
return (
<Link
to="/"
className="relative z-20 flex items-center space-x-2 py-1 text-sm font-normal text-black"
>
<div className="relative z-20 flex items-center space-x-2 py-1 text-sm font-normal text-black cursor-default">
<img
src="/lovable-uploads/7149adf3-440a-491e-83c2-d964a3348cc9.png"
alt="Finance Home Logo"
@ -58,12 +57,13 @@ const Logo = () => {
<span className="font-medium whitespace-pre text-blue-700 dark:text-white">
Finance Home
</span>
</Link>
</div>
);
};
export default function NewModernLayout({ children }: NewModernLayoutProps) {
const location = useLocation();
const { isAdmin, hideWhatsAppButton } = useAdminSettings();
const {
isOpen: tourOpen,
@ -117,15 +117,16 @@ export default function NewModernLayout({ children }: NewModernLayoutProps) {
];
const whatsappItems = [
{
...(hideWhatsAppButton ? [] : [{
title: "Conectar WhatsApp",
url: "/whatsapp",
icon: MessageSquareText,
},
}]),
{
title: "Grupos",
url: "/grupos-whatsapp",
icon: Users,
'data-tour': 'grupos-menu'
}
];
@ -137,6 +138,14 @@ export default function NewModernLayout({ children }: NewModernLayoutProps) {
}
];
const adminItems = isAdmin ? [
{
title: "Admin",
url: "/admin",
icon: Shield,
}
] : [];
return (
<div className="min-h-screen bg-background flex w-full">
<SidebarProvider defaultOpen={false}>
@ -173,6 +182,29 @@ export default function NewModernLayout({ children }: NewModernLayoutProps) {
<SidebarGroupContent>
<SidebarMenu>
{whatsappItems.map((item) => (
<SidebarMenuItem key={item.title}>
<SidebarMenuButton
asChild
tooltip={item.title}
isActive={location.pathname === item.url}
>
<Link to={item.url} data-tour={item['data-tour']}>
<item.icon />
<span>{item.title}</span>
</Link>
</SidebarMenuButton>
</SidebarMenuItem>
))}
</SidebarMenu>
</SidebarGroupContent>
</SidebarGroup>
{adminItems.length > 0 && (
<SidebarGroup>
<SidebarGroupLabel>Administração</SidebarGroupLabel>
<SidebarGroupContent>
<SidebarMenu>
{adminItems.map((item) => (
<SidebarMenuItem key={item.title}>
<SidebarMenuButton
asChild
@ -189,6 +221,7 @@ export default function NewModernLayout({ children }: NewModernLayoutProps) {
</SidebarMenu>
</SidebarGroupContent>
</SidebarGroup>
)}
</SidebarContent>
<SidebarFooter>

View File

@ -23,34 +23,34 @@ const OnboardingTour: React.FC<OnboardingTourProps> = ({
const steps = [
{
title: "Passo 1 - Conecte seu WhatsApp",
title: "Bem-vindo ao Finance Home! 🎉",
content: (
<>
<p className="mb-4">
Para que o app funcione, você precisa integrar seu WhatsApp com nossa plataforma.
Olá! Este é o Finance Home, sua ferramenta para controle financeiro automático.
</p>
<p className="mb-4">
Isso permitirá que suas transações feitas no WhatsApp apareçam automaticamente aqui no Finance Home.
Vamos te ensinar como navegar pelo sistema e usar suas principais funcionalidades.
</p>
<p className="font-semibold text-blue-600">
👉 Clique em "Conectar WhatsApp", digite o código da cidade + seu número de WhatsApp e depois clique em "Criar Instância".
👉 Use o botão do menu () no canto superior esquerdo para abrir e retrair o menu lateral.
</p>
</>
),
spotlight: 'whatsapp-menu'
spotlight: null
},
{
title: "Passo 2 - Crie seu Grupo com a IA",
title: "Passo 1 - Crie seu Grupo com a IA",
content: (
<>
<p className="mb-4">
Agora você precisa criar um grupo no WhatsApp com o nosso bot de IA chamado Angelina.
Para começar, você precisa criar um grupo no WhatsApp com o nosso bot de IA chamado Angelina.
</p>
<p className="mb-4">
Nele, você vai mandar mensagens, áudios ou comprovantes, e o sistema vai registrar automaticamente suas despesas e receitas.
</p>
<p className="font-semibold text-blue-600">
👉 Clique em "Grupos", escolha o nome do grupo e clique em "Cadastrar Grupo".
👉 Clique em "Grupos" no menu lateral, escolha o nome do grupo e clique em "Cadastrar Grupo".
</p>
</>
),
@ -63,9 +63,12 @@ const OnboardingTour: React.FC<OnboardingTourProps> = ({
<p className="text-lg mb-4">
Agora você pode começar a usar o Finance Home de forma automática.
</p>
<p className="text-base">
<p className="text-base mb-4">
Tudo que você enviar para o grupo vai ser registrado no app.
</p>
<p className="text-sm text-gray-600">
💡 Dica: Use o menu lateral para navegar entre as diferentes seções do sistema.
</p>
</>
),
spotlight: null
@ -87,7 +90,6 @@ const OnboardingTour: React.FC<OnboardingTourProps> = ({
});
};
// Aplicar highlight no elemento
const highlightElementStyle = () => {
const currentStepData = steps[currentStep];
if (!currentStepData?.spotlight) return;
@ -112,7 +114,6 @@ const OnboardingTour: React.FC<OnboardingTourProps> = ({
return;
}
// Limpar estilos anteriores
removeHighlightStyle();
const currentStepData = steps[currentStep];
@ -134,7 +135,6 @@ const OnboardingTour: React.FC<OnboardingTourProps> = ({
};
}, [isOpen, currentStep]);
// Cleanup quando o componente for desmontado
useEffect(() => {
return () => {
removeHighlightStyle();

View File

@ -0,0 +1,60 @@
import { useState, useEffect } from 'react';
import { supabase } from '@/integrations/supabase/client';
const ADMIN_EMAIL = 'rodrigobm10@gmail.com';
const SETTINGS_KEY = 'admin_hide_whatsapp_button';
export const useAdminSettings = () => {
const [isAdmin, setIsAdmin] = useState(false);
const [hideWhatsAppButton, setHideWhatsAppButton] = useState(false);
const [loading, setLoading] = useState(true);
useEffect(() => {
checkAdminStatus();
loadSettings();
}, []);
const checkAdminStatus = async () => {
try {
const { data: { user } } = await supabase.auth.getUser();
if (user?.email === ADMIN_EMAIL) {
setIsAdmin(true);
}
} catch (error) {
console.error('Erro ao verificar status admin:', error);
}
};
const loadSettings = async () => {
try {
const savedSetting = localStorage.getItem(SETTINGS_KEY);
if (savedSetting) {
setHideWhatsAppButton(JSON.parse(savedSetting));
}
} catch (error) {
console.error('Erro ao carregar configurações:', error);
} finally {
setLoading(false);
}
};
const toggleWhatsAppButton = async () => {
if (!isAdmin) return;
try {
const newValue = !hideWhatsAppButton;
setHideWhatsAppButton(newValue);
localStorage.setItem(SETTINGS_KEY, JSON.stringify(newValue));
} catch (error) {
console.error('Erro ao salvar configuração:', error);
}
};
return {
isAdmin,
hideWhatsAppButton,
loading,
toggleWhatsAppButton
};
};

View File

@ -27,7 +27,7 @@ export const useOnboardingTour = () => {
console.log('🔍 [TOUR] Iniciando verificação de condições do tour...');
// Só mostrar o tour na página inicial (dashboard)
if (location.pathname !== '/') {
if (location.pathname !== '/dashboard') {
console.log('❌ [TOUR] Tour só aparece no dashboard, página atual:', location.pathname);
setShouldShowTour(false);
return;
@ -108,7 +108,7 @@ export const useOnboardingTour = () => {
// Verificar condições quando a localização mudar ou na inicialização
useEffect(() => {
if (!tourCheckedRef.current && location.pathname === '/') {
if (!tourCheckedRef.current && location.pathname === '/dashboard') {
console.log('🔄 [TOUR] useEffect disparado - verificando condições do tour');
tourCheckedRef.current = true;
const timer = setTimeout(() => {
@ -123,7 +123,7 @@ export const useOnboardingTour = () => {
useEffect(() => {
const handleUserLoggedIn = (event: CustomEvent) => {
console.log('🎉 [TOUR] Evento de login recebido:', event.detail);
if (location.pathname === '/') {
if (location.pathname === '/dashboard') {
console.log('📧 [TOUR] Login detectado no dashboard, re-verificando condições do tour');
tourCheckedRef.current = false;
sessionStorage.removeItem(TOUR_SESSION_KEY);

35
src/pages/Admin.tsx Normal file
View File

@ -0,0 +1,35 @@
import AdminPanel from '@/components/admin/AdminPanel';
import { useAdminSettings } from '@/hooks/useAdminSettings';
import { Navigate } from 'react-router-dom';
const Admin = () => {
const { isAdmin, loading } = useAdminSettings();
if (loading) {
return (
<div className="flex items-center justify-center min-h-[200px]">
<div className="text-center">
<div className="animate-spin rounded-full h-8 w-8 border-b-2 border-blue-600 mx-auto"></div>
<p className="mt-2 text-gray-600">Carregando...</p>
</div>
</div>
);
}
if (!isAdmin) {
return <Navigate to="/dashboard" replace />;
}
return (
<div className="space-y-6">
<div className="flex justify-between items-center">
<h1 className="text-2xl font-bold tracking-tight">Administração</h1>
</div>
<AdminPanel />
</div>
);
};
export default Admin;