5f209a2fcc
Раунд 2 минор-фиксы (Playwright-аудит): - RuDateField (новый): даты дд.мм.гггг через ru date-picker вместо нативного <input type=date> (показывал мм/дд/гггг на en-локали) — Отчёты + Сделки. - BalanceCapacityIndicator: разделитель тысяч «1 000 ₽», эмодзи→mdi. - dealsApiMapper/DealDetailBody: статус-смена в активности русскими метками (было «viewed → new» сырыми слагами). - ProfileTab: инлайн-валидация Имя/Фамилия (под полем, как в Реквизитах). - RequisitesTab: проверка формата телефона на клиенте. - ApiTab: eye-toggle с aria-label (показать/скрыть ключ и секрет). - DashboardView: «3 / 0» → скрываем «/ N» и «лимит тарифа» при лимите 0. - KanbanView: тост-подтверждение при смене статуса (+ цветной фейл-тост). - NotificationsTab: убран жаргон «users.notification_preferences в БД». - Админка: TenantsTable «ИНН не указан» вместо пустого «ИНН »; PricingTiers epoch-дата «1970»→«начала» + ru-формат цены; Incidents empty-state «Инцидентов нет»; SupplierIntegration/PdSubjectRequests — window.confirm/alert → v-dialog/snackbar. Верификация: type-check, build, Playwright (даты дд.мм.гггг подтверждены). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
76 lines
3.0 KiB
Vue
76 lines
3.0 KiB
Vue
<script setup lang="ts">
|
|
/**
|
|
* Постоянная подсказка под балансом (Billing v2 Spec C §3.6, Task 1.10).
|
|
*
|
|
* Чистый presentational-компонент: показывает, на сколько дней хватит ёмкости
|
|
* баланса (в лидах) при текущем дневном заказе всех eligible-проектов.
|
|
* Зелёный — хватает на 3+ дня; жёлтый — меньше 3 дней; красный — не хватает.
|
|
*/
|
|
import { computed } from 'vue';
|
|
|
|
const props = defineProps<{
|
|
/** Баланс в рублях (строка scale 2, например "1000.00"). */
|
|
balanceRub: string;
|
|
/** Сколько лидов покрывает баланс по текущему тарифу. */
|
|
capacityLeads: number;
|
|
/** Суммарный дневной заказ всех активных проектов (лидов/день). */
|
|
requiredLeadsPerDay: number;
|
|
}>();
|
|
|
|
const daysLeft = computed(() =>
|
|
props.requiredLeadsPerDay > 0 ? props.capacityLeads / props.requiredLeadsPerDay : Infinity,
|
|
);
|
|
|
|
const statusClass = computed(() => {
|
|
if (props.requiredLeadsPerDay > 0 && props.capacityLeads < props.requiredLeadsPerDay) {
|
|
return 'capacity-insufficient';
|
|
}
|
|
if (daysLeft.value < 3) return 'capacity-warning';
|
|
return 'capacity-ok';
|
|
});
|
|
|
|
const daysLabel = computed(() => (Number.isFinite(daysLeft.value) ? daysLeft.value.toFixed(1) : '∞'));
|
|
|
|
// Разделитель тысяч + без лишних копеек: "1000.00" → "1 000".
|
|
const balanceFormatted = computed(() =>
|
|
new Intl.NumberFormat('ru-RU', { maximumFractionDigits: 0 }).format(parseFloat(props.balanceRub) || 0),
|
|
);
|
|
</script>
|
|
|
|
<template>
|
|
<div class="balance-capacity text-body-2" :class="statusClass" data-testid="balance-capacity-indicator">
|
|
<div>Баланс: {{ balanceFormatted }} ₽ = до {{ capacityLeads }} лидов по тарифу</div>
|
|
<div>Проекты заказывают: {{ requiredLeadsPerDay }} лидов в день</div>
|
|
<div v-if="statusClass === 'capacity-insufficient'" class="capacity-note">
|
|
<v-icon size="14" class="mr-1">mdi-alert-outline</v-icon>Не хватает — пополните счёт
|
|
</div>
|
|
<div v-else-if="statusClass === 'capacity-warning'" class="capacity-note">
|
|
Хватит на ~{{ daysLabel }} дн. — скоро потребуется пополнение
|
|
</div>
|
|
<div v-else class="capacity-note">
|
|
<v-icon size="14" class="mr-1">mdi-check-circle-outline</v-icon>Хватит на ~{{ daysLabel }} дн.
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<style scoped>
|
|
.balance-capacity {
|
|
display: flex;
|
|
flex-direction: column;
|
|
gap: 2px;
|
|
line-height: 1.4;
|
|
}
|
|
.capacity-note {
|
|
font-weight: 600;
|
|
}
|
|
.capacity-ok .capacity-note {
|
|
color: rgb(var(--v-theme-success));
|
|
}
|
|
.capacity-warning .capacity-note {
|
|
color: rgb(var(--v-theme-warning));
|
|
}
|
|
.capacity-insufficient .capacity-note {
|
|
color: rgb(var(--v-theme-error));
|
|
}
|
|
</style>
|