Files
portal/app/resources/js/stores/projectsStore.ts
T
Дмитрий cb05657f30 chore(format): prettier --write across 37 .vue/.ts files
Phase 1B audit found 48 files failing `prettier --check`. Auto-apply
via `npx prettier --write resources/js/**/*.{ts,vue,css}` produced
style-only changes:
- consistent quote style
- trailing comma normalization
- spaces around : in v-card style="position: relative" attrs
- explicit ; insertion

No semantic changes. No code-behavior changes. Production-code only;
test files batched separately into `test(frontend):` commit.

Verification:
- npx vitest run → 79/79 files, 614/614 + 3 skipped (no regression).
- npx vue-tsc --noEmit → 0 errors.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-12 20:24:33 +03:00

210 lines
6.5 KiB
TypeScript

import { defineStore } from 'pinia';
import { ref, reactive, watch } from 'vue';
import axios from 'axios';
export interface Project {
id: number;
name: string;
signal_type: 'site' | 'call' | 'sms';
signal_identifier?: string | null;
sms_senders?: string[] | null;
sms_keyword?: string | null;
daily_limit_target: number;
delivered_today: number;
delivered_in_month?: number;
is_active: boolean;
archived_at: string | null;
region_mask?: number;
region_mode?: string;
delivery_days_mask?: number;
sync_status: 'ok' | 'pending' | 'failed';
last_synced_at?: string | null;
}
export const useProjectsStore = defineStore('projects', () => {
const items = ref<Project[]>([]);
const total = ref(0);
const filters = reactive({ signal_type: '', status: '', search: '', page: 1, per_page: 20 });
const selectedIds = ref<Set<number>>(new Set());
const pendingIds = ref<Set<number>>(new Set());
const loading = ref(false);
const selectAllByFilter = ref<boolean>(false);
// Closure state for polling — kept outside returned store surface.
let pollTimeout: ReturnType<typeof setTimeout> | null = null;
let currentDelay = 5000;
const DELAY_OK = 5000;
const DELAY_MAX = 30000;
async function fetch() {
loading.value = true;
try {
const params: Record<string, unknown> = { page: filters.page, per_page: filters.per_page };
if (filters.signal_type) params.signal_type = filters.signal_type;
if (filters.status) params.status = filters.status;
if (filters.search) params.search = filters.search;
const { data } = await axios.get('/api/projects', { params });
items.value = data.data;
total.value = data.meta.total;
} finally {
loading.value = false;
}
}
async function create(payload: Partial<Project>) {
const { data } = await axios.post('/api/projects', payload);
pendingIds.value.add(data.data.id);
await fetch();
return data.data;
}
async function update(id: number, payload: Partial<Project>) {
const { data } = await axios.patch(`/api/projects/${id}`, payload);
await fetch();
return data.data;
}
async function archive(id: number) {
await axios.delete(`/api/projects/${id}`);
await fetch();
}
async function syncNow(id: number) {
await axios.post(`/api/projects/${id}/sync`);
pendingIds.value.add(id);
await fetch();
}
async function toggleActive(project: Project) {
await axios.patch(`/api/projects/${project.id}/toggle-active`, { is_active: !project.is_active });
await fetch();
}
function toggleSelect(id: number) {
selectAllByFilter.value = false; // user opted into manual mode
if (selectedIds.value.has(id)) {
selectedIds.value.delete(id);
} else {
selectedIds.value.add(id);
}
}
function clearSelection() {
selectedIds.value.clear();
}
async function bulkAction(action: 'pause' | 'resume' | 'archive') {
const ids = Array.from(selectedIds.value);
if (!ids.length) return;
await axios.post('/api/projects/bulk', { action, ids });
clearSelection();
await fetch();
}
interface BulkPayload {
action: 'pause' | 'resume' | 'archive' | 'update_regions' | 'update_days' | 'update_limit';
add?: number;
remove?: number;
delta?: number;
replace?: number;
}
interface BulkResponse {
updated: number;
skipped: Array<{ id: number; reason: string }>;
warnings: string[];
}
async function bulkUpdate(payload: BulkPayload): Promise<BulkResponse> {
const body: Record<string, unknown> = { ...payload };
if (selectAllByFilter.value) {
const f: Record<string, unknown> = {};
if (filters.signal_type) f.signal_type = filters.signal_type;
if (filters.status) f.status = filters.status;
if (filters.search) f.search = filters.search;
body.scope = { filter: f };
} else {
body.ids = Array.from(selectedIds.value);
}
const { data } = await axios.post('/api/projects/bulk', body);
clearSelection();
selectAllByFilter.value = false;
await fetch();
return data;
}
// Watch filter changes — clear selection on switch
watch(
() => [filters.signal_type, filters.status, filters.search] as const,
() => {
clearSelection();
selectAllByFilter.value = false;
},
);
function scheduleNext() {
pollTimeout = setTimeout(async () => {
pollTimeout = null;
if (pendingIds.value.size === 0) {
currentDelay = DELAY_OK;
return;
}
try {
const ids = Array.from(pendingIds.value).join(',');
const { data } = await axios.get<{ data: Project[] }>('/api/projects', { params: { ids } });
for (const project of data.data) {
const idx = items.value.findIndex((i) => i.id === project.id);
if (idx !== -1) items.value[idx] = project;
if (project.sync_status === 'ok' || project.sync_status === 'failed') {
pendingIds.value.delete(project.id);
}
}
currentDelay = DELAY_OK;
} catch {
// Exponential backoff to avoid hammering on transient errors.
currentDelay = Math.min(currentDelay * 2, DELAY_MAX);
}
if (pendingIds.value.size > 0) {
scheduleNext();
}
}, currentDelay);
}
function startPolling() {
if (pollTimeout !== null) return;
scheduleNext();
}
function stopPolling() {
if (pollTimeout !== null) {
clearTimeout(pollTimeout);
pollTimeout = null;
}
currentDelay = DELAY_OK;
}
return {
items,
total,
filters,
selectedIds,
pendingIds,
loading,
selectAllByFilter,
fetch,
create,
update,
archive,
syncNow,
toggleActive,
toggleSelect,
clearSelection,
bulkAction,
bulkUpdate,
startPolling,
stopPolling,
};
});