|
1 | 1 | import { isEscapeKey } from './util.js'; |
2 | | -import { showSuccessMessage, showErrorMessage } from './form-message.js'; |
| 2 | +import { initEffectSlider, resetEffects } from './effects.js'; |
3 | 3 |
|
4 | 4 | const fileInput = document.querySelector('#upload-file'); |
5 | 5 | const formOverlay = document.querySelector('.img-upload__overlay'); |
6 | 6 | const cancelButton = document.querySelector('.img-upload__cancel'); |
7 | 7 | const form = document.querySelector('.img-upload__form'); |
8 | | -const body = document.body; |
9 | 8 |
|
10 | | -const hashtagInput = form.querySelector('.text__hashtags'); |
11 | | -const commentInput = form.querySelector('.text__description'); |
12 | | - |
13 | | -const pristine = new Pristine(form, { |
14 | | - classTo: 'img-upload__field-wrapper', |
15 | | - errorClass: 'img-upload__field-wrapper--error', |
16 | | - successClass: 'img-upload__field-wrapper--success', |
17 | | - errorTextParent: 'img-upload__field-wrapper', |
18 | | - errorTextTag: 'div', |
19 | | - errorTextClass: 'pristine-error', |
20 | | -}); |
21 | | - |
22 | | -// Валидация хэштегов |
23 | | -const validateHashtags = (value) => { |
24 | | - if (!value.trim()) { |
25 | | - return true; |
26 | | - } |
27 | | - |
28 | | - const hashtags = value.trim().toLowerCase().split(/\s+/); |
29 | | - if (hashtags.length > 5) { |
30 | | - return false; |
31 | | - } |
32 | | - |
33 | | - const uniqueHashtags = new Set(hashtags); |
34 | | - if (uniqueHashtags.size !== hashtags.length) { |
35 | | - return false; |
36 | | - } |
37 | | - |
38 | | - return hashtags.every((tag) => /^#[a-zа-яё0-9]{1,19}$/i.test(tag)); |
39 | | -}; |
40 | | - |
41 | | -const getHashtagErrorMessage = (value) => { |
42 | | - const hashtags = value.trim().toLowerCase().split(/\s+/); |
43 | | - |
44 | | - if (hashtags.length > 5) { |
45 | | - return 'Нельзя указать больше 5 хэштегов'; |
46 | | - } |
47 | | - |
48 | | - const uniqueHashtags = new Set(hashtags); |
49 | | - if (uniqueHashtags.size !== hashtags.length) { |
50 | | - return 'Хэштеги не должны повторяться'; |
51 | | - } |
52 | | - |
53 | | - if (!hashtags.every((tag) => /^#[a-zа-яё0-9]{1,19}$/i.test(tag))) { |
54 | | - return 'Хэштег начинается с # и может содержать только буквы и цифры, не более 20 символов'; |
55 | | - } |
56 | | - |
57 | | - return ''; |
58 | | -}; |
59 | | - |
60 | | -// Валидация комментария |
61 | | -const validateComment = (value) => value.length <= 140; |
62 | | - |
63 | | -pristine.addValidator(hashtagInput, validateHashtags, getHashtagErrorMessage); |
64 | | -pristine.addValidator(commentInput, validateComment, 'Комментарий не может быть длиннее 140 символов'); |
65 | | - |
66 | | -// Обработчик Escape (с учётом фокуса) |
67 | | -const onDocumentKeydown = (evt) => { |
68 | | - if (isEscapeKey(evt)) { |
69 | | - const isInHashtag = document.activeElement === hashtagInput; |
70 | | - const isInComment = document.activeElement === commentInput; |
71 | | - |
72 | | - if (isInHashtag || isInComment) { |
73 | | - evt.stopPropagation(); // предотвращаем закрытие формы |
74 | | - } else { |
75 | | - evt.preventDefault(); |
76 | | - closeForm(); |
77 | | - } |
78 | | - } |
79 | | -}; |
80 | | - |
81 | | -// Показ формы |
82 | | -const openForm = () => { |
| 9 | +const showUploadForm = () => { |
83 | 10 | formOverlay.classList.remove('hidden'); |
84 | | - body.classList.add('modal-open'); |
| 11 | + document.body.classList.add('modal-open'); |
85 | 12 | document.addEventListener('keydown', onDocumentKeydown); |
| 13 | + initEffectSlider(); |
86 | 14 | }; |
87 | 15 |
|
88 | | -// Закрытие формы |
89 | | -const closeForm = () => { |
| 16 | +const hideUploadForm = () => { |
90 | 17 | formOverlay.classList.add('hidden'); |
91 | | - body.classList.remove('modal-open'); |
92 | | - form.reset(); // сброс полей |
93 | | - pristine.reset(); // сброс ошибок |
94 | | - fileInput.value = ''; // сброс выбранного файла |
| 18 | + document.body.classList.remove('modal-open'); |
95 | 19 | document.removeEventListener('keydown', onDocumentKeydown); |
| 20 | + form.reset(); |
| 21 | + resetEffects(); |
96 | 22 | }; |
97 | 23 |
|
98 | | -// Отправка формы |
99 | | -form.addEventListener('submit', (evt) => { |
100 | | - evt.preventDefault(); |
101 | | - |
102 | | - if (!pristine.validate()) { |
103 | | - return; |
| 24 | +function onDocumentKeydown(evt) { |
| 25 | + if (isEscapeKey(evt)) { |
| 26 | + evt.preventDefault(); |
| 27 | + hideUploadForm(); |
104 | 28 | } |
| 29 | +} |
105 | 30 |
|
106 | | - const submitButton = form.querySelector('#upload-submit'); |
107 | | - submitButton.disabled = true; |
108 | | - |
109 | | - const formData = new FormData(form); |
110 | | - |
111 | | - fetch('https://27.javascript.pages.academy/kekstagram', { |
112 | | - method: 'POST', |
113 | | - body: formData, |
114 | | - }) |
115 | | - .then((response) => { |
116 | | - if (!response.ok) { |
117 | | - throw new Error('Не удалось отправить форму'); |
118 | | - } |
119 | | - |
120 | | - closeForm(); |
121 | | - showSuccessMessage(); |
122 | | - }) |
123 | | - .catch(() => { |
124 | | - showErrorMessage(); |
125 | | - }) |
126 | | - .finally(() => { |
127 | | - submitButton.disabled = false; |
128 | | - }); |
129 | | -}); |
130 | | - |
131 | | -// Слушатели |
132 | 31 | const initFormListeners = () => { |
133 | | - fileInput.addEventListener('change', openForm); |
134 | | - cancelButton.addEventListener('click', closeForm); |
| 32 | + fileInput.addEventListener('change', () => { |
| 33 | + showUploadForm(); |
| 34 | + }); |
| 35 | + |
| 36 | + cancelButton.addEventListener('click', () => { |
| 37 | + hideUploadForm(); |
| 38 | + }); |
135 | 39 | }; |
136 | 40 |
|
137 | 41 | export { initFormListeners }; |
0 commit comments