19096552b4
- RegisterView: email + password (strength-meter 0..4) + 2 click-wrap'а (оферта + ПДн). 3-й «маркетинг» из handoff НЕ реализован (расхождение #2 реестра v1.13 - handoff противоречит ТЗ §1.5/§4.1). - TwoFactorView: 6 input-cell с auto-focus вперёд при вводе цифры, Backspace назад при empty, paste 6 цифр заполняет все. - ForgotPasswordView: email + alert «5 попыток / 15 минут» по ТЗ §1.7. - RecoveryCodesView: 8 кодов в 2-column grid + Скачать .txt (Blob/URL.createObjectURL) + Копировать (navigator.clipboard) + warning о невозможности повторного просмотра. Router: 4 новых маршрута (/register, /2fa, /forgot, /recovery), все meta.layout='auth', lazy-imports. Vitest +14 тестов (всего 24/24 за 3.29s): - RegisterView 4 (вкл. assertion на отсутствие маркетингового click-wrap) - TwoFactorView 3, ForgotPasswordView 3, RecoveryCodesView 4 Stories +4 (Histoire 6/6 за 29.17s). Регресс: lint+type-check+format OK; vitest 24/24; vite build 5 lazy-chunks для views + Vuetify в отдельные chunks (app chunk 198KB→78KB); Pest 48/48 за 4.85s. CLAUDE.md v1.19→v1.20, реестр Открытых_вопросов v1.28→v1.29. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
37 lines
1.5 KiB
TypeScript
37 lines
1.5 KiB
TypeScript
import { describe, it, expect } from 'vitest';
|
|
import { mount } from '@vue/test-utils';
|
|
import { createVuetify } from 'vuetify';
|
|
import RecoveryCodesView from '../../resources/js/views/auth/RecoveryCodesView.vue';
|
|
|
|
describe('RecoveryCodesView.vue', () => {
|
|
const factory = () =>
|
|
mount(RecoveryCodesView, {
|
|
global: { plugins: [createVuetify()] },
|
|
});
|
|
|
|
it('монтируется и содержит заголовок «Резервные коды»', () => {
|
|
const wrapper = factory();
|
|
expect(wrapper.text()).toContain('Резервные коды');
|
|
});
|
|
|
|
it('рендерит ровно 8 кодов в формате XXXX-XXXX', () => {
|
|
const wrapper = factory();
|
|
const codes = wrapper.findAll('.code-item');
|
|
expect(codes).toHaveLength(8);
|
|
codes.forEach((c) => {
|
|
expect(c.text()).toMatch(/^[A-Z0-9]{4}-[A-Z0-9]{4}$/);
|
|
});
|
|
});
|
|
|
|
it('содержит warning о невозможности повторного просмотра', () => {
|
|
const wrapper = factory();
|
|
expect(wrapper.text()).toContain('После закрытия страницы коды нельзя посмотреть снова');
|
|
});
|
|
|
|
it('содержит кнопки «Скачать .txt» и «Копировать»', () => {
|
|
const wrapper = factory();
|
|
expect(wrapper.text()).toContain('Скачать');
|
|
expect(wrapper.text()).toContain('Копировать');
|
|
});
|
|
});
|