1
0
mirror of https://github.com/wbbaddons/Tims-Chat.git synced 2024-10-31 14:10:08 +00:00
Tims-Chat/file/js/TimWolla.WCF.Chat.coffee

453 lines
14 KiB
CoffeeScript
Raw Normal View History

2011-12-19 14:46:40 +00:00
###
2011-12-19 14:52:00 +00:00
# TimWolla.WCF.Chat
2011-12-26 19:02:23 +00:00
#
# @author Tim Düsterhus
2012-01-28 16:50:33 +00:00
# @copyright 2010-2012 Tim Düsterhus
2011-12-26 19:02:23 +00:00
# @license Creative Commons Attribution-NonCommercial-ShareAlike <http://creativecommons.org/licenses/by-nc-sa/3.0/legalcode>
# @package timwolla.wcf.chat
2011-12-19 14:46:40 +00:00
###
TimWolla ?= {}
TimWolla.WCF ?= {}
consoleMock ?=
log: () ->,
warn: () ->
(($, window, console) ->
TimWolla.WCF.Chat =
2012-01-28 15:09:24 +00:00
# Templates
titleTemplate: null
messageTemplate: null
2012-01-28 15:09:24 +00:00
# Notifications
2011-12-26 21:16:37 +00:00
newMessageCount: null
2012-01-28 13:08:51 +00:00
isActive: true
2012-01-28 15:09:24 +00:00
# Autocompleter
2012-01-28 13:08:51 +00:00
autocompleteOffset: 0
autocompleteValue: null
2012-01-28 15:09:24 +00:00
# Autoscroll
oldScrollTop: null
2012-01-28 15:09:24 +00:00
# Events
2012-01-12 19:04:28 +00:00
events:
newMessage: $.Callbacks()
userMenu: $.Callbacks()
2011-12-26 15:03:16 +00:00
init: () ->
2012-01-30 16:03:49 +00:00
console.log '[TimWolla.WCF.Chat] Initializing'
2011-12-24 16:47:07 +00:00
@bindEvents()
@events.newMessage.add $.proxy @notify, @
new WCF.PeriodicalExecuter $.proxy(@refreshRoomList, @), 60e3
2012-01-28 15:09:24 +00:00
new WCF.PeriodicalExecuter $.proxy(@getMessages, @), @config.reloadTime * 1e3
@refreshRoomList()
@getMessages()
2011-12-26 19:02:23 +00:00
console.log '[TimWolla.WCF.Chat] Finished initializing'
2011-12-19 14:52:00 +00:00
###
2012-01-28 13:08:51 +00:00
# Autocompletes a username
###
autocomplete: (firstChars, offset = @autocompleteOffset) ->
users = []
2012-01-28 14:57:33 +00:00
# Search all matching users
2012-02-05 16:21:43 +00:00
for user in $ '.timsChatUser'
2012-01-28 13:08:51 +00:00
username = $(user).data('username');
if username.indexOf(firstChars) is 0
users.push username
2012-01-28 14:57:33 +00:00
# None found -> return firstChars
# otherwise return the user at the current offset
2012-01-28 13:08:51 +00:00
return if users.length is 0 then firstChars else users[offset % users.length]
###
2011-12-19 14:52:00 +00:00
# Binds all the events needed for Tims Chat.
###
bindEvents: () ->
2011-12-26 21:16:37 +00:00
$(window).focus $.proxy () ->
2012-02-05 16:21:43 +00:00
document.title = @titleTemplate.fetch
title: $('#timsChatRoomList .activeMenuItem a').text()
2011-12-26 21:16:37 +00:00
@newMessageCount = 0
@isActive = true
, @
2011-12-26 21:16:37 +00:00
$(window).blur $.proxy () ->
@isActive = false
, @
2011-12-26 21:16:37 +00:00
2012-01-28 14:57:33 +00:00
# Insert a smiley
$('.smiley').click $.proxy (event) ->
2011-12-24 16:47:07 +00:00
@insertText ' ' + $(event.target).attr('alt') + ' '
, @
2011-12-26 19:02:23 +00:00
2012-01-28 14:57:33 +00:00
# Switch sidebar tab
2012-02-05 16:21:43 +00:00
$('.timsChatSidebarTabs li').click $.proxy (event) ->
event.preventDefault()
2011-12-24 16:47:07 +00:00
@toggleSidebarContents $ event.target
, @
2011-12-26 19:02:23 +00:00
2012-01-28 14:57:33 +00:00
# Submit Handler
2012-02-05 16:21:43 +00:00
$('#timsChatForm').submit $.proxy (event) ->
event.preventDefault()
2011-12-24 16:47:07 +00:00
@submit $ event.target
, @
2012-01-28 14:57:33 +00:00
# Autocompleter
2012-02-05 16:21:43 +00:00
$('#timsChatInput').keydown $.proxy (event) ->
# tab key
2012-01-28 13:08:51 +00:00
if event.keyCode is 9
event.preventDefault()
if @autocompleteValue is null
2012-02-05 16:21:43 +00:00
@autocompleteValue = $('#timsChatInput').val()
2012-01-28 13:08:51 +00:00
firstChars = @autocompleteValue.substring(@autocompleteValue.lastIndexOf(' ')+1)
console.log '[TimWolla.WCF.Chat] Autocompleting "' + firstChars + '"'
return if firstChars.length is 0
2012-01-28 14:57:33 +00:00
# Insert name and increment offset
2012-02-05 16:21:43 +00:00
$('#timsChatInput').val(@autocompleteValue.substring(0, @autocompleteValue.lastIndexOf(' ') + 1) + @autocomplete(firstChars) + ', ')
2012-01-28 13:08:51 +00:00
@autocompleteOffset++
else
@autocompleteOffset = 0
@autocompleteValue = null
, @
2012-01-28 14:57:33 +00:00
# Refreshes the roomlist
2012-02-05 16:21:43 +00:00
$('#timsChatRoomList button').click $.proxy(@refreshRoomList, @)
2012-01-28 14:57:33 +00:00
# Clears the stream
2012-02-05 16:21:43 +00:00
$('#timsChatClear').click (event) ->
event.preventDefault()
$('.timsChatMessage').remove()
@oldScrollTop = $('.timsChatMessageContainer').scrollTop()
$('.timsChatMessageContainer').scrollTop $('.timsChatMessageContainer ul').height()
$('#timsChatInput').focus()
2012-01-28 14:57:33 +00:00
# Toggle Buttons
2012-02-05 16:21:43 +00:00
$('.timsChatToggle').click (event) ->
element = $ @
icon = element.find 'img'
2011-12-24 16:48:08 +00:00
if element.data('status') is 1
element.data 'status', 0
icon.attr 'src', icon.attr('src').replace /enabled(\d?).([a-z]{3})$/, 'disabled$1.$2'
element.attr 'title', element.data 'enableMessage'
else
element.data 'status', 1
icon.attr 'src', icon.attr('src').replace /disabled(\d?).([a-z]{3})$/, 'enabled$1.$2'
element.attr 'title', element.data 'disableMessage'
2012-01-28 14:57:33 +00:00
# Immediatly scroll down when activating autoscroll
2012-02-05 16:21:43 +00:00
$('#timsChatAutoscroll').click (event) ->
$(this).removeClass('hot')
if $(this).data 'status'
2012-02-05 16:21:43 +00:00
$('.timsChatMessageContainer').scrollTop $('.timsChatMessageContainer ul').height()
@oldScrollTop = $('.timsChatMessageContainer').scrollTop()
2012-01-28 20:48:06 +00:00
2012-01-28 14:57:33 +00:00
# Desktop Notifications
2012-01-28 15:09:24 +00:00
unless typeof window.webkitNotifications is 'undefined'
2012-02-05 16:21:43 +00:00
$('#timsChatNotify').click (event) ->
window.webkitNotifications.requestPermission() if $(this).data 'status'
2012-01-21 17:43:58 +00:00
2011-12-19 14:52:00 +00:00
###
# Changes the chat-room.
#
# @param jQuery-object target
###
changeRoom: (target) ->
window.history.replaceState {}, '', target.attr('href')
$.ajax target.attr('href'),
dataType: 'json'
data:
ajax: 1
type: 'POST'
success: $.proxy((data, textStatus, jqXHR) ->
2011-12-24 16:47:07 +00:00
@loading = false
target.parent().removeClass 'ajaxLoad'
2012-01-28 14:57:33 +00:00
# Mark as active
2012-02-05 16:21:43 +00:00
$('.activeMenuItem .timsChatRoom').parent().removeClass 'activeMenuItem'
target.parent().addClass 'activeMenuItem'
2011-12-26 19:02:23 +00:00
2012-01-28 14:57:33 +00:00
# Set new topic
2011-12-24 16:48:08 +00:00
if data.topic is ''
return if $('#topic').text().trim() is ''
$('#topic').wcfBlindOut 'vertical', () ->
$(@).text ''
else
$('#topic').text data.topic
2012-01-21 16:12:49 +00:00
$('#topic').wcfBlindIn() if $('#topic').text().trim() isnt '' and $('#topic').is(':hidden')
2012-02-05 16:21:43 +00:00
$('.timsChatMessage').animate('opacity', .8);
@handleMessages data.messages
document.title = @titleTemplate.fetch data
, @)
error: () ->
2012-01-28 14:57:33 +00:00
# Reload the page to change the room the old fashion-way
# inclusive the error-message :)
window.location.reload true
beforeSend: $.proxy(() ->
2011-12-24 16:47:07 +00:00
return false if @loading or target.parent().hasClass 'activeMenuItem'
2011-12-24 16:47:07 +00:00
@loading = true
target.parent().addClass 'ajaxLoad'
, @)
2011-12-19 14:52:00 +00:00
###
2011-12-24 16:28:36 +00:00
# Frees the fish
###
freeTheFish: () ->
return if $.wcfIsset('fish')
console.warn '[TimWolla.WCF.Chat] Freeing the fish'
2012-01-28 14:57:33 +00:00
fish = $ '<div id="fish">' + WCF.String.escapeHTML('><((((\u00B0>') + '</div>'
2011-12-24 16:28:36 +00:00
fish.css
position: 'absolute'
top: '150px'
left: '400px'
color: 'black'
textShadow: '1px 1px white'
zIndex: 9999
fish.appendTo $ 'body'
new WCF.PeriodicalExecuter(() ->
2012-01-28 15:09:24 +00:00
left = Math.random() * 100 - 50
top = Math.random() * 100 - 50
fish = $('#fish')
2011-12-24 16:28:36 +00:00
left *= -1 unless fish.width() < (fish.position().left + left) < ($(document).width() - fish.width())
top *= -1 unless fish.height() < (fish.position().top + top) < ($(document).height() - fish.height())
2011-12-24 16:28:36 +00:00
2012-01-28 15:09:24 +00:00
fish.text('><((((\u00B0>') if left > 0
fish.text('<\u00B0))))><') if left < 0
2011-12-24 16:28:36 +00:00
fish.animate
top: '+=' + top
2011-12-24 16:28:36 +00:00
left: '+=' + left
, 1e3
, 1.5e3);
2011-12-24 16:28:36 +00:00
###
2011-12-19 14:52:00 +00:00
# Loads new messages.
###
getMessages: () ->
2011-12-26 16:01:24 +00:00
$.ajax 'index.php/Chat/Message/',
dataType: 'json'
type: 'POST'
success: $.proxy((data, textStatus, jqXHR) ->
@handleMessages(data.messages)
2012-01-12 19:04:28 +00:00
@handleUsers(data.users)
, @)
2011-12-19 14:52:00 +00:00
###
# Inserts the new messages.
#
# @param array<object> messages
###
handleMessages: (messages) ->
2012-01-28 14:57:33 +00:00
# Disable scrolling automagically when user manually scrolled
unless @oldScrollTop is null
2012-02-05 16:21:43 +00:00
if $('.timsChatMessageContainer').scrollTop() < @oldScrollTop
if $('#timsChatAutoscroll').data('status') is 1
$('#timsChatAutoscroll').click()
$('#timsChatAutoscroll').addClass('hot').fadeOut('slow').fadeIn('slow')
2012-01-28 14:57:33 +00:00
# Insert the messages
for message in messages
continue if $.wcfIsset 'timsChatMessage'+message.messageID # Prevent problems with race condition
2011-12-27 16:07:12 +00:00
@events.newMessage.fire message
2011-12-24 16:47:07 +00:00
output = @messageTemplate.fetch message
li = $ '<li></li>'
li.attr 'id', 'timsChatMessage'+message.messageID
2012-02-05 16:21:43 +00:00
li.addClass 'timsChatMessage timsChatMessage'+message.type
2011-12-24 16:48:08 +00:00
li.addClass 'ownMessage' if message.sender is WCF.User.userID
li.append output
2012-02-05 16:21:43 +00:00
li.appendTo $ '.timsChatMessageContainer ul'
2012-01-28 14:57:33 +00:00
# Autoscroll down
2012-02-05 16:21:43 +00:00
if $('#timsChatAutoscroll').data('status') is 1
$('.timsChatMessageContainer').scrollTop $('.timsChatMessageContainer ul').height()
@oldScrollTop = $('.timsChatMessageContainer').scrollTop()
2012-01-28 14:57:33 +00:00
###
# Builds the userlist.
#
# @param array<object> users
###
2012-01-12 19:04:28 +00:00
handleUsers: (users) ->
foundUsers = { }
2012-01-12 19:04:28 +00:00
for user in users
2012-02-05 16:21:43 +00:00
id = 'timsChatUser-'+user.userID
2012-01-12 19:04:28 +00:00
element = $('#'+id)
2012-01-28 14:57:33 +00:00
# Move the user to the correct position
2012-01-12 19:04:28 +00:00
if element[0]
console.log '[TimWolla.WCF.Chat] Moving User: "' + user.username + '"'
2012-01-12 19:04:28 +00:00
element = element.detach()
2012-02-05 16:21:43 +00:00
$('#timsChatUserList').append element
2012-01-28 14:57:33 +00:00
# Insert the user
2012-01-12 19:04:28 +00:00
else
console.log '[TimWolla.WCF.Chat] Inserting User: "' + user.username + '"'
2012-01-12 19:04:28 +00:00
li = $ '<li></li>'
li.attr 'id', id
2012-02-05 16:21:43 +00:00
li.addClass 'timsChatUser'
2012-01-28 13:08:51 +00:00
li.data 'username', user.username
2012-01-12 19:04:28 +00:00
a = $ '<a href="javascript:;">'+user.username+'</a>'
a.click $.proxy (event) ->
event.preventDefault()
@toggleUserMenu $ event.target
, @
2012-01-12 19:04:28 +00:00
li.append a
menu = $ '<ul></ul>'
2012-02-05 16:21:43 +00:00
menu.addClass 'timsChatUserMenu'
menu.append $ '<li><a href="javascript:;">' + WCF.Language.get('wcf.chat.query') + '</a></li>'
menu.append $ '<li><a href="javascript:;">' + WCF.Language.get('wcf.chat.kick') + '</a></li>'
menu.append $ '<li><a href="javascript:;">' + WCF.Language.get('wcf.chat.ban') + '</a></li>'
menu.append $ '<li><a href="index.php/User/' + user.userID + '">' + WCF.Language.get('wcf.chat.profile') + '</a></li>'
2012-01-12 19:04:28 +00:00
@events.userMenu.fire user, menu
li.append menu
2012-02-05 16:21:43 +00:00
li.appendTo $ '#timsChatUserList'
2012-01-12 19:04:28 +00:00
foundUsers[id] = true
# Remove users that were not found
2012-02-05 16:21:43 +00:00
$('.timsChatUser').each () ->
if typeof foundUsers[$(@).attr('id')] is 'undefined'
console.log '[TimWolla.WCF.Chat] Removing User: "' + $(@).data('username') + '"'
$(@).remove();
2012-01-12 19:04:28 +00:00
$('#toggleUsers .badge').text(users.length);
2011-12-19 14:52:00 +00:00
###
# Inserts text into our input.
#
# @param string text
# @param object options
###
insertText: (text, options) ->
options = $.extend
append: true
submit: false
, options or {}
2012-02-05 16:21:43 +00:00
text = $('#timsChatInput').val() + text if options.append
$('#timsChatInput').val(text)
$('#timsChatInput').keyup()
if (options.submit)
2012-02-05 16:21:43 +00:00
$('#timsChatForm').submit()
else
2012-02-05 16:21:43 +00:00
$('#timsChatInput').focus()
2011-12-19 14:52:00 +00:00
###
2012-01-21 17:43:58 +00:00
# Sends a notification about a message.
#
# @param object message
###
notify: (message) ->
2012-02-05 16:21:43 +00:00
return if @isActive or $('#timsChatNotify').data('status') is 0
2012-01-21 17:43:58 +00:00
@newMessageCount++
2012-02-05 16:21:43 +00:00
document.title = '(' + @newMessageCount + ') ' + @titleTemplate.fetch
title: $('#timsChatRoomList .activeMenuItem a').text()
2012-01-21 17:43:58 +00:00
2012-01-28 14:57:33 +00:00
# Desktop Notifications
2012-01-21 17:43:58 +00:00
if typeof window.webkitNotifications isnt 'undefined'
if window.webkitNotifications.checkPermission() is 0
2012-02-05 16:21:43 +00:00
title = WCF.Language.get('wcf.chat.newMessages')
icon = WCF.Icon.get('timwolla.wcf.chat.chat')
content = message.username + message.separator + ' ' + message.message
notification = window.webkitNotifications.createNotification icon, title, content
2012-01-21 17:43:58 +00:00
notification.show()
setTimeout(() ->
notification.cancel()
2012-01-29 14:23:21 +00:00
, 5e3)
2012-01-21 17:43:58 +00:00
###
2011-12-19 14:52:00 +00:00
# Refreshes the room-list.
###
refreshRoomList: () ->
2012-01-28 14:57:33 +00:00
console.log '[TimWolla.WCF.Chat] Refreshing the roomlist'
$('#toggleRooms a').addClass 'ajaxLoad'
$.ajax $('#toggleRooms a').data('refreshUrl'),
dataType: 'json'
type: 'POST'
success: $.proxy((data, textStatus, jqXHR) ->
2012-02-05 16:21:43 +00:00
$('#timsChatRoomList li').remove()
$('#toggleRooms a').removeClass 'ajaxLoad'
2012-01-07 15:45:14 +00:00
$('#toggleRooms .badge').text(data.length);
for room in data
li = $ '<li></li>'
li.addClass 'activeMenuItem' if room.active
2012-02-05 16:21:43 +00:00
$('<a href="' + room.link + '">' + room.title + '</a>').addClass('timsChatRoom').appendTo li
$('#timsChatRoomList ul').append li
2012-02-05 16:21:43 +00:00
$('.timsChatRoom').click $.proxy (event) ->
2011-12-24 16:48:08 +00:00
return if typeof window.history.replaceState is 'undefined'
event.preventDefault()
2011-12-24 16:47:07 +00:00
@changeRoom $ event.target
, @
console.log '[TimWolla.WCF.Chat] Found ' + data.length + ' rooms'
, @)
2011-12-19 14:52:00 +00:00
###
# Handles submitting of messages.
#
# @param jQuery-object target
###
submit: (target) ->
2012-01-28 14:57:33 +00:00
# Break if input contains only whitespace
2012-02-05 16:21:43 +00:00
return false if $('#timsChatInput').val().trim().length is 0
2012-01-28 14:57:33 +00:00
# Finally free the fish
2012-02-05 16:21:43 +00:00
@freeTheFish() if $('#timsChatInput').val().trim().toLowerCase() is '/free the fish'
2011-12-24 16:28:36 +00:00
2012-02-05 16:21:43 +00:00
$.ajax $('#timsChatForm').attr('action'),
data:
2012-02-05 16:21:43 +00:00
text: $('#timsChatInput').val(),
smilies: $('#timsChatSmilies').data('status')
type: 'POST',
beforeSend: (jqXHR) ->
2012-02-05 16:21:43 +00:00
$('#timsChatInput').addClass 'ajaxLoad'
success: $.proxy((data, textStatus, jqXHR) ->
2011-12-24 16:47:07 +00:00
@getMessages()
2012-02-05 16:21:43 +00:00
$('#timsChatInput').val('').focus()
$('#timsChatInput').keyup()
, @)
complete: () ->
2012-02-05 16:21:43 +00:00
$('#timsChatInput').removeClass 'ajaxLoad'
2011-12-19 14:52:00 +00:00
###
# Toggles between user- and room-list.
#
# @param jQuery-object target
###
toggleSidebarContents: (target) ->
return if target.parents('li').hasClass 'active'
if target.parents('li').attr('id') is 'toggleUsers'
$('#toggleUsers').addClass 'active'
$('#toggleRooms').removeClass 'active'
2012-02-05 16:21:43 +00:00
$('#timsChatRoomList').hide()
$('#timsChatUserList').show()
else if target.parents('li').attr('id') is 'toggleRooms'
$('#toggleRooms').addClass 'active'
$('#toggleUsers').removeClass 'active'
2012-02-05 16:21:43 +00:00
$('#timsChatUserList').hide()
$('#timsChatRoomList').show()
2011-12-19 14:52:00 +00:00
###
# Toggles the user-menu.
#
# @param jQuery-object target
###
toggleUserMenu: (target) ->
2012-01-12 19:04:28 +00:00
li = target.parent()
2012-01-12 19:04:28 +00:00
if li.hasClass 'activeMenuItem'
2012-02-05 16:21:43 +00:00
li.find('.timsChatUserMenu').wcfBlindOut 'vertical', () ->
2012-01-12 19:04:28 +00:00
li.removeClass 'activeMenuItem'
else
2012-01-12 19:04:28 +00:00
li.addClass 'activeMenuItem'
2012-02-05 16:21:43 +00:00
li.find('.timsChatUserMenu').wcfBlindIn 'vertical'
)(jQuery, @, consoleMock)