View = require 'views/kinds/CocoView'
template = require 'templates/play/level/playback'
{me} = require 'lib/auth'
EditorConfigModal = require './modal/editor_config_modal'
KeyboardShortcutsModal = require './modal/keyboard_shortcuts_modal'
module.exports = class PlaybackView extends View
id: 'playback-view'
template: template
subscriptions:
'level-disable-controls': 'onDisableControls'
'level-enable-controls': 'onEnableControls'
'level-set-playing': 'onSetPlaying'
'level-toggle-playing': 'onTogglePlay'
'level-scrub-forward': 'onScrubForward'
'level-scrub-back': 'onScrubBack'
'level-set-volume': 'onSetVolume'
'level-set-debug': 'onSetDebug'
'level-set-grid': 'onSetGrid'
'level-toggle-grid': 'onToggleGrid'
'surface:frame-changed': 'onFrameChanged'
'god:new-world-created': 'onNewWorld'
'level-set-letterbox': 'onSetLetterbox'
'tome:cast-spells': 'onCastSpells'
events:
'click #debug-toggle': 'onToggleDebug'
'click #grid-toggle': 'onToggleGrid'
'click #edit-wizard-settings': 'onEditWizardSettings'
'click #edit-editor-config': 'onEditEditorConfig'
'click #view-keyboard-shortcuts': 'onViewKeyboardShortcuts'
'click #music-button': 'onToggleMusic'
'click #zoom-in-button': -> Backbone.Mediator.publish('camera-zoom-in') unless @shouldIgnore()
'click #zoom-out-button': -> Backbone.Mediator.publish('camera-zoom-out') unless @shouldIgnore()
'click #volume-button': 'onToggleVolume'
'click #play-button': 'onTogglePlay'
'click': -> Backbone.Mediator.publish 'tome:focus-editor'
'mouseenter #timeProgress': 'onProgressEnter'
'mouseleave #timeProgress': 'onProgressLeave'
'mousemove #timeProgress': 'onProgressHover'
shortcuts:
'⌘+p, p, ctrl+p': 'onTogglePlay'
'⌘+[, ctrl+[': 'onScrubBack'
'⌘+⇧+[, ctrl+⇧+[': 'onSingleScrubBack'
'⌘+], ctrl+]': 'onScrubForward'
'⌘+⇧+], ctrl+⇧+]': 'onSingleScrubForward'
# popover that shows at the current mouse position on the progressbar, using the bootstrap popover.
# Could make this into a jQuery plugins itself theoretically.
class HoverPopup extends $.fn.popover.Constructor
constructor: () ->
@enabled = true
@shown = false
@type = 'HoverPopup'
@options =
placement: 'top'
container: 'body'
animation: true
html: true
delay:
show: 400
@$element = $('#timeProgress')
@$tip = $('#timePopover')
@content = ''
getContent: -> @content
show: ->
unless @shown
super()
@shown = true
updateContent: (@content) ->
@setContent()
@$tip.addClass('fade top in')
onHover: (@e) ->
pos = @getPosition()
actualWidth = @$tip[0].offsetWidth
actualHeight = @$tip[0].offsetHeight
calculatedOffset =
top: pos.top - actualHeight
left: pos.left + pos.width / 2 - actualWidth / 2
this.applyPlacement(calculatedOffset, 'top')
getPosition: ->
top: @$element.offset().top
left: if @e? then @e.pageX else @$element.offset().left
height: 0
width: 0
hide: ->
super()
@shown = false
disable: ->
super()
@hide()
constructor: ->
super(arguments...)
me.on('change:music', @updateMusicButton, @)
afterRender: ->
super()
@$progressScrubber = $('.scrubber .progress', @$el)
@hookUpScrubber()
@updateMusicButton()
$(window).on('resize', @onWindowResize)
ua = navigator.userAgent.toLowerCase()
if /safari/.test(ua) and not /chrome/.test(ua)
@$el.find('.toggle-fullscreen').hide()
updatePopupContent: ->
@timePopup?.updateContent "
#{@timeToString @newTime}
#{@formatTime(@current, @currentTime)}
#{@formatTime(@total, @totalTime)}"
# These functions could go to some helper class
pad2: (num) ->
if not num? or num is 0 then '00' else ((if num < 10 then '0' else '') + num)
formatTime: (text, time) =>
"#{text}\t#{@timeToString time}"
timeToString: (time=0, withUnits=false) ->
mins = Math.floor(time / 60)
secs = (time - mins * 60).toFixed(1)
if withUnits
ret = ''
ret = (mins + ' ' + (if mins is 1 then @minute else @minutes)) if (mins > 0)
ret = (ret + ' ' + secs + ' ' + (if secs is 1 then @second else @seconds)) if (secs > 0 or mins is 0)
else
"#{mins}:#{@pad2 secs}"
# callbacks
updateMusicButton: ->
@$el.find('#music-button').toggleClass('music-on', me.get('music'))
onSetLetterbox: (e) ->
buttons = @$el.find '#play-button, .scrubber-handle'
buttons.css 'visibility', if e.on then 'hidden' else 'visible'
@disabled = e.on
onWindowResize: (s...) =>
@barWidth = $('.progress', @$el).width()
onNewWorld: (e) ->
@totalTime = e.world.totalFrames / e.world.frameRate
pct = parseInt(100 * e.world.totalFrames / e.world.maxTotalFrames) + '%'
@barWidth = $('.progress', @$el).css('width', pct).show().width()
@casting = false
$('.scrubber .progress', @$el).slider('enable', true)
@newTime = 0
@currentTime = 0
@timePopup ?= new HoverPopup
t = $.i18n.t
@second = t 'units.second'
@seconds = t 'units.seconds'
@minute = t 'units.minute'
@minutes = t 'units.minutes'
@goto = t 'play_level.time_goto'
@current = t 'play_level.time_current'
@total = t 'play_level.time_total'
onToggleDebug: ->
return if @shouldIgnore()
flag = $('#debug-toggle i.icon-ok')
Backbone.Mediator.publish('level-set-debug', {debug: flag.hasClass('invisible')})
onToggleGrid: ->
return if @shouldIgnore()
flag = $('#grid-toggle i.icon-ok')
Backbone.Mediator.publish('level-set-grid', {grid: flag.hasClass('invisible')})
onEditWizardSettings: ->
Backbone.Mediator.publish 'edit-wizard-settings'
onEditEditorConfig: ->
@openModalView new EditorConfigModal session: @options.session
onViewKeyboardShortcuts: ->
@openModalView new KeyboardShortcutsModal()
onCastSpells: (e) ->
return if e.preload
@casting = true
@$progressScrubber.slider('disable', true)
onDisableControls: (e) ->
if not e.controls or 'playback' in e.controls
@disabled = true
$('button', @$el).addClass('disabled')
try
@$progressScrubber.slider('disable', true)
catch error
console.warn('error disabling scrubber', error)
@timePopup?.disable()
$('#volume-button', @$el).removeClass('disabled')
onEnableControls: (e) ->
if not e.controls or 'playback' in e.controls
@disabled = false
$('button', @$el).removeClass('disabled')
try
@$progressScrubber.slider('enable', true)
catch error
console.warn('error enabling scrubber', error)
@timePopup?.enable()
onSetPlaying: (e) ->
@playing = (e ? {}).playing ? true
button = @$el.find '#play-button'
ended = button.hasClass 'ended'
button.toggleClass('playing', @playing and not ended).toggleClass('paused', not @playing and not ended)
return # don't stripe the bar
bar = @$el.find '.scrubber .progress'
bar.toggleClass('progress-striped', @playing and not ended).toggleClass('active', @playing and not ended)
onSetVolume: (e) ->
classes = ['vol-off', 'vol-down', 'vol-up']
button = $('#volume-button', @$el)
button.removeClass(c) for c in classes
button.addClass(classes[0]) if e.volume <= 0.0
button.addClass(classes[1]) if e.volume > 0.0 and e.volume < 1.0
button.addClass(classes[2]) if e.volume >= 1.0
onScrub: (e, options) ->
e?.preventDefault()
options.scrubDuration = 500
Backbone.Mediator.publish('level-set-time', options)
onScrubForward: (e) ->
@onScrub e, ratioOffset: 0.05
onSingleScrubForward: (e) ->
@onScrub e, frameOffset: 1
onScrubBack: (e) ->
@onScrub e, ratioOffset: -0.05
onSingleScrubBack: (e) ->
@onScrub e, frameOffset: -1
onFrameChanged: (e) ->
if e.progress isnt @lastProgress
@currentTime = e.frame / e.world.frameRate
# Game will sometimes stop at 29.97, but with only one digit, this is unnecesary.
# @currentTime = @totalTime if Math.abs(@totalTime - @currentTime) < 0.04
@updatePopupContent() if @timePopup?.shown
@updateProgress(e.progress)
@updatePlayButton(e.progress)
@lastProgress = e.progress
onProgressEnter: (e) ->
# Why it needs itself as parameter you ask? Ask Twitter instead.
@timePopup?.enter @timePopup
onProgressLeave: (e) ->
@timePopup?.leave @timePopup
onProgressHover: (e) ->
timeRatio = @$progressScrubber.width() / @totalTime
offsetX = e.offsetX or e.clientX - $(e.target).offset().left
@newTime = offsetX / timeRatio
@updatePopupContent()
@timePopup?.onHover e
# Show it instantaneously if close enough to current time.
if @timePopup and Math.abs(@currentTime - @newTime) < 1 and not @timePopup.shown
@timePopup.show()
updateProgress: (progress) ->
$('.scrubber .progress-bar', @$el).css('width', "#{progress*100}%")
updatePlayButton: (progress) ->
if progress >= 0.99 and @lastProgress < 0.99
$('#play-button').removeClass('playing').removeClass('paused').addClass('ended')
if progress < 0.99 and @lastProgress >= 0.99
b = $('#play-button').removeClass('ended')
if @playing then b.addClass('playing') else b.addClass('paused')
onSetDebug: (e) ->
flag = $('#debug-toggle i.icon-ok')
flag.toggleClass 'invisible', not e.debug
onSetGrid: (e) ->
flag = $('#grid-toggle i.icon-ok')
flag.toggleClass 'invisible', not e.grid
# to refactor
hookUpScrubber: ->
@sliderIncrements = 500 # max slider width before we skip pixels
@clickingSlider = false # whether the mouse has been pressed down without moving
@$progressScrubber.slider(
max: @sliderIncrements
animate: 'slow'
slide: (event, ui) =>
@scrubTo ui.value / @sliderIncrements
@slideCount += 1
start: (event, ui) =>
@slideCount = 0
@wasPlaying = @playing
Backbone.Mediator.publish 'level-set-playing', {playing: false}
stop: (event, ui) =>
@actualProgress = ui.value / @sliderIncrements
Backbone.Mediator.publish 'playback:manually-scrubbed', ratio: @actualProgress
Backbone.Mediator.publish 'level-set-playing', {playing: @wasPlaying}
if @slideCount < 3
@wasPlaying = false
Backbone.Mediator.publish 'level-set-playing', {playing: false}
@$el.find('.scrubber-handle').effect('bounce', {times: 2})
)
getScrubRatio: ->
@$progressScrubber.find('.progress-bar').width() / @$progressScrubber.width()
scrubTo: (ratio, duration=0) ->
return if @shouldIgnore()
Backbone.Mediator.publish 'level-set-time', ratio: ratio, scrubDuration: duration
shouldIgnore: -> return @disabled or @casting or false
onTogglePlay: (e) ->
e?.preventDefault()
return if @shouldIgnore()
button = $('#play-button')
willPlay = button.hasClass('paused') or button.hasClass('ended')
Backbone.Mediator.publish 'level-set-playing', playing: willPlay
$(document.activeElement).blur()
onToggleVolume: (e) ->
button = $(e.target).closest('#volume-button')
classes = ['vol-off', 'vol-down', 'vol-up']
volumes = [0, 0.4, 1.0]
for oldClass, i in classes
if button.hasClass oldClass
newI = (i + 1) % classes.length
break
else if i is classes.length - 1 # no oldClass
newI = 2
Backbone.Mediator.publish 'level-set-volume', volume: volumes[newI]
$(document.activeElement).blur()
onToggleMusic: (e) ->
e?.preventDefault()
me.set('music', not me.get('music'))
me.patch()
$(document.activeElement).blur()
destroy: ->
me.off('change:music', @updateMusicButton, @)
$(window).off('resize', @onWindowResize)
@onWindowResize = null
super()