diff --git a/6/bundle.25e23cc746eca4886d2b.js b/6/bundle.25e23cc746eca4886d2b.js
new file mode 100644
index 0000000..9d4db74
--- /dev/null
+++ b/6/bundle.25e23cc746eca4886d2b.js
@@ -0,0 +1 @@
+(()=>{var __webpack_modules__={89:(__unused_webpack_module,__unused_webpack___webpack_exports__,__webpack_require__)=>{"use strict";eval('\n// EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\nvar injectStylesIntoStyleTag = __webpack_require__(379);\nvar injectStylesIntoStyleTag_default = /*#__PURE__*/__webpack_require__.n(injectStylesIntoStyleTag);\n// EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/styleDomAPI.js\nvar styleDomAPI = __webpack_require__(795);\nvar styleDomAPI_default = /*#__PURE__*/__webpack_require__.n(styleDomAPI);\n// EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/insertBySelector.js\nvar insertBySelector = __webpack_require__(569);\nvar insertBySelector_default = /*#__PURE__*/__webpack_require__.n(insertBySelector);\n// EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\nvar setAttributesWithoutAttributes = __webpack_require__(565);\nvar setAttributesWithoutAttributes_default = /*#__PURE__*/__webpack_require__.n(setAttributesWithoutAttributes);\n// EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/insertStyleElement.js\nvar insertStyleElement = __webpack_require__(216);\nvar insertStyleElement_default = /*#__PURE__*/__webpack_require__.n(insertStyleElement);\n// EXTERNAL MODULE: ./node_modules/style-loader/dist/runtime/styleTagTransform.js\nvar styleTagTransform = __webpack_require__(589);\nvar styleTagTransform_default = /*#__PURE__*/__webpack_require__.n(styleTagTransform);\n// EXTERNAL MODULE: ./node_modules/css-loader/dist/cjs.js!./src/framework/view/abstract-view.css\nvar abstract_view = __webpack_require__(10);\n;// CONCATENATED MODULE: ./src/framework/view/abstract-view.css\n\n \n \n \n \n \n \n \n \n \n\nvar options = {};\n\noptions.styleTagTransform = (styleTagTransform_default());\noptions.setAttributes = (setAttributesWithoutAttributes_default());\n\n options.insert = insertBySelector_default().bind(null, "head");\n \noptions.domAPI = (styleDomAPI_default());\noptions.insertStyleElement = (insertStyleElement_default());\n\nvar update = injectStylesIntoStyleTag_default()(abstract_view/* default */.Z, options);\n\n\n\n\n /* harmony default export */ const view_abstract_view = (abstract_view/* default */.Z && abstract_view/* default.locals */.Z.locals ? abstract_view/* default.locals */.Z.locals : undefined);\n\n;// CONCATENATED MODULE: ./src/framework/view/abstract-view.js\n\n\n\n/** @const {string} Класс, реализующий эффект "покачивания головой" */\nconst SHAKE_CLASS_NAME = \'shake\';\n\n/** @const {number} Время анимации в миллисекундах */\nconst SHAKE_ANIMATION_TIMEOUT = 600;\n\n/**\n * Абстрактный класс представления\n */\nclass AbstractView {\n /** @type {HTMLElement|null} Элемент представления */\n #element = null;\n constructor() {\n if (new.target === AbstractView) {\n throw new Error(\'Can\\\'t instantiate AbstractView, only concrete one.\');\n }\n }\n\n /**\n * Геттер для получения элемента\n * @returns {HTMLElement} Элемент представления\n */\n get element() {\n if (!this.#element) {\n this.#element = createElement(this.template);\n }\n return this.#element;\n }\n\n /**\n * Геттер для получения разметки элемента\n * @abstract\n * @returns {string} Разметка элемента в виде строки\n */\n get template() {\n throw new Error(\'Abstract method not implemented: get template\');\n }\n\n /** Метод для удаления элемента */\n removeElement() {\n this.#element = null;\n }\n\n /**\n * Метод, реализующий эффект "покачивания головой"\n * @param {shakeCallback} [callback] Функция, которая будет вызвана после завершения анимации\n */\n shake(callback) {\n this.element.classList.add(SHAKE_CLASS_NAME);\n setTimeout(() => {\n this.element.classList.remove(SHAKE_CLASS_NAME);\n callback?.();\n }, SHAKE_ANIMATION_TIMEOUT);\n }\n}\n\n/**\n * Функция, которая будет вызвана методом shake после завершения анимации\n * @callback shakeCallback\n */\n;// CONCATENATED MODULE: ./src/framework/render.js\n\n\n/** @enum {string}\n * BEFOREBEGIN - вставить элемент перед началом контейнера\n * AFTERBEGIN - вставить элемент сразу после начала контейнера\n * BEFOREEND - вставить элемент перед концом контейнера\n * AFTEREND - вставить элемент сразу после конца контейнера\n*/\nconst RenderPosition = {\n BEFOREBEGIN: \'beforebegin\',\n AFTERBEGIN: \'afterbegin\',\n BEFOREEND: \'beforeend\',\n AFTEREND: \'afterend\'\n};\n\n/**\n * Функция для создания элемента на основе разметки\n * @param {string} template Разметка в виде строки\n * @returns {HTMLElement} Созданный элемент\n */\nfunction createElement(template) {\n const newElement = document.createElement(\'div\');\n newElement.innerHTML = template;\n return newElement.firstElementChild;\n}\n\n/**\n * Функция для отрисовки элемента\n * @param {AbstractView} component Компонент, который должен был отрисован\n * @param {HTMLElement} container Элемент в котором будет отрисован компонент\n * @param {string} place Позиция компонента относительно контейнера. По умолчанию - `beforeend`\n */\nfunction render(component, container, place = RenderPosition.BEFOREEND) {\n if (!(component instanceof AbstractView)) {\n throw new Error(\'Can render only components\');\n }\n if (container === null) {\n throw new Error(\'Container element doesn\\\'t exist\');\n }\n container.insertAdjacentElement(place, component.element);\n}\n\n/**\n * Функция для замены одного компонента на другой\n * @param {AbstractView} newComponent Компонент, который нужно показать\n * @param {AbstractView} oldComponent Компонент, который нужно скрыть\n */\nfunction replace(newComponent, oldComponent) {\n if (!(newComponent instanceof AbstractView && oldComponent instanceof AbstractView)) {\n throw new Error(\'Can replace only components\');\n }\n const newElement = newComponent.element;\n const oldElement = oldComponent.element;\n const parent = oldElement.parentElement;\n if (parent === null) {\n throw new Error(\'Parent element doesn\\\'t exist\');\n }\n parent.replaceChild(newElement, oldElement);\n}\n\n/**\n * Функция для удаления компонента\n * @param {AbstractView} component Компонент, который нужно удалить\n */\nfunction remove(component) {\n if (component === null) {\n return;\n }\n if (!(component instanceof AbstractView)) {\n throw new Error(\'Can remove only components\');\n }\n component.element.remove();\n component.removeElement();\n}\n\n;// CONCATENATED MODULE: ./src/utils/common.js\nconst ESCAPE_KEY_CODE = 27;\nconst isEscapeKey = evt => evt.keyCode === ESCAPE_KEY_CODE;\nconst getRandomNumber = (from, to) => {\n const lower = Math.ceil(Math.min(from, to));\n const upper = Math.floor(Math.max(from, to));\n const result = Math.random() * (upper - lower + 1) + lower;\n return Math.floor(result);\n};\nconst getRandomString = (desiredStringLength = 1) => {\n const primer = \' ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\';\n let randomString = \'\';\n for (let i = 0; i < desiredStringLength; i++) {\n randomString += primer.charAt(getRandomNumber(0, primer.length - 1));\n }\n return randomString;\n};\nconst getRandomBoolean = () => Math.random() >= 0.5;\nconst getRandomArrayElement = items => items[getRandomNumber(0, items.length - 1)];\nconst getRandomDate = () => {\n const today = new Date();\n const oneYearFromNow = new Date();\n oneYearFromNow.setFullYear(today.getFullYear() + 1);\n const randomTime = Math.random() * (oneYearFromNow - today) + today.getTime();\n return new Date(randomTime);\n};\nconst convertToCamelCase = str => str.replace(/(_\\w)/g, match => match[1].toUpperCase());\nconst convertKeysToCamelCase = items => {\n if (Array.isArray(items)) {\n return items.map(convertKeysToCamelCase);\n }\n if (items !== null && typeof items === \'object\') {\n return Object.entries(items).reduce((newItems, [key, value]) => {\n newItems[convertToCamelCase(key)] = convertKeysToCamelCase(value);\n return newItems;\n }, {});\n }\n return items;\n};\nconst updateItem = (items, update) => items.map(item => item.id === update.id ? update : item);\n\n;// CONCATENATED MODULE: ./src/const.js\nconst EVENTS_TYPES = [\'Taxi\', \'Bus\', \'Train\', \'Ship\', \'Drive\', \'Flight\', \'Check-in\', \'Sightseeing\', \'Restaurant\'];\nconst CITIES = [\'New York\', \'Tokyo\', \'Paris\', \'London\', \'Sydney\', \'Berlin\', \'Rio de Janeiro\', \'Moscow\', \'Cairo\', \'Cape Town\'];\nconst SENTENCES = [\'Lorem ipsum dolor sit amet, consectetur adipiscing elit. Cras aliquet varius magna, non porta ligula feugiat eget. \', \'Fusce tristique felis at fermentum pharetra. \', \'Aliquam id orci ut lectus varius viverra. \', \'Nullam nunc ex, convallis sed finibus eget, sollicitudin eget ante. \', \'Phasellus eros mauris, condimentum sed nibh vitae, sodales efficitur ipsum. \', \'Sed blandit, eros vel aliquam faucibus, purus ex euismod diam, eu luctus nunc ante ut dui. \', \'Sed sed nisi sed augue convallis suscipit in sed felis. \', \'Aliquam erat volutpat. \', \'Nunc fermentum tortor ac porta dapibus. \', \'In rutrum ac purus sit amet tempus.\'];\nconst FilterType = {\n EVERYTHING: \'everything\',\n FUTURE: \'future\',\n PRESENT: \'present\',\n PAST: \'past\'\n};\nconst SortType = {\n DEFAULT: \'default\',\n PRICE: \'price\',\n TIME: \'time\'\n};\n\n;// CONCATENATED MODULE: ./node_modules/nanoid/index.browser.js\n\nlet random = bytes => crypto.getRandomValues(new Uint8Array(bytes))\nlet customRandom = (alphabet, defaultSize, getRandom) => {\n let mask = (2 << (Math.log(alphabet.length - 1) / Math.LN2)) - 1\n let step = -~((1.6 * mask * defaultSize) / alphabet.length)\n return (size = defaultSize) => {\n let id = \'\'\n while (true) {\n let bytes = getRandom(step)\n let j = step\n while (j--) {\n id += alphabet[bytes[j] & mask] || \'\'\n if (id.length === size) return id\n }\n }\n }\n}\nlet customAlphabet = (alphabet, size = 21) =>\n customRandom(alphabet, size, random)\nlet nanoid = (size = 21) =>\n crypto.getRandomValues(new Uint8Array(size)).reduce((id, byte) => {\n byte &= 63\n if (byte < 36) {\n id += byte.toString(36)\n } else if (byte < 62) {\n id += (byte - 26).toString(36).toUpperCase()\n } else if (byte > 62) {\n id += \'-\'\n } else {\n id += \'_\'\n }\n return id\n }, \'\')\n\n;// CONCATENATED MODULE: ./src/mock/events.js\n\n\n\nconst TIME_SKIP = 125;\nconst getRandomEvent = (date, destinationsList) => {\n const firstDate = new Date(date);\n const secondDate = new Date(firstDate);\n const destinationsIds = destinationsList.map(destination => destination.id);\n secondDate.setMinutes(firstDate.getMinutes() + TIME_SKIP);\n const randomEvent = {\n \'id\': nanoid(),\n \'base_price\': getRandomNumber(499, 4999),\n \'date_from\': firstDate.toISOString(),\n \'date_to\': secondDate.toISOString(),\n \'destination\': getRandomArrayElement(destinationsIds),\n \'is_favorite\': getRandomBoolean(),\n \'offers\': [\'04c3e4e6-9053-42ce-b747-e281314baa31\', \'14c3e4e6-9053-42ce-b747-e281314baa31\', \'24c3e4e6-9053-42ce-b747-e281314baa31\', \'34c3e4e6-9053-42ce-b747-e281314baa31\'],\n \'type\': getRandomArrayElement(EVENTS_TYPES)\n };\n return randomEvent;\n};\nconst getRandomEvents = (count, destinationsList) => {\n const date = getRandomDate();\n const events = [];\n for (let i = 0; i < count; i++) {\n events.push(getRandomEvent(date, destinationsList));\n date.setMinutes(date.getMinutes() + TIME_SKIP);\n }\n return events;\n};\n\n;// CONCATENATED MODULE: ./src/mock/offers.js\n\nconst OFFERS_COUNT = 4;\nconst getRandomOffer = id => {\n const offer = {\n \'id\': `${id}4c3e4e6-9053-42ce-b747-e281314baa31`,\n \'title\': `Upgrade ${id}`,\n \'price\': getRandomNumber(19, 499)\n };\n return offer;\n};\nconst getRandomOffers = type => {\n const randomOffers = [{\n \'type\': type,\n \'offers\': []\n }];\n for (let i = 0; i < OFFERS_COUNT; i++) {\n randomOffers[0].offers.push(getRandomOffer(i));\n }\n return randomOffers;\n};\n\n;// CONCATENATED MODULE: ./src/mock/destinations.js\n\n\nconst getRandomDescription = () => {\n let randomDescription = \'\';\n for (let i = 0; i < getRandomNumber(1, 5); i++) {\n randomDescription += getRandomArrayElement(SENTENCES);\n }\n return randomDescription;\n};\nconst getRandomDestination = id => {\n const destination = {\n \'id\': `${id}fa5cb75-a1fe-4b77-a83c-0e528e910e04`,\n \'description\': getRandomDescription(),\n \'name\': CITIES[id],\n \'pictures\': [{\n \'src\': `https://loremflickr.com/248/152?random=${getRandomNumber(1, 100)}`,\n \'description\': getRandomDescription()\n }]\n };\n return destination;\n};\nconst getRandomDestinations = () => {\n const destinations = [];\n for (let i = 0; i < CITIES.length; i++) {\n destinations.push(getRandomDestination(i));\n }\n return destinations;\n};\n\n;// CONCATENATED MODULE: ./src/model/events-connector.js\n\n\n\n\nconst EVENTS_COUNT = 5;\nconst createUserEvents = () => {\n const offersList = convertKeysToCamelCase(getRandomOffers());\n const destinationsList = convertKeysToCamelCase(getRandomDestinations());\n const eventsList = convertKeysToCamelCase(getRandomEvents(EVENTS_COUNT, destinationsList));\n const destinationsMap = new Map(destinationsList.map(destination => [destination.id, destination]));\n const offers = offersList[0].offers;\n const mergedEvents = eventsList.map(event => {\n const destinationData = destinationsMap.get(event.destination);\n const offersData = event.offers.map(offerId => offers.find(offer => offer.id === offerId)).filter(Boolean);\n return {\n ...event,\n destination: {\n id: destinationData.id,\n name: destinationData.name,\n description: destinationData.description,\n pictures: destinationData.pictures\n },\n offers: offersData\n };\n });\n return Array.from(new Map(mergedEvents.map(event => [event.id, event])).values());\n};\nclass EventsConnector {\n get userEvents() {\n return structuredClone(createUserEvents());\n }\n}\n;// CONCATENATED MODULE: ./src/model/events-model.js\n\nclass EventsModel {\n #EventsConnector = new EventsConnector();\n #eventsList = this.#EventsConnector.userEvents;\n get userEvents() {\n return this.#eventsList;\n }\n}\n;// CONCATENATED MODULE: ./src/view/new-trip-info-view.js\n\nfunction createNewTripInfoTemplate() {\n return ` 18 — 20 Mar \n Total: € 1230\n Amsterdam — Chamonix — Geneva
\n\n
\n \n —\n \n
\n${getTimeDifference(dateFrom, dateTo)}
\n\n € ${basePrice}\n
\nLoading...
\';\n}\nclass NoEventsView extends AbstractView {\n get template() {\n return createNoEventsViewTemplate();\n }\n}\n;// CONCATENATED MODULE: ./src/presenter/board-presenter.js\n\n\n\n\n\n\n\n\nclass BoardPresenter {\n #container = null;\n #eventsModel = null;\n #sortComponent = null;\n #eventsListComponent = new NewEventsListView();\n #eventsList = [];\n #eventPresenters = new Map();\n #currentSortType = SortType.DEFAULT;\n #sourcedBoardEvents = [];\n constructor({\n container,\n eventsModel\n }) {\n this.#container = container;\n this.#eventsModel = eventsModel;\n }\n init() {\n this.#eventsList = [...this.#eventsModel.userEvents];\n this.#sourcedBoardEvents = [...this.#eventsModel.userEvents];\n this.#renderBoard();\n this.#renderSort();\n }\n #renderEvent(inputUserEvent) {\n const eventPresenter = new EventPresenter({\n container: this.#eventsListComponent.element,\n onDataChange: this.#handleEventChange,\n onModeChange: this.#handleModeChange\n });\n eventPresenter.init(inputUserEvent);\n this.#eventPresenters.set(inputUserEvent.id, eventPresenter);\n }\n #renderNoEvents() {\n render(new NoEventsView(), this.#container);\n }\n #sortEvents(sortType) {\n switch (sortType) {\n case SortType.PRICE:\n this.#eventsList.sort(sortEventsPrice);\n break;\n case SortType.TIME:\n this.#eventsList.sort(sortEventsTime);\n break;\n default:\n this.#eventsList = [...this.#sourcedBoardEvents];\n }\n this.#currentSortType = sortType;\n }\n #handleSortTypeChange = sortType => {\n if (sortType === undefined || this.#currentSortType === sortType) {\n return;\n }\n this.#sortEvents(sortType);\n this.#clearEventList();\n this.#renderBoard();\n };\n #renderSort() {\n this.#sortComponent = new NewTripSortView({\n onSortTypeChange: this.#handleSortTypeChange\n });\n render(this.#sortComponent, this.#container, \'AFTERBEGIN\');\n }\n #renderBoard() {\n if (this.#eventsList.length === 0) {\n this.#renderNoEvents();\n return;\n }\n render(this.#eventsListComponent, this.#container);\n for (let i = 0; i < this.#eventsList.length; i++) {\n this.#renderEvent(this.#eventsList[i]);\n }\n }\n #handleEventChange = updatedEvent => {\n this.#eventsList = updateItem(this.#eventsList, updatedEvent);\n this.#sourcedBoardEvents = updateItem(this.#sourcedBoardEvents, updatedEvent);\n this.#eventPresenters.get(updatedEvent.id).init(updatedEvent);\n };\n #handleModeChange = () => {\n this.#eventPresenters.forEach(presenter => presenter.resetView());\n };\n #clearEventList() {\n this.#eventPresenters.forEach(presenter => presenter.destroy());\n this.#eventPresenters.clear();\n }\n}\n;// CONCATENATED MODULE: ./src/utils/filter.js\n\n\nconst filter = {\n [FilterType.EVERYTHING]: userEvents => userEvents,\n [FilterType.FUTURE]: userEvents => userEvents.filter(userEvent => isEventFuture(userEvent.dateFrom)),\n [FilterType.PRESENT]: userEvents => userEvents.filter(userEvent => isEventPresent(userEvent.dateFrom)),\n [FilterType.PAST]: userEvents => userEvents.filter(userEvent => isEventPast(userEvent.dateFrom))\n};\n\n;// CONCATENATED MODULE: ./src/mock/filter.js\n\nfunction generateFilter(userEvents) {\n return Object.entries(filter).map(([filterType, filterEvents]) => ({\n type: filterType,\n count: filterEvents(userEvents).length\n }));\n}\n\n;// CONCATENATED MODULE: ./src/main.js\n\n\n\n\n\n\nconst tripMainContainer = document.querySelector(\'.trip-main\');\nconst tripFiltersContainer = tripMainContainer.querySelector(\'.trip-controls__filters\');\nconst tripEventsContainer = document.querySelector(\'.trip-events\');\nconst eventsModel = new EventsModel();\nconst boardPresenter = new BoardPresenter({\n container: tripEventsContainer,\n eventsModel\n});\nconst filters = generateFilter(eventsModel.userEvents);\nrender(new NewTripInfoView(), tripMainContainer, \'AFTERBEGIN\');\nrender(new NewTripFiltersView({\n filters\n}), tripFiltersContainer);\nboardPresenter.init();//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"89.js","mappings":";;;;;;;;;;;;;;;;;;;;;;AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC1BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACzEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC7BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACjBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACPA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sources":["webpack://big-trip/./src/framework/view/abstract-view.css?dfa6","webpack://big-trip/./src/framework/view/abstract-view.js?4073","webpack://big-trip/./src/framework/render.js?b51e","webpack://big-trip/./src/utils/common.js?c8c3","webpack://big-trip/./src/const.js?dfb6","webpack://big-trip/./node_modules/nanoid/index.browser.js?b381","webpack://big-trip/./src/mock/events.js?648d","webpack://big-trip/./src/mock/offers.js?8d29","webpack://big-trip/./src/mock/destinations.js?d8bf","webpack://big-trip/./src/model/events-connector.js?ade8","webpack://big-trip/./src/model/events-model.js?e099","webpack://big-trip/./src/view/new-trip-info-view.js?2f58","webpack://big-trip/./src/view/new-filters-view.js?4335","webpack://big-trip/./src/view/new-sort-container-view.js?7e49","webpack://big-trip/./src/view/new-events-list-view.js?680d","webpack://big-trip/./src/utils/event.js?2a51","webpack://big-trip/./src/view/new-events-item-view.js?74bc","webpack://big-trip/./src/view/new-event-edit-element-view.js?4450","webpack://big-trip/./src/presenter/event-presenter.js?c5f5","webpack://big-trip/./src/view/no-events-view.js?2987","webpack://big-trip/./src/presenter/board-presenter.js?5a60","webpack://big-trip/./src/utils/filter.js?efa6","webpack://big-trip/./src/mock/filter.js?942b","webpack://big-trip/./src/main.js?8138"],"sourcesContent":["\n      import API from \"!../../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\";\n      import domAPI from \"!../../../node_modules/style-loader/dist/runtime/styleDomAPI.js\";\n      import insertFn from \"!../../../node_modules/style-loader/dist/runtime/insertBySelector.js\";\n      import setAttributes from \"!../../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\";\n      import insertStyleElement from \"!../../../node_modules/style-loader/dist/runtime/insertStyleElement.js\";\n      import styleTagTransformFn from \"!../../../node_modules/style-loader/dist/runtime/styleTagTransform.js\";\n      import content, * as namedExport from \"!!../../../node_modules/css-loader/dist/cjs.js!./abstract-view.css\";\n      \n      \n\nvar options = {};\n\noptions.styleTagTransform = styleTagTransformFn;\noptions.setAttributes = setAttributes;\n\n      options.insert = insertFn.bind(null, \"head\");\n    \noptions.domAPI = domAPI;\noptions.insertStyleElement = insertStyleElement;\n\nvar update = API(content, options);\n\n\n\nexport * from \"!!../../../node_modules/css-loader/dist/cjs.js!./abstract-view.css\";\n       export default content && content.locals ? content.locals : undefined;\n","import { createElement } from '../render.js';\nimport './abstract-view.css';\n\n/** @const {string} Класс, реализующий эффект \"покачивания головой\" */\nconst SHAKE_CLASS_NAME = 'shake';\n\n/** @const {number} Время анимации в миллисекундах */\nconst SHAKE_ANIMATION_TIMEOUT = 600;\n\n/**\n * Абстрактный класс представления\n */\nexport default class AbstractView {\n  /** @type {HTMLElement|null} Элемент представления */\n  #element = null;\n  constructor() {\n    if (new.target === AbstractView) {\n      throw new Error('Can\\'t instantiate AbstractView, only concrete one.');\n    }\n  }\n\n  /**\n   * Геттер для получения элемента\n   * @returns {HTMLElement} Элемент представления\n   */\n  get element() {\n    if (!this.#element) {\n      this.#element = createElement(this.template);\n    }\n    return this.#element;\n  }\n\n  /**\n   * Геттер для получения разметки элемента\n   * @abstract\n   * @returns {string} Разметка элемента в виде строки\n   */\n  get template() {\n    throw new Error('Abstract method not implemented: get template');\n  }\n\n  /** Метод для удаления элемента */\n  removeElement() {\n    this.#element = null;\n  }\n\n  /**\n   * Метод, реализующий эффект \"покачивания головой\"\n   * @param {shakeCallback} [callback] Функция, которая будет вызвана после завершения анимации\n   */\n  shake(callback) {\n    this.element.classList.add(SHAKE_CLASS_NAME);\n    setTimeout(() => {\n      this.element.classList.remove(SHAKE_CLASS_NAME);\n      callback?.();\n    }, SHAKE_ANIMATION_TIMEOUT);\n  }\n}\n\n/**\n * Функция, которая будет вызвана методом shake после завершения анимации\n * @callback shakeCallback\n */","import AbstractView from './view/abstract-view.js';\n\n/** @enum {string}\n * BEFOREBEGIN - вставить элемент перед началом контейнера\n * AFTERBEGIN - вставить элемент сразу после начала контейнера\n * BEFOREEND - вставить элемент перед концом контейнера\n * AFTEREND - вставить элемент сразу после конца контейнера\n*/\nconst RenderPosition = {\n  BEFOREBEGIN: 'beforebegin',\n  AFTERBEGIN: 'afterbegin',\n  BEFOREEND: 'beforeend',\n  AFTEREND: 'afterend'\n};\n\n/**\n * Функция для создания элемента на основе разметки\n * @param {string} template Разметка в виде строки\n * @returns {HTMLElement} Созданный элемент\n */\nfunction createElement(template) {\n  const newElement = document.createElement('div');\n  newElement.innerHTML = template;\n  return newElement.firstElementChild;\n}\n\n/**\n * Функция для отрисовки элемента\n * @param {AbstractView} component Компонент, который должен был отрисован\n * @param {HTMLElement} container Элемент в котором будет отрисован компонент\n * @param {string} place Позиция компонента относительно контейнера. По умолчанию - `beforeend`\n */\nfunction render(component, container, place = RenderPosition.BEFOREEND) {\n  if (!(component instanceof AbstractView)) {\n    throw new Error('Can render only components');\n  }\n  if (container === null) {\n    throw new Error('Container element doesn\\'t exist');\n  }\n  container.insertAdjacentElement(place, component.element);\n}\n\n/**\n * Функция для замены одного компонента на другой\n * @param {AbstractView} newComponent Компонент, который нужно показать\n * @param {AbstractView} oldComponent Компонент, который нужно скрыть\n */\nfunction replace(newComponent, oldComponent) {\n  if (!(newComponent instanceof AbstractView && oldComponent instanceof AbstractView)) {\n    throw new Error('Can replace only components');\n  }\n  const newElement = newComponent.element;\n  const oldElement = oldComponent.element;\n  const parent = oldElement.parentElement;\n  if (parent === null) {\n    throw new Error('Parent element doesn\\'t exist');\n  }\n  parent.replaceChild(newElement, oldElement);\n}\n\n/**\n * Функция для удаления компонента\n * @param {AbstractView} component Компонент, который нужно удалить\n */\nfunction remove(component) {\n  if (component === null) {\n    return;\n  }\n  if (!(component instanceof AbstractView)) {\n    throw new Error('Can remove only components');\n  }\n  component.element.remove();\n  component.removeElement();\n}\nexport { RenderPosition, createElement, render, replace, remove };","const ESCAPE_KEY_CODE = 27;\nconst isEscapeKey = evt => evt.keyCode === ESCAPE_KEY_CODE;\nconst getRandomNumber = (from, to) => {\n  const lower = Math.ceil(Math.min(from, to));\n  const upper = Math.floor(Math.max(from, to));\n  const result = Math.random() * (upper - lower + 1) + lower;\n  return Math.floor(result);\n};\nconst getRandomString = (desiredStringLength = 1) => {\n  const primer = ' ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';\n  let randomString = '';\n  for (let i = 0; i < desiredStringLength; i++) {\n    randomString += primer.charAt(getRandomNumber(0, primer.length - 1));\n  }\n  return randomString;\n};\nconst getRandomBoolean = () => Math.random() >= 0.5;\nconst getRandomArrayElement = items => items[getRandomNumber(0, items.length - 1)];\nconst getRandomDate = () => {\n  const today = new Date();\n  const oneYearFromNow = new Date();\n  oneYearFromNow.setFullYear(today.getFullYear() + 1);\n  const randomTime = Math.random() * (oneYearFromNow - today) + today.getTime();\n  return new Date(randomTime);\n};\nconst convertToCamelCase = str => str.replace(/(_\\w)/g, match => match[1].toUpperCase());\nconst convertKeysToCamelCase = items => {\n  if (Array.isArray(items)) {\n    return items.map(convertKeysToCamelCase);\n  }\n  if (items !== null && typeof items === 'object') {\n    return Object.entries(items).reduce((newItems, [key, value]) => {\n      newItems[convertToCamelCase(key)] = convertKeysToCamelCase(value);\n      return newItems;\n    }, {});\n  }\n  return items;\n};\nconst updateItem = (items, update) => items.map(item => item.id === update.id ? update : item);\nexport { updateItem, getRandomNumber, getRandomString, getRandomBoolean, getRandomDate, getRandomArrayElement, convertKeysToCamelCase, isEscapeKey };","const EVENTS_TYPES = ['Taxi', 'Bus', 'Train', 'Ship', 'Drive', 'Flight', 'Check-in', 'Sightseeing', 'Restaurant'];\nconst CITIES = ['New York', 'Tokyo', 'Paris', 'London', 'Sydney', 'Berlin', 'Rio de Janeiro', 'Moscow', 'Cairo', 'Cape Town'];\nconst SENTENCES = ['Lorem ipsum dolor sit amet, consectetur adipiscing elit. Cras aliquet varius magna, non porta ligula feugiat eget. ', 'Fusce tristique felis at fermentum pharetra. ', 'Aliquam id orci ut lectus varius viverra. ', 'Nullam nunc ex, convallis sed finibus eget, sollicitudin eget ante. ', 'Phasellus eros mauris, condimentum sed nibh vitae, sodales efficitur ipsum. ', 'Sed blandit, eros vel aliquam faucibus, purus ex euismod diam, eu luctus nunc ante ut dui. ', 'Sed sed nisi sed augue convallis suscipit in sed felis. ', 'Aliquam erat volutpat. ', 'Nunc fermentum tortor ac porta dapibus. ', 'In rutrum ac purus sit amet tempus.'];\nconst FilterType = {\n  EVERYTHING: 'everything',\n  FUTURE: 'future',\n  PRESENT: 'present',\n  PAST: 'past'\n};\nconst SortType = {\n  DEFAULT: 'default',\n  PRICE: 'price',\n  TIME: 'time'\n};\nexport { EVENTS_TYPES, CITIES, SENTENCES, FilterType, SortType };","export { urlAlphabet } from './url-alphabet/index.js'\nexport let random = bytes => crypto.getRandomValues(new Uint8Array(bytes))\nexport let customRandom = (alphabet, defaultSize, getRandom) => {\n  let mask = (2 << (Math.log(alphabet.length - 1) / Math.LN2)) - 1\n  let step = -~((1.6 * mask * defaultSize) / alphabet.length)\n  return (size = defaultSize) => {\n    let id = ''\n    while (true) {\n      let bytes = getRandom(step)\n      let j = step\n      while (j--) {\n        id += alphabet[bytes[j] & mask] || ''\n        if (id.length === size) return id\n      }\n    }\n  }\n}\nexport let customAlphabet = (alphabet, size = 21) =>\n  customRandom(alphabet, size, random)\nexport let nanoid = (size = 21) =>\n  crypto.getRandomValues(new Uint8Array(size)).reduce((id, byte) => {\n    byte &= 63\n    if (byte < 36) {\n      id += byte.toString(36)\n    } else if (byte < 62) {\n      id += (byte - 26).toString(36).toUpperCase()\n    } else if (byte > 62) {\n      id += '-'\n    } else {\n      id += '_'\n    }\n    return id\n  }, '')\n","import { getRandomArrayElement, getRandomNumber, getRandomBoolean, getRandomDate } from '../utils/common';\nimport { EVENTS_TYPES } from '../const';\nimport { nanoid } from 'nanoid';\nconst TIME_SKIP = 125;\nconst getRandomEvent = (date, destinationsList) => {\n  const firstDate = new Date(date);\n  const secondDate = new Date(firstDate);\n  const destinationsIds = destinationsList.map(destination => destination.id);\n  secondDate.setMinutes(firstDate.getMinutes() + TIME_SKIP);\n  const randomEvent = {\n    'id': nanoid(),\n    'base_price': getRandomNumber(499, 4999),\n    'date_from': firstDate.toISOString(),\n    'date_to': secondDate.toISOString(),\n    'destination': getRandomArrayElement(destinationsIds),\n    'is_favorite': getRandomBoolean(),\n    'offers': ['04c3e4e6-9053-42ce-b747-e281314baa31', '14c3e4e6-9053-42ce-b747-e281314baa31', '24c3e4e6-9053-42ce-b747-e281314baa31', '34c3e4e6-9053-42ce-b747-e281314baa31'],\n    'type': getRandomArrayElement(EVENTS_TYPES)\n  };\n  return randomEvent;\n};\nconst getRandomEvents = (count, destinationsList) => {\n  const date = getRandomDate();\n  const events = [];\n  for (let i = 0; i < count; i++) {\n    events.push(getRandomEvent(date, destinationsList));\n    date.setMinutes(date.getMinutes() + TIME_SKIP);\n  }\n  return events;\n};\nexport { getRandomEvents };","import { getRandomNumber } from '../utils/common';\nconst OFFERS_COUNT = 4;\nconst getRandomOffer = id => {\n  const offer = {\n    'id': `${id}4c3e4e6-9053-42ce-b747-e281314baa31`,\n    'title': `Upgrade ${id}`,\n    'price': getRandomNumber(19, 499)\n  };\n  return offer;\n};\nconst getRandomOffers = type => {\n  const randomOffers = [{\n    'type': type,\n    'offers': []\n  }];\n  for (let i = 0; i < OFFERS_COUNT; i++) {\n    randomOffers[0].offers.push(getRandomOffer(i));\n  }\n  return randomOffers;\n};\nexport { getRandomOffers };","import { getRandomArrayElement, getRandomNumber } from '../utils/common';\nimport { SENTENCES, CITIES } from '../const';\nconst getRandomDescription = () => {\n  let randomDescription = '';\n  for (let i = 0; i < getRandomNumber(1, 5); i++) {\n    randomDescription += getRandomArrayElement(SENTENCES);\n  }\n  return randomDescription;\n};\nconst getRandomDestination = id => {\n  const destination = {\n    'id': `${id}fa5cb75-a1fe-4b77-a83c-0e528e910e04`,\n    'description': getRandomDescription(),\n    'name': CITIES[id],\n    'pictures': [{\n      'src': `https://loremflickr.com/248/152?random=${getRandomNumber(1, 100)}`,\n      'description': getRandomDescription()\n    }]\n  };\n  return destination;\n};\nconst getRandomDestinations = () => {\n  const destinations = [];\n  for (let i = 0; i < CITIES.length; i++) {\n    destinations.push(getRandomDestination(i));\n  }\n  return destinations;\n};\nexport { getRandomDestinations };","import { getRandomEvents } from '../mock/events';\nimport { getRandomOffers } from '../mock/offers';\nimport { getRandomDestinations } from '../mock/destinations';\nimport { convertKeysToCamelCase } from '../utils/common';\nconst EVENTS_COUNT = 5;\nconst createUserEvents = () => {\n  const offersList = convertKeysToCamelCase(getRandomOffers());\n  const destinationsList = convertKeysToCamelCase(getRandomDestinations());\n  const eventsList = convertKeysToCamelCase(getRandomEvents(EVENTS_COUNT, destinationsList));\n  const destinationsMap = new Map(destinationsList.map(destination => [destination.id, destination]));\n  const offers = offersList[0].offers;\n  const mergedEvents = eventsList.map(event => {\n    const destinationData = destinationsMap.get(event.destination);\n    const offersData = event.offers.map(offerId => offers.find(offer => offer.id === offerId)).filter(Boolean);\n    return {\n      ...event,\n      destination: {\n        id: destinationData.id,\n        name: destinationData.name,\n        description: destinationData.description,\n        pictures: destinationData.pictures\n      },\n      offers: offersData\n    };\n  });\n  return Array.from(new Map(mergedEvents.map(event => [event.id, event])).values());\n};\nexport default class EventsConnector {\n  get userEvents() {\n    return structuredClone(createUserEvents());\n  }\n}","import EventsConnector from './events-connector';\nexport default class EventsModel {\n  #EventsConnector = new EventsConnector();\n  #eventsList = this.#EventsConnector.userEvents;\n  get userEvents() {\n    return this.#eventsList;\n  }\n}","import AbstractView from '../framework/view/abstract-view';\nfunction createNewTripInfoTemplate() {\n  return `<section class=\"trip-main__trip-info  trip-info\">\n            <div class=\"trip-info__main\">\n              <h1 class=\"trip-info__title\">Amsterdam &mdash; Chamonix &mdash; Geneva</h1>\n\n              <p class=\"trip-info__dates\">18&nbsp;&mdash;&nbsp;20 Mar</p>\n            </div>\n\n            <p class=\"trip-info__cost\">\n              Total: &euro;&nbsp;<span class=\"trip-info__cost-value\">1230</span>\n            </p>\n          </section>`;\n}\nexport default class NewTripInfoView extends AbstractView {\n  get template() {\n    return createNewTripInfoTemplate();\n  }\n}","import AbstractView from '../framework/view/abstract-view';\nconst createFilterItemTemplate = (filter, isChecked) => {\n  const {\n    type,\n    count\n  } = filter;\n  return `<div class=\"trip-filters__filter\">\n                  <input\n                  id=\"filter-${type}\"\n                  class=\"trip-filters__filter-input visually-hidden\"\n                  type=\"radio\"\n                  name=\"trip-filter\"\n                  value=\"${type}\"\n                  ${isChecked ? 'checked' : ''}\n                  ${count === 0 ? 'disabled' : ''}>\n                  <label class=\"trip-filters__filter-label\" for=\"filter-${type}\">${type}</label>\n                </div>`;\n};\nconst createNewTripFiltersTemplate = filterItems => {\n  const filterItemsTemplate = filterItems.map((filter, index) => createFilterItemTemplate(filter, index === 0)).join('');\n  return `<form class=\"trip-filters\" action=\"#\" method=\"get\">\n  ${filterItemsTemplate}\n  <button class=\"visually-hidden\" type=\"submit\">Accept filter</button></form>`;\n};\nexport default class NewTripFiltersView extends AbstractView {\n  #filters = null;\n  constructor(filters) {\n    super();\n    this.#filters = filters;\n  }\n  get template() {\n    return createNewTripFiltersTemplate(this.#filters.filters);\n  }\n}","import AbstractView from '../framework/view/abstract-view';\nimport { SortType } from '../const';\nfunction createNewTripSortTemplate() {\n  return `<form class=\"trip-events__trip-sort  trip-sort\" action=\"#\" method=\"get\">\n            <div class=\"trip-sort__item  trip-sort__item--day\">\n              <input id=\"sort-day\" class=\"trip-sort__input  visually-hidden\" type=\"radio\" name=\"trip-sort\" value=\"sort-day\" checked>\n              <label class=\"trip-sort__btn\" for=\"sort-day\" data-sort-type=\"${SortType.DEFAULT}\">Day</label>\n            </div>\n\n            <div class=\"trip-sort__item  trip-sort__item--event\">\n              <input id=\"sort-event\" class=\"trip-sort__input  visually-hidden\" type=\"radio\" name=\"trip-sort\" value=\"sort-event\" disabled>\n              <label class=\"trip-sort__btn\" for=\"sort-event\">Event</label>\n            </div>\n\n            <div class=\"trip-sort__item  trip-sort__item--time\">\n              <input id=\"sort-time\" class=\"trip-sort__input  visually-hidden\" type=\"radio\" name=\"trip-sort\" value=\"sort-time\">\n              <label class=\"trip-sort__btn\" for=\"sort-time\" data-sort-type=\"${SortType.TIME}\">Time</label>\n            </div>\n\n            <div class=\"trip-sort__item  trip-sort__item--price\">\n              <input id=\"sort-price\" class=\"trip-sort__input  visually-hidden\" type=\"radio\" name=\"trip-sort\" value=\"sort-price\">\n              <label class=\"trip-sort__btn\" for=\"sort-price\" data-sort-type=\"${SortType.PRICE}\">Price</label>\n            </div>\n\n            <div class=\"trip-sort__item  trip-sort__item--offer\">\n              <input id=\"sort-offer\" class=\"trip-sort__input  visually-hidden\" type=\"radio\" name=\"trip-sort\" value=\"sort-offer\" disabled>\n              <label class=\"trip-sort__btn\" for=\"sort-offer\">Offers</label>\n            </div>\n          </form>`;\n}\nexport default class NewTripSortView extends AbstractView {\n  #handleSortTypeChange = null;\n  constructor({\n    onSortTypeChange\n  }) {\n    super();\n    this.#handleSortTypeChange = onSortTypeChange;\n    this.element.addEventListener('click', this.#sortTypeChangeHandler);\n  }\n  get template() {\n    return createNewTripSortTemplate();\n  }\n  #sortTypeChangeHandler = evt => {\n    if (evt.target.tagName !== 'LABEL') {\n      return;\n    }\n    evt.preventDefault();\n    this.#handleSortTypeChange(evt.target.dataset.sortType);\n  };\n}","import AbstractView from '../framework/view/abstract-view';\nfunction createNewTripEventsListTemplate() {\n  return '<ul class=\"trip-events__list\"></ul>';\n}\nexport default class NewEventsListView extends AbstractView {\n  get template() {\n    return createNewTripEventsListTemplate();\n  }\n}","import dayjs from 'dayjs';\nconst humanizeDueDate = (dueDate, dateFormat) => dueDate ? dayjs(dueDate).format(dateFormat) : '';\nconst isEventFuture = dateFrom => dateFrom && dayjs().isAfter(dateFrom, 'D');\nconst isEventPresent = (dateFrom, dateTo) => {\n  const now = dayjs();\n  return dateFrom && dateTo && now.isAfter(dateFrom, 'D') && now.isBefore(dateTo, 'D') || now.isSame(dateTo, 'D');\n};\nconst isEventPast = dateTo => dateTo && dayjs().isAfter(dateTo, 'D');\nconst sortEventsPrice = (eventA, eventB) => eventB.basePrice - eventA.basePrice;\nconst sortEventsTime = (eventA, eventB) => {\n  const dateFromA = dayjs(eventA.dateFrom);\n  const dateToA = dayjs(eventA.dateTo);\n  const dateFromB = dayjs(eventB.dateFrom);\n  const dateToB = dayjs(eventB.dateTo);\n  const durationA = dateToA.diff(dateFromA);\n  const durationB = dateToB.diff(dateFromB);\n  return durationB - durationA;\n};\nexport { humanizeDueDate, isEventFuture, isEventPresent, isEventPast, sortEventsPrice, sortEventsTime };","import dayjs from 'dayjs';\nimport AbstractView from '../framework/view/abstract-view';\nimport { humanizeDueDate } from '../utils/event';\nconst TIME_PATTERN = 'hh:mm';\nconst HUMANIZED_EVENT_DATE_PATTERN = 'MMM DD';\nconst getTimeDifference = (timeStart, timeEnd) => {\n  const start = dayjs(timeStart);\n  const end = dayjs(timeEnd);\n  const differenceInDays = end.diff(start, 'day');\n  const differenceInHours = end.diff(start, 'hour') % 24;\n  const differenceInMinutes = end.diff(start, 'minute') % 60;\n  if (differenceInDays > 0) {\n    return `${String(differenceInDays).padStart(2, '0')}D ${String(differenceInHours).padStart(2, '0')}H ${String(differenceInMinutes).padStart(2, '0')}M`;\n  } else if (differenceInHours > 0) {\n    return `${String(differenceInHours).padStart(2, '0')}H ${String(differenceInMinutes).padStart(2, '0')}M`;\n  } else {\n    return `${String(differenceInMinutes).padStart(2, '0')}M`;\n  }\n};\nconst createNewOffer = offer => {\n  const {\n    title,\n    price\n  } = offer;\n  return `<li class=\"event__offer\">\n                    <span class=\"event__offer-title\">${title}</span>\n                    &plus;&euro;&nbsp;\n                    <span class=\"event__offer-price\">${price}</span>\n                  </li>`;\n};\nconst createOffers = offers => {\n  let offersHTML = '';\n  offers.forEach(offer => {\n    offersHTML += createNewOffer(offer);\n  });\n  return offersHTML;\n};\nconst createNewTripEventsItemTemplate = eventData => {\n  const {\n    basePrice,\n    dateFrom,\n    dateTo,\n    isFavorite,\n    type,\n    offers,\n    destination\n  } = eventData;\n  const favoriteButtonClassName = isFavorite ? 'event__favorite-btn--active' : '';\n  const humanizedEventDate = humanizeDueDate(dateFrom, HUMANIZED_EVENT_DATE_PATTERN);\n  const humanizedStartTime = humanizeDueDate(dateFrom, TIME_PATTERN);\n  const humanizedEndTime = humanizeDueDate(dateTo, TIME_PATTERN);\n  return `<li class=\"trip-events__item\">\n              <div class=\"event\">\n                <time class=\"event__date\" datetime=\"${dateFrom}\">${humanizedEventDate}</time>\n                <div class=\"event__type\">\n                  <img class=\"event__type-icon\" width=\"42\" height=\"42\" src=\"img/icons/${type}.png\" alt=\"Event type icon\">\n                </div>\n                <h3 class=\"event__title\">${type} ${destination.name}</h3>\n                <div class=\"event__schedule\">\n                  <p class=\"event__time\">\n                    <time class=\"event__start-time\" datetime=\"${dateFrom}\">${humanizedStartTime}</time>\n                    &mdash;\n                    <time class=\"event__end-time\" datetime=\"${dateTo}\">${humanizedEndTime}</time>\n                  </p>\n                  <p class=\"event__duration\">${getTimeDifference(dateFrom, dateTo)}</p>\n                </div>\n                <p class=\"event__price\">\n                  &euro;&nbsp;<span class=\"event__price-value\">${basePrice}</span>\n                </p>\n                <h4 class=\"visually-hidden\">Offers:</h4>\n                <ul class=\"event__selected-offers\">\n                ${createOffers(offers)}\n                </ul>\n                <button class=\"event__favorite-btn ${favoriteButtonClassName}\" type=\"button\">\n                  <span class=\"visually-hidden\">Add to favorite</span>\n                  <svg class=\"event__favorite-icon\" width=\"28\" height=\"28\" viewBox=\"0 0 28 28\">\n                    <path d=\"M14 21l-8.22899 4.3262 1.57159-9.1631L.685209 9.67376 9.8855 8.33688 14 0l4.1145 8.33688 9.2003 1.33688-6.6574 6.48934 1.5716 9.1631L14 21z\"/>\n                  </svg>\n                </button>\n\n                <button class=\"event__rollup-btn\" type=\"button\">\n                  <span class=\"visually-hidden\">Open event</span>\n                </button>\n\n              </div>\n            </li>`;\n};\nexport default class NewEventsItemView extends AbstractView {\n  #eventData = null;\n  #handleClick = null;\n  #rollupButton = null;\n  #handleFavoriteClick = null;\n  constructor({\n    userEvent,\n    onClick,\n    onFavoriteClick\n  }) {\n    super();\n    this.#eventData = userEvent;\n    this.#handleClick = onClick;\n    this.#handleFavoriteClick = onFavoriteClick;\n    this.#rollupButton = this.element.querySelector('.event__rollup-btn');\n    this.#rollupButton.addEventListener('click', this.#clickHandler);\n    this.element.querySelector('.event__favorite-icon').addEventListener('click', this.#favoriteClickHandler);\n  }\n  get template() {\n    return createNewTripEventsItemTemplate(this.#eventData);\n  }\n  #clickHandler = evt => {\n    evt.preventDefault();\n    this.#handleClick();\n  };\n  removeEventListeners() {\n    this.#rollupButton.removeEventListener('click', this.#clickHandler);\n  }\n  #favoriteClickHandler = evt => {\n    evt.preventDefault();\n    this.#handleFavoriteClick();\n  };\n}","import AbstractView from '../framework/view/abstract-view';\nimport { humanizeDueDate } from '../utils/event';\nconst TIME_PATTERN = 'DD/MM/YY hh:mm';\nconst createNewOffer = offer => {\n  const {\n    title,\n    price\n  } = offer;\n  return `<div class=\"event__offer-selector\">\n                        <input class=\"event__offer-checkbox  visually-hidden\" id=\"event-offer-luggage-1\" type=\"checkbox\" name=\"event-offer-luggage\" checked>\n                        <label class=\"event__offer-label\" for=\"event-offer-luggage-1\">\n                          <span class=\"event__offer-title\">${title}</span>\n                          &plus;&euro;&nbsp;\n                          <span class=\"event__offer-price\">${price}</span>\n                        </label>\n                      </div>`;\n};\nconst createOffers = offers => {\n  let offersHTML = '';\n  offers.forEach(offer => {\n    offersHTML += createNewOffer(offer);\n  });\n  return offersHTML;\n};\nconst createNewEventEditElementTemplate = eventData => {\n  const {\n    basePrice,\n    dateFrom,\n    dateTo,\n    type,\n    offers,\n    destination\n  } = eventData;\n  const dateStart = humanizeDueDate(dateFrom, TIME_PATTERN);\n  const dateEnd = humanizeDueDate(dateTo, TIME_PATTERN);\n  return `<li class=\"trip-events__item\">\n              <form class=\"event event--edit\" action=\"#\" method=\"post\">\n                <header class=\"event__header\">\n                  <div class=\"event__type-wrapper\">\n                    <label class=\"event__type  event__type-btn\" for=\"event-type-toggle-1\">\n                      <span class=\"visually-hidden\">Choose event type</span>\n                      <img class=\"event__type-icon\" width=\"17\" height=\"17\" src=\"img/icons/${type}.png\" alt=\"Event type icon\">\n                    </label>\n                    <input class=\"event__type-toggle  visually-hidden\" id=\"event-type-toggle-1\" type=\"checkbox\">\n\n                    <div class=\"event__type-list\">\n                      <fieldset class=\"event__type-group\">\n                        <legend class=\"visually-hidden\">Event type</legend>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-taxi-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"taxi\">\n                          <label class=\"event__type-label  event__type-label--taxi\" for=\"event-type-taxi-1\">Taxi</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-bus-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"bus\">\n                          <label class=\"event__type-label  event__type-label--bus\" for=\"event-type-bus-1\">Bus</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-train-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"train\">\n                          <label class=\"event__type-label  event__type-label--train\" for=\"event-type-train-1\">Train</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-ship-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"ship\">\n                          <label class=\"event__type-label  event__type-label--ship\" for=\"event-type-ship-1\">Ship</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-drive-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"drive\">\n                          <label class=\"event__type-label  event__type-label--drive\" for=\"event-type-drive-1\">Drive</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-flight-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"flight\" checked>\n                          <label class=\"event__type-label  event__type-label--flight\" for=\"event-type-flight-1\">Flight</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-check-in-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"check-in\">\n                          <label class=\"event__type-label  event__type-label--check-in\" for=\"event-type-check-in-1\">Check-in</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-sightseeing-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"sightseeing\">\n                          <label class=\"event__type-label  event__type-label--sightseeing\" for=\"event-type-sightseeing-1\">Sightseeing</label>\n                        </div>\n\n                        <div class=\"event__type-item\">\n                          <input id=\"event-type-restaurant-1\" class=\"event__type-input  visually-hidden\" type=\"radio\" name=\"event-type\" value=\"restaurant\">\n                          <label class=\"event__type-label  event__type-label--restaurant\" for=\"event-type-restaurant-1\">Restaurant</label>\n                        </div>\n                      </fieldset>\n                    </div>\n                  </div>\n\n                  <div class=\"event__field-group  event__field-group--destination\">\n                    <label class=\"event__label  event__type-output\" for=\"event-destination-1\">\n                      ${type}\n                    </label>\n                    <input class=\"event__input  event__input--destination\" id=\"event-destination-1\" type=\"text\" name=\"event-destination\" value=\"${destination.name}\" list=\"destination-list-1\">\n                    <datalist id=\"destination-list-1\">\n                      <option value=\"Amsterdam\"></option>\n                      <option value=\"Geneva\"></option>\n                      <option value=\"Chamonix\"></option>\n                    </datalist>\n                  </div>\n\n                  <div class=\"event__field-group  event__field-group--time\">\n                    <label class=\"visually-hidden\" for=\"event-start-time-1\">From</label>\n                    <input class=\"event__input  event__input--time\" id=\"event-start-time-1\" type=\"text\" name=\"event-start-time\" value=\"${dateStart}\">\n                    &mdash;\n                    <label class=\"visually-hidden\" for=\"event-end-time-1\">To</label>\n                    <input class=\"event__input  event__input--time\" id=\"event-end-time-1\" type=\"text\" name=\"event-end-time\" value=\"${dateEnd}\">\n                  </div>\n\n                  <div class=\"event__field-group  event__field-group--price\">\n                    <label class=\"event__label\" for=\"event-price-1\">\n                      <span class=\"visually-hidden\">Price</span>\n                      &euro;\n                    </label>\n                    <input class=\"event__input  event__input--price\" id=\"event-price-1\" type=\"text\" name=\"event-price\" value=\"${basePrice}\">\n                  </div>\n\n                  <button class=\"event__save-btn  btn  btn--blue\" type=\"submit\">Save</button>\n                  <button class=\"event__reset-btn\" type=\"reset\">Delete</button>\n                  <button class=\"event__rollup-btn\" type=\"button\">\n                    <span class=\"visually-hidden\">Open event</span>\n                  </button>\n                </header>\n                <section class=\"event__details\">\n                  <section class=\"event__section  event__section--offers\">\n                    <h3 class=\"event__section-title  event__section-title--offers\">Offers</h3>\n\n                    <div class=\"event__available-offers\">\n                    ${createOffers(offers)}\n                    </div>\n                  </section>\n\n                  <section class=\"event__section  event__section--destination\">\n                    <h3 class=\"event__section-title  event__section-title--destination\">Destination</h3>\n                    <p class=\"event__destination-description\">${destination.description}</p>\n                  </section>\n                </section>\n              </form>\n            </li>`;\n};\nexport default class NewEventEditElementView extends AbstractView {\n  #eventData = null;\n  #handleClick = null;\n  #rollupButton = null;\n  #handleSubmit = null;\n  #formElement = null;\n  constructor({\n    userEvent,\n    onClick,\n    onSubmit\n  }) {\n    super();\n    this.#eventData = userEvent;\n    this.#handleClick = onClick;\n    this.#handleSubmit = onSubmit;\n    this.#rollupButton = this.element.querySelector('.event__rollup-btn');\n    this.#rollupButton.addEventListener('click', this.#clickHandler);\n    this.#formElement = this.element.querySelector('.event--edit');\n    this.#formElement.addEventListener('submit', this.#submitHandler);\n  }\n  get template() {\n    return createNewEventEditElementTemplate(this.#eventData);\n  }\n  #clickHandler = evt => {\n    evt.preventDefault();\n    this.#handleClick(this.#eventData);\n  };\n  #submitHandler = evt => {\n    evt.preventDefault();\n    this.#handleClick(this.#eventData);\n  };\n  removeEventListeners() {\n    this.#rollupButton.removeEventListener('click', this.#clickHandler);\n    this.#formElement.removeEventListener('submit', this.#submitHandler);\n  }\n}","import { render, replace, remove } from '../framework/render.js';\nimport { isEscapeKey } from '../utils/common.js';\nimport NewEventsItemView from '../view/new-events-item-view.js';\nimport NewEventEditElementView from '../view/new-event-edit-element-view.js';\nconst Mode = {\n  DEFAULT: 'DEFAULT',\n  EDITING: 'EDITING'\n};\nexport default class EventPresenter {\n  #container = null;\n  #handleDataChange = null;\n  #handleModeChange = null;\n  #eventComponent = null;\n  #eventEditComponent = null;\n  #eventItem = null;\n  #mode = Mode.DEFAULT;\n  constructor({\n    container,\n    onDataChange,\n    onModeChange\n  }) {\n    this.#container = container;\n    this.#handleDataChange = onDataChange;\n    this.#handleModeChange = onModeChange;\n  }\n  init(eventItem) {\n    this.#eventItem = eventItem;\n    const prevEventComponent = this.#eventComponent;\n    const prevEventEditComponent = this.#eventEditComponent;\n    this.#eventComponent = new NewEventsItemView({\n      userEvent: this.#eventItem,\n      onClick: this.#handleEditClick,\n      onFavoriteClick: this.#handleFavoriteClick\n    });\n    this.#eventEditComponent = new NewEventEditElementView({\n      userEvent: this.#eventItem,\n      onClick: this.#handleSaveClick\n    });\n    if (prevEventComponent === null || prevEventEditComponent === null) {\n      render(this.#eventComponent, this.#container);\n      return;\n    }\n    if (this.#mode === Mode.DEFAULT) {\n      replace(this.#eventComponent, prevEventComponent);\n    }\n    if (this.#mode === Mode.EDITING) {\n      replace(this.#eventEditComponent, prevEventEditComponent);\n    }\n    remove(prevEventComponent);\n    remove(prevEventEditComponent);\n  }\n  #replaceEventCardToEditForm() {\n    replace(this.#eventEditComponent, this.#eventComponent);\n    document.addEventListener('keydown', this.#escKeyDownHandler);\n    this.#handleModeChange();\n    this.#mode = Mode.EDITING;\n  }\n  #replaceEditFormToEventCard() {\n    replace(this.#eventComponent, this.#eventEditComponent);\n    document.removeEventListener('keydown', this.#escKeyDownHandler);\n    this.#mode = Mode.DEFAULT;\n  }\n  destroy() {\n    remove(this.#eventComponent);\n    remove(this.#eventEditComponent);\n  }\n  resetView() {\n    if (this.#mode !== Mode.DEFAULT) {\n      this.#replaceEditFormToEventCard();\n    }\n  }\n  #escKeyDownHandler = evt => {\n    if (isEscapeKey(evt)) {\n      evt.preventDefault();\n      this.#replaceEditFormToEventCard();\n    }\n  };\n  #handleEditClick = () => {\n    this.#replaceEventCardToEditForm();\n  };\n  #handleFavoriteClick = () => {\n    this.#handleDataChange({\n      ...this.#eventItem,\n      isFavorite: !this.#eventItem.isFavorite\n    });\n  };\n  #handleSaveClick = eventItem => {\n    this.#handleDataChange(eventItem);\n    this.#replaceEditFormToEventCard();\n  };\n}","import AbstractView from '../framework/view/abstract-view';\nfunction createNoEventsViewTemplate() {\n  return '<p class=\"trip-events__msg\">Loading...</p>';\n}\nexport default class NoEventsView extends AbstractView {\n  get template() {\n    return createNoEventsViewTemplate();\n  }\n}","import { render } from '../framework/render.js';\nimport { updateItem } from '../utils/common.js';\nimport NewTripSortView from '../view/new-sort-container-view.js';\nimport NewEventsListView from '../view/new-events-list-view.js';\nimport EventPresenter from './event-presenter.js';\nimport NoEventsView from '../view/no-events-view.js';\nimport { SortType } from '../const.js';\nimport { sortEventsPrice, sortEventsTime } from '../utils/event.js';\nexport default class BoardPresenter {\n  #container = null;\n  #eventsModel = null;\n  #sortComponent = null;\n  #eventsListComponent = new NewEventsListView();\n  #eventsList = [];\n  #eventPresenters = new Map();\n  #currentSortType = SortType.DEFAULT;\n  #sourcedBoardEvents = [];\n  constructor({\n    container,\n    eventsModel\n  }) {\n    this.#container = container;\n    this.#eventsModel = eventsModel;\n  }\n  init() {\n    this.#eventsList = [...this.#eventsModel.userEvents];\n    this.#sourcedBoardEvents = [...this.#eventsModel.userEvents];\n    this.#renderBoard();\n    this.#renderSort();\n  }\n  #renderEvent(inputUserEvent) {\n    const eventPresenter = new EventPresenter({\n      container: this.#eventsListComponent.element,\n      onDataChange: this.#handleEventChange,\n      onModeChange: this.#handleModeChange\n    });\n    eventPresenter.init(inputUserEvent);\n    this.#eventPresenters.set(inputUserEvent.id, eventPresenter);\n  }\n  #renderNoEvents() {\n    render(new NoEventsView(), this.#container);\n  }\n  #sortEvents(sortType) {\n    switch (sortType) {\n      case SortType.PRICE:\n        this.#eventsList.sort(sortEventsPrice);\n        break;\n      case SortType.TIME:\n        this.#eventsList.sort(sortEventsTime);\n        break;\n      default:\n        this.#eventsList = [...this.#sourcedBoardEvents];\n    }\n    this.#currentSortType = sortType;\n  }\n  #handleSortTypeChange = sortType => {\n    if (sortType === undefined || this.#currentSortType === sortType) {\n      return;\n    }\n    this.#sortEvents(sortType);\n    this.#clearEventList();\n    this.#renderBoard();\n  };\n  #renderSort() {\n    this.#sortComponent = new NewTripSortView({\n      onSortTypeChange: this.#handleSortTypeChange\n    });\n    render(this.#sortComponent, this.#container, 'AFTERBEGIN');\n  }\n  #renderBoard() {\n    if (this.#eventsList.length === 0) {\n      this.#renderNoEvents();\n      return;\n    }\n    render(this.#eventsListComponent, this.#container);\n    for (let i = 0; i < this.#eventsList.length; i++) {\n      this.#renderEvent(this.#eventsList[i]);\n    }\n  }\n  #handleEventChange = updatedEvent => {\n    this.#eventsList = updateItem(this.#eventsList, updatedEvent);\n    this.#sourcedBoardEvents = updateItem(this.#sourcedBoardEvents, updatedEvent);\n    this.#eventPresenters.get(updatedEvent.id).init(updatedEvent);\n  };\n  #handleModeChange = () => {\n    this.#eventPresenters.forEach(presenter => presenter.resetView());\n  };\n  #clearEventList() {\n    this.#eventPresenters.forEach(presenter => presenter.destroy());\n    this.#eventPresenters.clear();\n  }\n}","import { FilterType } from '../const';\nimport { isEventFuture, isEventPresent, isEventPast } from './event';\nconst filter = {\n  [FilterType.EVERYTHING]: userEvents => userEvents,\n  [FilterType.FUTURE]: userEvents => userEvents.filter(userEvent => isEventFuture(userEvent.dateFrom)),\n  [FilterType.PRESENT]: userEvents => userEvents.filter(userEvent => isEventPresent(userEvent.dateFrom)),\n  [FilterType.PAST]: userEvents => userEvents.filter(userEvent => isEventPast(userEvent.dateFrom))\n};\nexport { filter };","import { filter } from '../utils/filter.js';\nfunction generateFilter(userEvents) {\n  return Object.entries(filter).map(([filterType, filterEvents]) => ({\n    type: filterType,\n    count: filterEvents(userEvents).length\n  }));\n}\nexport { generateFilter };","import { render } from './framework/render.js';\nimport EventsModel from './model/events-model.js';\nimport NewTripInfoView from './view/new-trip-info-view.js';\nimport NewTripFiltersView from './view/new-filters-view.js';\nimport BoardPresenter from './presenter/board-presenter.js';\nimport { generateFilter } from './mock/filter.js';\nconst tripMainContainer = document.querySelector('.trip-main');\nconst tripFiltersContainer = tripMainContainer.querySelector('.trip-controls__filters');\nconst tripEventsContainer = document.querySelector('.trip-events');\nconst eventsModel = new EventsModel();\nconst boardPresenter = new BoardPresenter({\n  container: tripEventsContainer,\n  eventsModel\n});\nconst filters = generateFilter(eventsModel.userEvents);\nrender(new NewTripInfoView(), tripMainContainer, 'AFTERBEGIN');\nrender(new NewTripFiltersView({\n  filters\n}), tripFiltersContainer);\nboardPresenter.init();"],"names":[],"sourceRoot":""}\n//# sourceURL=webpack-internal:///89\n')},10:(module,__webpack_exports__,__webpack_require__)=>{"use strict";eval('/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ "Z": () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_noSourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(81);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_noSourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_noSourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(645);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_noSourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, ".shake {\\n animation: shake 0.6s;\\n position: relative;\\n z-index: 10;\\n}\\n\\n@keyframes shake {\\n 0%,\\n 100% {\\n transform: translateX(0);\\n }\\n\\n 10%,\\n 30%,\\n 50%,\\n 70%,\\n 90% {\\n transform: translateX(-5px);\\n }\\n\\n 20%,\\n 40%,\\n 60%,\\n 80% {\\n transform: translateX(5px);\\n }\\n}\\n", ""]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiMTAuanMiLCJtYXBwaW5ncyI6Ijs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9iaWctdHJpcC8uL3NyYy9mcmFtZXdvcmsvdmlldy9hYnN0cmFjdC12aWV3LmNzcz9kZjNkIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEltcG9ydHNcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9OT19TT1VSQ0VNQVBfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvbm9Tb3VyY2VNYXBzLmpzXCI7XG5pbXBvcnQgX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fIGZyb20gXCIuLi8uLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzXCI7XG52YXIgX19fQ1NTX0xPQURFUl9FWFBPUlRfX18gPSBfX19DU1NfTE9BREVSX0FQSV9JTVBPUlRfX18oX19fQ1NTX0xPQURFUl9BUElfTk9fU09VUkNFTUFQX0lNUE9SVF9fXyk7XG4vLyBNb2R1bGVcbl9fX0NTU19MT0FERVJfRVhQT1JUX19fLnB1c2goW21vZHVsZS5pZCwgXCIuc2hha2Uge1xcbiAgYW5pbWF0aW9uOiBzaGFrZSAwLjZzO1xcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xcbiAgei1pbmRleDogMTA7XFxufVxcblxcbkBrZXlmcmFtZXMgc2hha2Uge1xcbiAgMCUsXFxuICAxMDAlIHtcXG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKDApO1xcbiAgfVxcblxcbiAgMTAlLFxcbiAgMzAlLFxcbiAgNTAlLFxcbiAgNzAlLFxcbiAgOTAlIHtcXG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKC01cHgpO1xcbiAgfVxcblxcbiAgMjAlLFxcbiAgNDAlLFxcbiAgNjAlLFxcbiAgODAlIHtcXG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKDVweCk7XFxuICB9XFxufVxcblwiLCBcIlwiXSk7XG4vLyBFeHBvcnRzXG5leHBvcnQgZGVmYXVsdCBfX19DU1NfTE9BREVSX0VYUE9SVF9fXztcbiJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///10\n')},645:module=>{"use strict";eval('\n\n/*\n MIT License http://www.opensource.org/licenses/mit-license.php\n Author Tobias Koppers @sokra\n*/\nmodule.exports = function (cssWithMappingToString) {\n var list = [];\n\n // return the list of modules as css string\n list.toString = function toString() {\n return this.map(function (item) {\n var content = "";\n var needLayer = typeof item[5] !== "undefined";\n if (item[4]) {\n content += "@supports (".concat(item[4], ") {");\n }\n if (item[2]) {\n content += "@media ".concat(item[2], " {");\n }\n if (needLayer) {\n content += "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {");\n }\n content += cssWithMappingToString(item);\n if (needLayer) {\n content += "}";\n }\n if (item[2]) {\n content += "}";\n }\n if (item[4]) {\n content += "}";\n }\n return content;\n }).join("");\n };\n\n // import a list of modules into the list\n list.i = function i(modules, media, dedupe, supports, layer) {\n if (typeof modules === "string") {\n modules = [[null, modules, undefined]];\n }\n var alreadyImportedModules = {};\n if (dedupe) {\n for (var k = 0; k < this.length; k++) {\n var id = this[k][0];\n if (id != null) {\n alreadyImportedModules[id] = true;\n }\n }\n }\n for (var _k = 0; _k < modules.length; _k++) {\n var item = [].concat(modules[_k]);\n if (dedupe && alreadyImportedModules[item[0]]) {\n continue;\n }\n if (typeof layer !== "undefined") {\n if (typeof item[5] === "undefined") {\n item[5] = layer;\n } else {\n item[1] = "@layer".concat(item[5].length > 0 ? " ".concat(item[5]) : "", " {").concat(item[1], "}");\n item[5] = layer;\n }\n }\n if (media) {\n if (!item[2]) {\n item[2] = media;\n } else {\n item[1] = "@media ".concat(item[2], " {").concat(item[1], "}");\n item[2] = media;\n }\n }\n if (supports) {\n if (!item[4]) {\n item[4] = "".concat(supports);\n } else {\n item[1] = "@supports (".concat(item[4], ") {").concat(item[1], "}");\n item[4] = supports;\n }\n }\n list.push(item);\n }\n };\n return list;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiNjQ1LmpzIiwibWFwcGluZ3MiOiJBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYmlnLXRyaXAvLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzPzI0ZmIiXSwic291cmNlc0NvbnRlbnQiOlsiXCJ1c2Ugc3RyaWN0XCI7XG5cbi8qXG4gIE1JVCBMaWNlbnNlIGh0dHA6Ly93d3cub3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvbWl0LWxpY2Vuc2UucGhwXG4gIEF1dGhvciBUb2JpYXMgS29wcGVycyBAc29rcmFcbiovXG5tb2R1bGUuZXhwb3J0cyA9IGZ1bmN0aW9uIChjc3NXaXRoTWFwcGluZ1RvU3RyaW5nKSB7XG4gIHZhciBsaXN0ID0gW107XG5cbiAgLy8gcmV0dXJuIHRoZSBsaXN0IG9mIG1vZHVsZXMgYXMgY3NzIHN0cmluZ1xuICBsaXN0LnRvU3RyaW5nID0gZnVuY3Rpb24gdG9TdHJpbmcoKSB7XG4gICAgcmV0dXJuIHRoaXMubWFwKGZ1bmN0aW9uIChpdGVtKSB7XG4gICAgICB2YXIgY29udGVudCA9IFwiXCI7XG4gICAgICB2YXIgbmVlZExheWVyID0gdHlwZW9mIGl0ZW1bNV0gIT09IFwidW5kZWZpbmVkXCI7XG4gICAgICBpZiAoaXRlbVs0XSkge1xuICAgICAgICBjb250ZW50ICs9IFwiQHN1cHBvcnRzIChcIi5jb25jYXQoaXRlbVs0XSwgXCIpIHtcIik7XG4gICAgICB9XG4gICAgICBpZiAoaXRlbVsyXSkge1xuICAgICAgICBjb250ZW50ICs9IFwiQG1lZGlhIFwiLmNvbmNhdChpdGVtWzJdLCBcIiB7XCIpO1xuICAgICAgfVxuICAgICAgaWYgKG5lZWRMYXllcikge1xuICAgICAgICBjb250ZW50ICs9IFwiQGxheWVyXCIuY29uY2F0KGl0ZW1bNV0ubGVuZ3RoID4gMCA/IFwiIFwiLmNvbmNhdChpdGVtWzVdKSA6IFwiXCIsIFwiIHtcIik7XG4gICAgICB9XG4gICAgICBjb250ZW50ICs9IGNzc1dpdGhNYXBwaW5nVG9TdHJpbmcoaXRlbSk7XG4gICAgICBpZiAobmVlZExheWVyKSB7XG4gICAgICAgIGNvbnRlbnQgKz0gXCJ9XCI7XG4gICAgICB9XG4gICAgICBpZiAoaXRlbVsyXSkge1xuICAgICAgICBjb250ZW50ICs9IFwifVwiO1xuICAgICAgfVxuICAgICAgaWYgKGl0ZW1bNF0pIHtcbiAgICAgICAgY29udGVudCArPSBcIn1cIjtcbiAgICAgIH1cbiAgICAgIHJldHVybiBjb250ZW50O1xuICAgIH0pLmpvaW4oXCJcIik7XG4gIH07XG5cbiAgLy8gaW1wb3J0IGEgbGlzdCBvZiBtb2R1bGVzIGludG8gdGhlIGxpc3RcbiAgbGlzdC5pID0gZnVuY3Rpb24gaShtb2R1bGVzLCBtZWRpYSwgZGVkdXBlLCBzdXBwb3J0cywgbGF5ZXIpIHtcbiAgICBpZiAodHlwZW9mIG1vZHVsZXMgPT09IFwic3RyaW5nXCIpIHtcbiAgICAgIG1vZHVsZXMgPSBbW251bGwsIG1vZHVsZXMsIHVuZGVmaW5lZF1dO1xuICAgIH1cbiAgICB2YXIgYWxyZWFkeUltcG9ydGVkTW9kdWxlcyA9IHt9O1xuICAgIGlmIChkZWR1cGUpIHtcbiAgICAgIGZvciAodmFyIGsgPSAwOyBrIDwgdGhpcy5sZW5ndGg7IGsrKykge1xuICAgICAgICB2YXIgaWQgPSB0aGlzW2tdWzBdO1xuICAgICAgICBpZiAoaWQgIT0gbnVsbCkge1xuICAgICAgICAgIGFscmVhZHlJbXBvcnRlZE1vZHVsZXNbaWRdID0gdHJ1ZTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgICBmb3IgKHZhciBfayA9IDA7IF9rIDwgbW9kdWxlcy5sZW5ndGg7IF9rKyspIHtcbiAgICAgIHZhciBpdGVtID0gW10uY29uY2F0KG1vZHVsZXNbX2tdKTtcbiAgICAgIGlmIChkZWR1cGUgJiYgYWxyZWFkeUltcG9ydGVkTW9kdWxlc1tpdGVtWzBdXSkge1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cbiAgICAgIGlmICh0eXBlb2YgbGF5ZXIgIT09IFwidW5kZWZpbmVkXCIpIHtcbiAgICAgICAgaWYgKHR5cGVvZiBpdGVtWzVdID09PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgICAgICAgaXRlbVs1XSA9IGxheWVyO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGl0ZW1bMV0gPSBcIkBsYXllclwiLmNvbmNhdChpdGVtWzVdLmxlbmd0aCA+IDAgPyBcIiBcIi5jb25jYXQoaXRlbVs1XSkgOiBcIlwiLCBcIiB7XCIpLmNvbmNhdChpdGVtWzFdLCBcIn1cIik7XG4gICAgICAgICAgaXRlbVs1XSA9IGxheWVyO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBpZiAobWVkaWEpIHtcbiAgICAgICAgaWYgKCFpdGVtWzJdKSB7XG4gICAgICAgICAgaXRlbVsyXSA9IG1lZGlhO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGl0ZW1bMV0gPSBcIkBtZWRpYSBcIi5jb25jYXQoaXRlbVsyXSwgXCIge1wiKS5jb25jYXQoaXRlbVsxXSwgXCJ9XCIpO1xuICAgICAgICAgIGl0ZW1bMl0gPSBtZWRpYTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgaWYgKHN1cHBvcnRzKSB7XG4gICAgICAgIGlmICghaXRlbVs0XSkge1xuICAgICAgICAgIGl0ZW1bNF0gPSBcIlwiLmNvbmNhdChzdXBwb3J0cyk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgaXRlbVsxXSA9IFwiQHN1cHBvcnRzIChcIi5jb25jYXQoaXRlbVs0XSwgXCIpIHtcIikuY29uY2F0KGl0ZW1bMV0sIFwifVwiKTtcbiAgICAgICAgICBpdGVtWzRdID0gc3VwcG9ydHM7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIGxpc3QucHVzaChpdGVtKTtcbiAgICB9XG4gIH07XG4gIHJldHVybiBsaXN0O1xufTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///645\n')},81:module=>{"use strict";eval("\n\nmodule.exports = function (i) {\n return i[1];\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiODEuanMiLCJtYXBwaW5ncyI6IkFBQUE7QUFDQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2JpZy10cmlwLy4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL25vU291cmNlTWFwcy5qcz8yN2M4Il0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG5tb2R1bGUuZXhwb3J0cyA9IGZ1bmN0aW9uIChpKSB7XG4gIHJldHVybiBpWzFdO1xufTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///81\n")},484:function(module){eval('!function(t,e){ true?module.exports=e():0}(this,(function(){"use strict";var t=1e3,e=6e4,n=36e5,r="millisecond",i="second",s="minute",u="hour",a="day",o="week",f="month",h="quarter",c="year",d="date",l="Invalid Date",$=/^(\\d{4})[-/]?(\\d{1,2})?[-/]?(\\d{0,2})[Tt\\s]*(\\d{1,2})?:?(\\d{1,2})?:?(\\d{1,2})?[.:]?(\\d+)?$/,y=/\\[([^\\]]+)]|Y{1,4}|M{1,4}|D{1,2}|d{1,4}|H{1,2}|h{1,2}|a|A|m{1,2}|s{1,2}|Z{1,2}|SSS/g,M={name:"en",weekdays:"Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"),months:"January_February_March_April_May_June_July_August_September_October_November_December".split("_"),ordinal:function(t){var e=["th","st","nd","rd"],n=t%100;return"["+t+(e[(n-20)%10]||e[n]||e[0])+"]"}},m=function(t,e,n){var r=String(t);return!r||r.length>=e?t:""+Array(e+1-r.length).join(n)+t},v={s:m,z:function(t){var e=-t.utcOffset(),n=Math.abs(e),r=Math.floor(n/60),i=n%60;return(e<=0?"+":"-")+m(r,2,"0")+":"+m(i,2,"0")},m:function t(e,n){if(e.date()