2013-06-20 04:42:20 +00:00
/* window focus/blur */
$ ( window ) . focus ( function ( ) {
FOCUSED = true ;
clearInterval ( TITLE _BLINK ) ;
TITLE _BLINK = false ;
document . title = PAGETITLE ;
} ) . blur ( function ( ) {
FOCUSED = false ;
} ) ;
2013-10-03 03:26:28 +00:00
$ ( "#togglemotd" ) . click ( function ( ) {
var hidden = $ ( "#motd" ) . css ( "display" ) === "none" ;
$ ( "#motd" ) . toggle ( ) ;
if ( hidden ) {
2014-01-31 05:02:58 +00:00
$ ( "#togglemotd" ) . find ( ".glyphicon-plus" )
. removeClass ( "glyphicon-plus" )
. addClass ( "glyphicon-minus" ) ;
2013-10-03 03:26:28 +00:00
} else {
2014-01-31 05:02:58 +00:00
$ ( "#togglemotd" ) . find ( ".glyphicon-minus" )
. removeClass ( "glyphicon-minus" )
. addClass ( "glyphicon-plus" ) ;
2013-10-03 03:26:28 +00:00
}
} ) ;
2013-06-07 03:13:24 +00:00
/* chatbox */
2013-08-08 14:39:58 +00:00
2013-09-21 07:22:51 +00:00
$ ( "#modflair" ) . click ( function ( ) {
var m = $ ( "#modflair" ) ;
if ( m . hasClass ( "label-success" ) ) {
USEROPTS . modhat = false ;
2016-03-30 05:30:16 +00:00
m . removeClass ( "label-success" ) ;
if ( SUPERADMIN ) {
USEROPTS . adminhat = true ;
m . addClass ( "label-danger" ) ;
} else {
m . addClass ( "label-default" ) ;
}
} else if ( m . hasClass ( "label-danger" ) ) {
2013-09-21 07:22:51 +00:00
USEROPTS . adminhat = false ;
2013-12-19 17:14:48 +00:00
m . removeClass ( "label-danger" )
2016-03-30 05:30:16 +00:00
. addClass ( "label-default" ) ;
2013-09-21 07:22:51 +00:00
} else {
2016-03-30 05:30:16 +00:00
USEROPTS . modhat = true ;
2013-09-21 07:22:51 +00:00
m . removeClass ( "label-default" )
2016-03-30 05:30:16 +00:00
. addClass ( "label-success" ) ;
2013-09-21 07:22:51 +00:00
}
2016-07-12 05:14:26 +00:00
$ ( "#us-modflair" ) . prop ( "checked" , USEROPTS . modhat ) ;
setOpt ( 'modhat' , USEROPTS . modhat ) ;
2013-09-21 07:22:51 +00:00
} ) ;
$ ( "#usercount" ) . mouseenter ( function ( ev ) {
2013-08-08 14:39:58 +00:00
var breakdown = calcUserBreakdown ( ) ;
// re-using profile-box class for convenience
var popup = $ ( "<div/>" )
. addClass ( "profile-box" )
2014-01-30 04:50:14 +00:00
. css ( "top" , ( ev . clientY + 5 ) + "px" )
. css ( "left" , ( ev . clientX ) + "px" )
2013-09-21 07:22:51 +00:00
. appendTo ( $ ( "#usercount" ) ) ;
2013-08-08 14:39:58 +00:00
var contents = "" ;
for ( var key in breakdown ) {
contents += "<strong>" + key + ": </strong>" + breakdown [ key ] ;
contents += "<br>"
}
popup . html ( contents ) ;
} ) ;
2013-09-21 07:22:51 +00:00
$ ( "#usercount" ) . mousemove ( function ( ev ) {
var popup = $ ( "#usercount" ) . find ( ".profile-box" ) ;
2013-08-08 14:39:58 +00:00
if ( popup . length == 0 )
return ;
2014-01-30 04:50:14 +00:00
popup . css ( "top" , ( ev . clientY + 5 ) + "px" ) ;
popup . css ( "left" , ( ev . clientX ) + "px" ) ;
2013-08-08 14:39:58 +00:00
} ) ;
2013-09-21 07:22:51 +00:00
$ ( "#usercount" ) . mouseleave ( function ( ) {
$ ( "#usercount" ) . find ( ".profile-box" ) . remove ( ) ;
2013-08-08 14:39:58 +00:00
} ) ;
2015-12-06 01:57:33 +00:00
$ ( "#messagebuffer" ) . scroll ( function ( ev ) {
if ( IGNORE _SCROLL _EVENT ) {
// Skip event, this was triggered by scrollChat() and not by a user action.
// Reset for next event.
IGNORE _SCROLL _EVENT = false ;
return ;
}
2015-11-29 17:28:03 +00:00
var m = $ ( "#messagebuffer" ) ;
2015-12-06 02:05:23 +00:00
var lastChildHeight = 0 ;
var messages = m . children ( ) ;
if ( messages . length > 0 ) {
lastChildHeight = messages [ messages . length - 1 ] . clientHeight || 0 ;
}
var isCaughtUp = m . height ( ) + m . scrollTop ( ) >= m . prop ( "scrollHeight" ) - lastChildHeight ;
2015-11-29 17:28:03 +00:00
if ( isCaughtUp ) {
SCROLLCHAT = true ;
$ ( "#newmessages-indicator" ) . remove ( ) ;
} else {
SCROLLCHAT = false ;
}
} ) ;
2013-06-20 04:42:20 +00:00
2014-01-06 15:55:12 +00:00
$ ( "#guestname" ) . keydown ( function ( ev ) {
if ( ev . keyCode === 13 ) {
socket . emit ( "login" , {
name : $ ( "#guestname" ) . val ( )
} ) ;
}
} ) ;
2014-01-31 05:02:58 +00:00
function chatTabComplete ( ) {
var words = $ ( "#chatline" ) . val ( ) . split ( " " ) ;
var current = words [ words . length - 1 ] . toLowerCase ( ) ;
2014-01-31 05:17:19 +00:00
if ( ! current . match ( /^[\w-]{1,20}$/ ) ) {
return ;
}
2014-01-31 17:01:51 +00:00
var _ _slice = Array . prototype . slice ;
2014-03-20 03:17:57 +00:00
var usersWithCap = _ _slice . call ( $ ( "#userlist" ) . children ( ) ) . map ( function ( elem ) {
return elem . children [ 1 ] . innerHTML ;
} ) ;
var users = _ _slice . call ( usersWithCap ) . map ( function ( user ) {
return user . toLowerCase ( ) ;
2014-01-31 17:01:51 +00:00
} ) . filter ( function ( name ) {
2014-02-28 06:09:20 +00:00
return name . indexOf ( current ) === 0 ;
2014-01-31 05:02:58 +00:00
} ) ;
// users now contains a list of names that start with current word
2014-01-31 17:01:51 +00:00
2014-01-31 05:02:58 +00:00
if ( users . length === 0 ) {
return ;
}
2014-01-31 17:01:51 +00:00
// trim possible names to the shortest possible completion
var min = Math . min . apply ( Math , users . map ( function ( name ) {
2014-01-31 05:02:58 +00:00
return name . length ;
} ) ) ;
2014-01-31 17:01:51 +00:00
users = users . map ( function ( name ) {
return name . substring ( 0 , min ) ;
} ) ;
// continually trim off letters until all prefixes are the same
var changed = true ;
var iter = 21 ;
while ( changed ) {
changed = false ;
var first = users [ 0 ] ;
for ( var i = 1 ; i < users . length ; i ++ ) {
if ( users [ i ] !== first ) {
changed = true ;
2014-02-01 18:42:49 +00:00
break ;
2014-01-31 17:01:51 +00:00
}
}
if ( changed ) {
2014-02-01 18:42:49 +00:00
users = users . map ( function ( name ) {
return name . substring ( 0 , name . length - 1 ) ;
} ) ;
2014-01-31 17:01:51 +00:00
}
// In the event something above doesn't generate a break condition, limit
// the maximum number of repetitions
if ( -- iter < 0 ) {
break ;
}
}
2014-01-31 05:02:58 +00:00
current = users [ 0 ] . substring ( 0 , min ) ;
2014-03-20 03:17:57 +00:00
for ( var i = 0 ; i < usersWithCap . length ; i ++ ) {
if ( usersWithCap [ i ] . toLowerCase ( ) === current ) {
current = usersWithCap [ i ] ;
break ;
}
}
2014-01-31 05:02:58 +00:00
if ( users . length === 1 ) {
if ( words . length === 1 ) {
current += ":" ;
}
current += " " ;
}
words [ words . length - 1 ] = current ;
$ ( "#chatline" ) . val ( words . join ( " " ) ) ;
}
2013-06-09 18:03:41 +00:00
$ ( "#chatline" ) . keydown ( function ( ev ) {
2013-12-15 03:59:47 +00:00
// Enter/return
2013-06-07 03:13:24 +00:00
if ( ev . keyCode == 13 ) {
2013-11-19 21:14:40 +00:00
if ( CHATTHROTTLE ) {
return ;
}
2013-06-07 03:13:24 +00:00
var msg = $ ( "#chatline" ) . val ( ) ;
if ( msg . trim ( ) ) {
2013-11-17 19:12:56 +00:00
var meta = { } ;
2013-09-21 07:22:51 +00:00
if ( USEROPTS . adminhat && CLIENT . rank >= 255 ) {
msg = "/a " + msg ;
2013-11-17 19:12:56 +00:00
} else if ( USEROPTS . modhat && CLIENT . rank >= Rank . Moderator ) {
meta . modflair = CLIENT . rank ;
2013-06-07 03:13:24 +00:00
}
2013-12-15 03:59:47 +00:00
// The /m command no longer exists, so emulate it clientside
2013-11-17 19:12:56 +00:00
if ( CLIENT . rank >= 2 && msg . indexOf ( "/m " ) === 0 ) {
meta . modflair = CLIENT . rank ;
2013-11-21 23:46:33 +00:00
msg = msg . substring ( 3 ) ;
2013-11-17 19:12:56 +00:00
}
2013-06-07 03:13:24 +00:00
socket . emit ( "chatMsg" , {
2013-11-17 19:12:56 +00:00
msg : msg ,
meta : meta
2013-06-07 03:13:24 +00:00
} ) ;
CHATHIST . push ( $ ( "#chatline" ) . val ( ) ) ;
2013-06-19 21:54:27 +00:00
CHATHISTIDX = CHATHIST . length ;
2013-06-07 03:13:24 +00:00
$ ( "#chatline" ) . val ( "" ) ;
}
return ;
}
2013-12-15 03:59:47 +00:00
else if ( ev . keyCode == 9 ) { // Tab completion
2014-01-31 05:02:58 +00:00
chatTabComplete ( ) ;
2013-06-07 03:13:24 +00:00
ev . preventDefault ( ) ;
return false ;
}
2013-12-15 03:59:47 +00:00
else if ( ev . keyCode == 38 ) { // Up arrow (input history)
2013-06-07 03:13:24 +00:00
if ( CHATHISTIDX == CHATHIST . length ) {
CHATHIST . push ( $ ( "#chatline" ) . val ( ) ) ;
}
if ( CHATHISTIDX > 0 ) {
CHATHISTIDX -- ;
$ ( "#chatline" ) . val ( CHATHIST [ CHATHISTIDX ] ) ;
}
ev . preventDefault ( ) ;
return false ;
}
2013-12-15 03:59:47 +00:00
else if ( ev . keyCode == 40 ) { // Down arrow (input history)
2013-06-07 03:13:24 +00:00
if ( CHATHISTIDX < CHATHIST . length - 1 ) {
CHATHISTIDX ++ ;
$ ( "#chatline" ) . val ( CHATHIST [ CHATHISTIDX ] ) ;
}
ev . preventDefault ( ) ;
return false ;
}
} ) ;
/* poll controls */
$ ( "#newpollbtn" ) . click ( showPollMenu ) ;
/* search controls */
$ ( "#library_search" ) . click ( function ( ) {
2014-09-05 01:53:18 +00:00
if ( ! hasPermission ( "seeplaylist" ) ) {
$ ( "#searchcontrol .alert" ) . remove ( ) ;
var al = makeAlert ( "Permission Denied" ,
"This channel does not allow you to search its library" ,
"alert-danger" ) ;
al . find ( ".alert" ) . insertAfter ( $ ( "#library_query" ) . parent ( ) ) ;
return ;
}
2013-06-07 03:13:24 +00:00
socket . emit ( "searchMedia" , {
source : "library" ,
query : $ ( "#library_query" ) . val ( ) . toLowerCase ( )
} ) ;
} ) ;
$ ( "#library_query" ) . keydown ( function ( ev ) {
if ( ev . keyCode == 13 ) {
2014-09-05 01:53:18 +00:00
if ( ! hasPermission ( "seeplaylist" ) ) {
$ ( "#searchcontrol .alert" ) . remove ( ) ;
var al = makeAlert ( "Permission Denied" ,
"This channel does not allow you to search its library" ,
"alert-danger" ) ;
al . find ( ".alert" ) . insertAfter ( $ ( "#library_query" ) . parent ( ) ) ;
return ;
}
2013-06-07 03:13:24 +00:00
socket . emit ( "searchMedia" , {
source : "library" ,
query : $ ( "#library_query" ) . val ( ) . toLowerCase ( )
} ) ;
}
} ) ;
2013-08-07 17:34:14 +00:00
$ ( "#youtube_search" ) . click ( function ( ) {
var query = $ ( "#library_query" ) . val ( ) . toLowerCase ( ) ;
if ( parseMediaLink ( query ) . type !== null ) {
makeAlert ( "Media Link" , "If you already have the link, paste it " +
"in the 'Media URL' box under Playlist Controls. This " +
"searchbar works like YouTube's search function." ,
2013-12-15 03:59:47 +00:00
"alert-danger" )
2013-08-07 17:34:14 +00:00
. insertBefore ( $ ( "#library" ) ) ;
}
2013-06-07 03:13:24 +00:00
socket . emit ( "searchMedia" , {
source : "yt" ,
2013-08-07 17:34:14 +00:00
query : query
2013-06-07 03:13:24 +00:00
} ) ;
} ) ;
/* user playlists */
$ ( "#userpl_save" ) . click ( function ( ) {
if ( $ ( "#userpl_name" ) . val ( ) . trim ( ) == "" ) {
2013-12-15 03:59:47 +00:00
makeAlert ( "Invalid Name" , "Playlist name cannot be empty" , "alert-danger" )
2013-06-07 03:13:24 +00:00
. insertAfter ( $ ( "#userpl_save" ) . parent ( ) ) ;
return ;
}
2014-02-02 18:41:41 +00:00
socket . emit ( "clonePlaylist" , {
2013-06-07 03:13:24 +00:00
name : $ ( "#userpl_name" ) . val ( )
} ) ;
} ) ;
2013-06-23 18:21:21 +00:00
/* video controls */
2013-06-30 19:56:41 +00:00
$ ( "#mediarefresh" ) . click ( function ( ) {
2015-05-02 22:55:00 +00:00
PLAYER . mediaType = "" ;
PLAYER . mediaId = "" ;
2013-12-15 03:59:47 +00:00
// playerReady triggers the server to send a changeMedia.
// the changeMedia handler then reloads the player
2013-06-30 19:56:41 +00:00
socket . emit ( "playerReady" ) ;
} ) ;
2013-06-07 03:13:24 +00:00
/* playlist controls */
2013-06-07 22:09:36 +00:00
2013-06-11 19:41:03 +00:00
$ ( "#queue" ) . sortable ( {
start : function ( ev , ui ) {
2013-06-29 22:09:20 +00:00
PL _FROM = ui . item . data ( "uid" ) ;
2013-06-11 19:41:03 +00:00
} ,
update : function ( ev , ui ) {
2013-06-27 22:15:29 +00:00
var prev = ui . item . prevAll ( ) ;
if ( prev . length == 0 )
2013-06-29 22:09:20 +00:00
PL _AFTER = "prepend" ;
2013-06-27 22:15:29 +00:00
else
2013-06-29 22:09:20 +00:00
PL _AFTER = $ ( prev [ 0 ] ) . data ( "uid" ) ;
2013-06-27 22:15:29 +00:00
socket . emit ( "moveMedia" , {
from : PL _FROM ,
after : PL _AFTER
} ) ;
2013-10-04 03:11:47 +00:00
$ ( "#queue" ) . sortable ( "cancel" ) ;
2013-06-11 19:41:03 +00:00
}
2013-06-09 18:03:41 +00:00
} ) ;
2013-06-11 19:41:03 +00:00
$ ( "#queue" ) . disableSelection ( ) ;
2013-06-09 18:03:41 +00:00
2014-01-14 06:52:56 +00:00
function queue ( pos , src ) {
if ( ! src ) {
src = "url" ;
}
if ( src === "customembed" ) {
var title = $ ( "#customembed-title" ) . val ( ) ;
if ( ! title ) {
title = false ;
2013-08-22 22:33:03 +00:00
}
2014-01-14 06:52:56 +00:00
var content = $ ( "#customembed-content" ) . val ( ) ;
2013-08-03 19:10:06 +00:00
socket . emit ( "queue" , {
2014-01-14 06:52:56 +00:00
id : content ,
2013-08-22 22:33:03 +00:00
title : title ,
2014-01-14 06:52:56 +00:00
pos : pos ,
2014-02-09 06:24:20 +00:00
type : "cu" ,
temp : $ ( ".add-temp" ) . prop ( "checked" )
2013-08-03 19:10:06 +00:00
} ) ;
2014-01-14 06:52:56 +00:00
} else {
2015-08-20 06:27:05 +00:00
var linkList = $ ( "#mediaurl" ) . val ( ) ;
var links = linkList . split ( ",http" ) . map ( function ( link , i ) {
if ( i > 0 ) {
return "http" + link ;
} else {
return link ;
}
} ) ;
2015-07-17 02:14:55 +00:00
if ( pos === "next" ) links = links . reverse ( ) ;
if ( pos === "next" && $ ( "#queue li" ) . length === 0 ) links . unshift ( links . pop ( ) ) ;
var emitQueue = [ ] ;
var addTemp = $ ( ".add-temp" ) . prop ( "checked" ) ;
var notification = document . getElementById ( "addfromurl-queue" ) ;
if ( ! notification ) {
notification = document . createElement ( "div" ) ;
notification . id = "addfromurl-queue" ;
document . getElementById ( "addfromurl" ) . appendChild ( notification ) ;
2014-06-04 04:21:00 +00:00
}
2014-02-10 02:10:11 +00:00
2015-07-17 02:14:55 +00:00
links . forEach ( function ( link ) {
var data = parseMediaLink ( link ) ;
var duration = undefined ;
var title = undefined ;
if ( data . type === "fi" ) {
title = $ ( "#addfromurl-title-val" ) . val ( ) ;
}
if ( data . id == null || data . type == null ) {
makeAlert ( "Error" , "Failed to parse link " + link +
". Please check that it is correct" ,
2016-07-23 02:22:15 +00:00
"alert-danger" , true )
2015-07-17 02:14:55 +00:00
. insertAfter ( $ ( "#addfromurl" ) ) ;
} else {
emitQueue . push ( {
id : data . id ,
type : data . type ,
pos : pos ,
duration : duration ,
title : title ,
temp : addTemp ,
link : link
} ) ;
}
} ) ;
var nextQueueDelay = 1020 ;
function next ( ) {
var data = emitQueue . shift ( ) ;
if ( ! data ) {
$ ( "#mediaurl" ) . val ( "" ) ;
$ ( "#addfromurl-title" ) . remove ( ) ;
return ;
}
var link = data . link ;
delete data . link ;
socket . emit ( "queue" , data ) ;
2016-05-21 23:13:58 +00:00
startQueueSpinner ( data ) ;
2015-07-17 02:14:55 +00:00
if ( emitQueue . length > 0 ) {
notification . textContent = "Waiting to queue " + emitQueue [ 0 ] . link ;
} else {
notification . textContent = "" ;
}
setTimeout ( next , nextQueueDelay ) ;
2013-06-12 03:37:12 +00:00
}
2015-07-17 02:14:55 +00:00
next ( ) ;
2013-06-20 03:20:56 +00:00
}
2013-06-07 22:09:36 +00:00
}
2014-01-14 06:52:56 +00:00
$ ( "#queue_next" ) . click ( queue . bind ( this , "next" , "url" ) ) ;
$ ( "#queue_end" ) . click ( queue . bind ( this , "end" , "url" ) ) ;
$ ( "#ce_queue_next" ) . click ( queue . bind ( this , "next" , "customembed" ) ) ;
$ ( "#ce_queue_end" ) . click ( queue . bind ( this , "end" , "customembed" ) ) ;
2013-06-07 22:09:36 +00:00
2015-07-10 04:45:41 +00:00
$ ( "#mediaurl" ) . keyup ( function ( ev ) {
2014-01-14 06:52:56 +00:00
if ( ev . keyCode === 13 ) {
queue ( "end" , "url" ) ;
2014-02-10 02:10:11 +00:00
} else {
2014-06-04 04:21:00 +00:00
var url = $ ( "#mediaurl" ) . val ( ) . split ( "?" ) [ 0 ] ;
2015-12-13 19:18:28 +00:00
if ( url . match ( /^https?:\/\/(.*)?\.(flv|mp4|og[gv]|webm|mp3|mov)$/ ) ||
url . match ( /^fi:/ ) ) {
2014-06-04 04:21:00 +00:00
var title = $ ( "#addfromurl-title" ) ;
if ( title . length === 0 ) {
title = $ ( "<div/>" )
. attr ( "id" , "addfromurl-title" )
. appendTo ( $ ( "#addfromurl" ) ) ;
$ ( "<span/>" ) . text ( "Title (optional)" )
. appendTo ( title ) ;
$ ( "<input/>" ) . addClass ( "form-control" )
. attr ( "type" , "text" )
. attr ( "id" , "addfromurl-title-val" )
2015-02-13 19:40:58 +00:00
. keydown ( function ( ev ) {
2014-06-04 04:21:00 +00:00
if ( ev . keyCode === 13 ) {
queue ( "end" , "url" ) ;
}
} )
. appendTo ( $ ( "#addfromurl-title" ) ) ;
}
} else {
$ ( "#addfromurl-title" ) . remove ( ) ;
}
2014-01-14 06:52:56 +00:00
}
2013-06-07 22:09:36 +00:00
} ) ;
2014-01-14 06:52:56 +00:00
$ ( "#customembed-content" ) . keydown ( function ( ev ) {
if ( ev . keyCode === 13 ) {
queue ( "end" , "customembed" ) ;
2013-06-07 22:09:36 +00:00
}
} ) ;
$ ( "#qlockbtn" ) . click ( function ( ) {
socket . emit ( "togglePlaylistLock" ) ;
} ) ;
2013-06-18 20:18:41 +00:00
$ ( "#voteskip" ) . click ( function ( ) {
socket . emit ( "voteskip" ) ;
2013-07-24 19:11:50 +00:00
$ ( "#voteskip" ) . attr ( "disabled" , true ) ;
2013-06-18 20:18:41 +00:00
} ) ;
2013-06-07 22:09:36 +00:00
$ ( "#getplaylist" ) . click ( function ( ) {
var callback = function ( data ) {
2013-08-08 01:05:30 +00:00
hidePlayer ( ) ;
2016-02-07 03:40:50 +00:00
var idx = socket . listeners ( "errorMsg" ) . indexOf ( errCallback ) ;
if ( idx >= 0 ) {
socket . listeners ( "errorMsg" ) . splice ( idx ) ;
}
idx = socket . listeners ( "playlist" ) . indexOf ( callback ) ;
if ( idx >= 0 ) {
socket . listeners ( "playlist" ) . splice ( idx ) ;
}
2013-06-07 22:09:36 +00:00
var list = [ ] ;
2013-06-11 19:41:03 +00:00
for ( var i = 0 ; i < data . length ; i ++ ) {
2013-07-12 20:10:06 +00:00
var entry = formatURL ( data [ i ] . media ) ;
2013-06-07 22:09:36 +00:00
list . push ( entry ) ;
}
var urls = list . join ( "," ) ;
2013-12-15 03:59:47 +00:00
var outer = $ ( "<div/>" ) . addClass ( "modal fade" )
2013-06-07 22:09:36 +00:00
. appendTo ( $ ( "body" ) ) ;
2013-12-15 03:59:47 +00:00
modal = $ ( "<div/>" ) . addClass ( "modal-dialog" ) . appendTo ( outer ) ;
modal = $ ( "<div/>" ) . addClass ( "modal-content" ) . appendTo ( modal ) ;
2013-06-07 22:09:36 +00:00
var head = $ ( "<div/>" ) . addClass ( "modal-header" )
. appendTo ( modal ) ;
$ ( "<button/>" ) . addClass ( "close" )
. attr ( "data-dismiss" , "modal" )
. attr ( "aria-hidden" , "true" )
. html ( "×" )
. appendTo ( head ) ;
$ ( "<h3/>" ) . text ( "Playlist URLs" ) . appendTo ( head ) ;
var body = $ ( "<div/>" ) . addClass ( "modal-body" ) . appendTo ( modal ) ;
2013-12-15 03:59:47 +00:00
$ ( "<input/>" ) . addClass ( "form-control" ) . attr ( "type" , "text" )
2013-06-07 22:09:36 +00:00
. val ( urls )
. appendTo ( body ) ;
$ ( "<div/>" ) . addClass ( "modal-footer" ) . appendTo ( modal ) ;
2015-07-26 18:41:54 +00:00
outer . on ( "hidden.bs.modal" , function ( ) {
2013-12-15 03:59:47 +00:00
outer . remove ( ) ;
2013-08-08 01:05:30 +00:00
unhidePlayer ( ) ;
2013-06-07 22:09:36 +00:00
} ) ;
2013-12-15 03:59:47 +00:00
outer . modal ( ) ;
} ;
2013-06-07 22:09:36 +00:00
socket . on ( "playlist" , callback ) ;
2016-02-07 03:40:50 +00:00
var errCallback = function ( data ) {
if ( data . code !== "REQ_PLAYLIST_LIMIT_REACHED" ) {
return ;
}
var idx = socket . listeners ( "errorMsg" ) . indexOf ( errCallback ) ;
if ( idx >= 0 ) {
socket . listeners ( "errorMsg" ) . splice ( idx ) ;
}
idx = socket . listeners ( "playlist" ) . indexOf ( callback ) ;
if ( idx >= 0 ) {
socket . listeners ( "playlist" ) . splice ( idx ) ;
}
} ;
socket . on ( "errorMsg" , errCallback ) ;
2013-06-07 22:09:36 +00:00
socket . emit ( "requestPlaylist" ) ;
} ) ;
$ ( "#clearplaylist" ) . click ( function ( ) {
var clear = confirm ( "Are you sure you want to clear the playlist?" ) ;
if ( clear ) {
socket . emit ( "clearPlaylist" ) ;
}
} ) ;
$ ( "#shuffleplaylist" ) . click ( function ( ) {
2013-06-11 19:41:03 +00:00
var shuffle = confirm ( "Are you sure you want to shuffle the playlist?" ) ;
if ( shuffle ) {
2013-06-07 22:09:36 +00:00
socket . emit ( "shufflePlaylist" ) ;
}
} ) ;
2013-06-11 15:29:21 +00:00
/* load channel */
var loc = document . location + "" ;
2014-07-02 03:35:13 +00:00
var m = loc . match ( /\/r\/([a-zA-Z0-9-_]+)/ ) ;
2013-06-11 15:29:21 +00:00
if ( m ) {
CHANNEL . name = m [ 1 ] ;
2013-12-25 21:18:21 +00:00
if ( CHANNEL . name . indexOf ( "#" ) !== - 1 ) {
CHANNEL . name = CHANNEL . name . substring ( 0 , CHANNEL . name . indexOf ( "#" ) ) ;
}
2013-06-11 15:29:21 +00:00
}
2013-07-13 04:53:32 +00:00
2014-01-09 05:45:26 +00:00
/* channel ranks stuff */
2014-01-09 23:16:09 +00:00
function chanrankSubmit ( rank ) {
2014-01-09 05:45:26 +00:00
var name = $ ( "#cs-chanranks-name" ) . val ( ) ;
socket . emit ( "setChannelRank" , {
2014-05-21 03:11:40 +00:00
name : name ,
2014-01-09 05:45:26 +00:00
rank : rank
} ) ;
2014-01-09 23:16:09 +00:00
}
$ ( "#cs-chanranks-mod" ) . click ( chanrankSubmit . bind ( this , 2 ) ) ;
$ ( "#cs-chanranks-adm" ) . click ( chanrankSubmit . bind ( this , 3 ) ) ;
$ ( "#cs-chanranks-owner" ) . click ( chanrankSubmit . bind ( this , 4 ) ) ;
2014-01-14 00:31:12 +00:00
2014-02-02 18:41:41 +00:00
[ "#showmediaurl" , "#showsearch" , "#showcustomembed" , "#showplaylistmanager" ]
. forEach ( function ( id ) {
2014-01-14 00:31:12 +00:00
$ ( id ) . click ( function ( ) {
2014-01-22 05:04:06 +00:00
var wasActive = $ ( id ) . hasClass ( "active" ) ;
2014-01-14 00:31:12 +00:00
$ ( ".plcontrol-collapse" ) . collapse ( "hide" ) ;
2014-01-22 05:04:06 +00:00
$ ( "#plcontrol button.active" ) . button ( "toggle" ) ;
if ( ! wasActive ) {
$ ( id ) . button ( "toggle" ) ;
}
2014-01-14 00:31:12 +00:00
} ) ;
} ) ;
2014-01-22 05:04:06 +00:00
$ ( "#plcontrol button" ) . button ( ) ;
$ ( "#plcontrol button" ) . button ( "hide" ) ;
2014-01-14 00:31:12 +00:00
$ ( ".plcontrol-collapse" ) . collapse ( ) ;
$ ( ".plcontrol-collapse" ) . collapse ( "hide" ) ;
2014-01-15 06:16:29 +00:00
$ ( ".cs-checkbox" ) . change ( function ( ) {
var box = $ ( this ) ;
var key = box . attr ( "id" ) . replace ( "cs-" , "" ) ;
var value = box . prop ( "checked" ) ;
var data = { } ;
data [ key ] = value ;
socket . emit ( "setOptions" , data ) ;
} ) ;
2015-02-27 19:10:38 +00:00
$ ( ".cs-textbox" ) . keyup ( function ( ) {
2014-01-15 06:16:29 +00:00
var box = $ ( this ) ;
var key = box . attr ( "id" ) . replace ( "cs-" , "" ) ;
var value = box . val ( ) ;
var lastkey = Date . now ( ) ;
box . data ( "lastkey" , lastkey ) ;
setTimeout ( function ( ) {
if ( box . data ( "lastkey" ) !== lastkey || box . val ( ) !== value ) {
return ;
}
var data = { } ;
2014-01-16 17:53:34 +00:00
if ( key . match ( /chat_antiflood_(burst|sustained)/ ) ) {
data = {
chat _antiflood _params : {
burst : $ ( "#cs-chat_antiflood_burst" ) . val ( ) ,
sustained : $ ( "#cs-chat_antiflood_sustained" ) . val ( )
}
} ;
} else {
data [ key ] = value ;
}
2014-01-15 06:16:29 +00:00
socket . emit ( "setOptions" , data ) ;
} , 1000 ) ;
} ) ;
2014-01-16 17:53:34 +00:00
2016-08-08 05:07:52 +00:00
$ ( ".cs-textbox-timeinput" ) . keyup ( function ( event ) {
var box = $ ( this ) ;
var key = box . attr ( "id" ) . replace ( "cs-" , "" ) ;
var value = box . val ( ) ;
var lastkey = Date . now ( ) ;
box . data ( "lastkey" , lastkey ) ;
setTimeout ( function ( ) {
if ( box . data ( "lastkey" ) !== lastkey || box . val ( ) !== value ) {
return ;
}
$ ( "#cs-textbox-timeinput-validation-error-" + key ) . remove ( ) ;
$ ( event . target ) . parent ( ) . removeClass ( "has-error" ) ;
var data = { } ;
try {
data [ key ] = parseTimeout ( value ) ;
} catch ( error ) {
var msg = "Invalid timespan value '" + value + "'. Please use the format " +
"HH:MM:SS or enter a single number for the number of seconds." ;
var validationError = $ ( "<p/>" ) . addClass ( "text-danger" ) . text ( msg )
. attr ( "id" , "cs-textbox-timeinput-validation-error-" + key ) ;
validationError . insertAfter ( event . target ) ;
$ ( event . target ) . parent ( ) . addClass ( "has-error" ) ;
return ;
}
socket . emit ( "setOptions" , data ) ;
} , 1000 ) ;
} ) ;
2014-02-08 18:45:07 +00:00
$ ( "#cs-chanlog-refresh" ) . click ( function ( ) {
2014-01-16 17:53:34 +00:00
socket . emit ( "readChanLog" ) ;
} ) ;
2014-02-08 18:45:07 +00:00
$ ( "#cs-chanlog-filter" ) . change ( filterChannelLog ) ;
2014-01-16 17:53:34 +00:00
$ ( "#cs-motdsubmit" ) . click ( function ( ) {
socket . emit ( "setMotd" , {
motd : $ ( "#cs-motdtext" ) . val ( )
} ) ;
} ) ;
$ ( "#cs-csssubmit" ) . click ( function ( ) {
socket . emit ( "setChannelCSS" , {
css : $ ( "#cs-csstext" ) . val ( )
} ) ;
} ) ;
$ ( "#cs-jssubmit" ) . click ( function ( ) {
socket . emit ( "setChannelJS" , {
js : $ ( "#cs-jstext" ) . val ( )
} ) ;
} ) ;
2014-01-19 02:18:00 +00:00
$ ( "#cs-chatfilters-newsubmit" ) . click ( function ( ) {
var name = $ ( "#cs-chatfilters-newname" ) . val ( ) ;
var regex = $ ( "#cs-chatfilters-newregex" ) . val ( ) ;
var flags = $ ( "#cs-chatfilters-newflags" ) . val ( ) ;
var replace = $ ( "#cs-chatfilters-newreplace" ) . val ( ) ;
2014-04-13 07:14:34 +00:00
var entcheck = checkEntitiesInStr ( regex ) ;
if ( entcheck ) {
alert ( "Warning: " + entcheck . src + " will be replaced by " +
entcheck . replace + " in the message preprocessor. This " +
"regular expression may not match what you intended it to " +
"match." ) ;
}
2014-01-19 02:18:00 +00:00
2014-12-28 16:12:37 +00:00
socket . emit ( "addFilter" , {
2014-01-19 02:18:00 +00:00
name : name ,
source : regex ,
flags : flags ,
replace : replace ,
active : true
} ) ;
2014-12-28 16:12:37 +00:00
socket . once ( "addFilterSuccess" , function ( ) {
$ ( "#cs-chatfilters-newname" ) . val ( "" ) ;
$ ( "#cs-chatfilters-newregex" ) . val ( "" ) ;
$ ( "#cs-chatfilters-newflags" ) . val ( "" ) ;
$ ( "#cs-chatfilters-newreplace" ) . val ( "" ) ;
} ) ;
2014-01-19 02:18:00 +00:00
} ) ;
2014-02-13 05:33:42 +00:00
$ ( "#cs-emotes-newsubmit" ) . click ( function ( ) {
var name = $ ( "#cs-emotes-newname" ) . val ( ) ;
var image = $ ( "#cs-emotes-newimage" ) . val ( ) ;
socket . emit ( "updateEmote" , {
name : name ,
image : image ,
} ) ;
$ ( "#cs-emotes-newname" ) . val ( "" ) ;
$ ( "#cs-emotes-newimage" ) . val ( "" ) ;
} ) ;
2014-01-19 02:18:00 +00:00
$ ( "#cs-chatfilters-export" ) . click ( function ( ) {
var callback = function ( data ) {
socket . listeners ( "chatFilters" ) . splice (
socket . listeners ( "chatFilters" ) . indexOf ( callback )
) ;
$ ( "#cs-chatfilters-exporttext" ) . val ( JSON . stringify ( data ) ) ;
} ;
socket . on ( "chatFilters" , callback ) ;
socket . emit ( "requestChatFilters" ) ;
} ) ;
$ ( "#cs-chatfilters-import" ) . click ( function ( ) {
var text = $ ( "#cs-chatfilters-exporttext" ) . val ( ) ;
var choose = confirm ( "You are about to import filters from the contents of the textbox below the import button. If this is empty, it will clear all of your filters. Are you sure you want to continue?" ) ;
if ( ! choose ) {
return ;
}
if ( text . trim ( ) === "" ) {
text = "[]" ;
}
var data ;
try {
data = JSON . parse ( text ) ;
} catch ( e ) {
alert ( "Invalid import data: " + e ) ;
return ;
}
2014-01-19 07:45:20 +00:00
socket . emit ( "importFilters" , data ) ;
2014-01-19 02:18:00 +00:00
} ) ;
2014-01-25 19:49:34 +00:00
2014-02-13 05:33:42 +00:00
$ ( "#cs-emotes-export" ) . click ( function ( ) {
var em = CHANNEL . emotes . map ( function ( f ) {
return {
name : f . name ,
image : f . image
} ;
} ) ;
$ ( "#cs-emotes-exporttext" ) . val ( JSON . stringify ( em ) ) ;
} ) ;
$ ( "#cs-emotes-import" ) . click ( function ( ) {
var text = $ ( "#cs-emotes-exporttext" ) . val ( ) ;
var choose = confirm ( "You are about to import emotes from the contents of the textbox below the import button. If this is empty, it will clear all of your emotes. Are you sure you want to continue?" ) ;
if ( ! choose ) {
return ;
}
if ( text . trim ( ) === "" ) {
text = "[]" ;
}
var data ;
try {
data = JSON . parse ( text ) ;
} catch ( e ) {
alert ( "Invalid import data: " + e ) ;
return ;
}
socket . emit ( "importEmotes" , data ) ;
} ) ;
2014-01-25 19:49:34 +00:00
var toggleUserlist = function ( ) {
2015-02-05 07:14:51 +00:00
var direction = ! USEROPTS . layout . match ( /synchtube/ ) ? "glyphicon-chevron-right" : "glyphicon-chevron-left"
2014-01-25 19:49:34 +00:00
if ( $ ( "#userlist" ) . css ( "display" ) === "none" ) {
$ ( "#userlist" ) . show ( ) ;
2015-02-05 07:14:51 +00:00
$ ( "#userlisttoggle" ) . removeClass ( direction ) . addClass ( "glyphicon-chevron-down" ) ;
2014-01-25 19:49:34 +00:00
} else {
$ ( "#userlist" ) . hide ( ) ;
2015-02-05 07:14:51 +00:00
$ ( "#userlisttoggle" ) . removeClass ( "glyphicon-chevron-down" ) . addClass ( direction ) ;
2014-01-25 19:49:34 +00:00
}
scrollChat ( ) ;
} ;
$ ( "#usercount" ) . click ( toggleUserlist ) ;
$ ( "#userlisttoggle" ) . click ( toggleUserlist ) ;
2014-02-09 06:24:20 +00:00
$ ( ".add-temp" ) . change ( function ( ) {
$ ( ".add-temp" ) . prop ( "checked" , $ ( this ) . prop ( "checked" ) ) ;
} ) ;
2014-02-19 03:56:54 +00:00
2014-12-07 19:42:18 +00:00
/ *
* Fixes # 417 which is caused by changes in Bootstrap 3.3 . 0
* ( see twbs / bootstrap # 15136 )
*
* Whenever the active tab in channel options is changed ,
* the modal must be updated so that the backdrop is resized
* appropriately .
* /
$ ( "#channeloptions li > a[data-toggle='tab']" ) . on ( "shown.bs.tab" , function ( ) {
$ ( "#channeloptions" ) . data ( "bs.modal" ) . handleUpdate ( ) ;
} ) ;
2014-02-19 03:56:54 +00:00
applyOpts ( ) ;
2014-11-11 04:43:49 +00:00
( function ( ) {
2015-05-14 18:14:45 +00:00
var embed = document . querySelector ( "#videowrap .embed-responsive" ) ;
if ( ! embed ) {
return ;
}
2014-11-11 04:43:49 +00:00
if ( typeof window . MutationObserver === "function" ) {
var mr = new MutationObserver ( function ( records ) {
records . forEach ( function ( record ) {
if ( record . type !== "childList" ) return ;
if ( ! record . addedNodes || record . addedNodes . length === 0 ) return ;
var elem = record . addedNodes [ 0 ] ;
2014-11-12 01:48:08 +00:00
if ( elem . id === "ytapiplayer" ) handleVideoResize ( ) ;
2014-11-11 04:43:49 +00:00
} ) ;
} ) ;
2015-05-14 18:14:45 +00:00
mr . observe ( embed , { childList : true } ) ;
2014-11-11 04:43:49 +00:00
} else {
/ *
* DOMNodeInserted is deprecated . This code is here only as a fallback
* for browsers that do not support MutationObserver
* /
2015-05-14 18:14:45 +00:00
embed . addEventListener ( "DOMNodeInserted" , function ( ev ) {
2014-11-11 04:43:49 +00:00
if ( ev . target . id === "ytapiplayer" ) handleVideoResize ( ) ;
} ) ;
}
} ) ( ) ;
2015-05-12 18:50:59 +00:00
2016-03-30 06:31:02 +00:00
var EMOTELISTMODAL = $ ( "#emotelist" ) ;
EMOTELISTMODAL . on ( "hidden.bs.modal" , unhidePlayer ) ;
2015-05-12 18:50:59 +00:00
$ ( "#emotelistbtn" ) . click ( function ( ) {
2016-03-30 06:31:02 +00:00
EMOTELISTMODAL . modal ( ) ;
2015-05-12 18:50:59 +00:00
} ) ;
2015-05-13 17:17:32 +00:00
2016-03-30 06:31:02 +00:00
EMOTELISTMODAL . find ( ".emotelist-alphabetical" ) . change ( function ( ) {
2015-05-13 17:17:32 +00:00
USEROPTS . emotelist _sort = this . checked ;
setOpt ( "emotelist_sort" , USEROPTS . emotelist _sort ) ;
} ) ;
2016-03-30 06:31:02 +00:00
EMOTELISTMODAL . find ( ".emotelist-alphabetical" ) . prop ( "checked" , USEROPTS . emotelist _sort ) ;
2015-07-07 00:35:04 +00:00
$ ( "#fullscreenbtn" ) . click ( function ( ) {
var elem = document . querySelector ( "#videowrap .embed-responsive" ) ;
// this shit is why frontend web development sucks
var fn = elem . requestFullscreen ||
elem . mozRequestFullScreen || // Mozilla has to be different and use a capital 'S'
elem . webkitRequestFullscreen ||
elem . msRequestFullscreen ;
if ( fn ) {
fn . call ( elem ) ;
}
} ) ;
2015-12-13 00:49:40 +00:00
function handleCSSJSTooLarge ( selector ) {
if ( this . value . length > 20000 ) {
var warning = $ ( selector ) ;
if ( warning . length > 0 ) {
return ;
}
warning = makeAlert ( "Maximum Size Exceeded" , "Inline CSS and JavaScript are " +
"limited to 20,000 characters or less. If you need more room, you " +
"need to use the external CSS or JavaScript option." , "alert-danger" )
. attr ( "id" , selector . replace ( /#/ , "" ) ) ;
warning . insertBefore ( this ) ;
} else {
$ ( selector ) . remove ( ) ;
}
}
$ ( "#cs-csstext" ) . bind ( "input" , handleCSSJSTooLarge . bind ( $ ( "#cs-csstext" ) [ 0 ] ,
"#cs-csstext-too-big" ) ) ;
$ ( "#cs-jstext" ) . bind ( "input" , handleCSSJSTooLarge . bind ( $ ( "#cs-jstext" ) [ 0 ] ,
"#cs-jstext-too-big" ) ) ;