2014-06-26 01:56:39 -04:00
popoverTemplate = require ' templates/play/level/tome/spell_palette_entry_popover '
{ downTheChain } = require ' lib/world/world_utils '
window . Vector = require ' lib/world/vector ' # So we can document it
2014-10-28 16:44:49 -04:00
utils = require ' lib/utils '
2014-06-26 01:56:39 -04:00
safeJSONStringify = (input, maxDepth) ->
recursion = (input, path, depth) ->
output = { }
pPath = undefined
refIdx = undefined
2014-06-30 22:16:26 -04:00
path = path or ' '
2014-06-26 01:56:39 -04:00
depth = depth or 0
depth ++
2014-06-30 22:16:26 -04:00
return ' {depth over ' + maxDepth + ' } ' if maxDepth and depth > maxDepth
2014-06-26 01:56:39 -04:00
for p of input
2014-06-30 22:16:26 -04:00
pPath = ( ( if path then ( path + ' . ' ) else ' ' ) ) + p
if typeof input [ p ] is ' function '
output [ p ] = ' {function} '
else if typeof input [ p ] is ' object '
2014-06-26 01:56:39 -04:00
refIdx = refs . indexOf ( input [ p ] )
if - 1 isnt refIdx
2014-06-30 22:16:26 -04:00
output [ p ] = ' {reference to ' + refsPaths [ refIdx ] + ' } '
2014-06-26 01:56:39 -04:00
else
refs . push input [ p ]
refsPaths . push pPath
output [ p ] = recursion ( input [ p ] , pPath , depth )
else
output [ p ] = input [ p ]
output
refs = [ ]
refsPaths = [ ]
maxDepth = maxDepth or 5
2014-06-30 22:16:26 -04:00
if typeof input is ' object '
2014-06-26 01:56:39 -04:00
output = recursion ( input )
else
output = input
JSON . stringify output , null , 1
module.exports = class DocFormatter
constructor: (@options) ->
@doc = _ . cloneDeep options . doc
@ fillOutDoc ( )
fillOutDoc: ->
if _ . isString @ doc
@doc = name: @ doc , type: typeof @ options . thang [ @ doc ]
if @ options . isSnippet
@doc.type = ' snippet '
@doc.owner = ' snippets '
@doc.shortName = @doc.shorterName = @doc.title = @ doc . name
else
@ doc . owner ? = ' this '
ownerName = @doc.ownerName = if @ doc . owner isnt ' this ' then @ doc . owner else switch @ options . language
when ' python ' , ' lua ' then ' self '
when ' coffeescript ' then ' @ '
else ' this '
if @ doc . type is ' function '
sep = { clojure: ' ' } [ @ options . language ] ? ' , '
argNames = ( arg . name for arg in @ doc . args ? [ ] ) . join sep
argString = if argNames then ' __ARGS__ ' else ' '
@doc.shortName = switch @ options . language
when ' coffeescript ' then " #{ ownerName } #{ if ownerName is ' @ ' then ' ' else ' . ' } #{ @ doc . name } #{ if argString then ' ' + argString else ' () ' } "
when ' python ' then " #{ ownerName } . #{ @ doc . name } ( #{ argString } ) "
when ' lua ' then " #{ ownerName } : #{ @ doc . name } ( #{ argString } ) "
when ' clojure ' then " (. #{ @ doc . name } #{ ownerName } #{ if argNames then ' ' + argString else ' ' } ) "
when ' io ' then " #{ if ownerName is ' this ' then ' ' else ownerName + ' ' } #{ @ doc . name } #{ if argNames then ' ( ' + argNames + ' ) ' else ' ' } "
else " #{ ownerName } . #{ @ doc . name } ( #{ argString } ); "
else
@doc.shortName = switch @ options . language
when ' coffeescript ' then " #{ ownerName } #{ if ownerName is ' @ ' then ' ' else ' . ' } #{ @ doc . name } "
when ' python ' then " #{ ownerName } . #{ @ doc . name } "
when ' lua ' then " #{ ownerName } . #{ @ doc . name } "
when ' clojure ' then " (. #{ @ doc . name } #{ ownerName } ) "
when ' io ' then " #{ if ownerName is ' this ' then ' ' else ownerName + ' ' } #{ @ doc . name } "
else " #{ ownerName } . #{ @ doc . name } ; "
@doc.shorterName = @ doc . shortName
if @ doc . type is ' function ' and argString
@doc.shortName = @ doc . shorterName . replace argString , argNames
@doc.shorterName = @ doc . shorterName . replace argString , ( if argNames . length > 6 then ' ... ' else argNames )
if @ options . language is ' javascript '
@doc.shorterName = @ doc . shortName . replace ' ; ' , ' '
if @ doc . owner is ' this ' or @ options . tabbify
@doc.shorterName = @ doc . shorterName . replace / ^ this \ . / , ' '
@doc.title = if @ options . shortenize then @ doc . shorterName else @ doc . shortName
# Grab the language-specific documentation for some sub-properties, if we have it.
2014-10-17 18:25:31 -04:00
toTranslate = [ { obj: @ doc , prop: ' description ' } , { obj: @ doc , prop: ' example ' } ]
2014-06-26 01:56:39 -04:00
for arg in ( @ doc . args ? [ ] )
toTranslate . push { obj: arg , prop: ' example ' } , { obj: arg , prop: ' description ' }
2014-10-17 18:25:31 -04:00
if @ doc . returns
toTranslate . push { obj: @ doc . returns , prop: ' example ' } , { obj: @ doc . returns , prop: ' description ' }
2014-06-26 01:56:39 -04:00
for { obj , prop } in toTranslate
2014-10-28 16:44:49 -04:00
# Translate into chosen code language.
2014-06-26 01:56:39 -04:00
if val = obj [ prop ] ? [ @ options . language ]
obj [ prop ] = val
else unless _ . isString obj [ prop ]
obj [ prop ] = null
2014-10-28 16:44:49 -04:00
# Translate into chosen spoken language.
if val = obj [ prop ]
context = @ doc . context
obj [ prop ] = val = utils . i18n obj , prop
2014-11-23 13:21:58 -05:00
# For multiplexed-by-both-code-and-spoken-language objects, now also get code language again.
if _ . isObject val
obj [ prop ] = val = obj [ prop ] ? [ @ options . language ]
2014-10-28 16:44:49 -04:00
if @ doc . i18n
spokenLanguage = me . get ' preferredLanguage '
while spokenLanguage
spokenLanguage = spokenLanguage . substr 0 , spokenLanguage . lastIndexOf ( ' - ' ) if fallingBack ?
if spokenLanguageContext = @ doc . i18n [ spokenLanguage ] ? . context
2014-10-29 00:15:41 -04:00
context = _ . merge context , spokenLanguageContext
2014-10-28 16:44:49 -04:00
break
fallingBack = true
2014-10-29 15:08:03 -04:00
if context
try
obj [ prop ] = _ . template val , context
catch e
console . error " Couldn ' t create docs template of " , val , " \n with context " , context , " \n Error: " , e
2014-10-28 16:44:49 -04:00
2014-06-26 01:56:39 -04:00
formatPopover: ->
2014-10-17 00:38:11 -04:00
content = popoverTemplate doc: @ doc , language: @ options . language , value: @ formatValue ( ) , marked: marked , argumentExamples: ( arg . example or arg . default or arg . name for arg in @ doc . args ? [ ] ) , writable: @ options . writable , selectedMethod: @ options . selectedMethod , cooldowns: @ inferCooldowns ( ) , item: @ options . item
2014-06-26 01:56:39 -04:00
owner = if @ doc . owner is ' this ' then @ options . thang else window [ @ doc . owner ]
content = content . replace / #{spriteName}/g, @options.thang.type ? @options.thang.spriteName # Prefer type, and excluded the quotes we'd get with @formatValue
content . replace / \ #\{(.*?)\}/g, (s, properties) => @formatValue downTheChain(owner, properties.split('.'))
formatValue: (v) ->
return null if @ doc . type is ' snippet '
return @ options . thang . now ( ) if @ doc . name is ' now '
return ' [Function] ' if not v and @ doc . type is ' function '
unless v ?
if @ doc . owner is ' this '
v = @ options . thang [ @ doc . name ]
else
v = window [ @ doc . owner ] [ @ doc . name ] # grab Math or Vector
2014-10-09 12:53:57 -04:00
if @ doc . type is ' number ' and not _ . isNaN v
2014-11-23 00:22:46 -05:00
if v is Math . round v
2014-06-26 01:56:39 -04:00
return v
2014-10-09 12:53:57 -04:00
if _ . isNumber v
return v . toFixed 2
unless v
return ' null '
return ' ' + v
2014-06-26 01:56:39 -04:00
if _ . isString v
return " \" #{ v } \" "
if v ? . id
return v . id
if v ? . name
return v . name
if _ . isArray v
return ' [ ' + ( @ formatValue v2 for v2 in v ) . join ( ' , ' ) + ' ] '
if _ . isPlainObject v
return safeJSONStringify v , 2
v
2014-07-20 18:51:42 -04:00
inferCooldowns: ->
return null unless @ doc . type is ' function ' and @ doc . owner is ' this '
owner = @ options . thang
cooldowns = null
spellName = @ doc . name . match / ^ cast ( . + ) $ /
if spellName
actionName = _ . string . slugify _ . string . underscored spellName [ 1 ]
action = owner . spells ? [ actionName ]
type = ' spell '
else
actionName = _ . string . slugify _ . string . underscored @ doc . name
action = owner . actions ? [ actionName ]
type = ' action '
return null unless action
cooldowns = cooldown: action . cooldown , specificCooldown: action . specificCooldown , name: actionName , type: type
for prop in [ ' range ' , ' radius ' , ' duration ' , ' damage ' ]
2014-11-23 00:22:46 -05:00
cooldowns [ prop ] = v = owner [ _ . string . camelize actionName + _ . string . capitalize ( prop ) ]
if _ . isNumber ( v ) and v isnt Math . round v
cooldowns [ prop ] = v . toFixed 2
2014-07-20 18:51:42 -04:00
cooldowns