temp.coffee 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. # обязательно подключение глобальных массивов
  2. globalThis.renderFns = require 'pug.json'
  3. globalThis.stylFns = require 'styl.json'
  4. # подключение мета информации (строго в данном фиде)
  5. document.head.insertAdjacentHTML 'beforeend','<meta charset="UTF-8">'
  6. document.head.insertAdjacentHTML 'beforeend','<meta name="viewport" content="width=device-width, initial-scale=1.0">'
  7. document.head.insertAdjacentHTML('beforeend','<title>Кохи Борбад - Концертный зал Душанбе</title>')
  8. # Настройка tailwind
  9. tailwind.config = require 'tailwind.config.js'
  10. # подключение основных стилей
  11. ## tailwind
  12. document.head.insertAdjacentHTML('beforeend','<style type="text/tailwindcss" page="main">'+stylFns['main.css']+'</style>')
  13. ## базовой стиль приложения
  14. document.head.insertAdjacentHTML('beforeend','<style type="text/tailwindcss" page="root">'+stylFns['app/temp.styl']+'</style>')
  15. # CouchDB сервис
  16. class CouchDBService
  17. constructor: ->
  18. @baseUrl = 'https://couchdb.favt.ru.net'
  19. @dbName = 'kohi_borbad_events'
  20. @headers =
  21. 'Content-Type': 'application/json'
  22. 'Authorization': 'Basic ' + btoa('oleg:631074')
  23. getAllEvents: ->
  24. try
  25. response = await fetch("#{@baseUrl}/#{@dbName}/_all_docs?include_docs=true", {
  26. method: 'GET'
  27. headers: @headers
  28. })
  29. if response.ok
  30. data = await response.json()
  31. events = data.rows.map (row) -> row.doc
  32. return events.filter (event) -> !event._id.startsWith('_design/')
  33. else
  34. debug.log "Ошибка получения мероприятий: "+response.statusText
  35. return []
  36. catch error
  37. debug.log "Ошибка подключения к CouchDB: "+error
  38. return []
  39. getFeaturedEvents: ->
  40. events = await @getAllEvents()
  41. events
  42. .filter (event) -> event.isFeatured || false
  43. .slice(0, 6)
  44. getSliderEvents: ->
  45. events = await @getAllEvents()
  46. events
  47. .filter (event) -> event.inSlider || false
  48. .map (event) ->
  49. id: event._id
  50. image: event.image || '/images/default-event.jpg'
  51. title: event.title
  52. description: event.shortDescription || event.description
  53. cta: event.cta || 'Подробнее'
  54. category: event.category
  55. # Маршруты
  56. routes = [
  57. { path: '/', component: require 'app/pages/Home' }
  58. ]
  59. components = {}
  60. routes = [
  61. { path: '/', component: require 'app/pages/Home' }
  62. { path: '/events', component: require 'app/pages/Events' }
  63. { path: '/about', component: require 'app/pages/About' }
  64. { path: '/contacts', component: require 'app/pages/Contacts' }
  65. ]
  66. components =
  67. 'themetoggle': require 'app/shared/ThemeToggle'
  68. 'multilevelmenu': require 'app/shared/MultiLevelMenu'
  69. 'imageslider': require 'app/shared/ImageSlider'
  70. 'modalwindow': require 'app/shared/ModalWindow'
  71. 'formvalidator': require 'app/shared/FormValidator'
  72. 'filtersort': require 'app/shared/FilterSort'
  73. 'eventdetailmodal': require 'app/shared/EventDetailModal'
  74. 'successmodal': require 'app/shared/SuccessModal'
  75. # Глобальное определение vuejs приложения
  76. app = Vue.createApp
  77. name: 'app'
  78. data: ()->
  79. return
  80. theme: 'light'
  81. appState:
  82. events: []
  83. featuredEvents: []
  84. sliderEvents: []
  85. loading: true
  86. error: null
  87. modalState:
  88. currentModal: null
  89. modalProps: {}
  90. couchDBService: new CouchDBService()
  91. beforeMount: ()->
  92. debug.log "start beforeMount"
  93. # определение контекста vuejs приложения как глобальной переменной _
  94. globalThis._ = @
  95. render: (new Function '_ctx', '_cache', renderFns['app/temp.pug'])()
  96. mounted: ->
  97. # Предзагрузка темы
  98. if localStorage.theme == 'dark' || (!('theme' in localStorage) && window.matchMedia('(prefers-color-scheme: dark)').matches)
  99. @theme = 'dark'
  100. document.documentElement.classList.add('dark')
  101. else
  102. @theme = 'light'
  103. document.documentElement.classList.remove('dark')
  104. # Загрузка данных из CouchDB
  105. await @loadEventsData()
  106. # Обработчик открытия модальных окон
  107. EventBus.$on 'open-modal', (config) =>
  108. @modalState.currentModal = config.component
  109. @modalState.modalProps = config.props || {}
  110. methods:
  111. toggleTheme: ->
  112. @theme = if @theme == 'light' then 'dark' else 'light'
  113. localStorage.setItem 'theme', @theme
  114. document.documentElement.classList.toggle 'dark'
  115. @$emit 'theme-changed', @theme
  116. loadEventsData: ->
  117. @appState.loading = true
  118. try
  119. [events, featuredEvents, sliderEvents] = await Promise.all([
  120. @couchDBService.getAllEvents()
  121. @couchDBService.getFeaturedEvents()
  122. @couchDBService.getSliderEvents()
  123. ])
  124. @appState.events = events
  125. @appState.featuredEvents = featuredEvents
  126. @appState.sliderEvents = sliderEvents
  127. @appState.error = null
  128. catch error
  129. debug.log "Ошибка загрузки данных: "+error
  130. @appState.error = 'Не удалось загрузить данные мероприятий'
  131. @loadTestData()
  132. finally
  133. @appState.loading = false
  134. loadTestData: ->
  135. # Тестовые данные на случай недоступности CouchDB
  136. @appState.events = [
  137. {
  138. _id: '1'
  139. title: 'Концерт симфонического оркестра'
  140. date: '2025-10-15'
  141. time: '19:00'
  142. description: 'Произведения Чайковского и Рахманинова в исполнении Национального симфонического оркестра'
  143. image: '/images/event-classical.jpg'
  144. category: 'classical'
  145. price: 50
  146. venue: 'Большой зал'
  147. duration: '2 часа 30 минут'
  148. ageRestriction: '12+'
  149. availableTickets: 45
  150. isFeatured: true
  151. inSlider: true
  152. shortDescription: 'Шедевры классической музыки'
  153. cta: 'Купить билеты'
  154. }
  155. {
  156. _id: '2'
  157. title: 'Вечер таджикской народной музыки'
  158. date: '2025-10-20'
  159. time: '18:30'
  160. description: 'Выступление фольклорного ансамбля "Шашмаком" с программой традиционных мелодий и танцев'
  161. image: '/images/event-folk.jpg'
  162. category: 'folk'
  163. price: 30
  164. venue: 'Малый зал'
  165. duration: '2 часа'
  166. ageRestriction: '6+'
  167. availableTickets: 28
  168. isFeatured: true
  169. inSlider: true
  170. shortDescription: 'Традиционные мелодии и танцы Таджикистана'
  171. cta: 'Узнать больше'
  172. }
  173. {
  174. _id: '3'
  175. title: 'Джазовый фестиваль "Borbad Jazz"'
  176. date: '2025-10-25'
  177. time: '20:00'
  178. description: 'Международные джазовые коллективы из Европы и Азии в уникальной акустике зала'
  179. image: '/images/event-jazz.jpg'
  180. category: 'jazz'
  181. price: 70
  182. venue: 'Большой зал'
  183. duration: '3 часа'
  184. ageRestriction: '16+'
  185. availableTickets: 15
  186. isFeatured: true
  187. inSlider: true
  188. shortDescription: 'Международные джазовые коллективы'
  189. cta: 'Смотреть расписание'
  190. }
  191. ]
  192. @appState.featuredEvents = @appState.events.filter((event) -> event.isFeatured).slice(0, 6)
  193. @appState.sliderEvents = @appState.events.filter((event) -> event.inSlider).map (event) ->
  194. id: event._id
  195. image: event.image
  196. title: event.title
  197. description: event.shortDescription
  198. cta: event.cta
  199. category: event.category
  200. getEvents: -> @appState.events
  201. getFeaturedEvents: -> @appState.featuredEvents
  202. getSliderEvents: -> @appState.sliderEvents
  203. isLoading: -> @appState.loading
  204. hasError: -> @appState.error
  205. closeModal: ->
  206. @modalState.currentModal = null
  207. @modalState.modalProps = {}
  208. components: components
  209. # Создаем глобальную шину событий
  210. class AppEventBus
  211. constructor: ->
  212. @events = {}
  213. on: (event, callback) ->
  214. if !@events[event]
  215. @events[event] = []
  216. @events[event].push(callback)
  217. emit: (event, data) ->
  218. if @events[event]
  219. for callback in @events[event]
  220. try
  221. callback(data)
  222. catch error
  223. debug.log "Event bus error: " + error
  224. off: (event, callback) ->
  225. if @events[event]
  226. @events[event] = @events[event].filter (cb) -> cb != callback
  227. # Создаем глобально
  228. globalThis.EventBus = new AppEventBus()
  229. app.use(VueRouter.createRouter({
  230. routes: routes
  231. history: VueRouter.createWebHistory()
  232. scrollBehavior: (to, from, savedPosition) ->
  233. if savedPosition
  234. return savedPosition
  235. else
  236. return { x: 0, y: 0 }
  237. }))
  238. # Глобальные функции для работы с модальными окнами
  239. globalThis.openModal = (component, props = {}) ->
  240. EventBus.$emit 'open-modal', { component, props }
  241. globalThis.closeModal = ->
  242. EventBus.$emit 'close-modal'
  243. # подключаем в body ОБЯЗАТЕЛЬНО!!!
  244. app.mount('body')
  245. debug.log "Vue application initialized successfully"