temp.coffee 11 KB

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