2015-11-11 13:12:51 +00:00
|
|
|
class @AwardsHandler
|
2016-05-25 13:05:56 +00:00
|
|
|
|
2016-03-31 09:55:37 +00:00
|
|
|
constructor: ->
|
2016-05-25 13:05:56 +00:00
|
|
|
|
2016-03-31 09:55:37 +00:00
|
|
|
@aliases = emojiAliases()
|
|
|
|
|
2016-03-31 11:56:48 +00:00
|
|
|
$(document)
|
2016-05-25 13:05:56 +00:00
|
|
|
.off 'click', '.js-add-award'
|
|
|
|
.on 'click', '.js-add-award', (event) =>
|
2016-03-31 11:56:48 +00:00
|
|
|
event.stopPropagation()
|
|
|
|
event.preventDefault()
|
2016-02-19 04:02:22 +00:00
|
|
|
|
2016-03-31 11:56:48 +00:00
|
|
|
@showEmojiMenu $(event.currentTarget)
|
2016-04-25 08:09:39 +00:00
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
$('html').on 'click', (event) ->
|
|
|
|
unless $(event.target).closest('.emoji-menu').length
|
|
|
|
if $('.emoji-menu').is(':visible')
|
2016-04-01 17:18:45 +00:00
|
|
|
$('.js-add-award.is-active').removeClass 'is-active'
|
2016-05-25 13:05:56 +00:00
|
|
|
$('.emoji-menu').removeClass 'is-visible'
|
2015-12-17 12:29:55 +00:00
|
|
|
|
2016-03-31 09:55:37 +00:00
|
|
|
$(document)
|
2016-05-25 13:05:56 +00:00
|
|
|
.off 'click', '.js-emoji-btn'
|
|
|
|
.on 'click', '.js-emoji-btn', @handleClick
|
2015-11-11 13:12:51 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
|
2016-03-31 09:55:37 +00:00
|
|
|
handleClick: (e) =>
|
2016-05-23 18:33:58 +00:00
|
|
|
|
2016-03-01 11:25:59 +00:00
|
|
|
e.preventDefault()
|
2016-05-24 00:10:38 +00:00
|
|
|
|
2016-05-24 17:47:10 +00:00
|
|
|
emoji = $(e.currentTarget).find('.icon').data 'emoji'
|
|
|
|
@getVotesBlock().addClass 'js-awards-block'
|
|
|
|
@addAward @getAwardUrl(), emoji
|
2016-05-11 20:49:47 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
|
2016-03-31 09:55:37 +00:00
|
|
|
showEmojiMenu: ($addBtn) ->
|
2016-05-23 18:33:58 +00:00
|
|
|
|
2016-04-01 17:18:45 +00:00
|
|
|
$menu = $('.emoji-menu')
|
2016-03-31 09:55:37 +00:00
|
|
|
|
2016-04-01 17:18:45 +00:00
|
|
|
if $menu.length
|
|
|
|
$holder = $addBtn.closest('.js-award-holder')
|
2016-03-31 09:55:37 +00:00
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
if $menu.is '.is-visible'
|
|
|
|
$addBtn.removeClass 'is-active'
|
|
|
|
$menu.removeClass 'is-visible'
|
|
|
|
$('#emoji_search').blur()
|
2016-05-11 20:49:47 +00:00
|
|
|
else
|
2016-05-25 13:05:56 +00:00
|
|
|
$addBtn.addClass 'is-active'
|
2016-04-01 17:18:45 +00:00
|
|
|
@positionMenu($menu, $addBtn)
|
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
$menu.addClass 'is-visible'
|
|
|
|
$('#emoji_search').focus()
|
2016-05-11 20:49:47 +00:00
|
|
|
else
|
2016-05-23 18:33:58 +00:00
|
|
|
$addBtn.addClass 'is-loading is-active'
|
|
|
|
url = $addBtn.data 'award-menu-url'
|
2016-04-01 17:18:45 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
@createEmojiMenu url, =>
|
|
|
|
$addBtn.removeClass 'is-loading'
|
|
|
|
$menu = $('.emoji-menu')
|
2016-04-01 17:18:45 +00:00
|
|
|
@positionMenu($menu, $addBtn)
|
2016-03-31 09:55:37 +00:00
|
|
|
@renderFrequentlyUsedBlock()
|
2016-03-31 11:08:31 +00:00
|
|
|
|
2016-03-11 09:47:18 +00:00
|
|
|
setTimeout =>
|
2016-05-23 18:33:58 +00:00
|
|
|
$menu.addClass 'is-visible'
|
|
|
|
$('#emoji_search').focus()
|
2016-03-11 09:47:18 +00:00
|
|
|
@setupSearch()
|
|
|
|
, 200
|
2016-02-19 04:02:22 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
|
|
|
|
createEmojiMenu: (awardMenuUrl, callback) ->
|
|
|
|
|
|
|
|
$.get awardMenuUrl, (response) =>
|
|
|
|
$('body').append response
|
|
|
|
callback()
|
|
|
|
|
|
|
|
|
2016-04-01 17:18:45 +00:00
|
|
|
positionMenu: ($menu, $addBtn) ->
|
|
|
|
position = $addBtn.data('position')
|
|
|
|
|
|
|
|
# The menu could potentially be off-screen or in a hidden overflow element
|
|
|
|
# So we position the element absolute in the body
|
|
|
|
css =
|
|
|
|
top: "#{$addBtn.offset().top + $addBtn.outerHeight()}px"
|
|
|
|
|
|
|
|
if position? and position is 'right'
|
|
|
|
css.left = "#{($addBtn.offset().left - $menu.outerWidth()) + 20}px"
|
2016-05-25 13:05:56 +00:00
|
|
|
$menu.addClass 'is-aligned-right'
|
2016-04-01 17:18:45 +00:00
|
|
|
else
|
|
|
|
css.left = "#{$addBtn.offset().left}px"
|
2016-05-25 13:05:56 +00:00
|
|
|
$menu.removeClass 'is-aligned-right'
|
2016-04-01 17:18:45 +00:00
|
|
|
|
|
|
|
$menu.css(css)
|
|
|
|
|
2016-05-24 17:47:10 +00:00
|
|
|
|
|
|
|
addAward: (awardUrl, emoji, checkMutuality = yes) ->
|
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
emoji = @normilizeEmojiName(emoji)
|
|
|
|
@postEmoji awardUrl, emoji, =>
|
2016-05-24 17:47:10 +00:00
|
|
|
@addAwardToEmojiBar(emoji, checkMutuality)
|
2015-12-17 12:29:55 +00:00
|
|
|
|
2016-03-31 11:08:31 +00:00
|
|
|
$('.js-awards-block-current').removeClass 'js-awards-block-current'
|
|
|
|
|
2016-05-24 17:47:10 +00:00
|
|
|
$('.emoji-menu').removeClass 'is-visible'
|
2016-01-15 15:04:48 +00:00
|
|
|
|
2016-05-24 17:47:10 +00:00
|
|
|
|
|
|
|
addAwardToEmojiBar: (emoji, checkForMutuality = yes) ->
|
|
|
|
|
|
|
|
@checkMutuality emoji if checkForMutuality
|
2015-12-22 21:04:19 +00:00
|
|
|
@addEmojiToFrequentlyUsedList(emoji)
|
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
emoji = @normilizeEmojiName(emoji)
|
2016-03-31 11:56:48 +00:00
|
|
|
$emojiBtn = @findEmojiIcon(emoji).parent()
|
2016-03-31 11:08:31 +00:00
|
|
|
|
|
|
|
if $emojiBtn.length > 0
|
|
|
|
if @isActive($emojiBtn)
|
2016-03-31 11:56:48 +00:00
|
|
|
@decrementCounter($emojiBtn, emoji)
|
2015-11-17 14:44:58 +00:00
|
|
|
else
|
2016-05-23 18:25:45 +00:00
|
|
|
counter = $emojiBtn.find('.js-counter')
|
2016-05-11 20:49:47 +00:00
|
|
|
counter.text(parseInt(counter.text()) + 1)
|
2016-05-25 13:05:56 +00:00
|
|
|
$emojiBtn.addClass('active')
|
2016-03-31 08:21:33 +00:00
|
|
|
@addMeToUserList(emoji)
|
2015-11-17 14:44:58 +00:00
|
|
|
else
|
2015-12-17 12:29:55 +00:00
|
|
|
@createEmoji(emoji)
|
2015-11-11 13:12:51 +00:00
|
|
|
|
2016-05-24 17:47:10 +00:00
|
|
|
|
|
|
|
getVotesBlock: -> return $ '.awards.js-awards-block'
|
|
|
|
|
|
|
|
|
|
|
|
getAwardUrl: -> @getVotesBlock().data 'award-url'
|
|
|
|
|
|
|
|
|
|
|
|
checkMutuality: (emoji) ->
|
|
|
|
|
|
|
|
awardUrl = @getAwardUrl()
|
|
|
|
|
|
|
|
if emoji in [ 'thumbsup', 'thumbsdown' ]
|
|
|
|
mutualVote = if emoji is 'thumbsup' then 'thumbsdown' else 'thumbsup'
|
|
|
|
|
|
|
|
isAlreadyVoted = $("[data-emoji=#{mutualVote}]").parent().hasClass 'active'
|
|
|
|
@addAward awardUrl, mutualVote, no if isAlreadyVoted
|
|
|
|
|
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
isActive: ($emojiBtn) -> $emojiBtn.hasClass 'active'
|
|
|
|
|
2016-03-31 11:56:48 +00:00
|
|
|
|
|
|
|
decrementCounter: ($emojiBtn, emoji) ->
|
|
|
|
isntNoteBody = $emojiBtn.closest('.note-body').length is 0
|
|
|
|
counter = $('.js-counter', $emojiBtn)
|
|
|
|
counterNumber = parseInt(counter.text())
|
|
|
|
|
2016-03-31 12:17:57 +00:00
|
|
|
if !isntNoteBody
|
|
|
|
# If this is a note body, we just hide the award emoji row like the initial state
|
|
|
|
$emojiBtn.closest('.js-awards-block').addClass 'hidden'
|
|
|
|
|
2016-03-31 11:56:48 +00:00
|
|
|
if counterNumber > 1
|
|
|
|
counter.text(counterNumber - 1)
|
|
|
|
@removeMeFromUserList($emojiBtn, emoji)
|
2016-05-25 13:05:56 +00:00
|
|
|
else if (emoji == 'thumbsup' || emoji == 'thumbsdown') && isntNoteBody
|
|
|
|
$emojiBtn.tooltip('destroy')
|
2016-03-31 11:56:48 +00:00
|
|
|
counter.text('0')
|
|
|
|
@removeMeFromUserList($emojiBtn, emoji)
|
2015-11-11 13:12:51 +00:00
|
|
|
else
|
2016-05-25 13:05:56 +00:00
|
|
|
$emojiBtn.tooltip('destroy')
|
2016-03-31 11:56:48 +00:00
|
|
|
$emojiBtn.remove()
|
2016-05-11 20:49:47 +00:00
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
$emojiBtn.removeClass('active')
|
2016-03-31 11:56:48 +00:00
|
|
|
|
2016-05-24 00:44:19 +00:00
|
|
|
|
|
|
|
getAwardTooltip: ($awardBlock) ->
|
|
|
|
|
|
|
|
return $awardBlock.attr('data-original-title') or $awardBlock.attr('data-title')
|
|
|
|
|
|
|
|
|
2016-03-31 11:56:48 +00:00
|
|
|
removeMeFromUserList: ($emojiBtn, emoji) ->
|
2016-05-24 00:44:19 +00:00
|
|
|
|
|
|
|
awardBlock = $emojiBtn
|
|
|
|
originalTitle = @getAwardTooltip awardBlock
|
|
|
|
|
|
|
|
authors = originalTitle.split ', '
|
|
|
|
authors.splice authors.indexOf('me'), 1
|
|
|
|
|
|
|
|
newAuthors = authors.join ', '
|
|
|
|
|
|
|
|
awardBlock
|
|
|
|
.closest '.js-emoji-btn'
|
|
|
|
.removeData 'original-title'
|
|
|
|
.removeData 'title'
|
|
|
|
.attr 'data-original-title', newAuthors
|
|
|
|
.attr 'data-title', newAuthors
|
|
|
|
|
|
|
|
@resetTooltip(awardBlock)
|
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
|
|
|
|
addMeToUserList: (emoji) ->
|
2016-05-24 00:44:19 +00:00
|
|
|
|
|
|
|
awardBlock = @findEmojiIcon(emoji).parent()
|
|
|
|
origTitle = @getAwardTooltip awardBlock
|
|
|
|
users = []
|
|
|
|
|
2016-01-18 20:40:48 +00:00
|
|
|
if origTitle
|
2016-05-24 00:44:19 +00:00
|
|
|
users = origTitle.trim().split(', ')
|
|
|
|
|
|
|
|
users.push('me')
|
2016-05-25 13:05:56 +00:00
|
|
|
awardBlock.attr('title', users.join(', '))
|
2016-05-24 00:44:19 +00:00
|
|
|
|
|
|
|
@resetTooltip(awardBlock)
|
|
|
|
|
2015-11-18 13:43:53 +00:00
|
|
|
|
|
|
|
resetTooltip: (award) ->
|
2016-05-25 13:05:56 +00:00
|
|
|
award.tooltip('destroy')
|
2015-11-18 13:43:53 +00:00
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
# 'destroy' call is asynchronous and there is no appropriate callback on it, this is why we need to set timeout.
|
2015-11-18 13:43:53 +00:00
|
|
|
setTimeout (->
|
|
|
|
award.tooltip()
|
|
|
|
), 200
|
2016-01-15 15:04:48 +00:00
|
|
|
|
2016-05-11 20:33:27 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
createEmoji_: (emoji) ->
|
|
|
|
|
|
|
|
emojiCssClass = @resolveNameToCssClass emoji
|
2015-12-17 12:29:55 +00:00
|
|
|
|
2016-03-31 17:19:38 +00:00
|
|
|
buttonHtml = "<button class='btn award-control js-emoji-btn has-tooltip active' title='me' data-placement='bottom'>
|
2016-03-31 08:21:33 +00:00
|
|
|
<div class='icon emoji-icon #{emojiCssClass}' data-emoji='#{emoji}'></div>
|
|
|
|
<span class='award-control-text js-counter'>1</span>
|
|
|
|
</button>"
|
|
|
|
|
|
|
|
emoji_node = $(buttonHtml)
|
2016-05-23 18:33:58 +00:00
|
|
|
.insertBefore '.js-awards-block .js-award-holder:not(.js-award-action-btn)'
|
|
|
|
.find '.emoji-icon'
|
|
|
|
.data 'emoji', emoji
|
|
|
|
|
2016-03-01 11:25:59 +00:00
|
|
|
$('.award-control').tooltip()
|
2015-11-18 13:43:53 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
$currentBlock = $ '.js-awards-block'
|
|
|
|
|
|
|
|
if $currentBlock.is '.hidden'
|
2016-03-31 11:08:31 +00:00
|
|
|
$currentBlock.removeClass 'hidden'
|
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
|
|
|
|
createEmoji: (emoji) ->
|
|
|
|
|
|
|
|
return @createEmoji_ emoji if $('.emoji-menu').length
|
|
|
|
|
2016-05-24 16:34:08 +00:00
|
|
|
awardMenuUrl = gl.awardMenuUrl or '/emojis'
|
2016-05-23 18:33:58 +00:00
|
|
|
@createEmojiMenu awardMenuUrl, => @createEmoji emoji
|
|
|
|
|
|
|
|
|
2015-12-17 12:29:55 +00:00
|
|
|
resolveNameToCssClass: (emoji) ->
|
2016-05-23 18:33:58 +00:00
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
emoji_icon = $(".emoji-menu-content [data-emoji='#{emoji}']")
|
2015-11-18 21:59:58 +00:00
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
if emoji_icon.length > 0
|
2016-05-25 13:05:56 +00:00
|
|
|
unicodeName = emoji_icon.data('unicode-name')
|
2015-12-24 09:28:51 +00:00
|
|
|
else
|
|
|
|
# Find by alias
|
2016-05-25 13:05:56 +00:00
|
|
|
unicodeName = $(".emoji-menu-content [data-aliases*=':#{emoji}:']").data('unicode-name')
|
2015-12-24 09:28:51 +00:00
|
|
|
|
2016-05-23 18:33:58 +00:00
|
|
|
return "emoji-#{unicodeName}"
|
|
|
|
|
2015-11-11 13:12:51 +00:00
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
postEmoji: (awardUrl, emoji, callback) ->
|
|
|
|
$.post awardUrl, { name: emoji }, (data) ->
|
2015-11-11 13:12:51 +00:00
|
|
|
if data.ok
|
2015-11-17 14:44:58 +00:00
|
|
|
callback.call()
|
|
|
|
|
|
|
|
findEmojiIcon: (emoji) ->
|
2016-05-23 18:33:58 +00:00
|
|
|
$(".js-awards-block.awards > .js-emoji-btn [data-emoji='#{emoji}']")
|
2015-12-02 08:36:11 +00:00
|
|
|
|
|
|
|
scrollToAwards: ->
|
|
|
|
$('body, html').animate({
|
|
|
|
scrollTop: $('.awards').offset().top - 80
|
|
|
|
}, 200)
|
2015-12-11 11:10:00 +00:00
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
normilizeEmojiName: (emoji) ->
|
|
|
|
@aliases[emoji] || emoji
|
|
|
|
|
2015-12-22 21:04:19 +00:00
|
|
|
addEmojiToFrequentlyUsedList: (emoji) ->
|
2016-03-31 08:21:33 +00:00
|
|
|
frequently_used_emojis = @getFrequentlyUsedEmojis()
|
|
|
|
frequently_used_emojis.push(emoji)
|
2016-05-25 13:05:56 +00:00
|
|
|
$.cookie('frequently_used_emojis', frequently_used_emojis.join(','), { expires: 365 })
|
2015-12-22 21:04:19 +00:00
|
|
|
|
|
|
|
getFrequentlyUsedEmojis: ->
|
2016-05-25 13:05:56 +00:00
|
|
|
frequently_used_emojis = ($.cookie('frequently_used_emojis') || '').split(',')
|
2016-03-31 08:21:33 +00:00
|
|
|
_.compact(_.uniq(frequently_used_emojis))
|
2015-12-22 21:04:19 +00:00
|
|
|
|
|
|
|
renderFrequentlyUsedBlock: ->
|
2015-12-28 08:11:01 +00:00
|
|
|
if $.cookie('frequently_used_emojis')
|
2016-03-31 08:21:33 +00:00
|
|
|
frequently_used_emojis = @getFrequentlyUsedEmojis()
|
2015-12-22 21:04:19 +00:00
|
|
|
|
2016-03-31 09:55:37 +00:00
|
|
|
ul = $("<ul class='clearfix emoji-menu-list'>")
|
2015-12-22 21:04:19 +00:00
|
|
|
|
2016-03-31 08:21:33 +00:00
|
|
|
for emoji in frequently_used_emojis
|
2016-05-25 13:05:56 +00:00
|
|
|
$(".emoji-menu-content [data-emoji='#{emoji}']").closest('li').clone().appendTo(ul)
|
2015-12-22 21:04:19 +00:00
|
|
|
|
2016-05-25 13:05:56 +00:00
|
|
|
$('input.emoji-search').after(ul).after($('<h5>').text('Frequently used'))
|
2015-12-22 21:04:19 +00:00
|
|
|
|
2015-12-23 10:01:31 +00:00
|
|
|
setupSearch: ->
|
2016-05-25 13:05:56 +00:00
|
|
|
$('input.emoji-search').on 'keyup', (ev) =>
|
2015-12-23 10:01:31 +00:00
|
|
|
term = $(ev.target).val()
|
|
|
|
|
|
|
|
# Clean previous search results
|
2016-05-25 13:05:56 +00:00
|
|
|
$('ul.emoji-menu-search, h5.emoji-search').remove()
|
2015-12-23 10:01:31 +00:00
|
|
|
|
|
|
|
if term
|
2015-12-24 13:44:43 +00:00
|
|
|
# Generate a search result block
|
2016-05-25 13:05:56 +00:00
|
|
|
h5 = $('<h5>').text('Search results').addClass('emoji-search')
|
2016-03-31 08:21:33 +00:00
|
|
|
found_emojis = @searchEmojis(term).show()
|
2016-05-25 13:05:56 +00:00
|
|
|
ul = $('<ul>').addClass('emoji-menu-list emoji-menu-search').append(found_emojis)
|
|
|
|
$('.emoji-menu-content ul, .emoji-menu-content h5').hide()
|
|
|
|
$('.emoji-menu-content').append(h5).append(ul)
|
2015-12-23 10:01:31 +00:00
|
|
|
else
|
2016-05-25 13:05:56 +00:00
|
|
|
$('.emoji-menu-content').children().show()
|
2015-12-23 10:01:31 +00:00
|
|
|
|
|
|
|
searchEmojis: (term)->
|
2016-05-25 13:05:56 +00:00
|
|
|
$(".emoji-menu-content [data-emoji*='#{term}']").closest('li').clone()
|