feat: implement relational lead origins with team assignments
All checks were successful
Build and Deploy / build-and-push (push) Successful in 1m51s
All checks were successful
Build and Deploy / build-and-push (push) Successful in 1m51s
- Dropped simple origins table in favor of origin_groups and origin_items to match the Funnels architecture. - Added origin_group_id to teams table to assign specific origins to specific teams. - Updated /admin/origins page to support creating origin groups, adding origin items to them, and assigning teams to groups. - Updated Dashboard and UserDetail pages to dynamically load the exact origin items belonging to the active team/user.
This commit is contained in:
2
App.tsx
2
App.tsx
@@ -9,6 +9,7 @@ import { ApiKeys } from './pages/ApiKeys';
|
|||||||
import { TeamManagement } from './pages/TeamManagement';
|
import { TeamManagement } from './pages/TeamManagement';
|
||||||
import { Teams } from './pages/Teams';
|
import { Teams } from './pages/Teams';
|
||||||
import { Funnels } from './pages/Funnels';
|
import { Funnels } from './pages/Funnels';
|
||||||
|
import { Origins } from './pages/Origins';
|
||||||
import { Login } from './pages/Login';
|
import { Login } from './pages/Login';
|
||||||
import { ForgotPassword } from './pages/ForgotPassword';
|
import { ForgotPassword } from './pages/ForgotPassword';
|
||||||
import { ResetPassword } from './pages/ResetPassword';
|
import { ResetPassword } from './pages/ResetPassword';
|
||||||
@@ -96,6 +97,7 @@ const App: React.FC = () => {
|
|||||||
<Route path="/admin/users" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><TeamManagement /></AuthGuard>} />
|
<Route path="/admin/users" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><TeamManagement /></AuthGuard>} />
|
||||||
<Route path="/admin/teams" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><Teams /></AuthGuard>} />
|
<Route path="/admin/teams" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><Teams /></AuthGuard>} />
|
||||||
<Route path="/admin/funnels" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><Funnels /></AuthGuard>} />
|
<Route path="/admin/funnels" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><Funnels /></AuthGuard>} />
|
||||||
|
<Route path="/admin/origins" element={<AuthGuard roles={['super_admin', 'admin', 'manager']}><Origins /></AuthGuard>} />
|
||||||
<Route path="/users/:id" element={<AuthGuard><UserDetail /></AuthGuard>} />
|
<Route path="/users/:id" element={<AuthGuard><UserDetail /></AuthGuard>} />
|
||||||
<Route path="/attendances/:id" element={<AuthGuard><AttendanceDetail /></AuthGuard>} />
|
<Route path="/attendances/:id" element={<AuthGuard><AttendanceDetail /></AuthGuard>} />
|
||||||
<Route path="/super-admin" element={<AuthGuard roles={['super_admin']}><SuperAdmin /></AuthGuard>} />
|
<Route path="/super-admin" element={<AuthGuard roles={['super_admin']}><SuperAdmin /></AuthGuard>} />
|
||||||
|
|||||||
188
backend/index.js
188
backend/index.js
@@ -612,6 +612,128 @@ apiRouter.delete('/notifications/:id', async (req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// --- Origin Routes (Groups & Items) ---
|
||||||
|
apiRouter.get('/origins', async (req, res) => {
|
||||||
|
try {
|
||||||
|
const { tenantId } = req.query;
|
||||||
|
const effectiveTenantId = req.user.role === 'super_admin' ? tenantId : req.user.tenant_id;
|
||||||
|
if (!effectiveTenantId || effectiveTenantId === 'all') return res.json([]);
|
||||||
|
|
||||||
|
const [groups] = await pool.query('SELECT * FROM origin_groups WHERE tenant_id = ? ORDER BY created_at ASC', [effectiveTenantId]);
|
||||||
|
|
||||||
|
// Seed default origin group if none exists
|
||||||
|
if (groups.length === 0) {
|
||||||
|
const gid = `origrp_${crypto.randomUUID().split('-')[0]}`;
|
||||||
|
await pool.query('INSERT INTO origin_groups (id, tenant_id, name) VALUES (?, ?, ?)', [gid, effectiveTenantId, 'Origens Padrão']);
|
||||||
|
|
||||||
|
const defaultOrigins = ['WhatsApp', 'Instagram', 'Website', 'LinkedIn', 'Indicação'];
|
||||||
|
for (const name of defaultOrigins) {
|
||||||
|
const oid = `oriitm_${crypto.randomUUID().split('-')[0]}`;
|
||||||
|
await pool.query(
|
||||||
|
'INSERT INTO origin_items (id, origin_group_id, name) VALUES (?, ?, ?)',
|
||||||
|
[oid, gid, name]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update all teams of this tenant to use this origin group if they have none
|
||||||
|
await pool.query('UPDATE teams SET origin_group_id = ? WHERE tenant_id = ? AND origin_group_id IS NULL', [gid, effectiveTenantId]);
|
||||||
|
|
||||||
|
groups.push({ id: gid, tenant_id: effectiveTenantId, name: 'Origens Padrão' });
|
||||||
|
}
|
||||||
|
|
||||||
|
const [items] = await pool.query('SELECT * FROM origin_items WHERE origin_group_id IN (?) ORDER BY created_at ASC', [groups.map(g => g.id)]);
|
||||||
|
const [teams] = await pool.query('SELECT id, origin_group_id FROM teams WHERE tenant_id = ? AND origin_group_id IS NOT NULL', [effectiveTenantId]);
|
||||||
|
|
||||||
|
const result = groups.map(g => ({
|
||||||
|
...g,
|
||||||
|
items: items.filter(i => i.origin_group_id === g.id),
|
||||||
|
teamIds: teams.filter(t => t.origin_group_id === g.id).map(t => t.id)
|
||||||
|
}));
|
||||||
|
|
||||||
|
res.json(result);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("GET /origins error:", error);
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.post('/origins', requireRole(['admin', 'owner', 'manager', 'super_admin']), async (req, res) => {
|
||||||
|
const { name, tenantId } = req.body;
|
||||||
|
const effectiveTenantId = req.user.role === 'super_admin' ? tenantId : req.user.tenant_id;
|
||||||
|
try {
|
||||||
|
const gid = `origrp_${crypto.randomUUID().split('-')[0]}`;
|
||||||
|
await pool.query('INSERT INTO origin_groups (id, tenant_id, name) VALUES (?, ?, ?)', [gid, effectiveTenantId, name]);
|
||||||
|
res.status(201).json({ id: gid });
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.put('/origins/:id', requireRole(['admin', 'owner', 'manager', 'super_admin']), async (req, res) => {
|
||||||
|
const { name, teamIds } = req.body;
|
||||||
|
try {
|
||||||
|
if (name) {
|
||||||
|
await pool.query('UPDATE origin_groups SET name = ? WHERE id = ?', [name, req.params.id]);
|
||||||
|
}
|
||||||
|
if (teamIds && Array.isArray(teamIds)) {
|
||||||
|
await pool.query('UPDATE teams SET origin_group_id = NULL WHERE origin_group_id = ?', [req.params.id]);
|
||||||
|
if (teamIds.length > 0) {
|
||||||
|
await pool.query('UPDATE teams SET origin_group_id = ? WHERE id IN (?)', [req.params.id, teamIds]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
res.json({ message: 'Origin group updated.' });
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.delete('/origins/:id', requireRole(['admin', 'owner', 'manager', 'super_admin']), async (req, res) => {
|
||||||
|
try {
|
||||||
|
await pool.query('DELETE FROM origin_items WHERE origin_group_id = ?', [req.params.id]);
|
||||||
|
await pool.query('UPDATE teams SET origin_group_id = NULL WHERE origin_group_id = ?', [req.params.id]);
|
||||||
|
await pool.query('DELETE FROM origin_groups WHERE id = ?', [req.params.id]);
|
||||||
|
res.json({ message: 'Origin group deleted.' });
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.post('/origins/:id/items', requireRole(['admin', 'owner', 'manager', 'super_admin']), async (req, res) => {
|
||||||
|
const { name } = req.body;
|
||||||
|
try {
|
||||||
|
const oid = `oriitm_${crypto.randomUUID().split('-')[0]}`;
|
||||||
|
await pool.query(
|
||||||
|
'INSERT INTO origin_items (id, origin_group_id, name) VALUES (?, ?, ?)',
|
||||||
|
[oid, req.params.id, name]
|
||||||
|
);
|
||||||
|
res.status(201).json({ id: oid });
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.put('/origin_items/:id', requireRole(['admin', 'owner', 'manager', 'super_admin']), async (req, res) => {
|
||||||
|
const { name } = req.body;
|
||||||
|
try {
|
||||||
|
const [existing] = await pool.query('SELECT * FROM origin_items WHERE id = ?', [req.params.id]);
|
||||||
|
if (existing.length === 0) return res.status(404).json({ error: 'Origin item not found' });
|
||||||
|
|
||||||
|
await pool.query('UPDATE origin_items SET name = ? WHERE id = ?', [name || existing[0].name, req.params.id]);
|
||||||
|
res.json({ message: 'Origin item updated.' });
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.delete('/origin_items/:id', requireRole(['admin', 'owner', 'manager', 'super_admin']), async (req, res) => {
|
||||||
|
try {
|
||||||
|
await pool.query('DELETE FROM origin_items WHERE id = ?', [req.params.id]);
|
||||||
|
res.json({ message: 'Origin item deleted.' });
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// --- Funnel Routes ---
|
// --- Funnel Routes ---
|
||||||
apiRouter.get('/funnels', async (req, res) => {
|
apiRouter.get('/funnels', async (req, res) => {
|
||||||
try {
|
try {
|
||||||
@@ -952,6 +1074,37 @@ apiRouter.get('/integration/users', requireRole(['admin']), async (req, res) =>
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
apiRouter.get('/integration/origins', requireRole(['admin']), async (req, res) => {
|
||||||
|
if (!req.user.is_api_key) return res.status(403).json({ error: 'Endpoint restrito a chaves de API.' });
|
||||||
|
try {
|
||||||
|
const [origins] = await pool.query('SELECT name FROM origins WHERE tenant_id = ? ORDER BY created_at ASC', [req.user.tenant_id]);
|
||||||
|
res.json(origins.map(o => o.name));
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
apiRouter.get('/integration/funnels', requireRole(['admin']), async (req, res) => {
|
||||||
|
if (!req.user.is_api_key) return res.status(403).json({ error: 'Endpoint restrito a chaves de API.' });
|
||||||
|
try {
|
||||||
|
const [funnels] = await pool.query('SELECT id, name FROM funnels WHERE tenant_id = ?', [req.user.tenant_id]);
|
||||||
|
if (funnels.length === 0) return res.json([]);
|
||||||
|
|
||||||
|
const [stages] = await pool.query('SELECT funnel_id, name, order_index FROM funnel_stages WHERE funnel_id IN (?) ORDER BY order_index ASC', [funnels.map(f => f.id)]);
|
||||||
|
const [teams] = await pool.query('SELECT id as team_id, name as team_name, funnel_id FROM teams WHERE tenant_id = ? AND funnel_id IS NOT NULL', [req.user.tenant_id]);
|
||||||
|
|
||||||
|
const result = funnels.map(f => ({
|
||||||
|
funnel_name: f.name,
|
||||||
|
stages: stages.filter(s => s.funnel_id === f.id).map(s => s.name),
|
||||||
|
assigned_teams: teams.filter(t => t.funnel_id === f.id).map(t => ({ id: t.team_id, name: t.team_name }))
|
||||||
|
}));
|
||||||
|
|
||||||
|
res.json(result);
|
||||||
|
} catch (error) {
|
||||||
|
res.status(500).json({ error: error.message });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
apiRouter.post('/integration/attendances', requireRole(['admin']), async (req, res) => {
|
apiRouter.post('/integration/attendances', requireRole(['admin']), async (req, res) => {
|
||||||
if (!req.user.is_api_key) return res.status(403).json({ error: 'Endpoint restrito a chaves de API.' });
|
if (!req.user.is_api_key) return res.status(403).json({ error: 'Endpoint restrito a chaves de API.' });
|
||||||
|
|
||||||
@@ -1288,9 +1441,9 @@ const provisionSuperAdmin = async (retries = 10, delay = 10000) => {
|
|||||||
if (err.code !== 'ER_DUP_FIELDNAME') console.log('Schema update note (sound_enabled):', err.message);
|
if (err.code !== 'ER_DUP_FIELDNAME') console.log('Schema update note (sound_enabled):', err.message);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update origin enum
|
// Update origin to VARCHAR for custom origins
|
||||||
try {
|
try {
|
||||||
await connection.query("ALTER TABLE attendances MODIFY COLUMN origin ENUM('WhatsApp','Instagram','Website','LinkedIn','Indicação') NOT NULL");
|
await connection.query("ALTER TABLE attendances MODIFY COLUMN origin VARCHAR(255) NOT NULL");
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
console.log('Schema update note (origin):', err.message);
|
console.log('Schema update note (origin):', err.message);
|
||||||
}
|
}
|
||||||
@@ -1309,6 +1462,37 @@ const provisionSuperAdmin = async (retries = 10, delay = 10000) => {
|
|||||||
if (err.code !== 'ER_DUP_FIELDNAME') console.log('Schema update note (full_summary):', err.message);
|
if (err.code !== 'ER_DUP_FIELDNAME') console.log('Schema update note (full_summary):', err.message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create origin_groups table
|
||||||
|
await connection.query(`
|
||||||
|
CREATE TABLE IF NOT EXISTS origin_groups (
|
||||||
|
id varchar(36) NOT NULL,
|
||||||
|
tenant_id varchar(36) NOT NULL,
|
||||||
|
name varchar(255) NOT NULL,
|
||||||
|
created_at timestamp NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
PRIMARY KEY (id),
|
||||||
|
KEY tenant_id (tenant_id)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
|
||||||
|
`);
|
||||||
|
|
||||||
|
// Create origin_items table
|
||||||
|
await connection.query(`
|
||||||
|
CREATE TABLE IF NOT EXISTS origin_items (
|
||||||
|
id varchar(36) NOT NULL,
|
||||||
|
origin_group_id varchar(36) NOT NULL,
|
||||||
|
name varchar(255) NOT NULL,
|
||||||
|
created_at timestamp NULL DEFAULT CURRENT_TIMESTAMP,
|
||||||
|
PRIMARY KEY (id),
|
||||||
|
KEY origin_group_id (origin_group_id)
|
||||||
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;
|
||||||
|
`);
|
||||||
|
|
||||||
|
// Add origin_group_id to teams
|
||||||
|
try {
|
||||||
|
await connection.query("ALTER TABLE teams ADD COLUMN origin_group_id VARCHAR(36) DEFAULT NULL");
|
||||||
|
} catch (err) {
|
||||||
|
if (err.code !== 'ER_DUP_FIELDNAME') console.log('Schema update note (teams.origin_group_id):', err.message);
|
||||||
|
}
|
||||||
|
|
||||||
// Rename summary to title
|
// Rename summary to title
|
||||||
try {
|
try {
|
||||||
await connection.query("ALTER TABLE attendances RENAME COLUMN summary TO title");
|
await connection.query("ALTER TABLE attendances RENAME COLUMN summary TO title");
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ import { NavLink, useLocation, useNavigate } from 'react-router-dom';
|
|||||||
import {
|
import {
|
||||||
LayoutDashboard, Users, UserCircle, Bell, Search, Menu, X, LogOut,
|
LayoutDashboard, Users, UserCircle, Bell, Search, Menu, X, LogOut,
|
||||||
Hexagon, Settings, Building2, Sun, Moon, Loader2, Layers,
|
Hexagon, Settings, Building2, Sun, Moon, Loader2, Layers,
|
||||||
ChevronLeft, ChevronRight, Key
|
ChevronLeft, ChevronRight, Key, Target
|
||||||
} from 'lucide-react';
|
} from 'lucide-react';
|
||||||
import {
|
import {
|
||||||
getAttendances, getUsers, getUserById, logout, searchGlobal,
|
getAttendances, getUsers, getUserById, logout, searchGlobal,
|
||||||
@@ -240,6 +240,7 @@ export const Layout: React.FC<{ children: React.ReactNode }> = ({ children }) =>
|
|||||||
<SidebarItem to="/admin/users" icon={Users} label="Membros" collapsed={isSidebarCollapsed} />
|
<SidebarItem to="/admin/users" icon={Users} label="Membros" collapsed={isSidebarCollapsed} />
|
||||||
<SidebarItem to="/admin/teams" icon={Building2} label={currentUser.role === 'manager' ? 'Meu Time' : 'Times'} collapsed={isSidebarCollapsed} />
|
<SidebarItem to="/admin/teams" icon={Building2} label={currentUser.role === 'manager' ? 'Meu Time' : 'Times'} collapsed={isSidebarCollapsed} />
|
||||||
<SidebarItem to="/admin/funnels" icon={Layers} label="Gerenciar Funis" collapsed={isSidebarCollapsed} />
|
<SidebarItem to="/admin/funnels" icon={Layers} label="Gerenciar Funis" collapsed={isSidebarCollapsed} />
|
||||||
|
<SidebarItem to="/admin/origins" icon={Target} label="Origens de Lead" collapsed={isSidebarCollapsed} />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
|||||||
@@ -5,9 +5,9 @@ import {
|
|||||||
import {
|
import {
|
||||||
Users, Clock, Phone, TrendingUp, Filter
|
Users, Clock, Phone, TrendingUp, Filter
|
||||||
} from 'lucide-react';
|
} from 'lucide-react';
|
||||||
import { getAttendances, getUsers, getTeams, getUserById, getFunnels } from '../services/dataService';
|
import { getAttendances, getUsers, getTeams, getUserById, getFunnels, getOrigins } from '../services/dataService';
|
||||||
import { COLORS } from '../constants';
|
import { COLORS } from '../constants';
|
||||||
import { Attendance, DashboardFilter, FunnelStage, User, FunnelStageDef } from '../types';
|
import { Attendance, DashboardFilter, FunnelStage, User, FunnelStageDef, OriginItemDef } from '../types';
|
||||||
import { KPICard } from '../components/KPICard';
|
import { KPICard } from '../components/KPICard';
|
||||||
import { DateRangePicker } from '../components/DateRangePicker';
|
import { DateRangePicker } from '../components/DateRangePicker';
|
||||||
import { SellersTable } from '../components/SellersTable';
|
import { SellersTable } from '../components/SellersTable';
|
||||||
@@ -29,6 +29,7 @@ export const Dashboard: React.FC = () => {
|
|||||||
const [users, setUsers] = useState<User[]>([]);
|
const [users, setUsers] = useState<User[]>([]);
|
||||||
const [teams, setTeams] = useState<any[]>([]);
|
const [teams, setTeams] = useState<any[]>([]);
|
||||||
const [funnelDefs, setFunnelDefs] = useState<FunnelStageDef[]>([]);
|
const [funnelDefs, setFunnelDefs] = useState<FunnelStageDef[]>([]);
|
||||||
|
const [originDefs, setOriginDefs] = useState<OriginItemDef[]>([]);
|
||||||
const [currentUser, setCurrentUser] = useState<User | null>(null);
|
const [currentUser, setCurrentUser] = useState<User | null>(null);
|
||||||
|
|
||||||
const [filters, setFilters] = useState<DashboardFilter>({
|
const [filters, setFilters] = useState<DashboardFilter>({
|
||||||
@@ -57,12 +58,13 @@ export const Dashboard: React.FC = () => {
|
|||||||
const prevFilters = { ...filters, dateRange: { start: prevStart, end: prevEnd } };
|
const prevFilters = { ...filters, dateRange: { start: prevStart, end: prevEnd } };
|
||||||
|
|
||||||
// Fetch users, attendances, teams, funnels and current user in parallel
|
// Fetch users, attendances, teams, funnels and current user in parallel
|
||||||
const [fetchedUsers, fetchedData, prevFetchedData, fetchedTeams, fetchedFunnels, me] = await Promise.all([
|
const [fetchedUsers, fetchedData, prevFetchedData, fetchedTeams, fetchedFunnels, fetchedOrigins, me] = await Promise.all([
|
||||||
getUsers(tenantId),
|
getUsers(tenantId),
|
||||||
getAttendances(tenantId, filters),
|
getAttendances(tenantId, filters),
|
||||||
getAttendances(tenantId, prevFilters),
|
getAttendances(tenantId, prevFilters),
|
||||||
getTeams(tenantId),
|
getTeams(tenantId),
|
||||||
getFunnels(tenantId),
|
getFunnels(tenantId),
|
||||||
|
getOrigins(tenantId),
|
||||||
storedUserId ? getUserById(storedUserId) : null
|
storedUserId ? getUserById(storedUserId) : null
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@@ -70,6 +72,7 @@ export const Dashboard: React.FC = () => {
|
|||||||
setData(fetchedData);
|
setData(fetchedData);
|
||||||
setPrevData(prevFetchedData);
|
setPrevData(prevFetchedData);
|
||||||
setTeams(fetchedTeams);
|
setTeams(fetchedTeams);
|
||||||
|
setOriginDefs(fetchedOrigins);
|
||||||
if (me) setCurrentUser(me);
|
if (me) setCurrentUser(me);
|
||||||
|
|
||||||
// Determine which funnel to display
|
// Determine which funnel to display
|
||||||
@@ -82,6 +85,14 @@ export const Dashboard: React.FC = () => {
|
|||||||
|
|
||||||
setFunnelDefs(activeFunnel && activeFunnel.stages ? activeFunnel.stages.sort((a: any, b: any) => a.order_index - b.order_index) : []);
|
setFunnelDefs(activeFunnel && activeFunnel.stages ? activeFunnel.stages.sort((a: any, b: any) => a.order_index - b.order_index) : []);
|
||||||
|
|
||||||
|
// Determine which origins to display
|
||||||
|
let activeOriginGroup = fetchedOrigins[0];
|
||||||
|
if (targetTeamId) {
|
||||||
|
const matchedOrigin = fetchedOrigins.find(o => o.teamIds?.includes(targetTeamId));
|
||||||
|
if (matchedOrigin) activeOriginGroup = matchedOrigin;
|
||||||
|
}
|
||||||
|
setOriginDefs(activeOriginGroup && activeOriginGroup.items ? activeOriginGroup.items : []);
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error loading dashboard data:", error);
|
console.error("Error loading dashboard data:", error);
|
||||||
} finally {
|
} finally {
|
||||||
@@ -305,13 +316,12 @@ export const Dashboard: React.FC = () => {
|
|||||||
onChange={(e) => handleFilterChange('origin', e.target.value)}
|
onChange={(e) => handleFilterChange('origin', e.target.value)}
|
||||||
>
|
>
|
||||||
<option value="all">Todas Origens</option>
|
<option value="all">Todas Origens</option>
|
||||||
<option value="WhatsApp">WhatsApp</option>
|
{originDefs.length > 0 ? originDefs.map(o => (
|
||||||
<option value="Instagram">Instagram</option>
|
<option key={o.id} value={o.name}>{o.name}</option>
|
||||||
<option value="Website">Website</option>
|
)) : ['WhatsApp', 'Instagram', 'Website', 'LinkedIn', 'Indicação'].map(o => (
|
||||||
<option value="LinkedIn">LinkedIn</option>
|
<option key={o} value={o}>{o}</option>
|
||||||
<option value="Indicação">Indicação</option>
|
))}
|
||||||
</select>
|
</select> </div>
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
|||||||
301
pages/Origins.tsx
Normal file
301
pages/Origins.tsx
Normal file
@@ -0,0 +1,301 @@
|
|||||||
|
import React, { useState, useEffect } from 'react';
|
||||||
|
import { Target, Plus, Edit, Trash2, Loader2, X, Users } from 'lucide-react';
|
||||||
|
import { getOrigins, createOriginGroup, updateOriginGroup, deleteOriginGroup, createOriginItem, updateOriginItem, deleteOriginItem, getTeams } from '../services/dataService';
|
||||||
|
import { OriginGroupDef, OriginItemDef } from '../types';
|
||||||
|
|
||||||
|
export const Origins: React.FC = () => {
|
||||||
|
const [originGroups, setOriginGroups] = useState<OriginGroupDef[]>([]);
|
||||||
|
const [teams, setTeams] = useState<any[]>([]);
|
||||||
|
const [selectedGroupId, setSelectedGroupId] = useState<string | null>(null);
|
||||||
|
|
||||||
|
const [loading, setLoading] = useState(true);
|
||||||
|
const [isModalOpen, setIsModalOpen] = useState(false);
|
||||||
|
const [editingItem, setEditingItem] = useState<OriginItemDef | null>(null);
|
||||||
|
const [formData, setFormData] = useState({ name: '' });
|
||||||
|
const [isSaving, setIsSaving] = useState(false);
|
||||||
|
|
||||||
|
// Group creation state
|
||||||
|
const [isGroupModalOpen, setIsGroupModalOpen] = useState(false);
|
||||||
|
const [groupName, setGroupName] = useState('');
|
||||||
|
|
||||||
|
const tenantId = localStorage.getItem('ctms_tenant_id') || '';
|
||||||
|
|
||||||
|
const loadData = async () => {
|
||||||
|
setLoading(true);
|
||||||
|
const [fetchedGroups, fetchedTeams] = await Promise.all([
|
||||||
|
getOrigins(tenantId),
|
||||||
|
getTeams(tenantId)
|
||||||
|
]);
|
||||||
|
setOriginGroups(fetchedGroups);
|
||||||
|
setTeams(fetchedTeams);
|
||||||
|
if (!selectedGroupId && fetchedGroups.length > 0) {
|
||||||
|
setSelectedGroupId(fetchedGroups[0].id);
|
||||||
|
}
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
loadData();
|
||||||
|
}, [tenantId]);
|
||||||
|
|
||||||
|
const selectedGroup = originGroups.find(g => g.id === selectedGroupId);
|
||||||
|
|
||||||
|
// --- Group Handlers ---
|
||||||
|
const handleCreateGroup = async (e: React.FormEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
setIsSaving(true);
|
||||||
|
try {
|
||||||
|
const res = await createOriginGroup({ name: groupName, tenantId });
|
||||||
|
setSelectedGroupId(res.id);
|
||||||
|
setIsGroupModalOpen(false);
|
||||||
|
setGroupName('');
|
||||||
|
loadData();
|
||||||
|
} catch (err) {
|
||||||
|
alert("Erro ao criar grupo de origens.");
|
||||||
|
} finally {
|
||||||
|
setIsSaving(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteGroup = async (id: string) => {
|
||||||
|
if (originGroups.length <= 1) {
|
||||||
|
alert("Você precisa ter pelo menos um grupo de origens ativo.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (confirm('Tem certeza que deseja excluir este grupo e todas as suas origens?')) {
|
||||||
|
await deleteOriginGroup(id);
|
||||||
|
setSelectedGroupId(null);
|
||||||
|
loadData();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleToggleTeam = async (teamId: string) => {
|
||||||
|
if (!selectedGroup) return;
|
||||||
|
const currentTeamIds = selectedGroup.teamIds || [];
|
||||||
|
const newTeamIds = currentTeamIds.includes(teamId)
|
||||||
|
? currentTeamIds.filter(id => id !== teamId)
|
||||||
|
: [...currentTeamIds, teamId];
|
||||||
|
|
||||||
|
// Optimistic
|
||||||
|
const newGroups = [...originGroups];
|
||||||
|
const idx = newGroups.findIndex(g => g.id === selectedGroup.id);
|
||||||
|
newGroups[idx].teamIds = newTeamIds;
|
||||||
|
setOriginGroups(newGroups);
|
||||||
|
|
||||||
|
await updateOriginGroup(selectedGroup.id, { teamIds: newTeamIds });
|
||||||
|
};
|
||||||
|
|
||||||
|
// --- Item Handlers ---
|
||||||
|
const handleSaveItem = async (e: React.FormEvent) => {
|
||||||
|
e.preventDefault();
|
||||||
|
if (!selectedGroup) return;
|
||||||
|
setIsSaving(true);
|
||||||
|
try {
|
||||||
|
if (editingItem) {
|
||||||
|
await updateOriginItem(editingItem.id, formData);
|
||||||
|
} else {
|
||||||
|
await createOriginItem(selectedGroup.id, formData);
|
||||||
|
}
|
||||||
|
setIsModalOpen(false);
|
||||||
|
loadData();
|
||||||
|
} catch (err: any) {
|
||||||
|
alert(err.message || "Erro ao salvar origem.");
|
||||||
|
} finally {
|
||||||
|
setIsSaving(false);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteItem = async (id: string) => {
|
||||||
|
if (confirm('Tem certeza que deseja excluir esta origem?')) {
|
||||||
|
await deleteOriginItem(id);
|
||||||
|
loadData();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const openItemModal = (item?: OriginItemDef) => {
|
||||||
|
if (item) {
|
||||||
|
setEditingItem(item);
|
||||||
|
setFormData({ name: item.name });
|
||||||
|
} else {
|
||||||
|
setEditingItem(null);
|
||||||
|
setFormData({ name: '' });
|
||||||
|
}
|
||||||
|
setIsModalOpen(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (loading && originGroups.length === 0) return <div className="p-12 flex justify-center"><Loader2 className="animate-spin text-zinc-400" size={32} /></div>;
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="max-w-6xl mx-auto space-y-6 pb-12 transition-colors duration-300 flex flex-col md:flex-row gap-8">
|
||||||
|
|
||||||
|
{/* Sidebar: Groups List */}
|
||||||
|
<div className="w-full md:w-64 shrink-0 space-y-4">
|
||||||
|
<div className="flex items-center justify-between">
|
||||||
|
<h2 className="text-lg font-bold text-zinc-900 dark:text-zinc-50">Configurações</h2>
|
||||||
|
<button onClick={() => setIsGroupModalOpen(true)} className="p-1.5 bg-zinc-100 dark:bg-dark-bg text-zinc-600 dark:text-dark-muted rounded-lg hover:bg-zinc-200 dark:hover:bg-dark-border transition-colors">
|
||||||
|
<Plus size={16} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
{originGroups.map(g => (
|
||||||
|
<button
|
||||||
|
key={g.id}
|
||||||
|
onClick={() => setSelectedGroupId(g.id)}
|
||||||
|
className={`text-left px-4 py-3 rounded-xl text-sm font-medium transition-all ${selectedGroupId === g.id ? 'bg-zinc-900 dark:bg-brand-yellow text-white dark:text-zinc-950 shadow-md' : 'bg-white dark:bg-dark-card text-zinc-600 dark:text-zinc-400 hover:bg-zinc-50 dark:hover:bg-dark-bg border border-zinc-200 dark:border-dark-border'}`}
|
||||||
|
>
|
||||||
|
<div className="flex justify-between items-center">
|
||||||
|
<span className="truncate pr-2">{g.name}</span>
|
||||||
|
<span className={`text-[10px] px-1.5 py-0.5 rounded-full shrink-0 ${selectedGroupId === g.id ? 'bg-white/20 dark:bg-black/20' : 'bg-zinc-100 dark:bg-dark-bg'}`}>{g.items?.length || 0}</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Main Content: Selected Group Details */}
|
||||||
|
<div className="flex-1 space-y-6">
|
||||||
|
{selectedGroup ? (
|
||||||
|
<>
|
||||||
|
<div className="flex flex-col sm:flex-row justify-between items-start sm:items-center gap-4 border-b border-zinc-200 dark:border-dark-border pb-6">
|
||||||
|
<div>
|
||||||
|
<h1 className="text-2xl font-bold text-zinc-900 dark:text-zinc-50 tracking-tight">{selectedGroup.name}</h1>
|
||||||
|
<p className="text-zinc-500 dark:text-zinc-400 text-sm">Gerencie as origens deste grupo e quais times as utilizam.</p>
|
||||||
|
</div>
|
||||||
|
<button onClick={() => handleDeleteGroup(selectedGroup.id)} className="text-red-500 hover:text-red-700 bg-red-50 dark:bg-red-900/20 px-3 py-2 rounded-lg text-sm font-semibold transition-colors flex items-center gap-2">
|
||||||
|
<Trash2 size={16} /> Excluir Grupo
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Teams Assignment */}
|
||||||
|
<div className="bg-white dark:bg-dark-card rounded-2xl border border-zinc-200 dark:border-dark-border shadow-sm overflow-hidden">
|
||||||
|
<div className="px-6 py-4 border-b border-zinc-100 dark:border-dark-border bg-zinc-50/50 dark:bg-dark-bg/50">
|
||||||
|
<h3 className="font-bold text-zinc-900 dark:text-zinc-50 flex items-center gap-2">
|
||||||
|
<Users className="text-brand-yellow" size={18} /> Times Atribuídos
|
||||||
|
</h3>
|
||||||
|
</div>
|
||||||
|
<div className="p-6">
|
||||||
|
{teams.length === 0 ? (
|
||||||
|
<p className="text-sm text-zinc-500">Nenhum time cadastrado na organização.</p>
|
||||||
|
) : (
|
||||||
|
<div className="flex flex-wrap gap-3">
|
||||||
|
{teams.map(t => {
|
||||||
|
const isAssigned = selectedGroup.teamIds?.includes(t.id);
|
||||||
|
return (
|
||||||
|
<button
|
||||||
|
key={t.id}
|
||||||
|
onClick={() => handleToggleTeam(t.id)}
|
||||||
|
className={`px-3 py-1.5 rounded-lg text-sm font-medium border transition-all ${isAssigned ? 'bg-brand-yellow/10 border-brand-yellow text-zinc-900 dark:text-zinc-100' : 'bg-white dark:bg-dark-input border-zinc-200 dark:border-dark-border text-zinc-500 dark:text-zinc-400 hover:border-zinc-300 dark:hover:border-zinc-700'}`}
|
||||||
|
>
|
||||||
|
{t.name}
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<p className="text-xs text-zinc-400 mt-4">Times não atribuídos a um grupo específico usarão o grupo padrão.</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Origin Items */}
|
||||||
|
<div className="bg-white dark:bg-dark-card rounded-2xl border border-zinc-200 dark:border-dark-border shadow-sm overflow-hidden">
|
||||||
|
<div className="px-6 py-4 border-b border-zinc-100 dark:border-dark-border bg-zinc-50/50 dark:bg-dark-bg/50 flex justify-between items-center">
|
||||||
|
<h3 className="font-bold text-zinc-900 dark:text-zinc-50 flex items-center gap-2">
|
||||||
|
<Target className="text-brand-yellow" size={18} /> Fontes de Tráfego
|
||||||
|
</h3>
|
||||||
|
<button onClick={() => openItemModal()} className="text-sm font-bold text-brand-yellow hover:underline flex items-center gap-1">
|
||||||
|
<Plus size={16} /> Nova Origem
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="divide-y divide-zinc-100 dark:divide-dark-border">
|
||||||
|
{selectedGroup.items?.map((o) => (
|
||||||
|
<div key={o.id} className="p-4 px-6 flex items-center justify-between group hover:bg-zinc-50 dark:hover:bg-dark-bg transition-colors">
|
||||||
|
<div className="font-medium text-zinc-800 dark:text-zinc-200">{o.name}</div>
|
||||||
|
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<button onClick={() => openItemModal(o)} className="p-2 text-zinc-400 hover:text-brand-yellow hover:bg-zinc-100 dark:hover:bg-dark-input rounded-lg transition-colors">
|
||||||
|
<Edit size={16} />
|
||||||
|
</button>
|
||||||
|
<button onClick={() => handleDeleteItem(o.id)} className="p-2 text-zinc-400 hover:text-red-600 hover:bg-red-50 dark:hover:bg-red-900/30 rounded-lg transition-colors">
|
||||||
|
<Trash2 size={16} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
{(!selectedGroup.items || selectedGroup.items.length === 0) && (
|
||||||
|
<div className="p-8 text-center text-zinc-500 dark:text-dark-muted">Nenhuma origem configurada neste grupo.</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<div className="p-12 text-center text-zinc-500">Selecione ou crie um grupo de origens.</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{/* Group Creation Modal */}
|
||||||
|
{isGroupModalOpen && (
|
||||||
|
<div className="fixed inset-0 z-50 flex items-center justify-center p-4 bg-zinc-950/80 backdrop-blur-sm">
|
||||||
|
<div className="bg-white dark:bg-dark-card rounded-xl shadow-xl w-full max-w-sm overflow-hidden animate-in fade-in zoom-in duration-200">
|
||||||
|
<div className="px-6 py-4 border-b border-zinc-100 dark:border-dark-border flex justify-between items-center bg-zinc-50/50 dark:bg-dark-bg/50">
|
||||||
|
<h3 className="font-bold text-zinc-900 dark:text-zinc-50">Novo Grupo</h3>
|
||||||
|
<button onClick={() => setIsGroupModalOpen(false)} className="text-zinc-400 hover:text-zinc-600 dark:hover:text-zinc-300"><X size={20} /></button>
|
||||||
|
</div>
|
||||||
|
<form onSubmit={handleCreateGroup} className="p-6 space-y-4">
|
||||||
|
<div>
|
||||||
|
<label className="text-xs font-bold text-zinc-500 dark:text-dark-muted uppercase mb-1 block">Nome do Grupo</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={groupName}
|
||||||
|
onChange={e => setGroupName(e.target.value)}
|
||||||
|
placeholder="Ex: Origens B2B"
|
||||||
|
className="w-full bg-white dark:bg-dark-input border border-zinc-200 dark:border-dark-border p-3 rounded-lg text-sm text-zinc-900 dark:text-zinc-100 focus:ring-2 focus:ring-brand-yellow/20 outline-none transition-all"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="pt-4 flex justify-end gap-3 mt-6 border-t border-zinc-100 dark:border-dark-border pt-6">
|
||||||
|
<button type="button" onClick={() => setIsGroupModalOpen(false)} className="px-4 py-2 text-zinc-600 dark:text-zinc-400 hover:bg-zinc-100 dark:hover:bg-dark-border rounded-lg text-sm font-medium transition-colors">Cancelar</button>
|
||||||
|
<button type="submit" disabled={isSaving || !groupName.trim()} className="px-6 py-2 bg-zinc-900 dark:bg-brand-yellow text-white dark:text-zinc-950 rounded-lg text-sm font-bold flex items-center gap-2 hover:opacity-90 transition-all shadow-sm disabled:opacity-70">
|
||||||
|
{isSaving ? <Loader2 className="animate-spin" size={16} /> : 'Criar Grupo'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{/* Item Modal */}
|
||||||
|
{isModalOpen && (
|
||||||
|
<div className="fixed inset-0 z-50 flex items-center justify-center p-4 bg-zinc-950/80 backdrop-blur-sm">
|
||||||
|
<div className="bg-white dark:bg-dark-card rounded-xl shadow-xl w-full max-w-md overflow-hidden animate-in fade-in zoom-in duration-200">
|
||||||
|
<div className="px-6 py-4 border-b border-zinc-100 dark:border-dark-border flex justify-between items-center bg-zinc-50/50 dark:bg-dark-bg/50">
|
||||||
|
<h3 className="font-bold text-zinc-900 dark:text-zinc-50">{editingItem ? 'Editar Origem' : 'Nova Origem'}</h3>
|
||||||
|
<button onClick={() => setIsModalOpen(false)} className="text-zinc-400 hover:text-zinc-600 dark:hover:text-zinc-300"><X size={20} /></button>
|
||||||
|
</div>
|
||||||
|
<form onSubmit={handleSaveItem} className="p-6 space-y-4">
|
||||||
|
<div>
|
||||||
|
<label className="text-xs font-bold text-zinc-500 dark:text-dark-muted uppercase mb-1 block">Nome da Origem</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={formData.name}
|
||||||
|
onChange={e => setFormData({...formData, name: e.target.value})}
|
||||||
|
placeholder="Ex: Facebook Ads"
|
||||||
|
className="w-full bg-white dark:bg-dark-input border border-zinc-200 dark:border-dark-border p-3 rounded-lg text-sm text-zinc-900 dark:text-zinc-100 focus:ring-2 focus:ring-brand-yellow/20 outline-none transition-all"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="pt-4 flex justify-end gap-3 mt-6 border-t border-zinc-100 dark:border-dark-border pt-6">
|
||||||
|
<button type="button" onClick={() => setIsModalOpen(false)} className="px-4 py-2 text-zinc-600 dark:text-zinc-400 hover:bg-zinc-100 dark:hover:bg-dark-border rounded-lg text-sm font-medium transition-colors">Cancelar</button>
|
||||||
|
<button type="submit" disabled={isSaving} className="px-6 py-2 bg-zinc-900 dark:bg-brand-yellow text-white dark:text-zinc-950 rounded-lg text-sm font-bold flex items-center gap-2 hover:opacity-90 transition-all shadow-sm disabled:opacity-70">
|
||||||
|
{isSaving ? <Loader2 className="animate-spin" size={16} /> : 'Salvar Origem'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
import React, { useEffect, useState, useMemo } from 'react';
|
import React, { useEffect, useState, useMemo } from 'react';
|
||||||
import { useParams, Link } from 'react-router-dom';
|
import { useParams, Link } from 'react-router-dom';
|
||||||
import { getAttendances, getUserById, getFunnels } from '../services/dataService';
|
import { getAttendances, getUserById, getFunnels, getOrigins } from '../services/dataService';
|
||||||
import { Attendance, User, FunnelStage, DashboardFilter, FunnelStageDef } from '../types';
|
import { Attendance, User, FunnelStage, DashboardFilter, FunnelStageDef, OriginItemDef } from '../types';
|
||||||
import { ArrowLeft, Mail, Phone, Clock, MessageSquare, ChevronLeft, ChevronRight, Eye, Filter } from 'lucide-react';
|
import { ArrowLeft, Mail, Phone, Clock, MessageSquare, ChevronLeft, ChevronRight, Eye, Filter } from 'lucide-react';
|
||||||
import { DateRangePicker } from '../components/DateRangePicker';
|
import { DateRangePicker } from '../components/DateRangePicker';
|
||||||
|
|
||||||
@@ -12,6 +12,7 @@ export const UserDetail: React.FC = () => {
|
|||||||
const [user, setUser] = useState<User | undefined>();
|
const [user, setUser] = useState<User | undefined>();
|
||||||
const [attendances, setAttendances] = useState<Attendance[]>([]);
|
const [attendances, setAttendances] = useState<Attendance[]>([]);
|
||||||
const [funnelDefs, setFunnelDefs] = useState<FunnelStageDef[]>([]);
|
const [funnelDefs, setFunnelDefs] = useState<FunnelStageDef[]>([]);
|
||||||
|
const [originDefs, setOriginDefs] = useState<OriginItemDef[]>([]);
|
||||||
const [loading, setLoading] = useState(true);
|
const [loading, setLoading] = useState(true);
|
||||||
const [currentPage, setCurrentPage] = useState(1);
|
const [currentPage, setCurrentPage] = useState(1);
|
||||||
const [filters, setFilters] = useState<DashboardFilter>({
|
const [filters, setFilters] = useState<DashboardFilter>({
|
||||||
@@ -32,12 +33,13 @@ export const UserDetail: React.FC = () => {
|
|||||||
setUser(u);
|
setUser(u);
|
||||||
|
|
||||||
if (u && tenantId) {
|
if (u && tenantId) {
|
||||||
const [data, fetchedFunnels] = await Promise.all([
|
const [data, fetchedFunnels, fetchedOrigins] = await Promise.all([
|
||||||
getAttendances(tenantId, {
|
getAttendances(tenantId, {
|
||||||
...filters,
|
...filters,
|
||||||
userId: id
|
userId: id
|
||||||
}),
|
}),
|
||||||
getFunnels(tenantId)
|
getFunnels(tenantId),
|
||||||
|
getOrigins(tenantId)
|
||||||
]);
|
]);
|
||||||
setAttendances(data);
|
setAttendances(data);
|
||||||
|
|
||||||
@@ -48,6 +50,13 @@ export const UserDetail: React.FC = () => {
|
|||||||
if (matchedFunnel) activeFunnel = matchedFunnel;
|
if (matchedFunnel) activeFunnel = matchedFunnel;
|
||||||
}
|
}
|
||||||
setFunnelDefs(activeFunnel && activeFunnel.stages ? activeFunnel.stages.sort((a: any, b: any) => a.order_index - b.order_index) : []);
|
setFunnelDefs(activeFunnel && activeFunnel.stages ? activeFunnel.stages.sort((a: any, b: any) => a.order_index - b.order_index) : []);
|
||||||
|
|
||||||
|
let activeOriginGroup = fetchedOrigins[0];
|
||||||
|
if (targetTeamId) {
|
||||||
|
const matchedOrigin = fetchedOrigins.find(o => o.teamIds?.includes(targetTeamId));
|
||||||
|
if (matchedOrigin) activeOriginGroup = matchedOrigin;
|
||||||
|
}
|
||||||
|
setOriginDefs(activeOriginGroup && activeOriginGroup.items ? activeOriginGroup.items : []);
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error("Error loading user details", error);
|
console.error("Error loading user details", error);
|
||||||
@@ -160,13 +169,12 @@ export const UserDetail: React.FC = () => {
|
|||||||
onChange={(e) => handleFilterChange('origin', e.target.value)}
|
onChange={(e) => handleFilterChange('origin', e.target.value)}
|
||||||
>
|
>
|
||||||
<option value="all">Todas Origens</option>
|
<option value="all">Todas Origens</option>
|
||||||
<option value="WhatsApp">WhatsApp</option>
|
{originDefs.length > 0 ? originDefs.map(o => (
|
||||||
<option value="Instagram">Instagram</option>
|
<option key={o.id} value={o.name}>{o.name}</option>
|
||||||
<option value="Website">Website</option>
|
)) : ['WhatsApp', 'Instagram', 'Website', 'LinkedIn', 'Indicação'].map(o => (
|
||||||
<option value="LinkedIn">LinkedIn</option>
|
<option key={o} value={o}>{o}</option>
|
||||||
<option value="Indicação">Indicação</option>
|
))}
|
||||||
</select>
|
</select> </div>
|
||||||
</div>
|
|
||||||
|
|
||||||
{/* KPI Cards */}
|
{/* KPI Cards */}
|
||||||
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
|
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
|
||||||
|
|||||||
@@ -176,6 +176,100 @@ export const deleteFunnelStage = async (id: string): Promise<boolean> => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// --- Origins Functions ---
|
||||||
|
export const getOrigins = async (tenantId: string): Promise<any[]> => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`${API_URL}/origins?tenantId=${tenantId}`, {
|
||||||
|
headers: getHeaders()
|
||||||
|
});
|
||||||
|
if (!response.ok) throw new Error('Falha ao buscar origens');
|
||||||
|
return await response.json();
|
||||||
|
} catch (error) {
|
||||||
|
console.error("API Error (getOrigins):", error);
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createOriginGroup = async (data: { name: string, tenantId: string }): Promise<any> => {
|
||||||
|
const response = await fetch(`${API_URL}/origins`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: getHeaders(),
|
||||||
|
body: JSON.stringify(data)
|
||||||
|
});
|
||||||
|
if (!response.ok) {
|
||||||
|
const error = await response.json();
|
||||||
|
throw new Error(error.error || 'Erro ao criar grupo de origens');
|
||||||
|
}
|
||||||
|
return await response.json();
|
||||||
|
};
|
||||||
|
|
||||||
|
export const updateOriginGroup = async (id: string, data: { name?: string, teamIds?: string[] }): Promise<boolean> => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`${API_URL}/origins/${id}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: getHeaders(),
|
||||||
|
body: JSON.stringify(data)
|
||||||
|
});
|
||||||
|
return response.ok;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("API Error (updateOriginGroup):", error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const deleteOriginGroup = async (id: string): Promise<boolean> => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`${API_URL}/origins/${id}`, {
|
||||||
|
method: 'DELETE',
|
||||||
|
headers: getHeaders()
|
||||||
|
});
|
||||||
|
return response.ok;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("API Error (deleteOriginGroup):", error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createOriginItem = async (groupId: string, data: { name: string }): Promise<any> => {
|
||||||
|
const response = await fetch(`${API_URL}/origins/${groupId}/items`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: getHeaders(),
|
||||||
|
body: JSON.stringify(data)
|
||||||
|
});
|
||||||
|
if (!response.ok) {
|
||||||
|
const error = await response.json();
|
||||||
|
throw new Error(error.error || 'Erro ao criar item de origem');
|
||||||
|
}
|
||||||
|
return await response.json();
|
||||||
|
};
|
||||||
|
|
||||||
|
export const updateOriginItem = async (id: string, data: { name: string }): Promise<boolean> => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`${API_URL}/origin_items/${id}`, {
|
||||||
|
method: 'PUT',
|
||||||
|
headers: getHeaders(),
|
||||||
|
body: JSON.stringify(data)
|
||||||
|
});
|
||||||
|
return response.ok;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("API Error (updateOriginItem):", error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const deleteOriginItem = async (id: string): Promise<boolean> => {
|
||||||
|
try {
|
||||||
|
const response = await fetch(`${API_URL}/origin_items/${id}`, {
|
||||||
|
method: 'DELETE',
|
||||||
|
headers: getHeaders()
|
||||||
|
});
|
||||||
|
return response.ok;
|
||||||
|
} catch (error) {
|
||||||
|
console.error("API Error (deleteOriginItem):", error);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// --- API Keys Functions ---
|
// --- API Keys Functions ---
|
||||||
export const getApiKeys = async (tenantId: string): Promise<any[]> => {
|
export const getApiKeys = async (tenantId: string): Promise<any[]> => {
|
||||||
try {
|
try {
|
||||||
|
|||||||
16
types.ts
16
types.ts
@@ -23,6 +23,20 @@ export interface FunnelDef {
|
|||||||
teamIds: string[];
|
teamIds: string[];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface OriginItemDef {
|
||||||
|
id: string;
|
||||||
|
origin_group_id: string;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface OriginGroupDef {
|
||||||
|
id: string;
|
||||||
|
tenant_id: string;
|
||||||
|
name: string;
|
||||||
|
items: OriginItemDef[];
|
||||||
|
teamIds: string[];
|
||||||
|
}
|
||||||
|
|
||||||
export interface User {
|
export interface User {
|
||||||
id: string;
|
id: string;
|
||||||
tenant_id: string;
|
tenant_id: string;
|
||||||
@@ -50,7 +64,7 @@ export interface Attendance {
|
|||||||
first_response_time_min: number;
|
first_response_time_min: number;
|
||||||
handling_time_min: number;
|
handling_time_min: number;
|
||||||
funnel_stage: string;
|
funnel_stage: string;
|
||||||
origin: 'WhatsApp' | 'Instagram' | 'Website' | 'LinkedIn' | 'Indicação';
|
origin: string;
|
||||||
product_requested: string;
|
product_requested: string;
|
||||||
product_sold?: string;
|
product_sold?: string;
|
||||||
converted: boolean;
|
converted: boolean;
|
||||||
|
|||||||
Reference in New Issue
Block a user