7f05c4ab16
- api/imports.ts: типы ImportLogResource/UnknownStatus/StatusMapping, функции uploadImport/listImports/getImport/getUnknownStatuses/resolveUnknownStatuses (apiClient из ./client, стиль api/dashboard.ts) - UnknownStatusesDialog.vue: wizard маппинга незамапленных статусов воронки (ТЗ §6.4/§6.6), 14 канонических slug, defineExpose(selection, save) - Vitest 3/3 (tests/Frontend/UnknownStatusesDialog.spec.ts) Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
126 lines
4.8 KiB
Vue
126 lines
4.8 KiB
Vue
<script setup lang="ts">
|
|
/**
|
|
* Wizard маппинга неизвестных статусов воронки из CSV-импорта (ТЗ §6.4/§6.6).
|
|
*
|
|
* Для каждого незамапленного русского статуса пользователь выбирает один из
|
|
* 14 канонических slug'ов. Сохранение → POST /api/imports/unknown-statuses/resolve.
|
|
*/
|
|
import { computed, reactive, ref } from 'vue';
|
|
import { resolveUnknownStatuses, type StatusMapping, type UnknownStatus } from '../../api/imports';
|
|
|
|
const props = defineProps<{
|
|
modelValue: boolean;
|
|
statuses: UnknownStatus[];
|
|
}>();
|
|
|
|
const emit = defineEmits<{
|
|
'update:modelValue': [value: boolean];
|
|
resolved: [];
|
|
}>();
|
|
|
|
/** 14 канонических статусов воронки (ТЗ §6.4). */
|
|
const STATUS_OPTIONS: { value: string; title: string }[] = [
|
|
{ value: 'new', title: 'Новые' },
|
|
{ value: 'viewed', title: 'Просмотрено' },
|
|
{ value: 'worked', title: 'Проработан' },
|
|
{ value: 'base', title: 'База' },
|
|
{ value: 'missed', title: 'Недозвон' },
|
|
{ value: 'negotiations', title: 'Переговоры' },
|
|
{ value: 'waiting_payment', title: 'Ожидаем оплаты' },
|
|
{ value: 'partnership', title: 'Партнерка' },
|
|
{ value: 'paid', title: 'Оплачено' },
|
|
{ value: 'closed', title: 'Закрыто и не реализовано' },
|
|
{ value: 'test_drive', title: 'Тест драйв' },
|
|
{ value: 'hot', title: 'Горячий' },
|
|
{ value: 'replacement', title: 'На замену' },
|
|
{ value: 'final_missed', title: 'Конечный недозвон' },
|
|
];
|
|
|
|
const selection = reactive<Record<string, string | null>>({});
|
|
const saving = ref(false);
|
|
const error = ref<string | null>(null);
|
|
|
|
const dialogOpen = computed({
|
|
get: () => props.modelValue,
|
|
set: (v: boolean) => emit('update:modelValue', v),
|
|
});
|
|
|
|
const allMapped = computed(
|
|
() => props.statuses.length > 0 && props.statuses.every((s) => !!selection[s.status_ru]),
|
|
);
|
|
|
|
async function save(): Promise<void> {
|
|
if (!allMapped.value) {
|
|
return;
|
|
}
|
|
saving.value = true;
|
|
error.value = null;
|
|
try {
|
|
const mappings: StatusMapping[] = props.statuses.map((s) => ({
|
|
status_ru: s.status_ru,
|
|
slug: selection[s.status_ru] as string,
|
|
}));
|
|
await resolveUnknownStatuses(mappings);
|
|
emit('resolved');
|
|
} catch {
|
|
error.value = 'Не удалось сохранить маппинг. Повторите попытку.';
|
|
} finally {
|
|
saving.value = false;
|
|
}
|
|
}
|
|
|
|
defineExpose({ selection, save });
|
|
</script>
|
|
|
|
<template>
|
|
<v-dialog v-model="dialogOpen" max-width="640">
|
|
<v-card>
|
|
<v-card-title class="text-h6">Маппинг неизвестных статусов</v-card-title>
|
|
<v-card-text>
|
|
<p class="text-body-2 text-medium-emphasis mb-4">
|
|
Эти статусы из CSV не входят в стандартную воронку. Выберите
|
|
соответствие — повторный импорт применит маппинг автоматически.
|
|
</p>
|
|
<div
|
|
v-for="status in statuses"
|
|
:key="status.id"
|
|
class="d-flex align-center ga-3 mb-3"
|
|
>
|
|
<div class="flex-grow-1">
|
|
<strong>{{ status.status_ru }}</strong>
|
|
<span class="text-caption text-medium-emphasis ml-2">
|
|
({{ status.occurrences }} шт.)
|
|
</span>
|
|
</div>
|
|
<v-select
|
|
v-model="selection[status.status_ru]"
|
|
:items="STATUS_OPTIONS"
|
|
label="Статус воронки"
|
|
density="compact"
|
|
variant="outlined"
|
|
hide-details
|
|
style="max-width: 280px"
|
|
/>
|
|
</div>
|
|
<v-alert v-if="error" type="error" variant="tonal" class="mt-2">
|
|
{{ error }}
|
|
</v-alert>
|
|
</v-card-text>
|
|
<v-card-actions>
|
|
<v-spacer />
|
|
<v-btn variant="text" @click="dialogOpen = false">Отмена</v-btn>
|
|
<v-btn
|
|
data-test="save-mappings"
|
|
color="primary"
|
|
variant="flat"
|
|
:loading="saving"
|
|
:disabled="!allMapped"
|
|
@click="save"
|
|
>
|
|
Сохранить
|
|
</v-btn>
|
|
</v-card-actions>
|
|
</v-card>
|
|
</v-dialog>
|
|
</template>
|