guide.coffee 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. #
  2. # Config
  3. #
  4. #FETCH_URL = 'http://www.tvgids.nl/json/lists/programs.php'
  5. FETCH_URL = 'programs.php'
  6. HOUR_WIDTH = 200
  7. CHANNEL_LABEL_WIDTH = 180
  8. #SCROLL_MULTIPLIER = HOUR_WIDTH
  9. #
  10. # Utils
  11. #
  12. seconds_today = (time) -> (time - (new Date()).setHours(0, 0, 0, 0)) / 1000
  13. time2px = (seconds) -> HOUR_WIDTH / 3600 * seconds
  14. zeropad = (digit) -> if digit < 10 then '0' + digit else String(digit)
  15. format_time = (time) ->
  16. date = new Date(time)
  17. zeropad(date.getHours()) + ':' + zeropad(date.getMinutes())
  18. #
  19. # Models & collections
  20. #
  21. Channel = Backbone.Model.extend(
  22. defaults: ->
  23. id: null
  24. name: 'Some channel'
  25. visible: true
  26. programs: []
  27. )
  28. Program = Backbone.Model.extend(
  29. defaults: ->
  30. title: 'Some program'
  31. genre: ''
  32. sort: ''
  33. start: 0
  34. end: 0
  35. #article_id: null
  36. #article_title: null
  37. )
  38. ChannelList = Backbone.Collection.extend(
  39. model: Channel
  40. comparator: (a, b) -> parseInt(a.get('id')) - parseInt(b.get('id'))
  41. initialize: (models, options) ->
  42. @fetchVisible()
  43. fetch: ->
  44. @reset(CHANNELS)
  45. #@reset(CHANNELS.slice(0,3))
  46. #$.getJSON('channels.php', (data) => @reset(data))
  47. fetchVisible: ->
  48. visible = if localStorage.hasOwnProperty('channels') \
  49. then localStorage.getItem('channels').split(',') else @pluck('id')
  50. @setVisible(visible)
  51. saveVisible: ->
  52. selected = (c.id for c in @channels if c.visible)
  53. localStorage.setItem('channels', selected.join(','))
  54. setVisible: (visible, save=false) ->
  55. for id in visible
  56. @findWhere(id: id).set(visible: true)
  57. for id in _.difference(@pluck('id'), visible)
  58. @findWhere(id: id).set(visible: false)
  59. @saveVisible() if save
  60. fetchPrograms: (day) ->
  61. $.getJSON(
  62. FETCH_URL
  63. channels: @pluck('id').join(','), day: day
  64. (channels) ->
  65. _.each channels, (programs, id) ->
  66. channel = Channels.findWhere(id: id)
  67. channel.set('programs', (
  68. new Program(
  69. title: p.titel
  70. genre: p.genre
  71. sort: p.soort
  72. start: Date.parse(p.datum_start)
  73. end: Date.parse(p.datum_end)
  74. #article_id: p.artikel_id
  75. #article_title: p.artikel_titel
  76. ) for p in programs
  77. ))
  78. )
  79. )
  80. #
  81. # Views
  82. #
  83. ChannelView = Backbone.View.extend(
  84. tagName: 'div'
  85. className: 'channel'
  86. #template: _.template($('#channel-template').html())
  87. initialize: ->
  88. @listenTo(@model, 'change:programs', @render)
  89. @listenTo(@model, 'change:visible', @toggleVisible)
  90. #@$el.text(@model.get('title'))
  91. render: ->
  92. @$el.empty()
  93. _.each @model.get('programs'), (program) =>
  94. view = new ProgramView(model: program)
  95. view.render()
  96. @$el.append(view.el)
  97. toggleVisible: ->
  98. @$el.toggle(@model.get('visible'))
  99. )
  100. ProgramView = Backbone.View.extend(
  101. tagName: 'div'
  102. className: 'program'
  103. initialize: ->
  104. @$el.text(@model.get('title'))
  105. from = format_time(@model.get('start'))
  106. to = format_time(@model.get('end'))
  107. @$el.attr('title', @model.get('title') + " (#{from} - #{to})")
  108. left = time2px(Math.max(0, seconds_today(@model.get('start'))))
  109. width = time2px(seconds_today(@model.get('end'))) - left
  110. @$el.css(
  111. left: left + 'px'
  112. width: (width - 10) + 'px'
  113. )
  114. render: ->
  115. if @model.get('start') <= Date.now()
  116. if @model.get('end') < Date.now()
  117. @$el.removeClass('current').addClass('past')
  118. else
  119. @$el.addClass('current')
  120. )
  121. ChannelLabelView = Backbone.View.extend(
  122. el: $('.channel-labels')
  123. initialize: (options) ->
  124. @listenTo(Channels, 'reset', @addChannels)
  125. @listenTo(options.app, 'scroll', @moveTop)
  126. addChannels: ->
  127. @$el.empty()
  128. Channels.each((channel) ->
  129. @$el.append('<div class="label">' + channel.get('name') + '</div>')
  130. , @)
  131. moveTop: (delta) ->
  132. @$el.css('top', (@$el.position().top - delta) + 'px')
  133. )
  134. AppView = Backbone.View.extend(
  135. el: $('#guide')
  136. events:
  137. 'click #yesterday': -> @loadDay(-1)
  138. 'click #today': -> @loadDay(0)
  139. 'click #tomorrow': -> @loadDay(1)
  140. 'scroll': 'moveTimeline'
  141. moveTimeline: ->
  142. if @$el.scrollTop() != @prevScrollTop
  143. @trigger('scroll', @$el.scrollTop() - @prevScrollTop)
  144. @prevScrollTop = @$el.scrollTop()
  145. @$('.timeline').css('top', (@$el.scrollTop() + 37) + 'px')
  146. initialize: ->
  147. @prevScrollTop = null
  148. @listenTo(Channels, 'reset', @addChannels)
  149. @listenTo(Settings, 'change:day', @fetchPrograms)
  150. @labelview = new ChannelLabelView(app: @)
  151. @updateIndicator()
  152. @centerIndicator()
  153. Channels.fetch()
  154. setInterval((=> @updateIndicator()), 3600000 / HOUR_WIDTH)
  155. addChannels: ->
  156. @$('.channels > .channel').remove()
  157. Channels.each((channel) ->
  158. view = new ChannelView(model: channel)
  159. view.render()
  160. @$('.channels').append(view.el)
  161. , @)
  162. @$('.indicator').height(@$('.channels').height())
  163. @fetchPrograms()
  164. loadDay: (day) ->
  165. Settings.set(day: day)
  166. @$('.navbar .active').removeClass('active')
  167. $(@$('.navbar .navitem')[day + 1]).addClass('active')
  168. updateIndicator: ->
  169. left = time2px(seconds_today(Date.now())) + CHANNEL_LABEL_WIDTH - 1
  170. @$('.indicator').css('left', left + 'px')
  171. centerIndicator: ->
  172. @$el.scrollLeft(@$('.indicator').position().left - @$el.width() / 2)
  173. fetchPrograms: ->
  174. Channels.fetchPrograms(Settings.get('day'))
  175. )
  176. #
  177. # Main
  178. #
  179. Settings = new (Backbone.Model.extend(
  180. defaults: ->
  181. day: 0
  182. ))()
  183. Channels = new ChannelList()
  184. App = new AppView()