830a652588
Расширяет stages 5/6 (soft-delete + 8-сек undo) до постоянного доступа
к удалённым сделкам через отдельный view-mode.
Backend (DealController::index):
- Новый query-param only_deleted=true.
- withTrashed() + whereNotNull('deleted_at') — обход global scope
SoftDeletes + явный фильтр для NO-OP idempotency.
- Все остальные фильтры применимы и в trash-mode.
Pest +3 (DealIndexTest):
- only_deleted=true → только soft-deleted (alive скрыты).
- Без only_deleted → soft-deleted скрыты (default behavior).
- RLS+app-фильтр изолирует чужие удалённые.
Frontend:
- ListDealsParams.onlyDeleted?: boolean + axios mapping.
- DealsView: trashMode ref + toggleTrashMode (clear selected + reload) +
applyBulkRestoreFromTrash (optimistic remove + bulkRestoreDeals + toast).
- UI changes в trash-mode:
- Заголовок «Сделки» → «Корзина».
- Toggle-btn 'mdi-arrow-left К сделкам' (warning-flat) вместо
'mdi-trash-can-outline Корзина' (outlined).
- Скрыты Экспорт + Новая сделка.
- Скрыт chiprow filter-bar.
- Info-alert «Корзина: показаны удалённые сделки».
- Bulk-bar: только Восстановить (mdi-restore success-tonal) + clear;
status/export/delete скрыты.
Vitest +2 (DealsListIntegration):
- toggleTrashMode → trashMode=true + listDeals с onlyDeleted=true.
- applyBulkRestoreFromTrash → bulkRestoreDeals + remove from state +
toast «Восстановлено 2».
PHPStan baseline регенерирован.
Регресс:
- Lint+type-check+format passed.
- Vitest 321/321 за 19.60 сек (+2 от 319).
- Vite build 1.04 сек.
- Pint + PHPStan passed.
- Pest 269/269 за 29.12 сек (+3 от 266, 1009 assertions).
Реестр v1.72→v1.73 / CLAUDE.md v1.63→v1.64.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
236 lines
6.7 KiB
TypeScript
236 lines
6.7 KiB
TypeScript
import { apiClient, ensureCsrfCookie } from './client';
|
|
|
|
/**
|
|
* API-вызовы для DealController.
|
|
*
|
|
* На MVP без auth — tenant_id передаётся параметром (на prod возьмётся из
|
|
* middleware → backend параметр исчезнет).
|
|
*/
|
|
|
|
export interface CreateDealPayload {
|
|
tenant_id: number;
|
|
project_name: string;
|
|
phone: string;
|
|
contact_name?: string;
|
|
status?: string;
|
|
manager_id?: number;
|
|
comment?: string;
|
|
}
|
|
|
|
export interface CreatedDeal {
|
|
id: number;
|
|
tenant_id: number;
|
|
project_id: number;
|
|
phone: string;
|
|
status: string;
|
|
contact_name: string | null;
|
|
manager_id: number | null;
|
|
received_at: string;
|
|
}
|
|
|
|
export async function createDeal(payload: CreateDealPayload): Promise<CreatedDeal> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.post<{ deal: CreatedDeal; message: string }>('/api/deals', payload);
|
|
return data.deal;
|
|
}
|
|
|
|
export interface BulkDeleteDealsPayload {
|
|
tenant_id: number;
|
|
ids: number[];
|
|
}
|
|
|
|
export interface BulkDeleteDealsResponse {
|
|
deleted: number;
|
|
requested: number;
|
|
}
|
|
|
|
/** Bulk soft-delete. Возвращает {deleted, requested} — deleted может быть < requested
|
|
* если часть id чужие или уже удалены (NO-OP). */
|
|
export async function bulkDeleteDeals(payload: BulkDeleteDealsPayload): Promise<BulkDeleteDealsResponse> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.delete<BulkDeleteDealsResponse>('/api/deals', { data: payload });
|
|
return data;
|
|
}
|
|
|
|
export interface BulkRestoreDealsPayload {
|
|
tenant_id: number;
|
|
ids: number[];
|
|
}
|
|
|
|
export interface BulkRestoreDealsResponse {
|
|
restored: number;
|
|
requested: number;
|
|
}
|
|
|
|
/** Bulk restore soft-deleted сделок. NO-OP idempotent для не-удалённых. */
|
|
export async function bulkRestoreDeals(payload: BulkRestoreDealsPayload): Promise<BulkRestoreDealsResponse> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.post<BulkRestoreDealsResponse>('/api/deals/restore', payload);
|
|
return data;
|
|
}
|
|
|
|
export interface ExportDealsPayload {
|
|
tenant_id: number;
|
|
ids: number[];
|
|
format?: 'csv' | 'xlsx';
|
|
}
|
|
|
|
export interface UpdateDealPayload {
|
|
tenant_id: number;
|
|
comment?: string | null;
|
|
manager_id?: number | null;
|
|
status?: string;
|
|
}
|
|
|
|
export async function updateDeal(id: number, payload: UpdateDealPayload): Promise<ApiDealDetail> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.patch<{ deal: ApiDealDetail }>(`/api/deals/${id}`, payload);
|
|
return data.deal;
|
|
}
|
|
|
|
export interface TransitionDealsPayload {
|
|
tenant_id: number;
|
|
ids: number[];
|
|
status: string;
|
|
}
|
|
|
|
export interface TransitionDealsResponse {
|
|
updated: number;
|
|
requested: number;
|
|
status: string;
|
|
}
|
|
|
|
/** Bulk status-update. Возвращает {updated, requested} — updated может быть < requested
|
|
* если часть id'шников чужие или уже в этом статусе (NO-OP). */
|
|
export async function transitionDeals(payload: TransitionDealsPayload): Promise<TransitionDealsResponse> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.post<TransitionDealsResponse>('/api/deals/transition', payload);
|
|
return data;
|
|
}
|
|
|
|
/** CSV-export. Возвращает строку с BOM — caller заворачивает в Blob+download. */
|
|
export async function exportDeals(payload: ExportDealsPayload): Promise<string> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.post<string>(
|
|
'/api/deals/export',
|
|
{ ...payload, format: 'csv' },
|
|
{ responseType: 'text' },
|
|
);
|
|
return data;
|
|
}
|
|
|
|
/** XLSX-export. Возвращает Blob (binary) — caller триггерит download. */
|
|
export async function exportDealsXlsx(payload: Omit<ExportDealsPayload, 'format'>): Promise<Blob> {
|
|
await ensureCsrfCookie();
|
|
const { data } = await apiClient.post<Blob>(
|
|
'/api/deals/export',
|
|
{ ...payload, format: 'xlsx' },
|
|
{ responseType: 'blob' },
|
|
);
|
|
return data;
|
|
}
|
|
|
|
export interface ApiDeal {
|
|
id: number;
|
|
tenant_id: number;
|
|
project_id: number;
|
|
project_name: string | null;
|
|
phone: string;
|
|
contact_name: string | null;
|
|
status: string;
|
|
manager_id: number | null;
|
|
manager_name: string | null;
|
|
manager_initials: string | null;
|
|
received_at: string | null;
|
|
}
|
|
|
|
export interface ApiDealEvent {
|
|
id: number;
|
|
event: string;
|
|
context: Record<string, unknown> | null;
|
|
created_at: string | null;
|
|
actor: { id: number; name: string; initials: string } | null;
|
|
}
|
|
|
|
export interface ApiDealDetail extends ApiDeal {
|
|
comment: string | null;
|
|
assigned_at: string | null;
|
|
}
|
|
|
|
export interface GetDealResponse {
|
|
deal: ApiDealDetail;
|
|
events: ApiDealEvent[];
|
|
}
|
|
|
|
export async function getDeal(id: number, tenantId: number): Promise<GetDealResponse> {
|
|
const { data } = await apiClient.get<GetDealResponse>(`/api/deals/${id}`, {
|
|
params: { tenant_id: tenantId },
|
|
});
|
|
return data;
|
|
}
|
|
|
|
export interface ListDealsParams {
|
|
tenantId: number;
|
|
statusIn?: string[];
|
|
projectId?: number;
|
|
managerId?: number;
|
|
search?: string;
|
|
limit?: number;
|
|
offset?: number;
|
|
/** «Корзина» — вернуть ТОЛЬКО soft-deleted сделки. */
|
|
onlyDeleted?: boolean;
|
|
}
|
|
|
|
export interface ListDealsResponse {
|
|
deals: ApiDeal[];
|
|
total: number;
|
|
limit: number;
|
|
offset: number;
|
|
}
|
|
|
|
export async function listDeals(params: ListDealsParams): Promise<ListDealsResponse> {
|
|
const { data } = await apiClient.get<ListDealsResponse>('/api/deals', {
|
|
params: {
|
|
tenant_id: params.tenantId,
|
|
status_in: params.statusIn,
|
|
project_id: params.projectId,
|
|
manager_id: params.managerId,
|
|
search: params.search,
|
|
limit: params.limit,
|
|
offset: params.offset,
|
|
only_deleted: params.onlyDeleted ? 'true' : undefined,
|
|
},
|
|
});
|
|
return data;
|
|
}
|
|
|
|
export interface ApiManager {
|
|
id: number;
|
|
email: string;
|
|
first_name: string | null;
|
|
last_name: string | null;
|
|
name: string;
|
|
initials: string;
|
|
}
|
|
|
|
export async function listManagers(tenantId: number): Promise<ApiManager[]> {
|
|
const { data } = await apiClient.get<{ managers: ApiManager[] }>('/api/managers', {
|
|
params: { tenant_id: tenantId },
|
|
});
|
|
return data.managers;
|
|
}
|
|
|
|
export interface ApiProject {
|
|
id: number;
|
|
name: string;
|
|
tag: string | null;
|
|
type: string;
|
|
}
|
|
|
|
export async function listProjects(tenantId: number): Promise<ApiProject[]> {
|
|
const { data } = await apiClient.get<{ projects: ApiProject[] }>('/api/projects', {
|
|
params: { tenant_id: tenantId },
|
|
});
|
|
return data.projects;
|
|
}
|