Files
portal/app/resources/js/views/HelpView.vue
T
Дмитрий 1a0c9f5c8d
Accessibility (Pa11y live) / a11y (push) Has been cancelled
SAST — Semgrep / Semgrep SAST scan (push) Has been cancelled
fix(ui): иконки-? → Lucide, почта поддержки .app→.ru, без префикса text:, карточка Канбана 0₽→—, crm.bp-gr.ru
UI-аудит раунд 2 (Playwright, протыкивание форм):
- vuetify.ts: +13 mdi→Lucide маппингов — bulk-бар проектов / импорт / экспорт отчётов и сделок / помощь / действия админки больше не падают в HelpCircle-fallback «?»
- config/services.php + ErrorMeta/ErrorView/HelpView: support@liderra.appsupport@liderra.ru (домен продукта .ru); status.liderra.app → status.liderra.ru
- dealsApiMapper: ветка deal.commented — текст комментария в активности без служебного ключа «text:»
- KanbanCard: costKopecks null-aware — «—» вместо врущего «0 ₽» (как в drawer)
- DealsView: подзаголовок «crm.bp» → «crm.bp-gr.ru» (как в импорте/админке)

Верификация: type-check ✓, build ✓, переоткрыто в Playwright локально (иконки/почта/комментарий/карточка/подзаголовок).

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-21 14:34:03 +03:00

96 lines
3.7 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<script setup lang="ts">
import { ref } from 'vue';
import { useAuthStore } from '../stores/auth';
import { submitSupportRequest } from '../api/support';
const auth = useAuthStore();
const supportEmail =
document.querySelector('meta[name="support-email"]')?.getAttribute('content') ?? 'support@liderra.ru';
const name = ref(
[auth.user?.first_name, auth.user?.last_name].filter(Boolean).join(' ') || '',
);
const contact = ref(auth.user?.email ?? '');
const message = ref('');
const loading = ref(false);
const sent = ref(false);
const errorMsg = ref('');
const fieldErrors = ref<Record<string, string[]>>({});
async function submit() {
errorMsg.value = '';
fieldErrors.value = {};
if (!name.value.trim() || !contact.value.trim() || !message.value.trim()) {
errorMsg.value = 'Заполните все поля.';
return;
}
loading.value = true;
try {
await submitSupportRequest({ name: name.value, contact: contact.value, message: message.value });
sent.value = true;
message.value = '';
} catch (e: unknown) {
const err = e as { response?: { status?: number; data?: { errors?: Record<string, string[]> } } };
if (err.response?.status === 422 && err.response.data?.errors) {
fieldErrors.value = err.response.data.errors;
} else {
errorMsg.value = 'Не удалось отправить. Попробуйте ещё раз или напишите на почту.';
}
} finally {
loading.value = false;
}
}
</script>
<template>
<div class="help-view pa-6" data-testid="help-view">
<h1 class="text-h5 mb-1">Помощь</h1>
<p class="text-body-2 text-medium-emphasis mb-6">
Напишите нам ответим на ваш контакт. Можно по почте, через форму ниже или в чат справа.
</p>
<v-card variant="outlined" class="pa-5 mb-4" max-width="640">
<h3 class="text-subtitle-2 mb-2">Почта техподдержки</h3>
<a :href="`mailto:${supportEmail}`" class="text-primary" data-testid="support-email">{{ supportEmail }}</a>
</v-card>
<v-card variant="outlined" class="pa-5" max-width="640">
<h3 class="text-subtitle-2 mb-3">Оставить заявку</h3>
<v-alert v-if="sent" type="success" variant="tonal" class="mb-4" data-testid="support-sent">
Заявка отправлена. Мы свяжемся с вами по указанному контакту.
</v-alert>
<v-alert v-if="errorMsg" type="error" variant="tonal" class="mb-4">{{ errorMsg }}</v-alert>
<v-text-field
v-model="name"
label="Имя"
:error-messages="fieldErrors.name"
density="comfortable"
class="mb-2"
data-testid="support-name"
/>
<v-text-field
v-model="contact"
label="Контакт (телефон или email)"
:error-messages="fieldErrors.contact"
density="comfortable"
class="mb-2"
data-testid="support-contact"
/>
<v-textarea
v-model="message"
label="Сообщение"
:error-messages="fieldErrors.message"
rows="4"
density="comfortable"
class="mb-3"
data-testid="support-message"
/>
<v-btn color="primary" :loading="loading" data-testid="support-submit" @click="submit">Отправить</v-btn>
</v-card>
</div>
</template>