mirror of
https://github.com/codeninjasllc/codecombat.git
synced 2024-12-12 16:51:35 -05:00
73e49f9972
Removing social share buttons from base if IE, except Facebook. Unrolling restricted property setting loop (weird). Updating isIE check to support IE11.
337 lines
12 KiB
CoffeeScript
337 lines
12 KiB
CoffeeScript
SuperModel = require 'models/SuperModel'
|
|
utils = require 'lib/utils'
|
|
CocoClass = require 'lib/CocoClass'
|
|
loadingScreenTemplate = require 'templates/loading'
|
|
loadingErrorTemplate = require 'templates/loading_error'
|
|
|
|
lastToggleModalCall = 0
|
|
visibleModal = null
|
|
waitingModal = null
|
|
classCount = 0
|
|
makeScopeName = -> "view-scope-#{classCount++}"
|
|
doNothing = ->
|
|
|
|
module.exports = class CocoView extends Backbone.View
|
|
cache: false # signals to the router to keep this view around
|
|
template: -> ''
|
|
|
|
events:
|
|
'click .retry-loading-resource': 'onRetryResource'
|
|
'click .skip-loading-resource': 'onSkipResource'
|
|
|
|
subscriptions: {}
|
|
shortcuts: {}
|
|
|
|
# load progress properties
|
|
loadProgress:
|
|
progress: 0
|
|
|
|
# Setup, Teardown
|
|
|
|
constructor: (options) ->
|
|
@loadProgress = _.cloneDeep @loadProgress
|
|
@supermodel ?= new SuperModel()
|
|
@options = options
|
|
if options?.supermodel # kind of a hacky way to get each view to store its own progress
|
|
@supermodel.models = options.supermodel.models
|
|
@supermodel.collections = options.supermodel.collections
|
|
@supermodel.shouldSaveBackups = options.supermodel.shouldSaveBackups
|
|
|
|
@subscriptions = utils.combineAncestralObject(@, 'subscriptions')
|
|
@events = utils.combineAncestralObject(@, 'events')
|
|
@scope = makeScopeName()
|
|
@shortcuts = utils.combineAncestralObject(@, 'shortcuts')
|
|
@subviews = {}
|
|
@listenToShortcuts()
|
|
@updateProgressBar = _.debounce @updateProgressBar, 100
|
|
# Backbone.Mediator handles subscription setup/teardown automatically
|
|
|
|
@listenTo(@supermodel, 'loaded-all', @onLoaded)
|
|
@listenTo(@supermodel, 'update-progress', @updateProgress)
|
|
@listenTo(@supermodel, 'failed', @onResourceLoadFailed)
|
|
|
|
super options
|
|
|
|
destroy: ->
|
|
@stopListening()
|
|
@off()
|
|
@stopListeningToShortcuts()
|
|
@undelegateEvents() # removes both events and subs
|
|
view.destroy() for id, view of @subviews
|
|
$('#modal-wrapper .modal').off 'hidden.bs.modal', @modalClosed
|
|
@[key] = undefined for key, value of @
|
|
@destroyed = true
|
|
@off = doNothing
|
|
@destroy = doNothing
|
|
$.noty.closeAll()
|
|
|
|
destroyAceEditor: (editor) ->
|
|
# convenience method to make sure the ace editor is as destroyed as can be
|
|
return unless editor
|
|
session = editor.getSession()
|
|
session.setMode ''
|
|
editor.destroy()
|
|
|
|
afterInsert: ->
|
|
|
|
willDisappear: ->
|
|
# the router removes this view but this view will be cached
|
|
@undelegateEvents()
|
|
@hidden = true
|
|
@stopListeningToShortcuts()
|
|
view.willDisappear() for id, view of @subviews
|
|
$.noty.closeAll()
|
|
|
|
didReappear: ->
|
|
# the router brings back this view from the cache
|
|
@delegateEvents()
|
|
@hidden = false
|
|
@listenToShortcuts()
|
|
view.didReappear() for id, view of @subviews
|
|
|
|
# View Rendering
|
|
|
|
render: ->
|
|
return @ unless me
|
|
view.destroy() for id, view of @subviews
|
|
@subviews = {}
|
|
super()
|
|
return @template if _.isString(@template)
|
|
@$el.html @template(@getRenderData())
|
|
|
|
if not @supermodel.finished()
|
|
@showLoading()
|
|
else
|
|
@hideLoading()
|
|
|
|
@afterRender()
|
|
@$el.i18n()
|
|
@
|
|
|
|
getRenderData: (context) ->
|
|
context ?= {}
|
|
context.isProduction = document.location.href.search(/codecombat.com/) isnt -1
|
|
context.me = me
|
|
context.pathname = document.location.pathname # like '/play/level'
|
|
context.fbRef = context.pathname.replace(/[^a-zA-Z0-9+/=\-.:_]/g, '').slice(0, 40) or 'home'
|
|
context.isMobile = @isMobile()
|
|
context.isIE = @isIE()
|
|
context.moment = moment
|
|
context.translate = $.i18n.t
|
|
context
|
|
|
|
afterRender: ->
|
|
@renderScrollbar()
|
|
|
|
renderScrollbar: ->
|
|
#Defer the call till the content actually gets rendered, nanoscroller requires content to be visible
|
|
_.defer => @$el.find('.nano').nanoScroller() unless @destroyed
|
|
|
|
updateProgress: (progress) ->
|
|
@loadProgress.progress = progress if progress > @loadProgress.progress
|
|
@updateProgressBar(progress)
|
|
|
|
updateProgressBar: (progress) =>
|
|
prog = "#{parseInt(progress*100)}%"
|
|
@$el?.find('.loading-container .progress-bar').css('width', prog)
|
|
|
|
onLoaded: -> @render()
|
|
|
|
# Error handling for loading
|
|
onResourceLoadFailed: (e) ->
|
|
r = e.resource
|
|
@$el.find('.loading-container .errors').append(loadingErrorTemplate({
|
|
status: r.jqxhr?.status
|
|
name: r.name
|
|
resourceIndex: r.rid,
|
|
responseText: r.jqxhr?.responseText
|
|
})).i18n()
|
|
@$el.find('.progress').hide()
|
|
|
|
onRetryResource: (e) ->
|
|
res = @supermodel.getResource($(e.target).data('resource-index'))
|
|
# different views may respond to this call, and not all have the resource to reload
|
|
return unless res and res.isFailed
|
|
res.load()
|
|
@$el.find('.progress').show()
|
|
$(e.target).closest('.loading-error-alert').remove()
|
|
|
|
onSkipResource: (e) ->
|
|
res = @supermodel.getResource($(e.target).data('resource-index'))
|
|
return unless res and res.isFailed
|
|
res.markLoaded()
|
|
@$el.find('.progress').show()
|
|
$(e.target).closest('.loading-error-alert').remove()
|
|
|
|
# Modals
|
|
|
|
@lastToggleModalCall = 0
|
|
|
|
toggleModal: (e) ->
|
|
if $(e.currentTarget).prop('target') is '_blank'
|
|
return true
|
|
# special handler for opening modals that are dynamically loaded, rather than static in the page. It works (or should work) like Bootstrap's modals, except use coco-modal for the data-toggle value.
|
|
elem = $(e.target)
|
|
return unless elem.data('toggle') is 'coco-modal'
|
|
target = elem.data('target')
|
|
Modal = require 'views/'+target
|
|
e.stopPropagation()
|
|
@openModalView new Modal supermodel: @supermodal
|
|
|
|
openModalView: (modalView, softly=false) ->
|
|
return if waitingModal # can only have one waiting at once
|
|
if visibleModal
|
|
waitingModal = modalView
|
|
return if softly
|
|
return visibleModal.hide() if visibleModal.$el.is(':visible') # close, then this will get called again
|
|
return @modalClosed(visibleModal) # was closed, but modalClosed was not called somehow
|
|
modalView.render()
|
|
$('#modal-wrapper').empty().append modalView.el
|
|
modalView.afterInsert()
|
|
visibleModal = modalView
|
|
modalOptions = {show: true, backdrop: if modalView.closesOnClickOutside then true else 'static'}
|
|
$('#modal-wrapper .modal').modal(modalOptions).on 'hidden.bs.modal', @modalClosed
|
|
window.currentModal = modalView
|
|
@getRootView().stopListeningToShortcuts(true)
|
|
Backbone.Mediator.publish 'modal:opened', {}
|
|
|
|
modalClosed: =>
|
|
visibleModal.willDisappear() if visibleModal
|
|
visibleModal.destroy()
|
|
visibleModal = null
|
|
window.currentModal = null
|
|
#$('#modal-wrapper .modal').off 'hidden.bs.modal', @modalClosed
|
|
if waitingModal
|
|
wm = waitingModal
|
|
waitingModal = null
|
|
@openModalView(wm)
|
|
else
|
|
@getRootView().listenToShortcuts(true)
|
|
Backbone.Mediator.publish 'modal:closed', {}
|
|
|
|
# Loading RootViews
|
|
|
|
showLoading: ($el=@$el) ->
|
|
$el.find('>').addClass('hidden')
|
|
$el.append loadingScreenTemplate()
|
|
@_lastLoading = $el
|
|
|
|
hideLoading: ->
|
|
return unless @_lastLoading?
|
|
@_lastLoading.find('.loading-screen').remove()
|
|
@_lastLoading.find('>').removeClass('hidden')
|
|
@_lastLoading = null
|
|
|
|
showReadOnly: ->
|
|
return if me.isAdmin()
|
|
warning = $.i18n.t 'editor.read_only_warning2', defaultValue: 'Note: you can\'t save any edits here, because you\'re not logged in.'
|
|
noty text: warning, layout: 'center', type: 'information', killer: true, timeout: 5000
|
|
|
|
# Loading ModalViews
|
|
|
|
enableModalInProgress: (modal) ->
|
|
el = modal.find('.modal-content')
|
|
el.find('> div', modal).hide()
|
|
el.find('.wait', modal).show()
|
|
|
|
disableModalInProgress: (modal) ->
|
|
el = modal.find('.modal-content')
|
|
el.find('> div', modal).show()
|
|
el.find('.wait', modal).hide()
|
|
|
|
# Subscriptions
|
|
|
|
addNewSubscription: CocoClass.prototype.addNewSubscription
|
|
|
|
# Shortcuts
|
|
|
|
listenToShortcuts: (recurse) ->
|
|
return unless key
|
|
for shortcut, func of @shortcuts
|
|
func = utils.normalizeFunc(func, @)
|
|
key(shortcut, @scope, _.bind(func, @))
|
|
if recurse
|
|
for viewID, view of @subviews
|
|
view.listenToShortcuts()
|
|
|
|
stopListeningToShortcuts: (recurse) ->
|
|
return unless key
|
|
key.deleteScope(@scope)
|
|
if recurse
|
|
for viewID, view of @subviews
|
|
view.stopListeningToShortcuts()
|
|
|
|
# Subviews
|
|
|
|
insertSubView: (view, elToReplace=null) ->
|
|
# used to insert views with ids
|
|
key = @makeSubViewKey(view)
|
|
@subviews[key].destroy() if key of @subviews
|
|
elToReplace ?= @$el.find('#'+view.id)
|
|
elToReplace.after(view.el).remove()
|
|
@registerSubView(view, key)
|
|
view.render()
|
|
view.afterInsert()
|
|
view
|
|
|
|
registerSubView: (view, key) ->
|
|
# used to register views which are custom inserted into the view,
|
|
# like views where you add multiple instances of them
|
|
key = @makeSubViewKey(view)
|
|
view.parent = @
|
|
view.parentKey = key
|
|
@subviews[key] = view
|
|
view
|
|
|
|
makeSubViewKey: (view) ->
|
|
key = view.id or (view.constructor.name+classCount++)
|
|
key = _.string.underscored(key) # handy for autocomplete in dev console
|
|
key
|
|
|
|
removeSubView: (view) ->
|
|
view.$el.empty()
|
|
delete @subviews[view.parentKey]
|
|
view.destroy()
|
|
|
|
# Utilities
|
|
|
|
getQueryVariable: (param, defaultValue) -> CocoView.getQueryVariable(param, defaultValue)
|
|
@getQueryVariable: (param, defaultValue) ->
|
|
query = document.location.search.substring 1
|
|
pairs = (pair.split('=') for pair in query.split '&')
|
|
for pair in pairs when pair[0] is param
|
|
return {'true': true, 'false': false}[pair[1]] ? decodeURIComponent(pair[1])
|
|
defaultValue
|
|
|
|
getRootView: ->
|
|
view = @
|
|
view = view.parent while view.parent?
|
|
view
|
|
|
|
isMobile: ->
|
|
ua = navigator.userAgent or navigator.vendor or window.opera
|
|
return mobileRELong.test(ua) or mobileREShort.test(ua.substr(0, 4))
|
|
|
|
isIE: ->
|
|
# http://stackoverflow.com/questions/19999388/jquery-check-if-user-is-using-ie
|
|
navigator.userAgent.indexOf('MSIE') > 0 or !!navigator.userAgent.match(/Trident.*rv\:11\./)
|
|
|
|
isMac: ->
|
|
navigator.platform.toUpperCase().indexOf('MAC') isnt -1
|
|
|
|
isIPadApp: ->
|
|
return @_isIPadApp if @_isIPadApp?
|
|
return @_isIPadApp = webkit?.messageHandlers? and navigator.userAgent?.indexOf('iPad') isnt -1
|
|
|
|
initSlider: ($el, startValue, changeCallback) ->
|
|
slider = $el.slider({animate: 'fast'})
|
|
slider.slider('value', startValue)
|
|
slider.on('slide', changeCallback)
|
|
slider.on('slidechange', changeCallback)
|
|
slider
|
|
|
|
|
|
mobileRELong = /(android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od)|iris|kindle|lge |maemo|midp|mmp|mobile.+firefox|netfront|opera m(ob|in)i|palm( os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series(4|6)0|symbian|treo|up\.(browser|link)|vodafone|wap|windows (ce|phone)|xda|xiino/i
|
|
mobileREShort = /1207|6310|6590|3gso|4thp|50[1-6]i|770s|802s|a wa|abac|ac(er|oo|s\-)|ai(ko|rn)|al(av|ca|co)|amoi|an(ex|ny|yw)|aptu|ar(ch|go)|as(te|us)|attw|au(di|\-m|r |s )|avan|be(ck|ll|nq)|bi(lb|rd)|bl(ac|az)|br(e|v)w|bumb|bw\-(n|u)|c55\/|capi|ccwa|cdm\-|cell|chtm|cldc|cmd\-|co(mp|nd)|craw|da(it|ll|ng)|dbte|dc\-s|devi|dica|dmob|do(c|p)o|ds(12|\-d)|el(49|ai)|em(l2|ul)|er(ic|k0)|esl8|ez([4-7]0|os|wa|ze)|fetc|fly(\-|_)|g1 u|g560|gene|gf\-5|g\-mo|go(\.w|od)|gr(ad|un)|haie|hcit|hd\-(m|p|t)|hei\-|hi(pt|ta)|hp( i|ip)|hs\-c|ht(c(\-| |_|a|g|p|s|t)|tp)|hu(aw|tc)|i\-(20|go|ma)|i230|iac( |\-|\/)|ibro|idea|ig01|ikom|im1k|inno|ipaq|iris|ja(t|v)a|jbro|jemu|jigs|kddi|keji|kgt( |\/)|klon|kpt |kwc\-|kyo(c|k)|le(no|xi)|lg( g|\/(k|l|u)|50|54|\-[a-w])|libw|lynx|m1\-w|m3ga|m50\/|ma(te|ui|xo)|mc(01|21|ca)|m\-cr|me(rc|ri)|mi(o8|oa|ts)|mmef|mo(01|02|bi|de|do|t(\-| |o|v)|zz)|mt(50|p1|v )|mwbp|mywa|n10[0-2]|n20[2-3]|n30(0|2)|n50(0|2|5)|n7(0(0|1)|10)|ne((c|m)\-|on|tf|wf|wg|wt)|nok(6|i)|nzph|o2im|op(ti|wv)|oran|owg1|p800|pan(a|d|t)|pdxg|pg(13|\-([1-8]|c))|phil|pire|pl(ay|uc)|pn\-2|po(ck|rt|se)|prox|psio|pt\-g|qa\-a|qc(07|12|21|32|60|\-[2-7]|i\-)|qtek|r380|r600|raks|rim9|ro(ve|zo)|s55\/|sa(ge|ma|mm|ms|ny|va)|sc(01|h\-|oo|p\-)|sdk\/|se(c(\-|0|1)|47|mc|nd|ri)|sgh\-|shar|sie(\-|m)|sk\-0|sl(45|id)|sm(al|ar|b3|it|t5)|so(ft|ny)|sp(01|h\-|v\-|v )|sy(01|mb)|t2(18|50)|t6(00|10|18)|ta(gt|lk)|tcl\-|tdg\-|tel(i|m)|tim\-|t\-mo|to(pl|sh)|ts(70|m\-|m3|m5)|tx\-9|up(\.b|g1|si)|utst|v400|v750|veri|vi(rg|te)|vk(40|5[0-3]|\-v)|vm40|voda|vulc|vx(52|53|60|61|70|80|81|83|85|98)|w3c(\-| )|webc|whit|wi(g |nc|nw)|wmlb|wonu|x700|yas\-|your|zeto|zte\-/i
|
|
|
|
module.exports = CocoView
|