2013-06-20 00:42:20 -04:00
/* window focus/blur */
2018-11-15 22:40:01 -08:00
CyTube . ui . onPageFocus = function ( ) {
FOCUSED = true ;
clearInterval ( TITLE _BLINK ) ;
TITLE _BLINK = false ;
document . title = PAGETITLE ;
} ;
CyTube . ui . onPageBlur = function ( event ) {
FOCUSED = false ;
} ;
2022-01-22 16:18:51 -08:00
$ ( window ) . on ( 'focus' , CyTube . ui . onPageFocus ) . on ( 'blur' , CyTube . ui . onPageBlur ) ;
2018-11-15 22:40:01 -08:00
// See #783
2022-01-22 16:18:51 -08:00
$ ( ".modal" ) . on ( 'focus' , CyTube . ui . onPageFocus ) ;
2013-06-20 00:42:20 -04:00
2022-01-22 16:18:51 -08:00
$ ( "#togglemotd" ) . on ( 'click' , function ( ) {
2020-04-17 14:53:39 -07:00
var hidden = $ ( "#motd" ) [ 0 ] . style . display === "none" ;
2013-10-02 22:26:28 -05:00
$ ( "#motd" ) . toggle ( ) ;
if ( hidden ) {
2014-01-30 23:02:58 -06:00
$ ( "#togglemotd" ) . find ( ".glyphicon-plus" )
. removeClass ( "glyphicon-plus" )
. addClass ( "glyphicon-minus" ) ;
2013-10-02 22:26:28 -05:00
} else {
2014-01-30 23:02:58 -06:00
$ ( "#togglemotd" ) . find ( ".glyphicon-minus" )
. removeClass ( "glyphicon-minus" )
. addClass ( "glyphicon-plus" ) ;
2013-10-02 22:26:28 -05:00
}
} ) ;
2013-06-06 23:13:24 -04:00
/* chatbox */
2013-08-08 10:39:58 -04:00
2022-01-22 16:18:51 -08:00
$ ( "#modflair" ) . on ( 'click' , function ( ) {
2013-09-21 02:22:51 -05:00
var m = $ ( "#modflair" ) ;
if ( m . hasClass ( "label-success" ) ) {
USEROPTS . modhat = false ;
2016-03-29 22:30:16 -07: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 02:22:51 -05:00
USEROPTS . adminhat = false ;
2013-12-19 12:14:48 -05:00
m . removeClass ( "label-danger" )
2016-03-29 22:30:16 -07:00
. addClass ( "label-default" ) ;
2013-09-21 02:22:51 -05:00
} else {
2016-03-29 22:30:16 -07:00
USEROPTS . modhat = true ;
2013-09-21 02:22:51 -05:00
m . removeClass ( "label-default" )
2016-03-29 22:30:16 -07:00
. addClass ( "label-success" ) ;
2013-09-21 02:22:51 -05:00
}
2016-07-11 22:14:26 -07:00
$ ( "#us-modflair" ) . prop ( "checked" , USEROPTS . modhat ) ;
setOpt ( 'modhat' , USEROPTS . modhat ) ;
2013-09-21 02:22:51 -05:00
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#usercount" ) . on ( 'mouseenter' , function ( ev ) {
2013-08-08 10:39:58 -04:00
var breakdown = calcUserBreakdown ( ) ;
// re-using profile-box class for convenience
var popup = $ ( "<div/>" )
. addClass ( "profile-box" )
2014-01-29 22:50:14 -06:00
. css ( "top" , ( ev . clientY + 5 ) + "px" )
. css ( "left" , ( ev . clientX ) + "px" )
2013-09-21 02:22:51 -05:00
. appendTo ( $ ( "#usercount" ) ) ;
2013-08-08 10:39:58 -04:00
var contents = "" ;
for ( var key in breakdown ) {
contents += "<strong>" + key + ": </strong>" + breakdown [ key ] ;
contents += "<br>"
}
popup . html ( contents ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#usercount" ) . on ( 'mousemove' , function ( ev ) {
2013-09-21 02:22:51 -05:00
var popup = $ ( "#usercount" ) . find ( ".profile-box" ) ;
2013-08-08 10:39:58 -04:00
if ( popup . length == 0 )
return ;
2014-01-29 22:50:14 -06:00
popup . css ( "top" , ( ev . clientY + 5 ) + "px" ) ;
popup . css ( "left" , ( ev . clientX ) + "px" ) ;
2013-08-08 10:39:58 -04:00
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#usercount" ) . on ( 'mouseleave' , function ( ) {
2013-09-21 02:22:51 -05:00
$ ( "#usercount" ) . find ( ".profile-box" ) . remove ( ) ;
2013-08-08 10:39:58 -04:00
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#messagebuffer" ) . on ( 'scroll' , function ( ev ) {
2015-12-05 17:57:33 -08:00
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 09:28:03 -08:00
var m = $ ( "#messagebuffer" ) ;
2015-12-05 18:05:23 -08: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 09:28:03 -08:00
if ( isCaughtUp ) {
SCROLLCHAT = true ;
$ ( "#newmessages-indicator" ) . remove ( ) ;
} else {
SCROLLCHAT = false ;
}
} ) ;
2013-06-20 00:42:20 -04:00
2022-01-22 16:18:51 -08:00
$ ( "#guestname" ) . on ( 'keydown' , function ( ev ) {
2014-01-06 09:55:12 -06:00
if ( ev . keyCode === 13 ) {
socket . emit ( "login" , {
name : $ ( "#guestname" ) . val ( )
} ) ;
}
} ) ;
2017-01-10 22:26:46 -08:00
CyTube . chatTabCompleteData = {
context : { }
} ;
2020-03-14 01:43:25 +01:00
function chatTabComplete ( chatline ) {
2017-01-10 22:26:46 -08:00
if ( ! CyTube . tabCompleteMethods ) {
2017-07-15 14:48:53 -07:00
console . error ( 'Missing CyTube.tabCompleteMethods!' ) ;
2017-01-10 22:26:46 -08:00
return ;
}
var currentText = chatline . value ;
var currentPosition = chatline . selectionEnd ;
if ( typeof currentPosition !== 'number' || ! chatline . setSelectionRange ) {
// Bail, we're on IE8 or something similarly dysfunctional
return ;
}
var firstWord = ! /\s/ . test ( currentText . trim ( ) ) ;
var options = [ ] ;
var userlistElems = document . getElementById ( "userlist" ) . children ;
for ( var i = 0 ; i < userlistElems . length ; i ++ ) {
var username = userlistElems [ i ] . children [ 1 ] . textContent ;
if ( firstWord ) {
username += ':' ;
}
options . push ( username ) ;
}
CHANNEL . emotes . forEach ( function ( emote ) {
options . push ( emote . name ) ;
} ) ;
var method = USEROPTS . chat _tab _method ;
if ( ! CyTube . tabCompleteMethods [ method ] ) {
console . error ( "Unknown chat tab completion method '" + method + "', using default" ) ;
method = "Cycle options" ;
}
var result = CyTube . tabCompleteMethods [ method ] (
currentText ,
currentPosition ,
options ,
CyTube . chatTabCompleteData . context
) ;
chatline . value = result . text ;
chatline . setSelectionRange ( result . newPosition , result . newPosition ) ;
}
2026-04-21 17:37:45 +02:00
/* emote autocomplete */
var EMOTE _SUGGEST _IDX = 0 ;
2026-05-21 15:03:56 +02:00
var EMOTE _SUGGEST _CONTEXT = null ;
function getEmoteTriggerSymbols ( ) {
var raw = ( CHANNEL && CHANNEL . opts && CHANNEL . opts . emote _triggers ) || ":!#/" ;
if ( typeof raw !== "string" || raw . length === 0 ) {
return ":!#/" ;
}
return raw ;
}
function escapeForRegex ( s ) {
return s . replace ( /[.*+?^${}()|[\]\\]/g , "\\$&" ) ;
}
function emoteTokenAtCaret ( ) {
var cl = document . getElementById ( "chatline" ) ;
if ( ! cl ) return null ;
var caret = cl . selectionStart ;
if ( typeof caret !== "number" ) return null ;
var value = cl . value ;
var left = value . slice ( 0 , caret ) ;
var tokenStart = left . lastIndexOf ( " " ) + 1 ;
var token = left . slice ( tokenStart ) ;
if ( ! token ) return null ;
var triggers = getEmoteTriggerSymbols ( ) ;
var triggerClass = escapeForRegex ( triggers ) ;
var re = new RegExp ( "^([" + triggerClass + "])([^\\s]{2,})$" ) ;
var m = token . match ( re ) ;
if ( ! m ) return null ;
return {
tokenStart : tokenStart ,
trigger : m [ 1 ] ,
query : m [ 2 ] . toLowerCase ( )
} ;
2026-04-21 17:37:45 +02:00
}
2026-05-21 15:03:56 +02:00
2026-04-21 17:37:45 +02:00
function emoteAccept ( ) {
var item = $ ( "#emote-suggestions .active" ) ;
if ( ! item . length ) item = $ ( "#emote-suggestions" ) . children ( ) . first ( ) ;
2026-05-21 15:03:56 +02:00
if ( ! item . length || ! EMOTE _SUGGEST _CONTEXT ) return false ;
var cl = document . getElementById ( "chatline" ) ;
if ( ! cl ) return false ;
var value = cl . value ;
var start = EMOTE _SUGGEST _CONTEXT . tokenStart ;
var end = cl . selectionStart ;
var replacement = item . data ( "name" ) + " " ;
cl . value = value . slice ( 0 , start ) + replacement + value . slice ( end ) ;
var newPos = start + replacement . length ;
cl . setSelectionRange ( newPos , newPos ) ;
2026-04-21 17:37:45 +02:00
$ ( "#emote-suggestions" ) . hide ( ) ;
2026-05-21 15:03:56 +02:00
EMOTE _SUGGEST _CONTEXT = null ;
return true ;
2026-04-21 17:37:45 +02:00
}
function emoteRefresh ( ) {
2026-05-21 15:03:56 +02:00
var token = emoteTokenAtCaret ( ) ;
2026-04-21 17:37:45 +02:00
var popup = $ ( "#emote-suggestions" ) ;
2026-05-21 15:03:56 +02:00
if ( ! token || ! CHANNEL . emotes || ! CHANNEL . emotes . length ) {
EMOTE _SUGGEST _CONTEXT = null ;
popup . hide ( ) ;
return ;
}
var partial = token . query ;
EMOTE _SUGGEST _CONTEXT = token ;
2026-05-03 20:10:00 +01:00
var matches = CHANNEL . emotes
. filter ( e => e . name . toLowerCase ( ) . includes ( partial ) )
. sort ( ( a , b ) => {
const an = a . name . toLowerCase ( ) ;
const bn = b . name . toLowerCase ( ) ;
return ( bn . startsWith ( partial ) - an . startsWith ( partial ) ) || an . localeCompare ( bn ) ;
} )
. slice ( 0 , 8 ) ;
2026-04-21 17:37:45 +02:00
if ( ! matches . length ) { popup . hide ( ) ; return ; }
popup . empty ( ) ;
matches . forEach ( function ( e ) {
$ ( "<div>" ) . addClass ( "emote-suggest-item" )
. data ( "name" , e . name )
. html ( '<img src="' + e . image + '">' + e . name )
. appendTo ( popup ) ;
} ) ;
popup . children ( ) . first ( ) . addClass ( "active" ) ;
EMOTE _SUGGEST _IDX = 0 ;
var cl = $ ( "#chatline" ) , off = cl . offset ( ) ;
popup . css ( { left : off . left , width : cl . outerWidth ( ) } ) . show ( ) ;
popup . css ( "top" , off . top - popup . outerHeight ( ) - 2 ) ;
}
$ ( "#chatline" ) . on ( "input" , emoteRefresh ) ;
$ ( document ) . on ( "mousedown" , function ( e ) {
if ( ! $ ( e . target ) . closest ( "#emote-suggestions, #chatline" ) . length )
$ ( "#emote-suggestions" ) . hide ( ) ;
} ) ;
$ ( document ) . on ( "mousedown" , "#emote-suggestions .emote-suggest-item" , function ( ) {
EMOTE _SUGGEST _IDX = $ ( this ) . index ( ) ;
$ ( "#emote-suggestions .active" ) . removeClass ( "active" ) ;
$ ( this ) . addClass ( "active" ) ;
emoteAccept ( ) ;
$ ( "#chatline" ) . focus ( ) ;
} ) ;
$ ( "body" ) . append ( '<div id="emote-suggestions" style="display:none;position:absolute;z-index:9999;background:#272b30;border:1px solid #555;border-radius:4px;overflow-y:auto;max-height:220px;box-shadow:0 2px 8px rgba(0,0,0,.5);color:#c8c8c8"></div>' ) ;
2022-01-22 16:18:51 -08:00
$ ( "#chatline" ) . on ( 'keydown' , function ( ev ) {
2026-04-21 17:37:45 +02:00
var open = $ ( "#emote-suggestions" ) . is ( ":visible" ) ;
if ( open ) {
if ( ev . keyCode == 27 ) { // Escape
$ ( "#emote-suggestions" ) . hide ( ) ;
2026-05-21 15:03:56 +02:00
EMOTE _SUGGEST _CONTEXT = null ;
2026-04-21 17:37:45 +02:00
ev . preventDefault ( ) ; return false ;
2026-05-21 15:03:56 +02:00
} else if ( ev . keyCode == 13 ) { // Enter accept
if ( emoteAccept ( ) ) {
ev . preventDefault ( ) ; return false ;
}
2026-04-21 17:37:45 +02:00
} else if ( ev . keyCode == 9 || ( ev . keyCode == 39 && this . selectionStart === this . value . length ) ) { // Tab or right arrow at end
2026-05-21 15:03:56 +02:00
if ( emoteAccept ( ) ) {
ev . preventDefault ( ) ; return false ;
}
2026-04-21 17:37:45 +02:00
} else if ( ev . keyCode == 38 || ev . keyCode == 40 ) { // Up/down navigate
var items = $ ( "#emote-suggestions" ) . children ( ) ;
items . eq ( EMOTE _SUGGEST _IDX ) . removeClass ( "active" ) ;
EMOTE _SUGGEST _IDX = ev . keyCode == 38
? ( EMOTE _SUGGEST _IDX - 1 + items . length ) % items . length
: ( EMOTE _SUGGEST _IDX + 1 ) % items . length ;
items . eq ( EMOTE _SUGGEST _IDX ) . addClass ( "active" ) [ 0 ] . scrollIntoView ( { block : "nearest" } ) ;
ev . preventDefault ( ) ; return false ;
}
}
2013-12-14 21:59:47 -06:00
// Enter/return
2013-06-06 23:13:24 -04:00
if ( ev . keyCode == 13 ) {
2026-04-21 17:37:45 +02:00
$ ( "#emote-suggestions" ) . hide ( ) ;
2026-05-21 15:03:56 +02:00
EMOTE _SUGGEST _CONTEXT = null ;
2013-11-19 15:14:40 -06:00
if ( CHATTHROTTLE ) {
return ;
}
2013-06-06 23:13:24 -04:00
var msg = $ ( "#chatline" ) . val ( ) ;
if ( msg . trim ( ) ) {
2013-11-17 13:12:56 -06:00
var meta = { } ;
2013-09-21 02:22:51 -05:00
if ( USEROPTS . adminhat && CLIENT . rank >= 255 ) {
msg = "/a " + msg ;
2013-11-17 13:12:56 -06:00
} else if ( USEROPTS . modhat && CLIENT . rank >= Rank . Moderator ) {
meta . modflair = CLIENT . rank ;
2013-06-06 23:13:24 -04:00
}
2013-12-14 21:59:47 -06:00
// The /m command no longer exists, so emulate it clientside
2013-11-17 13:12:56 -06:00
if ( CLIENT . rank >= 2 && msg . indexOf ( "/m " ) === 0 ) {
meta . modflair = CLIENT . rank ;
2013-11-21 17:46:33 -06:00
msg = msg . substring ( 3 ) ;
2013-11-17 13:12:56 -06:00
}
2013-06-06 23:13:24 -04:00
socket . emit ( "chatMsg" , {
2013-11-17 13:12:56 -06:00
msg : msg ,
meta : meta
2013-06-06 23:13:24 -04:00
} ) ;
CHATHIST . push ( $ ( "#chatline" ) . val ( ) ) ;
2013-06-19 17:54:27 -04:00
CHATHISTIDX = CHATHIST . length ;
2013-06-06 23:13:24 -04:00
$ ( "#chatline" ) . val ( "" ) ;
}
return ;
}
2013-12-14 21:59:47 -06:00
else if ( ev . keyCode == 9 ) { // Tab completion
2017-01-10 22:26:46 -08:00
try {
2020-03-14 01:43:25 +01:00
chatTabComplete ( ev . target ) ;
2017-01-10 22:26:46 -08:00
} catch ( error ) {
console . error ( error ) ;
}
2013-06-06 23:13:24 -04:00
ev . preventDefault ( ) ;
return false ;
}
2013-12-14 21:59:47 -06:00
else if ( ev . keyCode == 38 ) { // Up arrow (input history)
2013-06-06 23:13:24 -04:00
if ( CHATHISTIDX == CHATHIST . length ) {
CHATHIST . push ( $ ( "#chatline" ) . val ( ) ) ;
}
if ( CHATHISTIDX > 0 ) {
CHATHISTIDX -- ;
$ ( "#chatline" ) . val ( CHATHIST [ CHATHISTIDX ] ) ;
}
ev . preventDefault ( ) ;
return false ;
}
2013-12-14 21:59:47 -06:00
else if ( ev . keyCode == 40 ) { // Down arrow (input history)
2013-06-06 23:13:24 -04:00
if ( CHATHISTIDX < CHATHIST . length - 1 ) {
CHATHISTIDX ++ ;
$ ( "#chatline" ) . val ( CHATHIST [ CHATHISTIDX ] ) ;
}
ev . preventDefault ( ) ;
return false ;
}
} ) ;
/* poll controls */
2022-01-22 16:18:51 -08:00
$ ( "#newpollbtn" ) . on ( 'click' , showPollMenu ) ;
2013-06-06 23:13:24 -04:00
/* search controls */
2022-01-22 16:18:51 -08:00
$ ( "#library_search" ) . on ( 'click' , function ( ) {
2014-09-04 20:53:18 -05: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-06 23:13:24 -04:00
socket . emit ( "searchMedia" , {
source : "library" ,
query : $ ( "#library_query" ) . val ( ) . toLowerCase ( )
} ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#library_query" ) . on ( 'keydown' , function ( ev ) {
2013-06-06 23:13:24 -04:00
if ( ev . keyCode == 13 ) {
2014-09-04 20:53:18 -05: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-06 23:13:24 -04:00
socket . emit ( "searchMedia" , {
source : "library" ,
query : $ ( "#library_query" ) . val ( ) . toLowerCase ( )
} ) ;
}
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#youtube_search" ) . on ( 'click' , function ( ) {
2013-08-07 13:34:14 -04:00
var query = $ ( "#library_query" ) . val ( ) . toLowerCase ( ) ;
2019-04-07 16:32:58 -07:00
try {
parseMediaLink ( query ) ;
2013-08-07 13:34:14 -04:00
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-14 21:59:47 -06:00
"alert-danger" )
2013-08-07 13:34:14 -04:00
. insertBefore ( $ ( "#library" ) ) ;
2019-04-07 16:32:58 -07:00
} catch ( e ) { }
2013-08-07 13:34:14 -04:00
2013-06-06 23:13:24 -04:00
socket . emit ( "searchMedia" , {
source : "yt" ,
2013-08-07 13:34:14 -04:00
query : query
2013-06-06 23:13:24 -04:00
} ) ;
} ) ;
/* user playlists */
2022-01-22 16:18:51 -08:00
$ ( "#userpl_save" ) . on ( 'click' , function ( ) {
2013-06-06 23:13:24 -04:00
if ( $ ( "#userpl_name" ) . val ( ) . trim ( ) == "" ) {
2013-12-14 21:59:47 -06:00
makeAlert ( "Invalid Name" , "Playlist name cannot be empty" , "alert-danger" )
2013-06-06 23:13:24 -04:00
. insertAfter ( $ ( "#userpl_save" ) . parent ( ) ) ;
return ;
}
2014-02-02 12:41:41 -06:00
socket . emit ( "clonePlaylist" , {
2013-06-06 23:13:24 -04:00
name : $ ( "#userpl_name" ) . val ( )
} ) ;
} ) ;
2013-06-23 14:21:21 -04:00
/* video controls */
2022-01-22 16:18:51 -08:00
$ ( "#mediarefresh" ) . on ( 'click' , function ( ) {
2015-05-02 17:55:00 -05:00
PLAYER . mediaType = "" ;
PLAYER . mediaId = "" ;
2013-12-14 21:59:47 -06:00
// playerReady triggers the server to send a changeMedia.
// the changeMedia handler then reloads the player
2013-06-30 15:56:41 -04:00
socket . emit ( "playerReady" ) ;
} ) ;
2013-06-06 23:13:24 -04:00
/* playlist controls */
2013-06-07 18:09:36 -04:00
2013-06-11 15:41:03 -04:00
$ ( "#queue" ) . sortable ( {
start : function ( ev , ui ) {
2013-06-29 18:09:20 -04:00
PL _FROM = ui . item . data ( "uid" ) ;
2013-06-11 15:41:03 -04:00
} ,
update : function ( ev , ui ) {
2013-06-27 18:15:29 -04:00
var prev = ui . item . prevAll ( ) ;
if ( prev . length == 0 )
2013-06-29 18:09:20 -04:00
PL _AFTER = "prepend" ;
2013-06-27 18:15:29 -04:00
else
2013-06-29 18:09:20 -04:00
PL _AFTER = $ ( prev [ 0 ] ) . data ( "uid" ) ;
2013-06-27 18:15:29 -04:00
socket . emit ( "moveMedia" , {
from : PL _FROM ,
after : PL _AFTER
} ) ;
2013-10-03 22:11:47 -05:00
$ ( "#queue" ) . sortable ( "cancel" ) ;
2013-06-11 15:41:03 -04:00
}
2013-06-09 14:03:41 -04:00
} ) ;
2013-06-11 15:41:03 -04:00
$ ( "#queue" ) . disableSelection ( ) ;
2013-06-09 14:03:41 -04:00
2014-01-14 00:52:56 -06:00
function queue ( pos , src ) {
if ( ! src ) {
src = "url" ;
}
if ( src === "customembed" ) {
var title = $ ( "#customembed-title" ) . val ( ) ;
if ( ! title ) {
title = false ;
2013-08-22 17:33:03 -05:00
}
2014-01-14 00:52:56 -06:00
var content = $ ( "#customembed-content" ) . val ( ) ;
2013-08-03 15:10:06 -04:00
socket . emit ( "queue" , {
2014-01-14 00:52:56 -06:00
id : content ,
2013-08-22 17:33:03 -05:00
title : title ,
2014-01-14 00:52:56 -06:00
pos : pos ,
2014-02-09 00:24:20 -06:00
type : "cu" ,
temp : $ ( ".add-temp" ) . prop ( "checked" )
2013-08-03 15:10:06 -04:00
} ) ;
2014-01-14 00:52:56 -06:00
} else {
2015-08-19 23:27:05 -07: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-16 19:14:55 -07: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-03 21:21:00 -07:00
}
2014-02-09 20:10:11 -06:00
2015-07-16 19:14:55 -07:00
links . forEach ( function ( link ) {
2019-04-07 16:32:58 -07:00
var data ;
2017-12-24 11:19:30 -08:00
2019-04-07 16:32:58 -07:00
try {
data = parseMediaLink ( link ) ;
} catch ( error ) {
2017-12-24 11:19:30 -08:00
Callbacks . queueFail ( {
link : link ,
2019-04-07 16:32:58 -07:00
msg : error . message
2017-12-24 11:19:30 -08:00
} ) ;
return ;
2015-07-16 19:14:55 -07:00
}
2019-04-07 16:32:58 -07:00
var duration = undefined ;
var title = undefined ;
if ( data . type === "fi" ) {
if ( data . id . match ( /^http:/ ) ) {
Callbacks . queueFail ( {
link : data . id ,
msg : "Raw files must begin with 'https'. Plain http is not supported."
} ) ;
return ;
}
// Explicit checks for kissanime and mega.nz since everyone
// asks about them
if ( data . id . match ( /kissanime|kimcartoon|kisscartoon/i ) ) {
Callbacks . queueFail ( {
link : data . id ,
msg : "Kisscartoon and Kissanime are not supported. See https://git.io/vxS9n" +
" for more information about why these cannot be supported."
} ) ;
return ;
} else if ( data . id . match ( /mega\.nz/ ) ) {
Callbacks . queueFail ( {
link : data . id ,
msg : "Mega.nz is not supported. See https://git.io/fx6fz" +
" for more information about why mega.nz cannot be supported."
} ) ;
return ;
}
// Raw files allow title overrides since the ffprobe tag data
// is not always correct.
title = $ ( "#addfromurl-title-val" ) . val ( ) ;
}
2015-07-16 19:14:55 -07:00
if ( data . id == null || data . type == null ) {
makeAlert ( "Error" , "Failed to parse link " + link +
". Please check that it is correct" ,
2016-07-22 19:22:15 -07:00
"alert-danger" , true )
2015-07-16 19:14:55 -07: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 16:13:58 -07:00
startQueueSpinner ( data ) ;
2015-07-16 19:14:55 -07:00
if ( emitQueue . length > 0 ) {
notification . textContent = "Waiting to queue " + emitQueue [ 0 ] . link ;
} else {
notification . textContent = "" ;
}
setTimeout ( next , nextQueueDelay ) ;
2013-06-11 23:37:12 -04:00
}
2015-07-16 19:14:55 -07:00
next ( ) ;
2013-06-19 23:20:56 -04:00
}
2013-06-07 18:09:36 -04:00
}
2022-01-22 16:18:51 -08:00
$ ( "#queue_next" ) . on ( 'click' , queue . bind ( this , "next" , "url" ) ) ;
$ ( "#queue_end" ) . on ( 'click' , queue . bind ( this , "end" , "url" ) ) ;
$ ( "#ce_queue_next" ) . on ( 'click' , queue . bind ( this , "next" , "customembed" ) ) ;
$ ( "#ce_queue_end" ) . on ( 'click' , queue . bind ( this , "end" , "customembed" ) ) ;
2013-06-07 18:09:36 -04:00
2022-01-22 16:18:51 -08:00
$ ( "#mediaurl" ) . on ( 'keyup' , function ( ev ) {
2014-01-14 00:52:56 -06:00
if ( ev . keyCode === 13 ) {
queue ( "end" , "url" ) ;
2014-02-09 20:10:11 -06:00
} else {
2019-04-07 16:32:58 -07:00
var editTitle = false ;
try {
if ( parseMediaLink ( $ ( "#mediaurl" ) . val ( ) ) . type === "fi" ) {
editTitle = true ;
}
} catch ( error ) {
}
if ( editTitle ) {
2014-06-03 21:21:00 -07:00
var title = $ ( "#addfromurl-title" ) ;
if ( title . length === 0 ) {
title = $ ( "<div/>" )
. attr ( "id" , "addfromurl-title" )
. appendTo ( $ ( "#addfromurl" ) ) ;
2019-04-07 16:32:58 -07:00
$ ( "<span/>" ) . text ( "Title (optional; for raw files only)" )
2014-06-03 21:21:00 -07:00
. appendTo ( title ) ;
$ ( "<input/>" ) . addClass ( "form-control" )
. attr ( "type" , "text" )
. attr ( "id" , "addfromurl-title-val" )
2022-01-22 16:18:51 -08:00
. on ( 'keydown' , function ( ev ) {
2014-06-03 21:21:00 -07:00
if ( ev . keyCode === 13 ) {
queue ( "end" , "url" ) ;
}
} )
. appendTo ( $ ( "#addfromurl-title" ) ) ;
}
} else {
$ ( "#addfromurl-title" ) . remove ( ) ;
}
2014-01-14 00:52:56 -06:00
}
2013-06-07 18:09:36 -04:00
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#customembed-content" ) . on ( 'keydown' , function ( ev ) {
2014-01-14 00:52:56 -06:00
if ( ev . keyCode === 13 ) {
queue ( "end" , "customembed" ) ;
2013-06-07 18:09:36 -04:00
}
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#qlockbtn" ) . on ( 'click' , function ( ) {
2013-06-07 18:09:36 -04:00
socket . emit ( "togglePlaylistLock" ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#voteskip" ) . on ( 'click' , function ( ) {
2013-06-18 16:18:41 -04:00
socket . emit ( "voteskip" ) ;
2013-07-24 15:11:50 -04:00
$ ( "#voteskip" ) . attr ( "disabled" , true ) ;
2013-06-18 16:18:41 -04:00
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#getplaylist" ) . on ( 'click' , function ( ) {
2013-06-07 18:09:36 -04:00
var callback = function ( data ) {
2016-02-06 19:40:50 -08: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 18:09:36 -04:00
var list = [ ] ;
2013-06-11 15:41:03 -04:00
for ( var i = 0 ; i < data . length ; i ++ ) {
2013-07-12 16:10:06 -04:00
var entry = formatURL ( data [ i ] . media ) ;
2013-06-07 18:09:36 -04:00
list . push ( entry ) ;
}
var urls = list . join ( "," ) ;
2013-12-14 21:59:47 -06:00
var outer = $ ( "<div/>" ) . addClass ( "modal fade" )
2013-06-07 18:09:36 -04:00
. appendTo ( $ ( "body" ) ) ;
2013-12-14 21:59:47 -06:00
modal = $ ( "<div/>" ) . addClass ( "modal-dialog" ) . appendTo ( outer ) ;
modal = $ ( "<div/>" ) . addClass ( "modal-content" ) . appendTo ( modal ) ;
2013-06-07 18:09:36 -04: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-14 21:59:47 -06:00
$ ( "<input/>" ) . addClass ( "form-control" ) . attr ( "type" , "text" )
2013-06-07 18:09:36 -04:00
. val ( urls )
. appendTo ( body ) ;
$ ( "<div/>" ) . addClass ( "modal-footer" ) . appendTo ( modal ) ;
2015-07-26 11:41:54 -07:00
outer . on ( "hidden.bs.modal" , function ( ) {
2013-12-14 21:59:47 -06:00
outer . remove ( ) ;
2013-06-07 18:09:36 -04:00
} ) ;
2013-12-14 21:59:47 -06:00
outer . modal ( ) ;
} ;
2013-06-07 18:09:36 -04:00
socket . on ( "playlist" , callback ) ;
2016-02-06 19:40:50 -08: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 18:09:36 -04:00
socket . emit ( "requestPlaylist" ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#clearplaylist" ) . on ( 'click' , function ( ) {
2013-06-07 18:09:36 -04:00
var clear = confirm ( "Are you sure you want to clear the playlist?" ) ;
if ( clear ) {
socket . emit ( "clearPlaylist" ) ;
}
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#shuffleplaylist" ) . on ( 'click' , function ( ) {
2013-06-11 15:41:03 -04:00
var shuffle = confirm ( "Are you sure you want to shuffle the playlist?" ) ;
if ( shuffle ) {
2013-06-07 18:09:36 -04:00
socket . emit ( "shufflePlaylist" ) ;
}
} ) ;
2013-06-11 11:29:21 -04:00
2014-01-08 23:45:26 -06:00
/* channel ranks stuff */
2014-01-09 17:16:09 -06:00
function chanrankSubmit ( rank ) {
2014-01-08 23:45:26 -06:00
var name = $ ( "#cs-chanranks-name" ) . val ( ) ;
socket . emit ( "setChannelRank" , {
2014-05-20 20:11:40 -07:00
name : name ,
2014-01-08 23:45:26 -06:00
rank : rank
} ) ;
2014-01-09 17:16:09 -06:00
}
2022-01-22 16:18:51 -08:00
$ ( "#cs-chanranks-mod" ) . on ( 'click' , chanrankSubmit . bind ( this , 2 ) ) ;
$ ( "#cs-chanranks-adm" ) . on ( 'click' , chanrankSubmit . bind ( this , 3 ) ) ;
$ ( "#cs-chanranks-owner" ) . on ( 'click' , chanrankSubmit . bind ( this , 4 ) ) ;
2014-01-13 18:31:12 -06:00
2014-02-02 12:41:41 -06:00
[ "#showmediaurl" , "#showsearch" , "#showcustomembed" , "#showplaylistmanager" ]
. forEach ( function ( id ) {
2022-01-22 16:18:51 -08:00
$ ( id ) . on ( 'click' , function ( ) {
2014-01-21 23:04:06 -06:00
var wasActive = $ ( id ) . hasClass ( "active" ) ;
2014-01-13 18:31:12 -06:00
$ ( ".plcontrol-collapse" ) . collapse ( "hide" ) ;
2014-01-21 23:04:06 -06:00
$ ( "#plcontrol button.active" ) . button ( "toggle" ) ;
if ( ! wasActive ) {
$ ( id ) . button ( "toggle" ) ;
}
2014-01-13 18:31:12 -06:00
} ) ;
} ) ;
2014-01-21 23:04:06 -06:00
$ ( "#plcontrol button" ) . button ( ) ;
$ ( "#plcontrol button" ) . button ( "hide" ) ;
2014-01-13 18:31:12 -06:00
$ ( ".plcontrol-collapse" ) . collapse ( ) ;
$ ( ".plcontrol-collapse" ) . collapse ( "hide" ) ;
2014-01-15 00:16:29 -06:00
2022-01-22 16:18:51 -08:00
$ ( ".cs-checkbox" ) . on ( 'change' , function ( ) {
2014-01-15 00:16:29 -06:00
var box = $ ( this ) ;
var key = box . attr ( "id" ) . replace ( "cs-" , "" ) ;
var value = box . prop ( "checked" ) ;
var data = { } ;
data [ key ] = value ;
socket . emit ( "setOptions" , data ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( ".cs-textbox" ) . on ( 'keyup' , function ( ) {
2014-01-15 00:16:29 -06: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 11:53:34 -06: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 00:16:29 -06:00
socket . emit ( "setOptions" , data ) ;
} , 1000 ) ;
} ) ;
2014-01-16 11:53:34 -06:00
2022-01-22 16:18:51 -08:00
$ ( ".cs-textbox-timeinput" ) . on ( 'keyup' , function ( event ) {
2016-08-07 22:07:52 -07: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 ;
}
$ ( "#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 ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-chanlog-refresh" ) . on ( 'click' , function ( ) {
2014-01-16 11:53:34 -06:00
socket . emit ( "readChanLog" ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-chanlog-filter" ) . on ( 'change' , filterChannelLog ) ;
2014-01-16 11:53:34 -06:00
2022-01-22 16:18:51 -08:00
$ ( "#cs-motdsubmit" ) . on ( 'click' , function ( ) {
2014-01-16 11:53:34 -06:00
socket . emit ( "setMotd" , {
motd : $ ( "#cs-motdtext" ) . val ( )
} ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-csssubmit" ) . on ( 'click' , function ( ) {
2014-01-16 11:53:34 -06:00
socket . emit ( "setChannelCSS" , {
css : $ ( "#cs-csstext" ) . val ( )
} ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-jssubmit" ) . on ( 'click' , function ( ) {
2014-01-16 11:53:34 -06:00
socket . emit ( "setChannelJS" , {
js : $ ( "#cs-jstext" ) . val ( )
} ) ;
} ) ;
2014-01-18 20:18:00 -06:00
2022-01-22 16:18:51 -08:00
$ ( "#cs-chatfilters-newsubmit" ) . on ( 'click' , function ( ) {
2014-01-18 20:18:00 -06:00
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 02:14:34 -05: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-18 20:18:00 -06:00
2014-12-28 11:12:37 -05:00
socket . emit ( "addFilter" , {
2014-01-18 20:18:00 -06:00
name : name ,
source : regex ,
flags : flags ,
replace : replace ,
active : true
} ) ;
2014-12-28 11:12:37 -05:00
socket . once ( "addFilterSuccess" , function ( ) {
$ ( "#cs-chatfilters-newname" ) . val ( "" ) ;
$ ( "#cs-chatfilters-newregex" ) . val ( "" ) ;
$ ( "#cs-chatfilters-newflags" ) . val ( "" ) ;
$ ( "#cs-chatfilters-newreplace" ) . val ( "" ) ;
} ) ;
2014-01-18 20:18:00 -06:00
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-emotes-newsubmit" ) . on ( 'click' , function ( ) {
2014-02-12 23:33:42 -06:00
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 ( "" ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-chatfilters-export" ) . on ( 'click' , function ( ) {
2014-01-18 20:18:00 -06:00
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" ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-chatfilters-import" ) . on ( 'click' , function ( ) {
2014-01-18 20:18:00 -06:00
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 01:45:20 -06:00
socket . emit ( "importFilters" , data ) ;
2014-01-18 20:18:00 -06:00
} ) ;
2014-01-25 13:49:34 -06:00
2022-01-22 16:18:51 -08:00
$ ( "#cs-emotes-export" ) . on ( 'click' , function ( ) {
2014-02-12 23:33:42 -06:00
var em = CHANNEL . emotes . map ( function ( f ) {
return {
name : f . name ,
image : f . image
} ;
} ) ;
$ ( "#cs-emotes-exporttext" ) . val ( JSON . stringify ( em ) ) ;
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#cs-emotes-import" ) . on ( 'click' , function ( ) {
2014-02-12 23:33:42 -06:00
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 13:49:34 -06:00
var toggleUserlist = function ( ) {
2015-02-04 23:14:51 -08:00
var direction = ! USEROPTS . layout . match ( /synchtube/ ) ? "glyphicon-chevron-right" : "glyphicon-chevron-left"
2020-04-17 14:53:39 -07:00
if ( $ ( "#userlist" ) [ 0 ] . style . display === "none" ) {
2014-01-25 13:49:34 -06:00
$ ( "#userlist" ) . show ( ) ;
2015-02-04 23:14:51 -08:00
$ ( "#userlisttoggle" ) . removeClass ( direction ) . addClass ( "glyphicon-chevron-down" ) ;
2014-01-25 13:49:34 -06:00
} else {
$ ( "#userlist" ) . hide ( ) ;
2015-02-04 23:14:51 -08:00
$ ( "#userlisttoggle" ) . removeClass ( "glyphicon-chevron-down" ) . addClass ( direction ) ;
2014-01-25 13:49:34 -06:00
}
scrollChat ( ) ;
} ;
2022-01-22 16:18:51 -08:00
$ ( "#usercount" ) . on ( 'click' , toggleUserlist ) ;
$ ( "#userlisttoggle" ) . on ( 'click' , toggleUserlist ) ;
2014-02-09 00:24:20 -06:00
2022-01-22 16:18:51 -08:00
$ ( ".add-temp" ) . on ( 'change' , function ( ) {
2014-02-09 00:24:20 -06:00
$ ( ".add-temp" ) . prop ( "checked" , $ ( this ) . prop ( "checked" ) ) ;
} ) ;
2014-02-18 21:56:54 -06:00
2014-12-07 13:42:18 -06: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-18 21:56:54 -06:00
applyOpts ( ) ;
2014-11-10 22:43:49 -06:00
( function ( ) {
2015-05-14 13:14:45 -05:00
var embed = document . querySelector ( "#videowrap .embed-responsive" ) ;
if ( ! embed ) {
return ;
}
2014-11-10 22:43:49 -06: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-11 19:48:08 -06:00
if ( elem . id === "ytapiplayer" ) handleVideoResize ( ) ;
2014-11-10 22:43:49 -06:00
} ) ;
} ) ;
2015-05-14 13:14:45 -05:00
mr . observe ( embed , { childList : true } ) ;
2014-11-10 22:43:49 -06:00
} else {
/ *
* DOMNodeInserted is deprecated . This code is here only as a fallback
* for browsers that do not support MutationObserver
* /
2015-05-14 13:14:45 -05:00
embed . addEventListener ( "DOMNodeInserted" , function ( ev ) {
2014-11-10 22:43:49 -06:00
if ( ev . target . id === "ytapiplayer" ) handleVideoResize ( ) ;
} ) ;
}
} ) ( ) ;
2015-05-12 13:50:59 -05:00
2016-03-29 23:31:02 -07:00
var EMOTELISTMODAL = $ ( "#emotelist" ) ;
2018-02-01 17:39:45 -08:00
EMOTELISTMODAL . find ( ".emotelist-alphabetical" ) . change ( function ( ) {
USEROPTS . emotelist _sort = this . checked ;
setOpt ( "emotelist_sort" , USEROPTS . emotelist _sort ) ;
} ) ;
EMOTELISTMODAL . find ( ".emotelist-alphabetical" ) . prop ( "checked" , USEROPTS . emotelist _sort ) ;
2026-04-21 17:51:03 +02:00
/* emote browser panel */
var EMOTE _BROWSER _OFFSET = 0 ;
var EMOTE _BROWSER _BATCH = 40 ;
var EMOTE _BROWSER _FILTER = '' ;
$ ( 'body' ) . append (
'<div id="emote-browser">' +
'<input id="emote-browser-search" class="form-control input-sm" type="text" placeholder="Search emotes…">' +
'<div id="emote-browser-grid"></div>' +
2026-05-21 13:46:51 +02:00
'<div class="emote-browser-resize-handle ne" data-dir="ne"></div>' +
'<div class="emote-browser-resize-handle nw" data-dir="nw"></div>' +
'<div class="emote-browser-resize-handle se" data-dir="se"></div>' +
'<div class="emote-browser-resize-handle sw" data-dir="sw"></div>' +
2026-04-21 17:51:03 +02:00
'</div>'
) ;
2026-05-21 13:46:51 +02:00
function updateEmoteBrowserScale ( ) {
var panel = document . getElementById ( 'emote-browser' ) ;
if ( ! panel ) return ;
var panelWidth = panel . clientWidth ;
var panelHeight = panel . clientHeight ;
var itemByWidth = Math . floor ( ( panelWidth - 48 ) / 6 ) ;
var itemByHeight = Math . floor ( ( panelHeight - 90 ) / 4 ) ;
var itemSize = Math . max ( 40 , Math . min ( 88 , itemByWidth , itemByHeight ) ) ;
var imageSize = Math . max ( 36 , itemSize - 4 ) ;
panel . style . setProperty ( '--emote-browser-item-size' , itemSize + 'px' ) ;
panel . style . setProperty ( '--emote-browser-image-size' , imageSize + 'px' ) ;
}
2026-04-21 17:51:03 +02:00
function emoteBrowserMatches ( ) {
if ( ! CHANNEL . emotes ) return [ ] ;
var f = EMOTE _BROWSER _FILTER . toLowerCase ( ) ;
return f ? CHANNEL . emotes . filter ( function ( e ) { return e . name . toLowerCase ( ) . indexOf ( f ) !== - 1 ; } )
: CHANNEL . emotes ;
}
function emoteBrowserRenderMore ( ) {
var matches = emoteBrowserMatches ( ) ;
var end = Math . min ( EMOTE _BROWSER _OFFSET + EMOTE _BROWSER _BATCH , matches . length ) ;
var grid = document . getElementById ( 'emote-browser-grid' ) ;
for ( var i = EMOTE _BROWSER _OFFSET ; i < end ; i ++ ) {
( function ( emote ) {
var item = document . createElement ( 'div' ) ;
item . className = 'emote-browser-item' ;
item . title = emote . name ;
var img = document . createElement ( 'img' ) ;
img . src = emote . image ;
item . appendChild ( img ) ;
item . addEventListener ( 'click' , function ( ) {
var cl = document . getElementById ( 'chatline' ) ;
var val = cl . value ;
if ( val && ! val . charAt ( val . length - 1 ) . match ( /\s/ ) ) val += ' ' ;
cl . value = val + emote . name ;
$ ( "#emote-browser" ) . hide ( ) ;
cl . focus ( ) ;
} ) ;
grid . appendChild ( item ) ;
} ) ( matches [ i ] ) ;
}
EMOTE _BROWSER _OFFSET = end ;
}
function emoteBrowserReset ( ) {
EMOTE _BROWSER _OFFSET = 0 ;
document . getElementById ( 'emote-browser-grid' ) . innerHTML = '' ;
emoteBrowserRenderMore ( ) ;
}
function emoteBrowserPosition ( ) {
var btn = $ ( "#emotelistbtn" ) , off = btn . offset ( ) ;
var panel = $ ( "#emote-browser" ) ;
var pw = panel . outerWidth ( ) , ph = panel . outerHeight ( ) ;
var ww = $ ( window ) . width ( ) , wh = $ ( window ) . height ( ) ;
var left = off . left ;
if ( left + pw > ww - 8 ) left = Math . max ( 8 , off . left + btn . outerWidth ( ) - pw ) ;
var top = off . top - ph - 4 ;
if ( top < 8 ) top = off . top + btn . outerHeight ( ) + 4 ;
panel . css ( { top : top , left : left } ) ;
}
2026-05-21 13:46:51 +02:00
function clampEmoteBrowserToViewport ( ) {
var panel = document . getElementById ( 'emote-browser' ) ;
if ( ! panel ) return ;
var rect = panel . getBoundingClientRect ( ) ;
var maxLeft = window . innerWidth - rect . width - 8 ;
var maxTop = window . innerHeight - rect . height - 8 ;
var left = Math . min ( Math . max ( rect . left , 8 ) , Math . max ( 8 , maxLeft ) ) ;
var top = Math . min ( Math . max ( rect . top , 8 ) , Math . max ( 8 , maxTop ) ) ;
panel . style . left = left + 'px' ;
panel . style . top = top + 'px' ;
}
$ ( document ) . on ( 'mousedown' , '#emote-browser .emote-browser-resize-handle' , function ( ev ) {
ev . preventDefault ( ) ;
ev . stopPropagation ( ) ;
var panel = document . getElementById ( 'emote-browser' ) ;
var dir = ev . target . getAttribute ( 'data-dir' ) ;
if ( ! panel || ! dir ) return ;
var startX = ev . clientX ;
var startY = ev . clientY ;
var startRect = panel . getBoundingClientRect ( ) ;
var minW = 260 , minH = 220 ;
var maxW = Math . floor ( window . innerWidth * 0.9 ) ;
var maxH = Math . floor ( window . innerHeight * 0.85 ) ;
function onMove ( moveEv ) {
var dx = moveEv . clientX - startX ;
var dy = moveEv . clientY - startY ;
var left = startRect . left ;
var top = startRect . top ;
var width = startRect . width ;
var height = startRect . height ;
if ( dir . indexOf ( 'e' ) !== - 1 ) {
width = Math . max ( minW , Math . min ( maxW , startRect . width + dx ) ) ;
}
if ( dir . indexOf ( 's' ) !== - 1 ) {
height = Math . max ( minH , Math . min ( maxH , startRect . height + dy ) ) ;
}
if ( dir . indexOf ( 'w' ) !== - 1 ) {
width = Math . max ( minW , Math . min ( maxW , startRect . width - dx ) ) ;
left = startRect . right - width ;
}
if ( dir . indexOf ( 'n' ) !== - 1 ) {
height = Math . max ( minH , Math . min ( maxH , startRect . height - dy ) ) ;
top = startRect . bottom - height ;
}
panel . style . left = Math . max ( 8 , left ) + 'px' ;
panel . style . top = Math . max ( 8 , top ) + 'px' ;
panel . style . width = width + 'px' ;
panel . style . height = height + 'px' ;
updateEmoteBrowserScale ( ) ;
clampEmoteBrowserToViewport ( ) ;
}
function onUp ( ) {
document . removeEventListener ( 'mousemove' , onMove ) ;
document . removeEventListener ( 'mouseup' , onUp ) ;
}
document . addEventListener ( 'mousemove' , onMove ) ;
document . addEventListener ( 'mouseup' , onUp ) ;
} ) ;
2026-04-21 17:51:03 +02:00
$ ( "#emotelistbtn" ) . on ( 'click' , function ( ) {
var panel = $ ( "#emote-browser" ) ;
if ( panel . is ( ':visible' ) ) { panel . hide ( ) ; return ; }
EMOTE _BROWSER _FILTER = '' ;
$ ( "#emote-browser-search" ) . val ( '' ) ;
emoteBrowserReset ( ) ;
panel . show ( ) ;
2026-05-21 13:46:51 +02:00
updateEmoteBrowserScale ( ) ;
2026-04-21 17:51:03 +02:00
emoteBrowserPosition ( ) ;
2026-05-21 13:46:51 +02:00
clampEmoteBrowserToViewport ( ) ;
2026-04-21 17:51:03 +02:00
document . getElementById ( 'emote-browser-search' ) . focus ( ) ;
} ) ;
$ ( document ) . on ( 'click.emotebrowser' , function ( e ) {
if ( ! $ ( e . target ) . closest ( '#emote-browser, #emotelistbtn' ) . length )
$ ( "#emote-browser" ) . hide ( ) ;
} ) ;
$ ( document ) . on ( 'input' , '#emote-browser-search' , function ( ) {
EMOTE _BROWSER _FILTER = this . value ;
emoteBrowserReset ( ) ;
} ) ;
document . getElementById ( 'emote-browser-grid' ) . addEventListener ( 'scroll' , function ( ) {
if ( this . scrollTop + this . clientHeight >= this . scrollHeight - 60 )
emoteBrowserRenderMore ( ) ;
} ) ;
2026-05-21 13:46:51 +02:00
$ ( window ) . on ( 'resize' , function ( ) {
updateEmoteBrowserScale ( ) ;
if ( $ ( "#emote-browser" ) . is ( ':visible' ) ) {
clampEmoteBrowserToViewport ( ) ;
}
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#fullscreenbtn" ) . on ( 'click' , function ( ) {
2015-07-06 17:35:04 -07:00
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-12 16:49:40 -08:00
function handleCSSJSTooLarge ( selector ) {
if ( this . value . length > 20000 ) {
2022-01-22 17:59:14 -08:00
let notice = document . querySelector ( selector ) ;
if ( notice !== null ) {
2015-12-12 16:49:40 -08:00
return ;
}
2022-01-22 17:59:14 -08:00
notice = makeAlert ( "Maximum Size Exceeded" , "Inline CSS and JavaScript are " +
2015-12-12 16:49:40 -08:00
"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 ( /#/ , "" ) ) ;
2022-01-22 17:59:14 -08:00
// makeAlert returns jQuery
this . parentNode . insertBefore ( notice [ 0 ] , this ) ;
2015-12-12 16:49:40 -08:00
} else {
2022-01-22 17:59:14 -08:00
let notice = document . querySelector ( selector ) ;
2022-01-31 17:02:10 -08:00
notice ? . remove ( ) ;
2015-12-12 16:49:40 -08:00
}
}
2022-01-22 17:59:14 -08:00
[ '#cs-csstext' , '#cs-jstext' ] . forEach ( ( selector ) => {
elem = document . querySelector ( selector ) ;
elem . addEventListener ( 'input' , handleCSSJSTooLarge . bind ( elem , ` ${ selector } -too-big ` ) ) ;
} ) ;
2017-01-23 21:47:21 -08:00
2022-01-22 16:18:51 -08:00
$ ( "#resize-video-larger" ) . on ( 'click' , function ( ) {
2017-01-23 21:47:21 -08:00
try {
CyTube . ui . changeVideoWidth ( 1 ) ;
} catch ( error ) {
console . error ( error ) ;
}
} ) ;
2022-01-22 16:18:51 -08:00
$ ( "#resize-video-smaller" ) . on ( 'click' , function ( ) {
2017-01-23 21:47:21 -08:00
try {
CyTube . ui . changeVideoWidth ( - 1 ) ;
} catch ( error ) {
console . error ( error ) ;
}
} ) ;
2026-05-04 16:07:59 +02:00
2026-05-21 16:23:30 +02:00
$ . ajaxPrefilter ( function ( options , _originalOptions , _jqXHR ) {
var url = String ( options . url || '' ) ;
if ( ! /\/api\/v1\// . test ( url ) ) {
return ;
}
var method = String ( options . type || options . method || 'GET' ) . toUpperCase ( ) ;
if ( method === 'GET' || method === 'HEAD' || method === 'OPTIONS' ) {
return ;
}
options . headers = options . headers || { } ;
if ( options . headers . Authorization || options . headers . authorization ) {
return ;
}
if ( typeof CSRF _TOKEN === 'string' && CSRF _TOKEN . length > 0 ) {
options . headers [ 'X-CSRF-Token' ] = CSRF _TOKEN ;
}
} ) ;
2026-05-04 16:07:59 +02:00
var CSTBots = ( function ( ) {
function apiBase ( ) {
return '/api/v1/channels/' + CHANNEL . name ;
}
function load ( ) {
$ . getJSON ( apiBase ( ) + '/bots' , function ( bots ) {
var tbody = $ ( '#cs-bots-list' ) . empty ( ) ;
bots . forEach ( function ( bot ) {
var lastConn = bot . last _connected
? new Date ( bot . last _connected ) . toLocaleString ( )
: 'Never' ;
var rankLabel = bot . rank >= 5 ? 'Creator' : bot . rank >= 4 ? 'Owner' : bot . rank >= 3 ? 'Admin' : 'Mod' ;
var row = $ ( '<tr>' ) ;
if ( bot . active ) {
row . append ( $ ( '<td>' ) . append (
$ ( '<button class="btn btn-xs btn-danger">' ) . text ( 'Revoke' )
. on ( 'click' , function ( ) { revoke ( bot . id ) ; } )
) ) ;
} else {
row . append ( $ ( '<td>' ) . append ( $ ( '<span class="text-muted">' ) . text ( 'Revoked' ) ) ) ;
}
row . append ( $ ( '<td>' ) . text ( bot . name ) ) ;
row . append ( $ ( '<td>' ) . text ( rankLabel + ' (' + bot . rank + ')' ) ) ;
row . append ( $ ( '<td>' ) . text ( bot . created _by ) ) ;
row . append ( $ ( '<td>' ) . text ( lastConn ) ) ;
tbody . append ( row ) ;
} ) ;
} ) . fail ( function ( ) {
$ ( '#cs-bots-list' ) . html ( '<tr><td colspan="5" class="text-danger">Failed to load bots</td></tr>' ) ;
} ) ;
}
function revoke ( id ) {
if ( ! confirm ( 'Revoke this bot token? Any connected bot will be disconnected immediately.' ) ) return ;
$ . ajax ( {
url : apiBase ( ) + '/bots/' + id ,
method : 'DELETE'
} ) . done ( function ( ) {
load ( ) ;
} ) . fail ( function ( xhr ) {
alert ( 'Failed to revoke: ' + ( xhr . responseJSON && xhr . responseJSON . error || xhr . statusText ) ) ;
} ) ;
}
$ ( '#cs-bots-issue' ) . on ( 'click' , function ( ) {
var name = $ ( '#cs-bots-name' ) . val ( ) . trim ( ) ;
var rank = parseInt ( $ ( '#cs-bots-rank' ) . val ( ) , 10 ) ;
if ( ! name ) { alert ( 'Bot name is required' ) ; return ; }
$ . ajax ( {
url : apiBase ( ) + '/bots' ,
method : 'POST' ,
contentType : 'application/json' ,
data : JSON . stringify ( { name : name , rank : rank } )
} ) . done ( function ( data ) {
$ ( '#cs-bots-token-value' ) . text ( data . token ) ;
$ ( '.cs-bots-token-result' ) . show ( ) ;
$ ( '#cs-bots-name' ) . val ( '' ) ;
load ( ) ;
} ) . fail ( function ( xhr ) {
alert ( 'Failed to create bot: ' + ( xhr . responseJSON && xhr . responseJSON . error || xhr . statusText ) ) ;
} ) ;
} ) ;
$ ( '.cs-bots-copy' ) . on ( 'click' , function ( ) {
var text = $ ( '#cs-bots-token-value' ) . text ( ) ;
navigator . clipboard . writeText ( text ) . catch ( function ( ) {
var el = document . getElementById ( 'cs-bots-token-value' ) ;
var range = document . createRange ( ) ;
range . selectNodeContents ( el ) ;
window . getSelection ( ) . removeAllRanges ( ) ;
window . getSelection ( ) . addRange ( range ) ;
} ) ;
} ) ;
return { load : load } ;
} ) ( ) ;
2026-05-20 20:52:26 +02:00
var CSTShows = ( function ( ) {
var selectedId = null ;
var draftPlaylist = [ ] ;
2026-05-20 21:00:48 +02:00
var timezoneOptionsLoaded = false ;
2026-05-20 21:10:49 +02:00
var resolvingTitles = false ;
2026-05-20 20:52:26 +02:00
function apiBase ( ) {
return '/api/v1/channels/' + CHANNEL . name + '/shows' ;
}
2026-05-20 21:00:48 +02:00
function loadTimezoneOptions ( ) {
if ( timezoneOptionsLoaded ) return ;
timezoneOptionsLoaded = true ;
var select = $ ( '#cs-shows-timezone' ) . empty ( ) ;
var tzs = [ ] ;
if ( typeof Intl !== 'undefined' && typeof Intl . supportedValuesOf === 'function' ) {
try {
tzs = Intl . supportedValuesOf ( 'timeZone' ) || [ ] ;
} catch ( _err ) {
tzs = [ ] ;
}
}
if ( ! tzs . length ) {
tzs = [
'UTC' ,
'Europe/Berlin' ,
'Europe/London' ,
'America/New_York' ,
'America/Chicago' ,
'America/Denver' ,
'America/Los_Angeles' ,
'Asia/Tokyo' ,
'Asia/Kolkata' ,
'Australia/Sydney'
] ;
}
tzs . forEach ( function ( tz ) {
$ ( '<option>' ) . attr ( 'value' , tz ) . text ( tz ) . appendTo ( select ) ;
} ) ;
}
2026-05-20 20:52:26 +02:00
function toLocalDateInput ( ms ) {
if ( ! ms ) return '' ;
var d = new Date ( ms ) ;
var pad = function ( n ) { return String ( n ) . padStart ( 2 , '0' ) ; } ;
return d . getFullYear ( ) + '-' +
pad ( d . getMonth ( ) + 1 ) + '-' +
pad ( d . getDate ( ) ) + 'T' +
pad ( d . getHours ( ) ) + ':' +
pad ( d . getMinutes ( ) ) ;
}
function renderDraftPlaylist ( ) {
var ul = $ ( '#cs-shows-playlist-list' ) . empty ( ) ;
if ( ! draftPlaylist . length ) {
ul . append ( '<li class="queue_entry text-muted">No items in show playlist</li>' ) ;
return ;
}
draftPlaylist . forEach ( function ( item , idx ) {
var li = $ ( '<li class="queue_entry">' ) . attr ( 'data-idx' , idx ) ;
2026-05-20 21:10:49 +02:00
var title = item . title || item . id || ( item . type + ':' + item . id ) ;
2026-05-20 20:52:26 +02:00
$ ( '<span>' ) . text ( '[' + item . type + '] ' + title ) . appendTo ( li ) ;
var controls = $ ( '<div class="btn-group pull-right">' ) . appendTo ( li ) ;
$ ( '<button class="btn btn-xs btn-default" type="button" title="Move up">' )
. html ( '<span class="glyphicon glyphicon-arrow-up"></span>' )
. on ( 'click' , function ( ) {
if ( idx <= 0 ) return ;
var tmp = draftPlaylist [ idx - 1 ] ;
draftPlaylist [ idx - 1 ] = draftPlaylist [ idx ] ;
draftPlaylist [ idx ] = tmp ;
renderDraftPlaylist ( ) ;
} )
. appendTo ( controls ) ;
$ ( '<button class="btn btn-xs btn-default" type="button" title="Move down">' )
. html ( '<span class="glyphicon glyphicon-arrow-down"></span>' )
. on ( 'click' , function ( ) {
if ( idx >= draftPlaylist . length - 1 ) return ;
var tmp = draftPlaylist [ idx + 1 ] ;
draftPlaylist [ idx + 1 ] = draftPlaylist [ idx ] ;
draftPlaylist [ idx ] = tmp ;
renderDraftPlaylist ( ) ;
} )
. appendTo ( controls ) ;
$ ( '<button class="btn btn-xs btn-danger" type="button" title="Remove">' )
. html ( '<span class="glyphicon glyphicon-remove"></span>' )
. on ( 'click' , function ( ) {
draftPlaylist . splice ( idx , 1 ) ;
renderDraftPlaylist ( ) ;
} )
. appendTo ( controls ) ;
ul . append ( li ) ;
} ) ;
}
2026-05-20 21:10:49 +02:00
function resolveDraftTitles ( ) {
if ( resolvingTitles || draftPlaylist . length === 0 ) {
return ;
}
var unresolved = draftPlaylist . filter ( function ( item ) {
return ! item . title || item . title === item . id ;
} ) . map ( function ( item ) {
return { id : item . id , type : item . type } ;
} ) ;
if ( unresolved . length === 0 ) {
return ;
}
resolvingTitles = true ;
$ . ajax ( {
url : apiBase ( ) + '/resolve-media' ,
method : 'POST' ,
contentType : 'application/json' ,
data : JSON . stringify ( { items : unresolved } )
} ) . done ( function ( data ) {
var map = { } ;
( data . items || [ ] ) . forEach ( function ( item ) {
map [ item . type + ':' + item . id ] = item . title || item . id ;
} ) ;
draftPlaylist . forEach ( function ( item ) {
var key = item . type + ':' + item . id ;
if ( map [ key ] ) {
item . title = map [ key ] ;
}
} ) ;
renderDraftPlaylist ( ) ;
} ) . always ( function ( ) {
resolvingTitles = false ;
} ) ;
}
2026-05-20 20:52:26 +02:00
function addUrlToDraft ( pos ) {
var raw = $ ( '#cs-shows-mediaurl' ) . val ( ) ;
if ( ! raw ) {
return ;
}
var links = raw . trim ( ) . split ( /\s+/ ) . filter ( function ( x ) { return x . trim ( ) !== '' ; } ) ;
if ( links . length === 0 ) return ;
var added = 0 ;
var duplicates = 0 ;
var parseFail = 0 ;
links . forEach ( function ( link ) {
var media = parseMediaLink ( link ) ;
if ( ! media || ! media . id || ! media . type ) {
parseFail ++ ;
return ;
}
var isDupe = draftPlaylist . some ( function ( item ) {
return item . id === media . id && item . type === media . type ;
} ) ;
if ( isDupe ) {
duplicates ++ ;
return ;
}
var entry = {
id : media . id ,
type : media . type ,
title : media . id ,
pos : pos === 'next' ? 'next' : 'end'
} ;
if ( pos === 'next' ) {
draftPlaylist . unshift ( entry ) ;
} else {
draftPlaylist . push ( entry ) ;
}
added ++ ;
} ) ;
$ ( '#cs-shows-mediaurl' ) . val ( '' ) ;
renderDraftPlaylist ( ) ;
2026-05-20 21:10:49 +02:00
resolveDraftTitles ( ) ;
2026-05-20 20:52:26 +02:00
if ( parseFail > 0 || duplicates > 0 ) {
var parts = [ ] ;
if ( added > 0 ) parts . push ( 'added ' + added ) ;
if ( duplicates > 0 ) parts . push ( 'skipped duplicates ' + duplicates ) ;
if ( parseFail > 0 ) parts . push ( 'failed to parse ' + parseFail ) ;
alert ( parts . join ( ', ' ) ) ;
}
}
function readFormPayload ( ) {
var scheduledRaw = $ ( '#cs-shows-scheduled-for' ) . val ( ) ;
var timezone = $ ( '#cs-shows-timezone' ) . val ( ) . trim ( ) ;
if ( ! timezone ) {
timezone = 'UTC' ;
}
return {
name : $ ( '#cs-shows-name' ) . val ( ) . trim ( ) ,
scheduled _for : scheduledRaw ? new Date ( scheduledRaw ) . toISOString ( ) : null ,
timezone : timezone ,
recurrence : $ ( '#cs-shows-recurrence' ) . val ( ) ,
fill _mode : $ ( '#cs-shows-fill-mode' ) . val ( ) ,
2026-05-21 14:14:47 +02:00
conflict _mode : $ ( '#cs-shows-conflict-skip' ) . prop ( 'checked' ) ? 'skip' : 'force' ,
2026-05-20 20:52:26 +02:00
start _playback : $ ( '#cs-shows-start-playback' ) . prop ( 'checked' ) ,
playlist : draftPlaylist . map ( function ( item ) {
return { id : item . id , type : item . type , pos : item . pos || 'end' } ;
} ) ,
status : 'scheduled'
} ;
}
function clearForm ( ) {
2026-05-20 21:00:48 +02:00
loadTimezoneOptions ( ) ;
2026-05-20 20:52:26 +02:00
selectedId = null ;
$ ( '#cs-shows-name' ) . val ( '' ) ;
$ ( '#cs-shows-scheduled-for' ) . val ( '' ) ;
var detectedTz = 'UTC' ;
if ( typeof Intl !== 'undefined' && Intl . DateTimeFormat ) {
detectedTz = Intl . DateTimeFormat ( ) . resolvedOptions ( ) . timeZone || 'UTC' ;
}
2026-05-20 21:00:48 +02:00
if ( $ ( '#cs-shows-timezone option[value="' + detectedTz + '"]' ) . length === 0 ) {
$ ( '<option>' ) . attr ( 'value' , detectedTz ) . text ( detectedTz ) . appendTo ( '#cs-shows-timezone' ) ;
}
2026-05-20 20:52:26 +02:00
$ ( '#cs-shows-timezone' ) . val ( detectedTz ) ;
$ ( '#cs-shows-recurrence' ) . val ( 'none' ) ;
$ ( '#cs-shows-fill-mode' ) . val ( 'append' ) ;
2026-05-21 14:14:47 +02:00
$ ( '#cs-shows-conflict-skip' ) . prop ( 'checked' , false ) ;
2026-05-20 20:52:26 +02:00
$ ( '#cs-shows-start-playback' ) . prop ( 'checked' , false ) ;
$ ( '#cs-shows-mediaurl' ) . val ( '' ) ;
draftPlaylist = [ ] ;
renderDraftPlaylist ( ) ;
}
function selectShow ( show ) {
2026-05-20 21:00:48 +02:00
loadTimezoneOptions ( ) ;
2026-05-20 20:52:26 +02:00
selectedId = show . id ;
$ ( '#cs-shows-name' ) . val ( show . name ) ;
$ ( '#cs-shows-scheduled-for' ) . val ( toLocalDateInput ( show . scheduled _for ) ) ;
2026-05-20 21:00:48 +02:00
var showTz = show . timezone || 'UTC' ;
if ( $ ( '#cs-shows-timezone option[value="' + showTz + '"]' ) . length === 0 ) {
$ ( '<option>' ) . attr ( 'value' , showTz ) . text ( showTz ) . appendTo ( '#cs-shows-timezone' ) ;
}
$ ( '#cs-shows-timezone' ) . val ( showTz ) ;
2026-05-20 20:52:26 +02:00
$ ( '#cs-shows-recurrence' ) . val ( show . recurrence || 'none' ) ;
$ ( '#cs-shows-fill-mode' ) . val ( show . fill _mode || 'append' ) ;
2026-05-21 14:14:47 +02:00
$ ( '#cs-shows-conflict-skip' ) . prop ( 'checked' , ( show . conflict _mode || 'force' ) === 'skip' ) ;
2026-05-20 20:52:26 +02:00
$ ( '#cs-shows-start-playback' ) . prop ( 'checked' , ! ! show . start _playback ) ;
draftPlaylist = ( show . playlist || [ ] ) . map ( function ( item ) {
return {
id : item . id ,
type : item . type ,
title : item . id ,
pos : item . pos || 'end'
} ;
} ) ;
renderDraftPlaylist ( ) ;
2026-05-20 21:10:49 +02:00
resolveDraftTitles ( ) ;
2026-05-20 20:52:26 +02:00
}
function action ( id , actionName ) {
$ . ajax ( {
url : apiBase ( ) + '/' + id + '/action' ,
method : 'POST' ,
contentType : 'application/json' ,
data : JSON . stringify ( { action : actionName } )
} ) . done ( function ( ) {
load ( ) ;
} ) . fail ( function ( xhr ) {
alert ( 'Failed action: ' + ( ( xhr . responseJSON && xhr . responseJSON . error ) || xhr . statusText ) ) ;
} ) ;
}
function render ( shows ) {
var tbody = $ ( '#cs-shows-list' ) . empty ( ) ;
if ( ! shows . length ) {
tbody . append ( '<tr><td colspan="6" class="text-muted">No shows configured</td></tr>' ) ;
return ;
}
shows . forEach ( function ( show ) {
var row = $ ( '<tr>' ) ;
row . append ( $ ( '<td>' ) . append (
$ ( '<a href=\"javascript:void(0)\">' ) . text ( show . name ) . on ( 'click' , function ( ) { selectShow ( show ) ; } )
) ) ;
row . append ( $ ( '<td>' ) . text ( show . status ) ) ;
row . append ( $ ( '<td>' ) . text ( show . next _run _at ? new Date ( show . next _run _at ) . toLocaleString ( undefined , { timeZone : show . timezone || 'UTC' } ) : 'N/A' ) ) ;
row . append ( $ ( '<td>' ) . text ( show . timezone || 'UTC' ) ) ;
row . append ( $ ( '<td>' ) . text ( show . recurrence || 'none' ) ) ;
var actions = $ ( '<td>' ) ;
$ ( '<button class=\"btn btn-xs btn-primary\" style=\"margin-right:4px\">Run</button>' )
. on ( 'click' , function ( ) { action ( show . id , 'run' ) ; } )
. appendTo ( actions ) ;
$ ( '<button class=\"btn btn-xs btn-default\" style=\"margin-right:4px\">Pause</button>' )
. on ( 'click' , function ( ) { action ( show . id , 'pause' ) ; } )
. appendTo ( actions ) ;
$ ( '<button class=\"btn btn-xs btn-success\" style=\"margin-right:4px\">Resume</button>' )
. on ( 'click' , function ( ) { action ( show . id , 'resume' ) ; } )
. appendTo ( actions ) ;
$ ( '<button class=\"btn btn-xs btn-warning\" style=\"margin-right:4px\">Cancel</button>' )
. on ( 'click' , function ( ) { action ( show . id , 'cancel' ) ; } )
. appendTo ( actions ) ;
$ ( '<button class=\"btn btn-xs btn-danger\">Delete</button>' )
. on ( 'click' , function ( ) {
if ( ! confirm ( 'Delete this show?' ) ) return ;
$ . ajax ( { url : apiBase ( ) + '/' + show . id , method : 'DELETE' } )
. done ( load )
. fail ( function ( xhr ) {
alert ( 'Delete failed: ' + ( ( xhr . responseJSON && xhr . responseJSON . error ) || xhr . statusText ) ) ;
} ) ;
} )
. appendTo ( actions ) ;
row . append ( actions ) ;
tbody . append ( row ) ;
} ) ;
}
function load ( ) {
$ . getJSON ( apiBase ( ) , render ) . fail ( function ( ) {
$ ( '#cs-shows-list' ) . html ( '<tr><td colspan=\"6\" class=\"text-danger\">Failed to load shows</td></tr>' ) ;
} ) ;
}
$ ( '#cs-shows-create' ) . on ( 'click' , function ( ) {
var payload = readFormPayload ( ) ;
$ . ajax ( {
url : apiBase ( ) ,
method : 'POST' ,
contentType : 'application/json' ,
data : JSON . stringify ( payload )
} ) . done ( function ( ) {
clearForm ( ) ;
load ( ) ;
} ) . fail ( function ( xhr ) {
alert ( 'Create failed: ' + ( ( xhr . responseJSON && xhr . responseJSON . error ) || xhr . statusText ) ) ;
} ) ;
} ) ;
$ ( '#cs-shows-update' ) . on ( 'click' , function ( ) {
if ( ! selectedId ) {
alert ( 'Select a show first' ) ;
return ;
}
var payload = readFormPayload ( ) ;
$ . ajax ( {
url : apiBase ( ) + '/' + selectedId ,
method : 'PUT' ,
contentType : 'application/json' ,
data : JSON . stringify ( payload )
} ) . done ( function ( ) {
load ( ) ;
} ) . fail ( function ( xhr ) {
alert ( 'Update failed: ' + ( ( xhr . responseJSON && xhr . responseJSON . error ) || xhr . statusText ) ) ;
} ) ;
} ) ;
$ ( '#cs-shows-add-next' ) . on ( 'click' , function ( ) { addUrlToDraft ( 'next' ) ; } ) ;
$ ( '#cs-shows-add-end' ) . on ( 'click' , function ( ) { addUrlToDraft ( 'end' ) ; } ) ;
$ ( '#cs-shows-mediaurl' ) . on ( 'keyup' , function ( ev ) {
if ( ev . keyCode === 13 ) {
addUrlToDraft ( 'end' ) ;
}
} ) ;
$ ( '#cs-shows-clear' ) . on ( 'click' , clearForm ) ;
$ ( '#cs-shows-playlist-list' ) . sortable ( {
update : function ( ) {
var nextDraft = [ ] ;
$ ( '#cs-shows-playlist-list > li' ) . each ( function ( ) {
var idx = parseInt ( $ ( this ) . attr ( 'data-idx' ) , 10 ) ;
if ( ! isNaN ( idx ) && draftPlaylist [ idx ] ) {
nextDraft . push ( draftPlaylist [ idx ] ) ;
}
} ) ;
if ( nextDraft . length === draftPlaylist . length ) {
draftPlaylist = nextDraft ;
renderDraftPlaylist ( ) ;
}
}
} ) . disableSelection ( ) ;
renderDraftPlaylist ( ) ;
clearForm ( ) ;
return { load : load } ;
} ) ( ) ;