gitlab-org--gitlab-foss/app/assets/javascripts/awards_handler.coffee

373 lines
9.8 KiB
CoffeeScript
Raw Normal View History

2015-11-11 13:12:51 +00:00
class @AwardsHandler
2016-05-25 13:05:56 +00:00
constructor: ->
2016-05-25 13:05:56 +00:00
2016-06-03 21:16:13 +00:00
@aliases = gl.emojiAliases()
2016-03-31 11:56:48 +00:00
$(document)
2016-05-25 13:05:56 +00:00
.off 'click', '.js-add-award'
2016-05-27 15:29:27 +00:00
.on 'click', '.js-add-award', (e) =>
e.stopPropagation()
e.preventDefault()
2016-05-27 15:29:27 +00:00
@showEmojiMenu $(e.currentTarget)
2016-05-27 15:29:27 +00:00
$('html').on 'click', (e) ->
$target = $ e.target
unless $target.closest('.emoji-menu-content').length
$('.js-awards-block.current').removeClass 'current'
unless $target.closest('.emoji-menu').length
2016-05-25 13:05:56 +00:00
if $('.emoji-menu').is(':visible')
$('.js-add-award.is-active').removeClass 'is-active'
2016-05-25 13:05:56 +00:00
$('.emoji-menu').removeClass 'is-visible'
$(document)
2016-05-25 13:05:56 +00:00
.off 'click', '.js-emoji-btn'
2016-05-27 15:29:27 +00:00
.on 'click', '.js-emoji-btn', (e) =>
e.preventDefault()
2016-05-27 15:29:27 +00:00
$target = $ e.currentTarget
emoji = $target.find('.icon').data 'emoji'
2016-05-27 15:29:27 +00:00
$target.closest('.js-awards-block').addClass 'current'
@addAward @getVotesBlock(), @getAwardUrl(), emoji
2016-05-11 20:49:47 +00:00
showEmojiMenu: ($addBtn) ->
2016-05-27 15:29:27 +00:00
$menu = $ '.emoji-menu'
2016-05-30 16:33:43 +00:00
if $addBtn.hasClass 'js-note-emoji'
2016-06-10 21:24:24 +00:00
$addBtn.closest('.note').find('.js-awards-block').addClass 'current'
2016-05-27 15:29:27 +00:00
else
$addBtn.closest('.js-awards-block').addClass 'current'
if $menu.length
$holder = $addBtn.closest('.js-award-holder')
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'
@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
$addBtn.addClass 'is-loading is-active'
2016-05-26 09:15:20 +00:00
url = @getAwardMenuUrl()
@createEmojiMenu url, =>
$addBtn.removeClass 'is-loading'
$menu = $('.emoji-menu')
@positionMenu($menu, $addBtn)
2016-06-09 00:02:49 +00:00
@renderFrequentlyUsedBlock() unless @frequentEmojiBlockRendered
2016-03-11 09:47:18 +00:00
setTimeout =>
$menu.addClass 'is-visible'
$('#emoji_search').focus()
2016-03-11 09:47:18 +00:00
@setupSearch()
, 200
createEmojiMenu: (awardMenuUrl, callback) ->
2016-05-27 15:29:27 +00:00
$.get awardMenuUrl, (response) ->
$('body').append response
callback()
positionMenu: ($menu, $addBtn) ->
2016-05-26 09:15:20 +00:00
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'
else
css.left = "#{$addBtn.offset().left}px"
2016-05-25 13:05:56 +00:00
$menu.removeClass 'is-aligned-right'
$menu.css(css)
addAward: (votesBlock, awardUrl, emoji, checkMutuality = true, callback) ->
2016-05-27 15:29:27 +00:00
emoji = @normilizeEmojiName emoji
2016-05-27 15:29:27 +00:00
@postEmoji awardUrl, emoji, =>
@addAwardToEmojiBar votesBlock, emoji, checkMutuality
callback?()
$('.emoji-menu').removeClass 'is-visible'
addAwardToEmojiBar: (votesBlock, emoji, checkForMutuality = true) ->
@checkMutuality votesBlock, emoji if checkForMutuality
@addEmojiToFrequentlyUsedList emoji
emoji = @normilizeEmojiName emoji
$emojiButton = @findEmojiIcon(votesBlock, emoji).parent()
if $emojiButton.length > 0
if @isActive $emojiButton
@decrementCounter $emojiButton, emoji
else
counter = $emojiButton.find '.js-counter'
2016-05-27 15:29:27 +00:00
counter.text parseInt(counter.text()) + 1
$emojiButton.addClass 'active'
@addMeToUserList votesBlock, emoji
@animateEmoji $emojiButton
else
votesBlock.removeClass 'hidden'
@createEmoji votesBlock, emoji
2015-11-11 13:12:51 +00:00
getVotesBlock: ->
currentBlock = $ '.js-awards-block.current'
return if currentBlock.length then currentBlock else $('.js-awards-block').eq 0
2016-05-27 15:29:27 +00:00
getAwardUrl: -> return @getVotesBlock().data 'award-url'
checkMutuality: (votesBlock, emoji) ->
awardUrl = @getAwardUrl()
if emoji in [ 'thumbsup', 'thumbsdown' ]
mutualVote = if emoji is 'thumbsup' then 'thumbsdown' else 'thumbsup'
$emojiButton = votesBlock.find("[data-emoji=#{mutualVote}]").parent()
isAlreadyVoted = $emojiButton.hasClass 'active'
if isAlreadyVoted
@showEmojiLoader $emojiButton
@addAward votesBlock, awardUrl, mutualVote, false, ->
$emojiButton.removeClass 'is-loading'
showEmojiLoader: ($emojiButton) ->
$loader = $emojiButton.find '.fa-spinner'
unless $loader.length
$emojiButton.append '<i class="fa fa-spinner fa-spin award-control-icon award-control-icon-loading"></i>'
$emojiButton.addClass 'is-loading'
isActive: ($emojiButton) -> $emojiButton.hasClass 'active'
2016-05-25 13:05:56 +00:00
2016-03-31 11:56:48 +00:00
decrementCounter: ($emojiButton, emoji) ->
2016-03-31 11:56:48 +00:00
2016-06-03 21:16:13 +00:00
counter = $ '.js-counter', $emojiButton
counterNumber = parseInt counter.text(), 10
2016-03-31 11:56:48 +00:00
if counterNumber > 1
counter.text counterNumber - 1
@removeMeFromUserList $emojiButton, emoji
2016-05-27 15:29:27 +00:00
else if emoji is 'thumbsup' or emoji is 'thumbsdown'
$emojiButton.tooltip 'destroy'
counter.text '0'
@removeMeFromUserList $emojiButton, emoji
@removeEmoji $emojiButton if $emojiButton.parents('.note').length
2015-11-11 13:12:51 +00:00
else
@removeEmoji $emojiButton
2016-05-11 20:49:47 +00:00
$emojiButton.removeClass 'active'
2016-03-31 11:56:48 +00:00
2016-05-24 00:44:19 +00:00
removeEmoji: ($emojiButton) ->
2016-05-27 15:29:27 +00:00
$emojiButton.tooltip('destroy')
$emojiButton.remove()
2016-05-27 15:29:27 +00:00
$votesBlock = @getVotesBlock()
if $votesBlock.find('.js-emoji-btn').length is 0
$votesBlock.addClass 'hidden'
2016-05-24 00:44:19 +00:00
getAwardTooltip: ($awardBlock) ->
2016-05-31 19:32:29 +00:00
return $awardBlock.attr('data-original-title') or $awardBlock.attr('data-title') or ''
2016-05-24 00:44:19 +00:00
removeMeFromUserList: ($emojiButton, emoji) ->
2016-05-24 00:44:19 +00:00
awardBlock = $emojiButton
2016-05-24 00:44:19 +00:00
originalTitle = @getAwardTooltip awardBlock
authors = originalTitle.split ', '
authors.splice authors.indexOf('me'), 1
newAuthors = authors.join ', '
awardBlock
.closest '.js-emoji-btn'
.removeData 'original-title'
.attr 'data-original-title', newAuthors
2016-05-31 19:32:29 +00:00
@resetTooltip awardBlock
2016-05-24 00:44:19 +00:00
addMeToUserList: (votesBlock, emoji) ->
2016-05-24 00:44:19 +00:00
awardBlock = @findEmojiIcon(votesBlock, emoji).parent()
2016-05-24 00:44:19 +00:00
origTitle = @getAwardTooltip awardBlock
users = []
if origTitle
2016-05-31 19:32:29 +00:00
users = origTitle.trim().split ', '
2016-05-24 00:44:19 +00:00
2016-05-31 19:32:29 +00:00
users.push 'me'
awardBlock.attr 'title', users.join ', '
2016-05-24 00:44:19 +00:00
2016-05-31 19:32:29 +00:00
@resetTooltip awardBlock
2016-05-24 00:44:19 +00:00
2015-11-18 13:43:53 +00:00
resetTooltip: (award) ->
2016-05-31 19:32:29 +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.
2016-05-31 19:32:29 +00:00
cb = -> award.tooltip()
setTimeout cb, 200
createEmoji_: (votesBlock, emoji) ->
emojiCssClass = @resolveNameToCssClass emoji
buttonHtml = "<button class='btn award-control js-emoji-btn has-tooltip active' title='me' data-placement='bottom'>
<div class='icon emoji-icon #{emojiCssClass}' data-emoji='#{emoji}'></div>
<span class='award-control-text js-counter'>1</span>
</button>"
2016-05-30 13:27:25 +00:00
$emojiButton = $ buttonHtml
2016-06-03 21:16:13 +00:00
$emojiButton
.insertBefore votesBlock.find '.js-award-holder'
.find '.emoji-icon'
.data 'emoji', emoji
2016-05-30 13:27:25 +00:00
@animateEmoji $emojiButton
$('.award-control').tooltip()
votesBlock.removeClass 'current'
2016-05-30 13:27:25 +00:00
animateEmoji: ($emoji) ->
className = 'pulse animated'
$emoji.addClass className
setTimeout (-> $emoji.removeClass className), 321
createEmoji: (votesBlock, emoji) ->
if $('.emoji-menu').length
return @createEmoji_ votesBlock, emoji
2016-06-03 21:16:13 +00:00
@createEmojiMenu @getAwardMenuUrl(), => @createEmoji_ votesBlock, emoji
2016-05-26 09:15:20 +00:00
getAwardMenuUrl: -> return gon.award_menu_url
resolveNameToCssClass: (emoji) ->
2016-06-03 21:16:13 +00:00
emojiIcon = $ ".emoji-menu-content [data-emoji='#{emoji}']"
2016-06-03 21:16:13 +00:00
if emojiIcon.length > 0
unicodeName = emojiIcon.data 'unicode-name'
2015-12-24 09:28:51 +00:00
else
# Find by alias
2016-06-03 21:16:13 +00:00
unicodeName = $(".emoji-menu-content [data-aliases*=':#{emoji}:']").data 'unicode-name'
2015-12-24 09:28:51 +00:00
return "emoji-#{unicodeName}"
2015-11-11 13:12:51 +00:00
postEmoji: (awardUrl, emoji, callback) ->
2016-05-27 15:29:27 +00:00
$.post awardUrl, { name: emoji }, (data) ->
2016-06-03 21:16:13 +00:00
callback() if data.ok
2016-05-27 15:29:27 +00:00
findEmojiIcon: (votesBlock, emoji) ->
2016-05-27 15:29:27 +00:00
return votesBlock.find ".js-emoji-btn [data-emoji='#{emoji}']"
2016-05-27 15:29:27 +00:00
scrollToAwards: ->
2015-12-11 11:10:00 +00:00
options = scrollTop: $('.awards').offset().top - 110
2016-05-27 15:29:27 +00:00
$('body, html').animate options, 200
normilizeEmojiName: (emoji) -> return @aliases[emoji] or emoji
addEmojiToFrequentlyUsedList: (emoji) ->
2016-05-27 15:29:27 +00:00
frequentlyUsedEmojis = @getFrequentlyUsedEmojis()
frequentlyUsedEmojis.push emoji
$.cookie 'frequently_used_emojis', frequentlyUsedEmojis.join(','), { expires: 365 }
getFrequentlyUsedEmojis: ->
2016-05-27 15:29:27 +00:00
frequentlyUsedEmojis = ($.cookie('frequently_used_emojis') or '').split(',')
return _.compact _.uniq frequentlyUsedEmojis
renderFrequentlyUsedBlock: ->
2016-05-27 15:29:27 +00:00
if $.cookie 'frequently_used_emojis'
frequentlyUsedEmojis = @getFrequentlyUsedEmojis()
ul = $("<ul class='clearfix emoji-menu-list frequent-emojis'>")
2016-05-27 15:29:27 +00:00
for emoji in frequentlyUsedEmojis
2016-05-25 13:05:56 +00:00
$(".emoji-menu-content [data-emoji='#{emoji}']").closest('li').clone().appendTo(ul)
$('.emoji-menu-content')
.prepend(ul)
.prepend($('<h5>').text('Frequently used'))
@frequentEmojiBlockRendered = true
2016-06-09 00:02:49 +00:00
2016-05-27 15:29:27 +00:00
2015-12-23 10:01:31 +00:00
setupSearch: ->
2016-05-27 15:29:27 +00:00
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
h5 = $('<h5>').text('Search results')
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
2016-05-27 15:29:27 +00:00
searchEmojis: (term) ->
$(".emoji-menu-list:not(.frequent-emojis) [data-emoji*='#{term}']").closest('li').clone()