2013-06-19 21:56:05 +00:00
|
|
|
/*
|
|
|
|
The MIT License (MIT)
|
|
|
|
Copyright (c) 2013 Calvin Montgomery
|
2013-08-31 03:12:28 +00:00
|
|
|
|
2013-06-19 21:56:05 +00:00
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
|
2013-08-31 03:12:28 +00:00
|
|
|
|
2013-06-19 21:56:05 +00:00
|
|
|
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
|
2013-08-31 03:12:28 +00:00
|
|
|
|
2013-06-19 21:56:05 +00:00
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
function makeAlert(title, text, klass) {
|
|
|
|
if(!klass) {
|
|
|
|
klass = "alert-info";
|
|
|
|
}
|
|
|
|
|
2014-01-26 20:15:50 +00:00
|
|
|
var wrap = $("<div/>").addClass("col-md-12");
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
var al = $("<div/>").addClass("alert")
|
|
|
|
.addClass(klass)
|
2014-01-26 20:15:50 +00:00
|
|
|
.html(text)
|
|
|
|
.appendTo(wrap);
|
2013-06-11 23:51:00 +00:00
|
|
|
$("<br/>").prependTo(al);
|
|
|
|
$("<strong/>").text(title).prependTo(al);
|
|
|
|
$("<button/>").addClass("close pull-right").html("×")
|
|
|
|
.click(function() {
|
2014-01-14 06:52:56 +00:00
|
|
|
al.hide("fade", function() {
|
2014-01-26 20:15:50 +00:00
|
|
|
wrap.remove();
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
})
|
|
|
|
.prependTo(al);
|
2014-01-26 20:15:50 +00:00
|
|
|
return wrap;
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function formatURL(data) {
|
|
|
|
switch(data.type) {
|
|
|
|
case "yt":
|
|
|
|
return "http://youtube.com/watch?v=" + data.id;
|
|
|
|
case "vi":
|
|
|
|
return "http://vimeo.com/" + data.id;
|
|
|
|
case "dm":
|
|
|
|
return "http://dailymotion.com/video/" + data.id;
|
|
|
|
case "sc":
|
|
|
|
return data.id;
|
|
|
|
case "li":
|
|
|
|
return "http://livestream.com/" + data.id;
|
|
|
|
case "tw":
|
|
|
|
return "http://twitch.tv/" + data.id;
|
|
|
|
case "jt":
|
|
|
|
return "http://justin.tv/" + data.id;
|
|
|
|
case "rt":
|
|
|
|
return data.id;
|
|
|
|
case "jw":
|
|
|
|
return data.id;
|
|
|
|
case "im":
|
|
|
|
return "http://imgur.com/a/" + data.id;
|
|
|
|
case "us":
|
|
|
|
return "http://ustream.tv/" + data.id;
|
2013-12-19 04:50:19 +00:00
|
|
|
case "gd":
|
|
|
|
return data.id;
|
2013-06-11 23:51:00 +00:00
|
|
|
default:
|
|
|
|
return "#";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-07 21:55:39 +00:00
|
|
|
function findUserlistItem(name) {
|
|
|
|
var children = $("#userlist .userlist_item");
|
2013-08-22 00:20:26 +00:00
|
|
|
if(children.length == 0)
|
|
|
|
return null;
|
|
|
|
name = name.toLowerCase();
|
2013-08-26 03:06:32 +00:00
|
|
|
// WARNING: Incoming hax because of jQuery and bootstrap bullshit
|
|
|
|
var keys = Object.keys(children);
|
|
|
|
for(var k in keys) {
|
|
|
|
var i = keys[k];
|
|
|
|
if(isNaN(parseInt(i))) {
|
2013-08-22 00:20:26 +00:00
|
|
|
continue;
|
2013-08-26 03:06:32 +00:00
|
|
|
}
|
|
|
|
var child = children[i];
|
2013-08-22 00:20:26 +00:00
|
|
|
if($(child.children[1]).text().toLowerCase() == name)
|
2013-08-07 21:55:39 +00:00
|
|
|
return $(child);
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2013-11-09 04:12:17 +00:00
|
|
|
function formatUserlistItem(div) {
|
|
|
|
var data = {
|
|
|
|
name: div.data("name") || "",
|
|
|
|
rank: div.data("rank"),
|
|
|
|
profile: div.data("profile") || { image: "", text: ""},
|
2013-11-09 18:33:18 +00:00
|
|
|
leader: div.data("leader") || false,
|
2013-11-15 01:50:17 +00:00
|
|
|
icon: div.data("icon") || false,
|
|
|
|
afk: div.data("afk") || false
|
2013-11-09 04:12:17 +00:00
|
|
|
};
|
2013-06-11 23:51:00 +00:00
|
|
|
var name = $(div.children()[1]);
|
|
|
|
name.removeClass();
|
|
|
|
name.css("font-style", "");
|
|
|
|
name.addClass(getNameColor(data.rank));
|
|
|
|
div.find(".profile-box").remove();
|
|
|
|
|
2013-08-29 04:30:54 +00:00
|
|
|
var profile = null;
|
2013-06-11 23:51:00 +00:00
|
|
|
name.mouseenter(function(ev) {
|
2013-08-29 04:30:54 +00:00
|
|
|
if (profile)
|
|
|
|
profile.remove();
|
|
|
|
|
2014-01-26 00:12:49 +00:00
|
|
|
var top = ev.clientY + 5// - name.position().top;
|
|
|
|
var left = ev.clientX;
|
2013-06-11 23:51:00 +00:00
|
|
|
profile = $("<div/>")
|
|
|
|
.addClass("profile-box")
|
2014-01-26 00:12:49 +00:00
|
|
|
.css("top", top + "px")
|
|
|
|
.css("left", left + "px")
|
2013-06-11 23:51:00 +00:00
|
|
|
.appendTo(div);
|
|
|
|
if(data.profile.image) {
|
|
|
|
$("<img/>").addClass("profile-image")
|
|
|
|
.attr("src", data.profile.image)
|
|
|
|
.appendTo(profile);
|
|
|
|
}
|
|
|
|
$("<strong/>").text(data.name).appendTo(profile);
|
2014-01-08 04:47:00 +00:00
|
|
|
|
|
|
|
var meta = div.data("meta") || {};
|
|
|
|
if (meta.ip) {
|
|
|
|
$("<br/>").appendTo(profile);
|
|
|
|
$("<em/>").text(meta.ip).appendTo(profile);
|
|
|
|
}
|
|
|
|
if (meta.aliases) {
|
2014-01-26 06:13:33 +00:00
|
|
|
$("<br/>").appendTo(profile);
|
|
|
|
$("<em/>").text("aliases: " + meta.aliases.join(", ")).appendTo(profile);
|
2014-01-08 04:47:00 +00:00
|
|
|
}
|
2014-01-26 06:13:33 +00:00
|
|
|
$("<hr/>").css("margin-top", "5px").css("margin-bottom", "5px").appendTo(profile);
|
2013-06-11 23:51:00 +00:00
|
|
|
$("<p/>").text(data.profile.text).appendTo(profile);
|
|
|
|
});
|
|
|
|
name.mousemove(function(ev) {
|
2014-01-26 00:12:49 +00:00
|
|
|
var top = ev.clientY + 5// - name.position().top;
|
|
|
|
var left = ev.clientX;
|
|
|
|
profile.css("top", top + "px")
|
|
|
|
.css("left", left + "px")
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
name.mouseleave(function() {
|
|
|
|
profile.remove();
|
|
|
|
});
|
2013-12-19 04:50:19 +00:00
|
|
|
var icon = div.children()[0];
|
|
|
|
icon.innerHTML = "";
|
2013-06-11 23:51:00 +00:00
|
|
|
// denote current leader with a star
|
|
|
|
if(data.leader) {
|
2013-12-19 04:50:19 +00:00
|
|
|
$("<span/>").addClass("glyphicon glyphicon-star-empty").appendTo(icon);
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
2013-11-15 01:50:17 +00:00
|
|
|
if(data.afk) {
|
2013-06-11 23:51:00 +00:00
|
|
|
name.css("font-style", "italic");
|
2013-12-19 04:50:19 +00:00
|
|
|
$("<span/>").addClass("glyphicon glyphicon-time").appendTo(icon);
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
2013-11-09 18:33:18 +00:00
|
|
|
if (data.icon) {
|
2014-01-19 02:18:00 +00:00
|
|
|
$("<span/>").addClass("glyphicon " + data.icon).prependTo(icon);
|
2013-06-25 14:18:33 +00:00
|
|
|
}
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function getNameColor(rank) {
|
|
|
|
if(rank >= Rank.Siteadmin)
|
|
|
|
return "userlist_siteadmin";
|
2013-06-18 14:46:28 +00:00
|
|
|
else if(rank >= Rank.Admin)
|
2013-06-11 23:51:00 +00:00
|
|
|
return "userlist_owner";
|
|
|
|
else if(rank >= Rank.Moderator)
|
|
|
|
return "userlist_op";
|
|
|
|
else if(rank == Rank.Guest)
|
|
|
|
return "userlist_guest";
|
|
|
|
else
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
2013-11-09 04:12:17 +00:00
|
|
|
function addUserDropdown(entry) {
|
|
|
|
var name = entry.data("name"),
|
|
|
|
rank = entry.data("rank"),
|
2014-01-19 22:50:14 +00:00
|
|
|
leader = entry.data("leader"),
|
|
|
|
meta = entry.data("meta") || {};
|
2013-06-11 23:51:00 +00:00
|
|
|
entry.find(".user-dropdown").remove();
|
2013-08-07 20:14:22 +00:00
|
|
|
var menu = $("<div/>")
|
|
|
|
.addClass("user-dropdown")
|
|
|
|
.appendTo(entry)
|
|
|
|
.hide();
|
2013-06-11 23:51:00 +00:00
|
|
|
|
|
|
|
$("<strong/>").text(name).appendTo(menu);
|
|
|
|
$("<br/>").appendTo(menu);
|
2013-08-07 20:14:22 +00:00
|
|
|
|
2014-01-12 05:55:52 +00:00
|
|
|
var btngroup = $("<div/>").addClass("btn-group-vertical").appendTo(menu);
|
2013-08-31 03:12:28 +00:00
|
|
|
|
2013-08-07 20:14:22 +00:00
|
|
|
/* ignore button */
|
2014-01-12 05:55:52 +00:00
|
|
|
var ignore = $("<button/>").addClass("btn btn-xs btn-default")
|
|
|
|
.appendTo(btngroup)
|
2013-08-07 20:14:22 +00:00
|
|
|
.click(function () {
|
|
|
|
if(IGNORED.indexOf(name) == -1) {
|
|
|
|
ignore.text("Unignore User");
|
|
|
|
IGNORED.push(name);
|
|
|
|
} else {
|
|
|
|
ignore.text("Ignore User");
|
|
|
|
IGNORED.splice(IGNORED.indexOf(name), 1);
|
|
|
|
}
|
|
|
|
});
|
2013-06-25 14:06:01 +00:00
|
|
|
if(IGNORED.indexOf(name) == -1) {
|
|
|
|
ignore.text("Ignore User");
|
2013-08-07 20:14:22 +00:00
|
|
|
} else {
|
2013-06-25 14:06:01 +00:00
|
|
|
ignore.text("Unignore User");
|
|
|
|
}
|
2013-08-07 20:14:22 +00:00
|
|
|
|
2014-01-23 22:03:50 +00:00
|
|
|
/* give/remove leader (moderator+ only) */
|
|
|
|
if (hasPermission("leaderctl")) {
|
2014-01-12 05:55:52 +00:00
|
|
|
var ldr = $("<button/>").addClass("btn btn-xs btn-default")
|
|
|
|
.appendTo(btngroup);
|
2013-08-07 20:14:22 +00:00
|
|
|
if(leader) {
|
|
|
|
ldr.text("Remove Leader");
|
|
|
|
ldr.click(function () {
|
|
|
|
socket.emit("assignLeader", {
|
|
|
|
name: ""
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
ldr.text("Give Leader");
|
|
|
|
ldr.click(function () {
|
2013-06-11 23:51:00 +00:00
|
|
|
socket.emit("assignLeader", {
|
|
|
|
name: name
|
|
|
|
});
|
2013-08-07 20:14:22 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* kick button */
|
|
|
|
if(hasPermission("kick")) {
|
2014-01-12 05:55:52 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default")
|
2013-08-07 20:14:22 +00:00
|
|
|
.text("Kick")
|
|
|
|
.click(function () {
|
2014-01-20 23:35:55 +00:00
|
|
|
var reason = prompt("Enter kick reason (optional)");
|
2013-08-07 20:14:22 +00:00
|
|
|
socket.emit("chatMsg", {
|
2014-01-20 23:35:55 +00:00
|
|
|
msg: "/kick " + name + " " + reason
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
})
|
2014-01-12 05:55:52 +00:00
|
|
|
.appendTo(btngroup);
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
2013-08-07 20:14:22 +00:00
|
|
|
|
2014-01-19 22:50:14 +00:00
|
|
|
/* mute buttons */
|
|
|
|
if (hasPermission("mute")) {
|
|
|
|
var mute = $("<button/>").addClass("btn btn-xs btn-default")
|
|
|
|
.text("Mute")
|
|
|
|
.click(function () {
|
|
|
|
socket.emit("chatMsg", {
|
|
|
|
msg: "/mute " + name
|
|
|
|
});
|
|
|
|
mute.hide();
|
|
|
|
smute.hide();
|
|
|
|
unmute.show();
|
|
|
|
})
|
|
|
|
.appendTo(btngroup);
|
|
|
|
var smute = $("<button/>").addClass("btn btn-xs btn-default")
|
|
|
|
.text("Shadow Mute")
|
|
|
|
.click(function () {
|
|
|
|
socket.emit("chatMsg", {
|
|
|
|
msg: "/smute " + name
|
|
|
|
});
|
|
|
|
mute.hide();
|
|
|
|
smute.hide();
|
|
|
|
unmute.show();
|
|
|
|
})
|
|
|
|
.appendTo(btngroup);
|
|
|
|
var unmute = $("<button/>").addClass("btn btn-xs btn-default")
|
|
|
|
.text("Unmute")
|
|
|
|
.click(function () {
|
|
|
|
socket.emit("chatMsg", {
|
|
|
|
msg: "/unmute " + name
|
|
|
|
});
|
|
|
|
unmute.hide();
|
|
|
|
mute.show();
|
|
|
|
smute.show();
|
|
|
|
})
|
|
|
|
.appendTo(btngroup);
|
|
|
|
if (meta.muted) {
|
|
|
|
mute.hide();
|
|
|
|
smute.hide();
|
|
|
|
} else {
|
|
|
|
unmute.hide();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-07 20:14:22 +00:00
|
|
|
/* ban buttons */
|
2013-06-11 23:51:00 +00:00
|
|
|
if(hasPermission("ban")) {
|
2014-01-12 05:55:52 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("Name Ban")
|
2013-08-07 20:14:22 +00:00
|
|
|
.click(function () {
|
2014-01-20 23:35:55 +00:00
|
|
|
var reason = prompt("Enter ban reason (optional)");
|
2013-06-11 23:51:00 +00:00
|
|
|
socket.emit("chatMsg", {
|
2014-01-20 23:35:55 +00:00
|
|
|
msg: "/ban " + name + " " + reason
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
})
|
2014-01-12 05:55:52 +00:00
|
|
|
.appendTo(btngroup);
|
|
|
|
$("<button/>").addClass("btn btn-xs btn-default")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("IP Ban")
|
2013-08-07 20:14:22 +00:00
|
|
|
.click(function () {
|
2014-01-20 23:35:55 +00:00
|
|
|
var reason = prompt("Enter ban reason (optional)");
|
2013-06-11 23:51:00 +00:00
|
|
|
socket.emit("chatMsg", {
|
2014-01-20 23:35:55 +00:00
|
|
|
msg: "/ipban " + name + " " + reason
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
})
|
2014-01-12 05:55:52 +00:00
|
|
|
.appendTo(btngroup);
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
entry.contextmenu(function(ev) {
|
|
|
|
ev.preventDefault();
|
|
|
|
if(menu.css("display") == "none") {
|
2013-10-17 04:10:59 +00:00
|
|
|
$(".user-dropdown").hide();
|
|
|
|
$(document).bind("mouseup.userlist-ddown", function (e) {
|
|
|
|
if (menu.has(e.target).length === 0 &&
|
|
|
|
entry.parent().has(e.target).length === 0) {
|
|
|
|
menu.hide();
|
|
|
|
$(document).unbind("mouseup.userlist-ddown");
|
|
|
|
}
|
|
|
|
});
|
2013-06-11 23:51:00 +00:00
|
|
|
menu.show();
|
2013-08-07 20:14:22 +00:00
|
|
|
} else {
|
2013-06-11 23:51:00 +00:00
|
|
|
menu.hide();
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2013-08-08 14:39:58 +00:00
|
|
|
function calcUserBreakdown() {
|
|
|
|
var breakdown = {
|
|
|
|
"Site Admins": 0,
|
|
|
|
"Channel Admins": 0,
|
|
|
|
"Moderators": 0,
|
|
|
|
"Regular Users": 0,
|
|
|
|
"Guests": 0,
|
2013-09-04 22:47:24 +00:00
|
|
|
"Anonymous": 0,
|
2013-08-08 14:39:58 +00:00
|
|
|
"AFK": 0
|
|
|
|
};
|
2013-09-04 22:47:24 +00:00
|
|
|
var total = 0;
|
2013-08-08 14:39:58 +00:00
|
|
|
$("#userlist .userlist_item").each(function (index, item) {
|
2013-11-09 04:12:17 +00:00
|
|
|
var data = {
|
|
|
|
rank: $(item).data("rank")
|
|
|
|
};
|
2013-11-16 05:44:53 +00:00
|
|
|
|
2013-08-08 14:39:58 +00:00
|
|
|
if(data.rank >= 255)
|
|
|
|
breakdown["Site Admins"]++;
|
|
|
|
else if(data.rank >= 3)
|
|
|
|
breakdown["Channel Admins"]++;
|
|
|
|
else if(data.rank == 2)
|
|
|
|
breakdown["Moderators"]++;
|
|
|
|
else if(data.rank >= 1)
|
|
|
|
breakdown["Regular Users"]++;
|
|
|
|
else
|
|
|
|
breakdown["Guests"]++;
|
|
|
|
|
2013-09-04 22:47:24 +00:00
|
|
|
total++;
|
|
|
|
|
2013-08-08 14:39:58 +00:00
|
|
|
if($(item).find(".icon-time").length > 0)
|
|
|
|
breakdown["AFK"]++;
|
|
|
|
});
|
2013-08-31 03:12:28 +00:00
|
|
|
|
2013-09-04 22:47:24 +00:00
|
|
|
breakdown["Anonymous"] = CHANNEL.usercount - total;
|
|
|
|
|
2013-08-08 14:39:58 +00:00
|
|
|
return breakdown;
|
|
|
|
}
|
|
|
|
|
2013-08-08 14:57:46 +00:00
|
|
|
function sortUserlist() {
|
|
|
|
var slice = Array.prototype.slice;
|
|
|
|
var list = slice.call($("#userlist .userlist_item"));
|
|
|
|
list.sort(function (a, b) {
|
2013-11-09 04:12:17 +00:00
|
|
|
var r1 = $(a).data("rank");
|
|
|
|
var r2 = $(b).data("rank");
|
2013-12-19 04:50:19 +00:00
|
|
|
var afk1 = $(a).find(".glyphicon-time").length > 0;
|
|
|
|
var afk2 = $(b).find(".glyphicon-time").length > 0;
|
2013-08-08 14:57:46 +00:00
|
|
|
var name1 = a.children[1].innerHTML.toLowerCase();
|
|
|
|
var name2 = b.children[1].innerHTML.toLowerCase();
|
|
|
|
|
|
|
|
if(USEROPTS.sort_afk) {
|
|
|
|
if(afk1 && !afk2)
|
|
|
|
return 1;
|
|
|
|
if(!afk1 && afk2)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(USEROPTS.sort_rank) {
|
|
|
|
if(r1 < r2)
|
|
|
|
return 1;
|
|
|
|
if(r1 > r2)
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return name1 === name2 ? 0 : (name1 < name2 ? -1 : 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
list.forEach(function (item) {
|
|
|
|
$(item).detach();
|
|
|
|
});
|
|
|
|
list.forEach(function (item) {
|
|
|
|
$(item).appendTo($("#userlist"));
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
/* queue stuff */
|
|
|
|
|
2013-07-03 21:29:49 +00:00
|
|
|
function scrollQueue() {
|
|
|
|
var li = playlistFind(PL_CURRENT);
|
|
|
|
if(!li)
|
|
|
|
return;
|
|
|
|
|
|
|
|
li = $(li);
|
|
|
|
$("#queue").scrollTop(0);
|
|
|
|
var scroll = li.position().top - $("#queue").position().top;
|
|
|
|
$("#queue").scrollTop(scroll);
|
|
|
|
}
|
|
|
|
|
2013-06-29 22:09:20 +00:00
|
|
|
function makeQueueEntry(item, addbtns) {
|
|
|
|
var video = item.media;
|
2013-06-11 23:51:00 +00:00
|
|
|
var li = $("<li/>");
|
|
|
|
li.addClass("queue_entry");
|
2013-06-29 22:09:20 +00:00
|
|
|
li.addClass("pluid-" + item.uid);
|
|
|
|
li.data("uid", item.uid);
|
2013-06-11 23:51:00 +00:00
|
|
|
li.data("media", video);
|
2013-06-29 22:09:20 +00:00
|
|
|
li.data("temp", item.temp);
|
2013-06-11 23:51:00 +00:00
|
|
|
if(video.thumb) {
|
|
|
|
$("<img/>").attr("src", video.thumb.url)
|
|
|
|
.css("float", "left")
|
|
|
|
.css("clear", "both")
|
|
|
|
.appendTo(li);
|
|
|
|
}
|
|
|
|
var title = $("<a/>").addClass("qe_title").appendTo(li)
|
|
|
|
.text(video.title)
|
|
|
|
.attr("href", formatURL(video))
|
|
|
|
.attr("target", "_blank");
|
|
|
|
var time = $("<span/>").addClass("qe_time").appendTo(li);
|
|
|
|
time.text(video.duration);
|
|
|
|
var clear = $("<div/>").addClass("qe_clear").appendTo(li);
|
2013-06-29 22:09:20 +00:00
|
|
|
if(item.temp) {
|
2013-06-11 23:51:00 +00:00
|
|
|
li.addClass("queue_temp");
|
|
|
|
}
|
|
|
|
|
|
|
|
if(addbtns)
|
|
|
|
addQueueButtons(li);
|
|
|
|
return li;
|
|
|
|
}
|
|
|
|
|
2013-06-29 22:09:20 +00:00
|
|
|
function makeSearchEntry(video) {
|
|
|
|
var li = $("<li/>");
|
|
|
|
li.addClass("queue_entry");
|
|
|
|
li.data("media", video);
|
|
|
|
if(video.thumb) {
|
|
|
|
$("<img/>").attr("src", video.thumb.url)
|
|
|
|
.css("float", "left")
|
|
|
|
.css("clear", "both")
|
|
|
|
.appendTo(li);
|
|
|
|
}
|
|
|
|
var title = $("<a/>").addClass("qe_title").appendTo(li)
|
|
|
|
.text(video.title)
|
|
|
|
.attr("href", formatURL(video))
|
|
|
|
.attr("target", "_blank");
|
|
|
|
var time = $("<span/>").addClass("qe_time").appendTo(li);
|
|
|
|
time.text(video.duration);
|
|
|
|
var clear = $("<div/>").addClass("qe_clear").appendTo(li);
|
|
|
|
|
|
|
|
return li;
|
|
|
|
}
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
function addQueueButtons(li) {
|
|
|
|
li.find(".btn-group").remove();
|
|
|
|
var menu = $("<div/>").addClass("btn-group").appendTo(li);
|
|
|
|
// Play
|
|
|
|
if(hasPermission("playlistjump")) {
|
2013-12-19 04:50:19 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default qbtn-play")
|
2013-12-19 17:14:48 +00:00
|
|
|
.html("<span class='glyphicon glyphicon-play'></span>Play")
|
2013-06-11 23:51:00 +00:00
|
|
|
.click(function() {
|
2013-06-29 22:09:20 +00:00
|
|
|
socket.emit("jumpTo", li.data("uid"));
|
2013-06-11 23:51:00 +00:00
|
|
|
})
|
|
|
|
.appendTo(menu);
|
|
|
|
}
|
|
|
|
// Queue next
|
2013-07-28 18:10:47 +00:00
|
|
|
if(hasPermission("playlistmove")) {
|
2013-12-19 04:50:19 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default qbtn-next")
|
2013-12-19 17:14:48 +00:00
|
|
|
.html("<span class='glyphicon glyphicon-share-alt'></span>Queue Next")
|
2013-06-11 23:51:00 +00:00
|
|
|
.click(function() {
|
|
|
|
socket.emit("moveMedia", {
|
2013-06-29 22:09:20 +00:00
|
|
|
from: li.data("uid"),
|
2013-10-04 03:11:47 +00:00
|
|
|
after: PL_CURRENT
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
})
|
|
|
|
.appendTo(menu);
|
|
|
|
}
|
|
|
|
// Temp/Untemp
|
|
|
|
if(hasPermission("settemp")) {
|
2013-06-29 22:09:20 +00:00
|
|
|
var tempstr = li.data("temp")?"Make Permanent":"Make Temporary";
|
2013-12-19 04:50:19 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default qbtn-tmp")
|
2013-12-19 17:14:48 +00:00
|
|
|
.html("<span class='glyphicon glyphicon-flag'></span>" + tempstr)
|
2013-06-11 23:51:00 +00:00
|
|
|
.click(function() {
|
|
|
|
socket.emit("setTemp", {
|
2013-06-29 22:09:20 +00:00
|
|
|
uid: li.data("uid"),
|
2013-07-09 17:11:44 +00:00
|
|
|
temp: !li.data("temp")
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
})
|
|
|
|
.appendTo(menu);
|
|
|
|
}
|
|
|
|
// Delete
|
|
|
|
if(hasPermission("playlistdelete")) {
|
2013-12-19 04:50:19 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default qbtn-delete")
|
2013-12-19 17:14:48 +00:00
|
|
|
.html("<span class='glyphicon glyphicon-trash'></span>Delete")
|
2013-06-11 23:51:00 +00:00
|
|
|
.click(function() {
|
2013-06-29 22:09:20 +00:00
|
|
|
socket.emit("delete", li.data("uid"));
|
2013-06-11 23:51:00 +00:00
|
|
|
})
|
|
|
|
.appendTo(menu);
|
|
|
|
}
|
|
|
|
|
2013-06-20 21:15:41 +00:00
|
|
|
if(USEROPTS.qbtn_hide && !USEROPTS.qbtn_idontlikechange
|
|
|
|
|| menu.find(".btn").length == 0)
|
2013-06-20 02:49:49 +00:00
|
|
|
menu.hide();
|
2013-06-11 23:51:00 +00:00
|
|
|
|
2013-06-20 20:50:12 +00:00
|
|
|
// I DON'T LIKE CHANGE
|
|
|
|
if(USEROPTS.qbtn_idontlikechange) {
|
|
|
|
menu.addClass("pull-left");
|
|
|
|
menu.detach().prependTo(li);
|
|
|
|
menu.find(".btn").each(function() {
|
|
|
|
// Clear icon
|
2013-12-25 21:18:21 +00:00
|
|
|
var icon = $(this).find(".glyphicon");
|
2013-06-20 20:50:12 +00:00
|
|
|
$(this).html("");
|
|
|
|
icon.appendTo(this);
|
|
|
|
});
|
|
|
|
menu.find(".qbtn-play").addClass("btn-success");
|
|
|
|
menu.find(".qbtn-delete").addClass("btn-danger");
|
|
|
|
}
|
2013-06-20 21:15:41 +00:00
|
|
|
else if(menu.find(".btn").length != 0) {
|
2013-06-25 14:38:19 +00:00
|
|
|
li.unbind("contextmenu");
|
2013-06-20 20:50:12 +00:00
|
|
|
li.contextmenu(function(ev) {
|
|
|
|
ev.preventDefault();
|
|
|
|
if(menu.css("display") == "none")
|
|
|
|
menu.show("blind");
|
|
|
|
else
|
|
|
|
menu.hide("blind");
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
}
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function rebuildPlaylist() {
|
2013-06-25 03:33:00 +00:00
|
|
|
var qli = $("#queue li");
|
|
|
|
if(qli.length == 0)
|
|
|
|
return;
|
2013-06-27 02:44:48 +00:00
|
|
|
REBUILDING = Math.random() + "";
|
|
|
|
var r = REBUILDING;
|
2013-06-25 02:45:44 +00:00
|
|
|
var i = 0;
|
|
|
|
qli.each(function() {
|
|
|
|
var li = $(this);
|
2013-06-27 02:44:48 +00:00
|
|
|
(function(i, r) {
|
2013-06-25 14:38:19 +00:00
|
|
|
setTimeout(function() {
|
2013-06-27 02:44:48 +00:00
|
|
|
// Stop if another rebuild is running
|
|
|
|
if(REBUILDING != r)
|
|
|
|
return;
|
2013-06-25 14:38:19 +00:00
|
|
|
addQueueButtons(li);
|
|
|
|
if(i == qli.length - 1) {
|
2013-09-02 17:09:14 +00:00
|
|
|
scrollQueue();
|
2013-06-25 14:38:19 +00:00
|
|
|
REBUILDING = false;
|
|
|
|
}
|
|
|
|
}, 10*i);
|
2013-06-27 02:44:48 +00:00
|
|
|
})(i, r);
|
2013-06-25 02:45:44 +00:00
|
|
|
i++;
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/* menus */
|
2013-08-31 17:37:37 +00:00
|
|
|
|
|
|
|
/* user settings menu */
|
2013-12-25 21:18:21 +00:00
|
|
|
function showUserOptions() {
|
2013-08-31 03:12:28 +00:00
|
|
|
hidePlayer();
|
2014-01-19 02:18:00 +00:00
|
|
|
$("#useroptions").on("hidden.bs.modal", function () {
|
|
|
|
console.log("unhiding");
|
2013-08-06 02:46:56 +00:00
|
|
|
unhidePlayer();
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
2013-08-31 17:37:37 +00:00
|
|
|
|
2013-12-25 21:18:21 +00:00
|
|
|
if (CLIENT.rank < 2) {
|
|
|
|
$("a[href='#us-mod']").parent().hide();
|
|
|
|
} else {
|
|
|
|
$("a[href='#us-mod']").parent().show();
|
|
|
|
}
|
|
|
|
|
|
|
|
$("#us-theme").val(USEROPTS.theme);
|
|
|
|
$("#us-layout").val(USEROPTS.layout);
|
|
|
|
$("#us-no-channelcss").prop("checked", USEROPTS.ignore_channelcss);
|
|
|
|
$("#us-no-channeljs").prop("checked", USEROPTS.ignore_channeljs);
|
|
|
|
$("#us-ssl").prop("checked", USEROPTS.secure_connection);
|
|
|
|
|
|
|
|
$("#us-synch").prop("checked", USEROPTS.synch);
|
|
|
|
$("#us-synch-accuracy").val(USEROPTS.synch_accuracy);
|
|
|
|
$("#us-wmode-transparent").prop("checked", USEROPTS.wmode_transparent);
|
|
|
|
$("#us-hidevideo").prop("checked", USEROPTS.hidevid);
|
|
|
|
$("#us-playlistbuttons").prop("checked", USEROPTS.qbtn_hide);
|
|
|
|
$("#us-oldbtns").prop("checked", USEROPTS.qbtn_idontlikechange);
|
2014-01-19 02:18:00 +00:00
|
|
|
$("#us-default-quality").val(USEROPTS.default_quality || "auto");
|
2013-12-25 21:18:21 +00:00
|
|
|
|
|
|
|
$("#us-chat-timestamp").prop("checked", USEROPTS.show_timestamps);
|
|
|
|
$("#us-sort-rank").prop("checked", USEROPTS.sort_rank);
|
|
|
|
$("#us-sort-afk").prop("checked", USEROPTS.sort_afk);
|
|
|
|
$("#us-chat-notice").prop("checked", USEROPTS.blink_title);
|
|
|
|
$("#us-boop").prop("checked", USEROPTS.boop);
|
|
|
|
$("#us-sendbtn").prop("checked", USEROPTS.chatbtn);
|
2014-01-08 04:47:00 +00:00
|
|
|
|
2013-12-25 21:18:21 +00:00
|
|
|
$("#us-modflair").prop("checked", USEROPTS.modhat);
|
|
|
|
$("#us-joinmessage").prop("checked", USEROPTS.joinmessage);
|
|
|
|
|
|
|
|
$("a[href='#us-general']").click();
|
|
|
|
$("#useroptions").modal();
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
2013-12-25 21:18:21 +00:00
|
|
|
function saveUserOptions() {
|
|
|
|
USEROPTS.theme = $("#us-theme").val();
|
2014-01-30 04:50:14 +00:00
|
|
|
createCookie("cytube-theme", USEROPTS.theme, 1000);
|
2013-12-25 21:18:21 +00:00
|
|
|
USEROPTS.layout = $("#us-layout").val();
|
|
|
|
USEROPTS.ignore_channelcss = $("#us-no-channelcss").prop("checked");
|
|
|
|
USEROPTS.ignore_channeljs = $("#us-no-channeljs").prop("checked");
|
|
|
|
USEROPTS.secure_connection = $("#us-ssl").prop("checked");
|
|
|
|
|
|
|
|
USEROPTS.synch = $("#us-synch").prop("checked");
|
|
|
|
USEROPTS.synch_accuracy = parseFloat($("#us-synch-accuracy").val()) || 2;
|
|
|
|
USEROPTS.wmode_transparent = $("#us-wmode-transparent").prop("checked");
|
|
|
|
USEROPTS.hidevid = $("#us-hidevideo").prop("checked");
|
|
|
|
USEROPTS.qbtn_hide = $("#us-playlistbuttons").prop("checked");
|
|
|
|
USEROPTS.qbtn_idontlikechange = $("#us-oldbtns").prop("checked");
|
2014-01-19 02:18:00 +00:00
|
|
|
USEROPTS.default_quality = $("#us-default-quality").val();
|
2013-12-25 21:18:21 +00:00
|
|
|
|
|
|
|
USEROPTS.show_timestamps = $("#us-chat-timestamp").prop("checked");
|
|
|
|
USEROPTS.sort_rank = $("#us-sort-rank").prop("checked");
|
|
|
|
USEROPTS.sort_afk = $("#us-sort-afk").prop("checked");
|
|
|
|
USEROPTS.blink_title = $("#us-chat-notice").prop("checked");
|
|
|
|
USEROPTS.boop = $("#us-boop").prop("checked");
|
|
|
|
USEROPTS.chatbtn = $("#us-sendbtn").prop("checked");
|
|
|
|
|
|
|
|
if (CLIENT.rank >= 2) {
|
|
|
|
USEROPTS.modhat = $("#us-modflair").prop("checked");
|
|
|
|
USEROPTS.joinmessage = $("#us-joinmessage").prop("checked");
|
|
|
|
}
|
|
|
|
|
|
|
|
storeOpts();
|
2014-01-30 04:50:14 +00:00
|
|
|
applyOpts();
|
2013-12-25 21:18:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function storeOpts() {
|
2013-06-11 23:51:00 +00:00
|
|
|
for(var key in USEROPTS) {
|
2013-07-03 02:40:40 +00:00
|
|
|
setOpt(key, USEROPTS[key]);
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function applyOpts() {
|
2014-01-30 04:50:14 +00:00
|
|
|
if ($("#usertheme").attr("href") !== USEROPTS.theme) {
|
|
|
|
$("#usertheme").remove();
|
|
|
|
if(USEROPTS.theme != "default") {
|
|
|
|
$("<link/>").attr("rel", "stylesheet")
|
|
|
|
.attr("type", "text/css")
|
|
|
|
.attr("id", "usertheme")
|
|
|
|
.attr("href", USEROPTS.theme)
|
|
|
|
.appendTo($("head"));
|
|
|
|
}
|
|
|
|
fixWeirdButtonAlignmentIssue();
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
2014-01-26 06:13:33 +00:00
|
|
|
switch (USEROPTS.layout) {
|
|
|
|
case "synchtube-fluid":
|
|
|
|
fluidLayout();
|
2013-06-11 23:51:00 +00:00
|
|
|
case "synchtube":
|
|
|
|
synchtubeLayout();
|
|
|
|
break;
|
|
|
|
case "fluid":
|
|
|
|
fluidLayout();
|
|
|
|
break;
|
2014-01-26 20:15:50 +00:00
|
|
|
case "hd":
|
|
|
|
hdLayout();
|
|
|
|
break;
|
2013-06-11 23:51:00 +00:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(USEROPTS.hidevid) {
|
2013-07-04 22:50:15 +00:00
|
|
|
$("#qualitywrap").html("");
|
2013-06-11 23:51:00 +00:00
|
|
|
$("#videowrap").remove();
|
2013-12-25 21:18:21 +00:00
|
|
|
$("#chatwrap").removeClass("col-lg-5 col-md-5").addClass("col-lg-12 col-md-12");
|
2014-01-26 06:13:33 +00:00
|
|
|
$("#chatline").removeClass().addClass("col-lg-12 col-md-12 form-control");
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
$("#chatbtn").remove();
|
|
|
|
if(USEROPTS.chatbtn) {
|
2013-12-25 21:18:21 +00:00
|
|
|
var btn = $("<button/>").addClass("btn btn-default btn-block")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("Send")
|
|
|
|
.attr("id", "chatbtn")
|
|
|
|
.appendTo($("#chatwrap"));
|
|
|
|
btn.click(function() {
|
|
|
|
if($("#chatline").val().trim()) {
|
|
|
|
socket.emit("chatMsg", {
|
|
|
|
msg: $("#chatline").val()
|
|
|
|
});
|
|
|
|
$("#chatline").val("");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2013-09-21 07:22:51 +00:00
|
|
|
|
|
|
|
if (USEROPTS.modhat) {
|
|
|
|
$("#modflair").removeClass("label-default")
|
|
|
|
.addClass("label-success");
|
|
|
|
} else {
|
|
|
|
$("#modflair").removeClass("label-success")
|
|
|
|
.addClass("label-default");
|
|
|
|
}
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
applyOpts();
|
|
|
|
|
|
|
|
function showPollMenu() {
|
|
|
|
$("#pollwrap .poll-menu").remove();
|
|
|
|
var menu = $("<div/>").addClass("well poll-menu")
|
2013-12-19 17:14:48 +00:00
|
|
|
.prependTo($("#pollwrap"));
|
2013-06-11 23:51:00 +00:00
|
|
|
|
2013-12-19 17:14:48 +00:00
|
|
|
$("<button/>").addClass("btn btn-sm btn-danger pull-right")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("Cancel")
|
|
|
|
.appendTo(menu)
|
|
|
|
.click(function() {
|
|
|
|
menu.remove();
|
|
|
|
});
|
|
|
|
|
|
|
|
$("<strong/>").text("Title").appendTo(menu);
|
|
|
|
|
2013-12-19 17:14:48 +00:00
|
|
|
var title = $("<input/>").addClass("form-control")
|
|
|
|
.attr("type", "text")
|
2013-06-11 23:51:00 +00:00
|
|
|
.appendTo(menu);
|
|
|
|
|
2013-09-12 01:22:00 +00:00
|
|
|
var lbl = $("<label/>").addClass("checkbox")
|
|
|
|
.text("Hide poll results")
|
|
|
|
.appendTo(menu);
|
|
|
|
var hidden = $("<input/>").attr("type", "checkbox")
|
|
|
|
.appendTo(lbl);
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
$("<strong/>").text("Options").appendTo(menu);
|
|
|
|
|
2013-12-19 17:14:48 +00:00
|
|
|
var addbtn = $("<button/>").addClass("btn btn-sm btn-default")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("Add Option")
|
|
|
|
.appendTo(menu);
|
|
|
|
|
|
|
|
function addOption() {
|
2013-12-19 17:14:48 +00:00
|
|
|
$("<input/>").addClass("form-control")
|
|
|
|
.attr("type", "text")
|
2013-06-11 23:51:00 +00:00
|
|
|
.addClass("poll-menu-option")
|
|
|
|
.insertBefore(addbtn);
|
|
|
|
}
|
|
|
|
|
|
|
|
addbtn.click(addOption);
|
|
|
|
addOption();
|
|
|
|
addOption();
|
|
|
|
|
2013-12-19 17:14:48 +00:00
|
|
|
$("<button/>").addClass("btn btn-default btn-block")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("Open Poll")
|
|
|
|
.appendTo(menu)
|
|
|
|
.click(function() {
|
|
|
|
var opts = []
|
|
|
|
menu.find(".poll-menu-option").each(function() {
|
|
|
|
if($(this).val() != "")
|
|
|
|
opts.push($(this).val());
|
|
|
|
});
|
|
|
|
socket.emit("newPoll", {
|
|
|
|
title: title.val(),
|
2013-09-12 01:22:00 +00:00
|
|
|
opts: opts,
|
|
|
|
obscured: hidden.prop("checked")
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
2013-06-22 23:02:55 +00:00
|
|
|
menu.remove();
|
2013-06-11 23:51:00 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
function scrollChat() {
|
|
|
|
$("#messagebuffer").scrollTop($("#messagebuffer").prop("scrollHeight"));
|
|
|
|
}
|
|
|
|
|
|
|
|
function hasPermission(key) {
|
|
|
|
if(key.indexOf("playlist") == 0 && CHANNEL.openqueue) {
|
|
|
|
var key2 = "o" + key;
|
|
|
|
var v = CHANNEL.perms[key2];
|
|
|
|
if(typeof v == "number" && CLIENT.rank >= v) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var v = CHANNEL.perms[key];
|
|
|
|
if(typeof v != "number") {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return CLIENT.rank >= v;
|
|
|
|
}
|
|
|
|
|
2013-06-20 18:54:15 +00:00
|
|
|
function setVisible(selector, bool) {
|
|
|
|
// I originally added this check because of a race condition
|
|
|
|
// Now it seems to work without but I don't trust it
|
|
|
|
if($(selector) && $(selector).attr("id") != selector.substring(1)) {
|
|
|
|
setTimeout(function() {
|
|
|
|
setVisible(selector, bool);
|
|
|
|
}, 100);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var disp = bool ? "" : "none";
|
|
|
|
$(selector).css("display", disp);
|
|
|
|
}
|
|
|
|
|
2014-01-15 06:16:29 +00:00
|
|
|
function setParentVisible(selector, bool) {
|
|
|
|
var disp = bool ? "" : "none";
|
|
|
|
$(selector).parent().css("display", disp);
|
|
|
|
}
|
|
|
|
|
2013-06-20 18:43:37 +00:00
|
|
|
function handleModPermissions() {
|
2014-01-09 23:16:09 +00:00
|
|
|
$("#cs-chanranks-adm").attr("disabled", CLIENT.rank < 4);
|
|
|
|
$("#cs-chanranks-owner").attr("disabled", CLIENT.rank < 4);
|
2013-06-20 18:43:37 +00:00
|
|
|
/* update channel controls */
|
2014-01-15 06:16:29 +00:00
|
|
|
$("#cs-pagetitle").val(CHANNEL.opts.pagetitle);
|
|
|
|
$("#cs-pagetitle").attr("disabled", CLIENT.rank < 3);
|
|
|
|
$("#cs-externalcss").val(CHANNEL.opts.externalcss);
|
|
|
|
$("#cs-externalcss").attr("disabled", CLIENT.rank < 3);
|
|
|
|
$("#cs-externaljs").val(CHANNEL.opts.externaljs);
|
|
|
|
$("#cs-externaljs").attr("disabled", CLIENT.rank < 3);
|
2014-01-16 17:53:34 +00:00
|
|
|
$("#cs-chat_antiflood").prop("checked", CHANNEL.opts.chat_antiflood);
|
2013-12-07 23:30:34 +00:00
|
|
|
if ("chat_antiflood_params" in CHANNEL.opts) {
|
2014-01-16 17:53:34 +00:00
|
|
|
$("#cs-chat_antiflood_burst").val(CHANNEL.opts.chat_antiflood_params.burst);
|
|
|
|
$("#cs-chat_antiflood_sustained").val(CHANNEL.opts.chat_antiflood_params.sustained);
|
2013-11-25 22:20:15 +00:00
|
|
|
}
|
2014-01-15 06:16:29 +00:00
|
|
|
$("#cs-show_public").prop("checked", CHANNEL.opts.show_public);
|
|
|
|
$("#cs-show_public").attr("disabled", CLIENT.rank < 3);
|
|
|
|
$("#cs-password").val(CHANNEL.opts.password || "");
|
|
|
|
$("#cs-password").attr("disabled", CLIENT.rank < 3);
|
|
|
|
$("#cs-enable_link_regex").prop("checked", CHANNEL.opts.enable_link_regex);
|
|
|
|
$("#cs-afk_timeout").val(CHANNEL.opts.afk_timeout);
|
|
|
|
$("#cs-allow_voteskip").prop("checked", CHANNEL.opts.allow_voteskip);
|
|
|
|
$("#cs-voteskip_ratio").val(CHANNEL.opts.voteskip_ratio);
|
2013-07-04 23:11:13 +00:00
|
|
|
(function() {
|
|
|
|
if(typeof CHANNEL.opts.maxlength != "number") {
|
2014-01-15 06:16:29 +00:00
|
|
|
$("#cs-maxlength").val("");
|
2013-07-04 23:11:13 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
var h = parseInt(CHANNEL.opts.maxlength / 3600);
|
|
|
|
h = ""+h;
|
|
|
|
if(h.length < 2) h = "0" + h;
|
|
|
|
var m = parseInt((CHANNEL.opts.maxlength % 3600) / 60);
|
|
|
|
m = ""+m;
|
|
|
|
if(m.length < 2) m = "0" + m;
|
|
|
|
var s = parseInt(CHANNEL.opts.maxlength % 60);
|
|
|
|
s = ""+s;
|
|
|
|
if(s.length < 2) s = "0" + s;
|
2014-01-15 06:16:29 +00:00
|
|
|
$("#cs-maxlength").val(h + ":" + m + ":" + s);
|
2013-07-04 23:11:13 +00:00
|
|
|
})();
|
2014-01-15 06:16:29 +00:00
|
|
|
$("#cs-csstext").val(CHANNEL.css);
|
|
|
|
$("#cs-jstext").val(CHANNEL.js);
|
|
|
|
$("#cs-motdtext").val(CHANNEL.motd_text);
|
|
|
|
setParentVisible("a[href='#cs-motdeditor']", hasPermission("motdedit"));
|
|
|
|
setParentVisible("a[href='#cs-permissions']", CLIENT.rank >= 3);
|
|
|
|
setParentVisible("a[href='#cs-banlist']", hasPermission("ban"));
|
|
|
|
setParentVisible("a[href='#cs-csseditor']", CLIENT.rank >= 3);
|
|
|
|
setParentVisible("a[href='#cs-jseditor']", CLIENT.rank >= 3);
|
|
|
|
setParentVisible("a[href='#cs-filtereditor']", CLIENT.rank >= 3);
|
|
|
|
setParentVisible("a[href='#cs-chanranks']", CLIENT.rank >= 3);
|
|
|
|
setParentVisible("a[href='#cs-chanlog']", CLIENT.rank >= 3);
|
2014-01-23 22:03:50 +00:00
|
|
|
$("#cs-chatfilters-import").attr("disabled", !hasPermission("filterimport"));
|
2013-06-20 18:43:37 +00:00
|
|
|
}
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
function handlePermissionChange() {
|
2013-12-19 17:14:48 +00:00
|
|
|
if(CLIENT.rank >= 2) {
|
|
|
|
handleModPermissions();
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
2014-01-24 04:59:08 +00:00
|
|
|
$("#qlockbtn").attr("disabled", !hasPermission("playlistlock"));
|
2014-01-19 07:45:20 +00:00
|
|
|
setVisible("#showchansettings", CLIENT.rank >= 2);
|
2013-12-19 17:14:48 +00:00
|
|
|
setVisible("#playlistmanagerwrap", CLIENT.rank >= 1);
|
2013-09-21 07:22:51 +00:00
|
|
|
setVisible("#modflair", CLIENT.rank >= 2);
|
|
|
|
setVisible("#adminflair", CLIENT.rank >= 255);
|
2013-12-19 17:14:48 +00:00
|
|
|
setVisible("#guestlogin", CLIENT.rank < 0);
|
|
|
|
setVisible("#chatline", CLIENT.rank >= 0);
|
2013-09-21 07:22:51 +00:00
|
|
|
|
2014-01-14 00:31:12 +00:00
|
|
|
setVisible("#showmediaurl", hasPermission("playlistadd"));
|
|
|
|
setVisible("#showcustomembed", hasPermission("playlistaddcustom"));
|
2013-06-11 23:51:00 +00:00
|
|
|
$("#queue_next").attr("disabled", !hasPermission("playlistnext"));
|
|
|
|
|
2013-06-23 18:35:40 +00:00
|
|
|
if(hasPermission("playlistadd") ||
|
|
|
|
hasPermission("playlistmove") ||
|
|
|
|
hasPermission("playlistjump") ||
|
|
|
|
hasPermission("playlistdelete") ||
|
|
|
|
hasPermission("settemp")) {
|
2013-07-30 00:06:01 +00:00
|
|
|
if(USEROPTS.first_visit && $("#plonotification").length == 0) {
|
2013-06-23 18:35:40 +00:00
|
|
|
var al = makeAlert("Playlist Options", [
|
|
|
|
"From the Options menu, you can choose to automatically",
|
|
|
|
" hide the buttons on each entry (and show them when",
|
|
|
|
" you right click). You can also choose to use the old",
|
|
|
|
" style of playlist buttons.",
|
|
|
|
"<br>"].join(""))
|
2013-07-30 00:06:01 +00:00
|
|
|
.attr("id", "plonotification")
|
2013-06-23 18:35:40 +00:00
|
|
|
.insertBefore($("#queue"));
|
|
|
|
|
|
|
|
al.find(".close").remove();
|
|
|
|
|
|
|
|
$("<button/>").addClass("btn btn-primary")
|
|
|
|
.text("Dismiss")
|
|
|
|
.appendTo(al)
|
|
|
|
.click(function() {
|
|
|
|
USEROPTS.first_visit = false;
|
2013-12-25 21:18:21 +00:00
|
|
|
storeOpts();
|
2014-01-14 06:52:56 +00:00
|
|
|
al.hide("fade", function() {
|
2013-06-23 18:35:40 +00:00
|
|
|
al.remove();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
if(hasPermission("playlistmove")) {
|
|
|
|
$("#queue").sortable("enable");
|
|
|
|
$("#queue").addClass("queue_sortable");
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$("#queue").sortable("disable");
|
|
|
|
$("#queue").removeClass("queue_sortable");
|
|
|
|
}
|
|
|
|
|
|
|
|
setVisible("#clearplaylist", hasPermission("playlistclear"));
|
|
|
|
setVisible("#shuffleplaylist", hasPermission("playlistshuffle"));
|
|
|
|
|
2014-01-30 04:50:14 +00:00
|
|
|
fixWeirdButtonAlignmentIssue();
|
2014-01-19 02:18:00 +00:00
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
setVisible("#newpollbtn", hasPermission("pollctl"));
|
2013-09-12 03:16:56 +00:00
|
|
|
$("#voteskip").attr("disabled", !hasPermission("voteskip") ||
|
|
|
|
!CHANNEL.opts.allow_voteskip);
|
2013-06-11 23:51:00 +00:00
|
|
|
|
2013-06-19 23:41:49 +00:00
|
|
|
$("#pollwrap .active").find(".btn-danger").remove();
|
2013-06-11 23:51:00 +00:00
|
|
|
if(hasPermission("pollctl")) {
|
2013-06-19 23:41:49 +00:00
|
|
|
var poll = $("#pollwrap .active");
|
2013-06-11 23:51:00 +00:00
|
|
|
if(poll.length > 0) {
|
|
|
|
$("<button/>").addClass("btn btn-danger pull-right")
|
|
|
|
.text("End Poll")
|
|
|
|
.insertAfter(poll.find(".close"))
|
|
|
|
.click(function() {
|
|
|
|
socket.emit("closePoll");
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2013-06-19 23:41:49 +00:00
|
|
|
var poll = $("#pollwrap .active");
|
2013-06-11 23:51:00 +00:00
|
|
|
if(poll.length > 0) {
|
|
|
|
poll.find(".btn").attr("disabled", !hasPermission("pollvote"));
|
|
|
|
}
|
|
|
|
var users = $("#userlist").children();
|
|
|
|
for(var i = 0; i < users.length; i++) {
|
2013-11-09 04:12:17 +00:00
|
|
|
addUserDropdown($(users[i]));
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
2013-06-20 19:02:53 +00:00
|
|
|
|
|
|
|
$("#chatline").attr("disabled", !hasPermission("chat"));
|
2013-06-11 23:51:00 +00:00
|
|
|
rebuildPlaylist();
|
|
|
|
}
|
|
|
|
|
2014-01-30 04:50:14 +00:00
|
|
|
function fixWeirdButtonAlignmentIssue() {
|
|
|
|
// Weird things happen to the alignment in chromium when I toggle visibility
|
|
|
|
// of the above buttons
|
|
|
|
// This fixes it?
|
|
|
|
var wtf = $("#videocontrols").removeClass("pull-right");
|
|
|
|
setTimeout(function () {
|
|
|
|
wtf.addClass("pull-right");
|
|
|
|
}, 1);
|
|
|
|
}
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
/* search stuff */
|
|
|
|
|
|
|
|
function clearSearchResults() {
|
|
|
|
$("#library").html("");
|
2013-06-22 23:02:55 +00:00
|
|
|
$("#search_clear").remove();
|
2013-07-31 15:05:07 +00:00
|
|
|
var p = $("#library").data("paginator");
|
|
|
|
if(p) {
|
|
|
|
p.paginator.html("");
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-23 21:25:45 +00:00
|
|
|
function addLibraryButtons(li, id, source) {
|
2013-06-11 23:51:00 +00:00
|
|
|
var btns = $("<div/>").addClass("btn-group")
|
|
|
|
.addClass("pull-left")
|
|
|
|
.prependTo(li);
|
|
|
|
|
2014-01-14 06:52:56 +00:00
|
|
|
var type = (source === "library") ? "lib" : source;
|
2013-09-23 21:25:45 +00:00
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
if(hasPermission("playlistadd")) {
|
|
|
|
if(hasPermission("playlistnext")) {
|
2013-12-19 17:14:48 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("Next")
|
|
|
|
.click(function() {
|
|
|
|
socket.emit("queue", {
|
|
|
|
id: id,
|
|
|
|
pos: "next",
|
|
|
|
type: type
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.appendTo(btns);
|
|
|
|
}
|
2013-12-19 17:14:48 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default")
|
2013-06-11 23:51:00 +00:00
|
|
|
.text("End")
|
|
|
|
.click(function() {
|
|
|
|
socket.emit("queue", {
|
|
|
|
id: id,
|
|
|
|
pos: "end",
|
|
|
|
type: type
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.appendTo(btns);
|
|
|
|
}
|
2013-09-23 21:25:45 +00:00
|
|
|
if(CLIENT.rank >= 2 && source === "library") {
|
2013-12-19 17:14:48 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-danger")
|
|
|
|
.html("<span class='glyphicon glyphicon-trash'></span>")
|
2013-06-11 23:51:00 +00:00
|
|
|
.click(function() {
|
|
|
|
socket.emit("uncache", {
|
|
|
|
id: id
|
|
|
|
});
|
2014-01-14 06:52:56 +00:00
|
|
|
li.hide("fade", function() {
|
2013-06-11 23:51:00 +00:00
|
|
|
li.remove();
|
|
|
|
});
|
|
|
|
})
|
|
|
|
.appendTo(btns);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* queue stuff */
|
|
|
|
|
2013-10-01 18:35:29 +00:00
|
|
|
var AsyncQueue = function () {
|
|
|
|
this._q = [];
|
|
|
|
this._lock = false;
|
|
|
|
this._tm = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
AsyncQueue.prototype.next = function () {
|
|
|
|
if (this._q.length > 0) {
|
|
|
|
if (!this.lock())
|
|
|
|
return;
|
|
|
|
var item = this._q.shift();
|
|
|
|
var fn = item[0], tm = item[1];
|
|
|
|
this._tm = Date.now() + item[1];
|
|
|
|
fn(this);
|
|
|
|
}
|
|
|
|
};
|
2013-07-03 15:26:10 +00:00
|
|
|
|
2013-10-01 18:35:29 +00:00
|
|
|
AsyncQueue.prototype.lock = function () {
|
|
|
|
if (this._lock) {
|
|
|
|
if (this._tm > 0 && Date.now() > this._tm) {
|
|
|
|
this._tm = 0;
|
|
|
|
return true;
|
2013-07-03 15:26:10 +00:00
|
|
|
}
|
2013-10-01 18:35:29 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._lock = true;
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
AsyncQueue.prototype.release = function () {
|
|
|
|
var self = this;
|
|
|
|
if (!self._lock)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
self._lock = false;
|
|
|
|
self.next();
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
AsyncQueue.prototype.queue = function (fn) {
|
|
|
|
var self = this;
|
|
|
|
self._q.push([fn, 20000]);
|
|
|
|
self.next();
|
|
|
|
};
|
|
|
|
|
|
|
|
AsyncQueue.prototype.reset = function () {
|
|
|
|
this._q = [];
|
|
|
|
this._lock = false;
|
|
|
|
};
|
|
|
|
|
|
|
|
var PL_ACTION_QUEUE = new AsyncQueue();
|
2013-07-03 15:26:10 +00:00
|
|
|
|
|
|
|
// Because jQuery UI does weird things
|
|
|
|
function playlistFind(uid) {
|
|
|
|
var children = document.getElementById("queue").children;
|
|
|
|
for(var i in children) {
|
|
|
|
if(typeof children[i].getAttribute != "function")
|
|
|
|
continue;
|
|
|
|
if(children[i].getAttribute("class").indexOf("pluid-" + uid) != -1)
|
|
|
|
return children[i];
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-10-01 18:35:29 +00:00
|
|
|
function playlistMove(from, after, cb) {
|
2013-06-29 22:09:20 +00:00
|
|
|
var lifrom = $(".pluid-" + from);
|
2013-10-01 18:35:29 +00:00
|
|
|
if(lifrom.length == 0) {
|
|
|
|
cb(false);
|
|
|
|
return;
|
|
|
|
}
|
2013-06-29 22:09:20 +00:00
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
var q = $("#queue");
|
|
|
|
|
2013-06-29 22:09:20 +00:00
|
|
|
if(after === "prepend") {
|
2014-01-23 04:43:17 +00:00
|
|
|
lifrom.hide("blind", function() {
|
2013-06-27 22:15:29 +00:00
|
|
|
lifrom.detach();
|
|
|
|
lifrom.prependTo(q);
|
2014-01-23 04:43:17 +00:00
|
|
|
lifrom.show("blind", cb);
|
2013-06-27 22:15:29 +00:00
|
|
|
});
|
|
|
|
}
|
2013-06-29 22:09:20 +00:00
|
|
|
else if(after === "append") {
|
2014-01-23 04:43:17 +00:00
|
|
|
lifrom.hide("blind", function() {
|
2013-06-29 22:09:20 +00:00
|
|
|
lifrom.detach();
|
|
|
|
lifrom.appendTo(q);
|
2014-01-23 04:43:17 +00:00
|
|
|
lifrom.show("blind", cb);
|
2013-06-29 22:09:20 +00:00
|
|
|
});
|
|
|
|
}
|
2013-06-27 22:15:29 +00:00
|
|
|
else {
|
2013-06-29 22:09:20 +00:00
|
|
|
var liafter = $(".pluid-" + after);
|
2013-10-01 18:35:29 +00:00
|
|
|
if(liafter.length == 0) {
|
|
|
|
cb(false);
|
|
|
|
return;
|
|
|
|
}
|
2014-01-23 04:43:17 +00:00
|
|
|
lifrom.hide("blind", function() {
|
2013-06-27 22:15:29 +00:00
|
|
|
lifrom.detach();
|
|
|
|
lifrom.insertAfter(liafter);
|
2014-01-23 04:43:17 +00:00
|
|
|
lifrom.show("blind", cb);
|
2013-06-27 22:15:29 +00:00
|
|
|
});
|
|
|
|
}
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function parseMediaLink(url) {
|
|
|
|
if(typeof url != "string") {
|
|
|
|
return {
|
|
|
|
id: null,
|
|
|
|
type: null
|
|
|
|
};
|
|
|
|
}
|
|
|
|
url = url.trim();
|
2013-06-12 03:37:12 +00:00
|
|
|
url = url.replace("feature=player_embedded&", "");
|
2013-06-11 23:51:00 +00:00
|
|
|
|
|
|
|
if(url.indexOf("jw:") == 0) {
|
|
|
|
return {
|
|
|
|
id: url.substring(3),
|
|
|
|
type: "jw"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if(url.indexOf("rtmp://") == 0) {
|
|
|
|
return {
|
|
|
|
id: url,
|
|
|
|
type: "rt"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
var m;
|
|
|
|
if((m = url.match(/youtube\.com\/watch\?v=([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "yt"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/youtu\.be\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "yt"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/youtube\.com\/playlist\?list=([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "yp"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/twitch\.tv\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "tw"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/justin\.tv\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "jt"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/livestream\.com\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "li"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/ustream\.tv\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "us"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/vimeo\.com\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "vi"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/dailymotion\.com\/video\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "dm"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/imgur\.com\/a\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "im"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
if((m = url.match(/soundcloud\.com\/([^&#]+)/))) {
|
|
|
|
return {
|
|
|
|
id: url,
|
|
|
|
type: "sc"
|
|
|
|
};
|
|
|
|
}
|
2013-06-12 03:37:12 +00:00
|
|
|
|
2013-11-07 23:19:36 +00:00
|
|
|
if ((m = url.match(/docs\.google\.com\/file\/d\/(.*?)\/edit/))) {
|
|
|
|
return {
|
|
|
|
id: m[1],
|
|
|
|
type: "gd"
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2013-06-12 03:37:12 +00:00
|
|
|
return {
|
|
|
|
id: null,
|
|
|
|
type: null
|
|
|
|
};
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function sendVideoUpdate() {
|
2013-11-15 04:23:33 +00:00
|
|
|
if (!CLIENT.leader) {
|
|
|
|
return;
|
|
|
|
}
|
2013-08-06 03:30:16 +00:00
|
|
|
PLAYER.getTime(function (seconds) {
|
2013-06-11 23:51:00 +00:00
|
|
|
socket.emit("mediaUpdate", {
|
2013-08-06 03:30:16 +00:00
|
|
|
id: PLAYER.videoId,
|
2013-06-11 23:51:00 +00:00
|
|
|
currentTime: seconds,
|
|
|
|
paused: PLAYER.paused,
|
|
|
|
type: PLAYER.type
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/* chat */
|
|
|
|
|
|
|
|
function formatChatMessage(data) {
|
2013-11-19 19:45:24 +00:00
|
|
|
// Backwards compat
|
|
|
|
if (!data.meta || data.msgclass) {
|
|
|
|
data.meta = {
|
2013-11-19 22:14:33 +00:00
|
|
|
addClass: data.msgclass,
|
|
|
|
// And the award for "variable name most like Java source code" goes to...
|
|
|
|
addClassToNameAndTimestamp: data.msgclass
|
2013-11-19 19:45:24 +00:00
|
|
|
};
|
|
|
|
}
|
2013-11-18 17:11:00 +00:00
|
|
|
// Phase 1: Determine whether to show the username or not
|
2013-11-17 19:12:56 +00:00
|
|
|
var skip = data.username === LASTCHATNAME;
|
2013-11-18 17:11:00 +00:00
|
|
|
if(data.meta.addClass === "server-whisper")
|
2013-11-17 19:12:56 +00:00
|
|
|
skip = true;
|
|
|
|
// Prevent impersonation by abuse of the bold filter
|
|
|
|
if(data.msg.match(/^\s*<strong>\w+\s*:\s*<\/strong>\s*/))
|
|
|
|
skip = false;
|
2013-11-18 17:11:00 +00:00
|
|
|
if (data.meta.forceShowName)
|
|
|
|
skip = false;
|
|
|
|
|
2013-11-17 19:12:56 +00:00
|
|
|
LASTCHATNAME = data.username;
|
|
|
|
LASTCHATTIME = data.time;
|
|
|
|
var div = $("<div/>");
|
2013-11-18 17:11:00 +00:00
|
|
|
/* drink is a special case because the entire container gets the class, not
|
|
|
|
just the message */
|
2013-11-17 21:32:19 +00:00
|
|
|
if (data.meta.addClass === "drink") {
|
|
|
|
div.addClass("drink");
|
|
|
|
data.meta.addClass = "";
|
|
|
|
}
|
2013-11-18 17:11:00 +00:00
|
|
|
|
|
|
|
// Add timestamps (unless disabled)
|
2013-11-17 19:12:56 +00:00
|
|
|
if (USEROPTS.show_timestamps) {
|
|
|
|
var time = $("<span/>").addClass("timestamp").appendTo(div);
|
|
|
|
var timestamp = new Date(data.time).toTimeString().split(" ")[0];
|
|
|
|
time.text("["+timestamp+"] ");
|
2013-11-19 22:14:33 +00:00
|
|
|
if (data.meta.addClass && data.meta.addClassToNameAndTimestamp) {
|
2013-11-17 19:12:56 +00:00
|
|
|
time.addClass(data.meta.addClass);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-18 17:11:00 +00:00
|
|
|
// Add username
|
2013-11-17 19:12:56 +00:00
|
|
|
var name = $("<span/>");
|
|
|
|
if (!skip) {
|
|
|
|
name.appendTo(div);
|
|
|
|
}
|
|
|
|
$("<strong/>").addClass("username").text(data.username + ": ").appendTo(name);
|
|
|
|
if (data.meta.modflair) {
|
|
|
|
name.addClass(getNameColor(data.meta.modflair));
|
|
|
|
}
|
2013-11-19 22:14:33 +00:00
|
|
|
if (data.meta.addClass && data.meta.addClassToNameAndTimestamp) {
|
2013-11-17 19:12:56 +00:00
|
|
|
name.addClass(data.meta.addClass);
|
|
|
|
}
|
|
|
|
if (data.meta.superadminflair) {
|
|
|
|
name.addClass("label")
|
|
|
|
.addClass(data.meta.superadminflair.labelclass);
|
2014-01-26 06:01:36 +00:00
|
|
|
$("<span/>").addClass(data.meta.superadminflair.icon)
|
|
|
|
.addClass("glyphicon")
|
2014-01-26 20:15:50 +00:00
|
|
|
.css("margin-right", "3px")
|
2013-11-17 19:12:56 +00:00
|
|
|
.prependTo(name);
|
|
|
|
}
|
|
|
|
|
2013-11-18 17:11:00 +00:00
|
|
|
// Add the message itself
|
2013-11-17 19:12:56 +00:00
|
|
|
var message = $("<span/>").appendTo(div);
|
|
|
|
message[0].innerHTML = data.msg;
|
|
|
|
|
2013-11-18 17:11:00 +00:00
|
|
|
// For /me the username is part of the message
|
2013-11-17 19:12:56 +00:00
|
|
|
if (data.meta.action) {
|
|
|
|
name.remove();
|
|
|
|
message[0].innerHTML = data.username + " " + data.msg;
|
|
|
|
}
|
|
|
|
if (data.meta.addClass) {
|
|
|
|
message.addClass(data.meta.addClass);
|
|
|
|
}
|
|
|
|
return div;
|
|
|
|
}
|
|
|
|
|
2013-06-11 23:51:00 +00:00
|
|
|
function addChatMessage(data) {
|
2013-11-18 17:11:00 +00:00
|
|
|
if(IGNORED.indexOf(data.username) !== -1) {
|
2013-06-11 23:51:00 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
var div = formatChatMessage(data);
|
2013-11-18 17:11:00 +00:00
|
|
|
// Incoming: a bunch of crap for the feature where if you hover over
|
|
|
|
// a message, it highlights messages from that user
|
2013-06-11 23:51:00 +00:00
|
|
|
div.data("sender", data.username);
|
|
|
|
div.appendTo($("#messagebuffer"));
|
|
|
|
div.mouseover(function() {
|
|
|
|
$("#messagebuffer").children().each(function() {
|
|
|
|
var name = $(this).data("sender");
|
|
|
|
if(name == data.username) {
|
|
|
|
$(this).addClass("nick-hover");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
div.mouseleave(function() {
|
|
|
|
$("#messagebuffer").children().each(function() {
|
|
|
|
$(this).removeClass("nick-hover");
|
|
|
|
});
|
|
|
|
});
|
|
|
|
// Cap chatbox at most recent 100 messages
|
|
|
|
if($("#messagebuffer").children().length > 100) {
|
|
|
|
$($("#messagebuffer").children()[0]).remove();
|
|
|
|
}
|
|
|
|
if(SCROLLCHAT)
|
|
|
|
scrollChat();
|
|
|
|
if(USEROPTS.blink_title && !FOCUSED && !TITLE_BLINK) {
|
2013-08-28 21:03:31 +00:00
|
|
|
USEROPTS.boop && CHATSOUND.play();
|
2013-06-11 23:51:00 +00:00
|
|
|
TITLE_BLINK = setInterval(function() {
|
|
|
|
if(document.title == "*Chat*")
|
|
|
|
document.title = PAGETITLE;
|
|
|
|
else
|
|
|
|
document.title = "*Chat*";
|
|
|
|
}, 1000);
|
|
|
|
}
|
|
|
|
if(CLIENT.name && data.username != CLIENT.name) {
|
|
|
|
if(data.msg.toUpperCase().indexOf(CLIENT.name.toUpperCase()) != -1) {
|
|
|
|
div.addClass("nick-highlight");
|
|
|
|
if(!FOCUSED && !TITLE_BLINK) {
|
2013-08-28 21:03:31 +00:00
|
|
|
USEROPTS.boop && CHATSOUND.play();
|
2013-06-11 23:51:00 +00:00
|
|
|
TITLE_BLINK = setInterval(function() {
|
|
|
|
if(document.title == "*Chat*")
|
|
|
|
document.title = PAGETITLE;
|
|
|
|
else
|
|
|
|
document.title = "*Chat*";
|
|
|
|
}, 1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* layouts */
|
|
|
|
|
|
|
|
function fluidLayout() {
|
2014-01-26 03:29:56 +00:00
|
|
|
$(".container").removeClass("container").addClass("container-fluid");
|
2014-01-26 20:15:50 +00:00
|
|
|
$("footer .container").removeClass("container-fluid").addClass("container");
|
|
|
|
$("body").addClass("fluid");
|
2014-01-26 03:48:57 +00:00
|
|
|
resizeStuff();
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function synchtubeLayout() {
|
|
|
|
$("#videowrap").detach().insertBefore($("#chatwrap"));
|
2013-12-20 03:33:24 +00:00
|
|
|
$("#rightcontrols").detach().insertBefore($("#leftcontrols"));
|
|
|
|
$("#rightpane").detach().insertBefore($("#leftpane"));
|
2013-08-30 20:49:30 +00:00
|
|
|
$("#userlist").css("float", "right");
|
2014-01-26 20:15:50 +00:00
|
|
|
$("body").addClass("synchtube");
|
|
|
|
}
|
|
|
|
|
|
|
|
function hdLayout() {
|
|
|
|
var videowrap = $("#videowrap"),
|
|
|
|
chatwrap = $("#chatwrap"),
|
|
|
|
playlist = $("#rightpane")
|
|
|
|
|
|
|
|
videowrap.detach().insertAfter($("#drinkbar"))
|
|
|
|
.removeClass()
|
|
|
|
.addClass("col-md-8 col-md-offset-2");
|
|
|
|
|
|
|
|
playlist.detach().insertBefore(chatwrap)
|
|
|
|
.removeClass()
|
|
|
|
.addClass("col-md-6");
|
|
|
|
|
|
|
|
chatwrap.removeClass()
|
|
|
|
.addClass("col-md-6");
|
|
|
|
|
|
|
|
var ch = "320px";
|
|
|
|
$("#messagebuffer").css("max-height", ch);
|
|
|
|
$("#userlist").css("max-height", ch);
|
|
|
|
$("#queue").css("max-height", "312px");
|
|
|
|
|
|
|
|
$("#leftcontrols").detach()
|
|
|
|
.insertAfter(chatwrap)
|
|
|
|
.removeClass()
|
|
|
|
.addClass("col-md-6");
|
|
|
|
|
|
|
|
$("#playlistmanagerwrap").detach()
|
|
|
|
.insertBefore($("#leftcontrols"))
|
|
|
|
.css("margin-top", "0")
|
|
|
|
.removeClass()
|
|
|
|
.addClass("col-md-6");
|
|
|
|
|
|
|
|
$("#showplaylistmanager").addClass("btn-sm");
|
|
|
|
|
|
|
|
var plcontrolwrap = $("<div/>").addClass("col-md-12")
|
|
|
|
.prependTo($("#rightpane-inner"));
|
|
|
|
|
|
|
|
$("#plcontrol").detach().appendTo(plcontrolwrap);
|
2014-01-30 04:50:14 +00:00
|
|
|
$("#videocontrols").detach()
|
2014-01-26 20:15:50 +00:00
|
|
|
.appendTo(plcontrolwrap);
|
|
|
|
|
|
|
|
$("#controlswrap").remove();
|
|
|
|
|
|
|
|
$("#pollwrap").detach()
|
|
|
|
.insertAfter($("#leftcontrols"))
|
|
|
|
.removeClass()
|
|
|
|
.addClass("col-md-6 col-md-offset-6");
|
|
|
|
|
2014-01-26 20:18:29 +00:00
|
|
|
$("#leftpane").remove();
|
2014-01-26 20:15:50 +00:00
|
|
|
$("nav.navbar-fixed-top").removeClass("navbar-fixed-top");
|
|
|
|
$("#mainpage").css("padding-top", "0");
|
|
|
|
|
|
|
|
$("body").addClass("hd");
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
function chatOnly() {
|
2013-12-20 03:33:24 +00:00
|
|
|
//fluidLayout();
|
2013-06-11 23:51:00 +00:00
|
|
|
$("#toprow").remove()
|
|
|
|
$("#announcements").remove();
|
|
|
|
$("#playlistrow").remove();
|
|
|
|
$("#videowrap").remove();
|
2013-12-20 03:33:24 +00:00
|
|
|
$("#controlsrow").remove();
|
|
|
|
$("#chatwrap").removeClass("col-lg-5 col-md-5").addClass("col-lg-12 col-md-12");
|
2013-06-11 23:51:00 +00:00
|
|
|
}
|
2013-06-17 22:16:59 +00:00
|
|
|
|
2014-01-26 03:48:57 +00:00
|
|
|
function resizeStuff() {
|
|
|
|
VWIDTH = $("#videowrap").width() + "";
|
|
|
|
VHEIGHT = Math.floor(parseInt(VWIDTH) * 9 / 16) + "";
|
|
|
|
$("#ytapiplayer").width(VWIDTH).height(VHEIGHT);
|
|
|
|
|
2014-01-26 20:15:50 +00:00
|
|
|
// Only execute if we are on a fluid layout
|
|
|
|
if ($("body").hasClass("fluid")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-01-26 03:48:57 +00:00
|
|
|
var h = parseInt(VHEIGHT) - 33;
|
|
|
|
$("#messagebuffer").height(h);
|
|
|
|
$("#userlist").height(h);
|
|
|
|
}
|
|
|
|
|
|
|
|
$(window).resize(resizeStuff);
|
|
|
|
|
2013-06-17 22:16:59 +00:00
|
|
|
/* channel administration stuff */
|
|
|
|
|
|
|
|
function genPermissionsEditor() {
|
2014-01-20 18:16:30 +00:00
|
|
|
$("#cs-permedit").html("");
|
2013-06-17 22:16:59 +00:00
|
|
|
var form = $("<form/>").addClass("form-horizontal")
|
|
|
|
.attr("action", "javascript:void(0)")
|
2014-01-20 18:16:30 +00:00
|
|
|
.appendTo($("#cs-permedit"));
|
2013-06-17 22:16:59 +00:00
|
|
|
|
|
|
|
function makeOption(text, key, permset, defval) {
|
2014-01-20 18:16:30 +00:00
|
|
|
var group = $("<div/>").addClass("form-group")
|
|
|
|
.appendTo(form);
|
|
|
|
$("<label/>").addClass("control-label col-sm-4")
|
2013-06-17 22:16:59 +00:00
|
|
|
.text(text)
|
|
|
|
.appendTo(group);
|
2014-01-20 18:16:30 +00:00
|
|
|
var controls = $("<div/>").addClass("col-sm-8")
|
2013-06-17 22:16:59 +00:00
|
|
|
.appendTo(group);
|
2014-01-20 18:16:30 +00:00
|
|
|
var select = $("<select/>").addClass("form-control")
|
|
|
|
.appendTo(controls)
|
|
|
|
.data("key", key);
|
|
|
|
|
|
|
|
for (var i = 0; i < permset.length; i++) {
|
2013-06-17 22:16:59 +00:00
|
|
|
$("<option/>").attr("value", permset[i][1])
|
|
|
|
.text(permset[i][0])
|
2014-01-20 18:16:30 +00:00
|
|
|
.attr("selected", defval === permset[i][1])
|
2013-06-17 22:16:59 +00:00
|
|
|
.appendTo(select);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-01-20 18:16:30 +00:00
|
|
|
function addDivider(text, nonewline) {
|
|
|
|
$("<hr/>").appendTo(form);
|
|
|
|
if (!nonewline) {
|
|
|
|
$("<h3/>").text(text).appendTo(form);
|
|
|
|
}
|
2013-06-17 22:16:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var standard = [
|
|
|
|
["Anonymous" , "-1"],
|
|
|
|
["Guest" , "0"],
|
|
|
|
["Registered" , "1"],
|
|
|
|
["Leader" , "1.5"],
|
|
|
|
["Moderator" , "2"],
|
2013-09-21 05:30:47 +00:00
|
|
|
["Channel Admin", "3"],
|
|
|
|
["Nobody" , "1000000"]
|
2013-06-17 22:16:59 +00:00
|
|
|
];
|
|
|
|
|
2013-06-22 23:02:55 +00:00
|
|
|
var noanon = [
|
|
|
|
["Guest" , "0"],
|
|
|
|
["Registered" , "1"],
|
|
|
|
["Leader" , "1.5"],
|
|
|
|
["Moderator" , "2"],
|
2013-09-21 05:30:47 +00:00
|
|
|
["Channel Admin", "3"],
|
|
|
|
["Nobody" , "1000000"]
|
2013-06-22 23:02:55 +00:00
|
|
|
];
|
|
|
|
|
2013-06-17 22:16:59 +00:00
|
|
|
var modleader = [
|
|
|
|
["Leader" , "1.5"],
|
|
|
|
["Moderator" , "2"],
|
2013-09-21 05:30:47 +00:00
|
|
|
["Channel Admin", "3"],
|
|
|
|
["Nobody" , "1000000"]
|
2013-06-17 22:16:59 +00:00
|
|
|
];
|
|
|
|
|
|
|
|
var modplus = [
|
|
|
|
["Moderator" , "2"],
|
2013-09-21 05:30:47 +00:00
|
|
|
["Channel Admin", "3"],
|
|
|
|
["Nobody" , "1000000"]
|
2013-06-17 22:16:59 +00:00
|
|
|
];
|
|
|
|
|
2014-01-20 18:16:30 +00:00
|
|
|
$("<h3/>").text("Open playlist permissions").appendTo(form);
|
2013-06-17 22:16:59 +00:00
|
|
|
makeOption("Add to playlist", "oplaylistadd", standard, CHANNEL.perms.oplaylistadd+"");
|
|
|
|
makeOption("Add/move to next", "oplaylistnext", standard, CHANNEL.perms.oplaylistnext+"");
|
|
|
|
makeOption("Move playlist items", "oplaylistmove", standard, CHANNEL.perms.oplaylistmove+"");
|
|
|
|
makeOption("Delete playlist items", "oplaylistdelete", standard, CHANNEL.perms.oplaylistdelete+"");
|
|
|
|
makeOption("Jump to video", "oplaylistjump", standard, CHANNEL.perms.oplaylistjump+"");
|
|
|
|
makeOption("Queue playlist", "oplaylistaddlist", standard, CHANNEL.perms.oplaylistaddlist+"");
|
|
|
|
|
|
|
|
addDivider("General playlist permissions");
|
|
|
|
makeOption("Add to playlist", "playlistadd", standard, CHANNEL.perms.playlistadd+"");
|
|
|
|
makeOption("Add/move to next", "playlistnext", standard, CHANNEL.perms.playlistnext+"");
|
|
|
|
makeOption("Move playlist items", "playlistmove", standard, CHANNEL.perms.playlistmove+"");
|
|
|
|
makeOption("Delete playlist items", "playlistdelete", standard, CHANNEL.perms.playlistdelete+"");
|
|
|
|
makeOption("Jump to video", "playlistjump", standard, CHANNEL.perms.playlistjump+"");
|
|
|
|
makeOption("Queue playlist", "playlistaddlist", standard, CHANNEL.perms.playlistaddlist+"");
|
|
|
|
makeOption("Queue livestream", "playlistaddlive", standard, CHANNEL.perms.playlistaddlive+"");
|
2013-08-08 22:25:56 +00:00
|
|
|
makeOption("Embed custom media", "playlistaddcustom", standard, CHANNEL.perms.playlistaddcustom + "");
|
2013-07-04 23:11:13 +00:00
|
|
|
makeOption("Exceed maximum media length", "exceedmaxlength", standard, CHANNEL.perms.exceedmaxlength+"");
|
2013-06-17 22:16:59 +00:00
|
|
|
makeOption("Add nontemporary media", "addnontemp", standard, CHANNEL.perms.addnontemp+"");
|
|
|
|
makeOption("Temp/untemp playlist item", "settemp", standard, CHANNEL.perms.settemp+"");
|
2014-01-23 22:03:50 +00:00
|
|
|
makeOption("Lock/unlock playlist", "playlistlock", modleader, CHANNEL.perms.playlistlock+"");
|
2013-06-17 22:16:59 +00:00
|
|
|
makeOption("Shuffle playlist", "playlistshuffle", standard, CHANNEL.perms.playlistshuffle+"");
|
|
|
|
makeOption("Clear playlist", "playlistclear", standard, CHANNEL.perms.playlistclear+"");
|
|
|
|
|
|
|
|
addDivider("Polls");
|
|
|
|
makeOption("Open/Close poll", "pollctl", modleader, CHANNEL.perms.pollctl+"");
|
|
|
|
makeOption("Vote", "pollvote", standard, CHANNEL.perms.pollvote+"");
|
2013-09-12 01:22:00 +00:00
|
|
|
makeOption("View hidden poll results", "viewhiddenpoll", standard, CHANNEL.perms.viewhiddenpoll+"");
|
2013-09-12 03:16:56 +00:00
|
|
|
makeOption("Voteskip", "voteskip", standard, CHANNEL.perms.voteskip+"");
|
2013-06-17 22:16:59 +00:00
|
|
|
|
|
|
|
addDivider("Moderation");
|
2014-01-23 22:03:50 +00:00
|
|
|
makeOption("Assign/Remove leader", "leaderctl", modplus, CHANNEL.perms.leaderctl+"");
|
2013-06-25 14:18:33 +00:00
|
|
|
makeOption("Mute users", "mute", modleader, CHANNEL.perms.mute+"");
|
2013-06-17 22:16:59 +00:00
|
|
|
makeOption("Kick users", "kick", modleader, CHANNEL.perms.kick+"");
|
|
|
|
makeOption("Ban users", "ban", modplus, CHANNEL.perms.ban+"");
|
|
|
|
makeOption("Edit MOTD", "motdedit", modplus, CHANNEL.perms.motdedit+"");
|
|
|
|
makeOption("Edit chat filters", "filteredit", modplus, CHANNEL.perms.filteredit+"");
|
2014-01-23 22:03:50 +00:00
|
|
|
makeOption("Import chat filters", "filterimport", modplus, CHANNEL.perms.filterimport+"");
|
2013-06-17 22:16:59 +00:00
|
|
|
|
|
|
|
addDivider("Misc");
|
|
|
|
makeOption("Drink calls", "drink", modleader, CHANNEL.perms.drink+"");
|
2013-06-22 23:02:55 +00:00
|
|
|
makeOption("Chat", "chat", noanon, CHANNEL.perms.chat+"");
|
2013-06-17 22:16:59 +00:00
|
|
|
|
2014-01-20 18:16:30 +00:00
|
|
|
var sgroup = $("<div/>").addClass("form-group").appendTo(form);
|
|
|
|
var sgroupinner = $("<div/>").addClass("col-sm-8 col-sm-offset-4").appendTo(sgroup);
|
|
|
|
var submit = $("<button/>").addClass("btn btn-primary").appendTo(sgroupinner);
|
2013-06-17 22:16:59 +00:00
|
|
|
submit.text("Save");
|
|
|
|
submit.click(function() {
|
|
|
|
var perms = {};
|
2014-01-20 18:16:30 +00:00
|
|
|
form.find("select").each(function() {
|
2013-06-17 22:16:59 +00:00
|
|
|
perms[$(this).data("key")] = parseFloat($(this).val());
|
|
|
|
});
|
|
|
|
socket.emit("setPermissions", perms);
|
|
|
|
});
|
2014-01-20 18:16:30 +00:00
|
|
|
|
|
|
|
var msggroup = $("<div/>").addClass("form-group").insertAfter(sgroup);
|
|
|
|
var msginner = $("<div/>").addClass("col-sm-8 col-sm-offset-4").appendTo(msggroup);
|
|
|
|
var text = $("<span/>").addClass("text-info").text("Permissions updated")
|
|
|
|
.appendTo(msginner);
|
|
|
|
|
|
|
|
setTimeout(function () {
|
|
|
|
msggroup.hide("fade", function () {
|
|
|
|
msggroup.remove();
|
|
|
|
});
|
|
|
|
}, 5000);
|
2013-06-17 22:16:59 +00:00
|
|
|
}
|
2013-06-18 14:46:28 +00:00
|
|
|
|
2013-07-28 14:49:12 +00:00
|
|
|
function waitUntilDefined(obj, key, fn) {
|
|
|
|
if(typeof obj[key] === "undefined") {
|
|
|
|
setTimeout(function () {
|
|
|
|
waitUntilDefined(obj, key, fn);
|
|
|
|
}, 100);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
fn();
|
|
|
|
}
|
2013-08-06 03:11:56 +00:00
|
|
|
|
|
|
|
function hidePlayer() {
|
|
|
|
if(!PLAYER)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if(!/(chrome|MSIE)/ig.test(navigator.userAgent))
|
|
|
|
return;
|
|
|
|
|
|
|
|
PLAYER.size = {
|
2014-01-19 02:18:00 +00:00
|
|
|
width: $("#ytapiplayer").width(),
|
|
|
|
height: $("#ytapiplayer").height()
|
2013-08-06 03:11:56 +00:00
|
|
|
};
|
2014-01-19 02:18:00 +00:00
|
|
|
|
2013-08-06 03:11:56 +00:00
|
|
|
$("#ytapiplayer").attr("width", 1)
|
|
|
|
.attr("height", 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
function unhidePlayer() {
|
|
|
|
if(!PLAYER)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if(!/(chrome|MSIE)/ig.test(navigator.userAgent))
|
|
|
|
return;
|
|
|
|
|
2014-01-19 02:18:00 +00:00
|
|
|
$("#ytapiplayer").width(PLAYER.size.width)
|
|
|
|
.height(PLAYER.size.height);
|
2013-08-06 03:11:56 +00:00
|
|
|
}
|
2013-10-17 04:22:37 +00:00
|
|
|
|
2013-11-25 22:20:15 +00:00
|
|
|
function chatDialog(div) {
|
|
|
|
var parent = $("<div/>").addClass("profile-box")
|
2014-01-26 03:29:56 +00:00
|
|
|
.css({
|
|
|
|
padding: "10px",
|
|
|
|
"z-index": "auto",
|
|
|
|
position: "absolute"
|
|
|
|
})
|
2013-11-25 22:20:15 +00:00
|
|
|
.appendTo($("body"));
|
|
|
|
|
|
|
|
div.appendTo(parent);
|
|
|
|
var cw = $("#chatwrap").width();
|
|
|
|
var ch = $("#chatwrap").height();
|
|
|
|
var cp = $("#chatwrap").offset();
|
|
|
|
var x = cp.left + cw/2 - parent.width()/2;
|
|
|
|
var y = cp.top + ch/2 - parent.height()/2;
|
|
|
|
parent.css("left", x + "px");
|
|
|
|
parent.css("top", y + "px");
|
|
|
|
return parent;
|
|
|
|
}
|
|
|
|
|
2013-10-17 04:22:37 +00:00
|
|
|
function errDialog(err) {
|
|
|
|
var div = $("<div/>").addClass("profile-box")
|
|
|
|
.css("padding", "10px")
|
|
|
|
.text(err)
|
|
|
|
.appendTo($("body"));
|
|
|
|
|
|
|
|
$("<br/>").appendTo(div);
|
2013-12-20 03:33:24 +00:00
|
|
|
$("<button/>").addClass("btn btn-xs btn-default")
|
2013-10-17 04:22:37 +00:00
|
|
|
.css("width", "100%")
|
|
|
|
.text("OK")
|
|
|
|
.click(function () { div.remove(); })
|
|
|
|
.appendTo(div);
|
|
|
|
var cw = $("#chatwrap").width();
|
|
|
|
var ch = $("#chatwrap").height();
|
|
|
|
var cp = $("#chatwrap").offset();
|
|
|
|
var x = cp.left + cw/2 - div.width()/2;
|
|
|
|
var y = cp.top + ch/2 - div.height()/2;
|
2014-01-30 04:50:14 +00:00
|
|
|
div.css("left", x + "px")
|
|
|
|
.css("top", y + "px")
|
|
|
|
.css("position", "absolute");
|
|
|
|
return div;
|
2013-10-17 04:22:37 +00:00
|
|
|
}
|
2013-11-11 04:26:30 +00:00
|
|
|
|
|
|
|
function queueMessage(data, type) {
|
|
|
|
if (!data)
|
|
|
|
data = { link: null };
|
|
|
|
if (!data.msg || data.msg === true) {
|
|
|
|
data.msg = "Queue failed. Check your link to make sure it is valid.";
|
|
|
|
}
|
2013-12-20 03:33:24 +00:00
|
|
|
var ltype = "label-danger";
|
2013-11-11 04:26:30 +00:00
|
|
|
var title = "Error";
|
2013-12-20 03:33:24 +00:00
|
|
|
if (type === "alert-warning") {
|
2013-11-11 04:26:30 +00:00
|
|
|
ltype = "label-warning";
|
|
|
|
title = "Warning";
|
2013-12-20 03:33:24 +00:00
|
|
|
}
|
2013-11-11 04:26:30 +00:00
|
|
|
|
|
|
|
var alerts = $(".qfalert.qf-" + type);
|
|
|
|
for (var i = 0; i < alerts.length; i++) {
|
|
|
|
var al = $(alerts[i]);
|
|
|
|
var cl = al.clone();
|
|
|
|
cl.children().remove();
|
|
|
|
if (cl.text() === data.msg) {
|
|
|
|
var tag = al.find("." + ltype);
|
|
|
|
if (tag.length > 0) {
|
|
|
|
var morelinks = al.find(".qflinks");
|
|
|
|
$("<a/>").attr("href", data.link)
|
|
|
|
.attr("target", "_blank")
|
|
|
|
.text(data.link)
|
|
|
|
.appendTo(morelinks);
|
|
|
|
$("<br/>").appendTo(morelinks);
|
|
|
|
var count = parseInt(tag.text().match(/\d+/)[0]) + 1;
|
|
|
|
tag.text(tag.text().replace(/\d+/, ""+count));
|
|
|
|
} else {
|
|
|
|
var tag = $("<span/>")
|
|
|
|
.addClass("label pull-right pointer " + ltype)
|
|
|
|
.text("+ 1 more")
|
|
|
|
.appendTo(al);
|
|
|
|
var morelinks = $("<div/>")
|
|
|
|
.addClass("qflinks")
|
|
|
|
.appendTo(al)
|
|
|
|
.hide();
|
|
|
|
$("<a/>").attr("href", data.link)
|
|
|
|
.attr("target", "_blank")
|
|
|
|
.text(data.link)
|
|
|
|
.appendTo(morelinks);
|
|
|
|
$("<br/>").appendTo(morelinks);
|
|
|
|
tag.click(function () {
|
|
|
|
morelinks.toggle();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var text = data.msg;
|
|
|
|
if (typeof data.link === "string") {
|
|
|
|
text += "<br><a href='" + data.link + "' target='_blank'>" +
|
|
|
|
data.link + "</a>";
|
|
|
|
}
|
|
|
|
makeAlert(title, text, type)
|
2013-12-20 03:33:24 +00:00
|
|
|
.addClass("qfalert qf-" + type)
|
2013-11-11 04:26:30 +00:00
|
|
|
.appendTo($("#queuefail"));
|
|
|
|
}
|
2013-11-14 04:36:43 +00:00
|
|
|
|
2013-11-16 02:23:57 +00:00
|
|
|
function filterChannelLog() {
|
2013-11-16 05:44:53 +00:00
|
|
|
var cc = $("#chanlog_contents");
|
|
|
|
if (!cc.data("log")) {
|
|
|
|
cc.data("log", cc.text());
|
|
|
|
}
|
|
|
|
var all = $("#filter_all").prop("checked");
|
|
|
|
var chat = $("#filter_chat").prop("checked");
|
|
|
|
var polls = $("#filter_polls").prop("checked");
|
|
|
|
var queue = $("#filter_queue").prop("checked");
|
|
|
|
var bans = $("#filter_bans").prop("checked");
|
|
|
|
var channelsettings = $("#filter_channelsettings").prop("checked");
|
|
|
|
var joinquit = $("#filter_joinquit").prop("checked");
|
|
|
|
|
|
|
|
var lines = cc.data("log").split("\n");
|
|
|
|
var include = [];
|
|
|
|
lines.forEach(function (line) {
|
|
|
|
if (line.trim() === "") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (all) {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var pre = line.split(" ")[5];
|
|
|
|
if (pre === undefined) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-11-20 16:10:01 +00:00
|
|
|
if (chat && pre.match(/<[\w-]+(\.\w*)?>/)) {
|
2013-11-16 05:44:53 +00:00
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (polls && pre === "***" && (line.indexOf("Opened Poll") >= 0 ||
|
|
|
|
line.indexOf("closed the active poll") >= 0)) {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (queue && pre === "###") {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (channelsettings && pre === "%%%") {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (joinquit) {
|
|
|
|
if (pre === "+++" || pre === "---") {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pre.match(/(\d{1,3}\.){3}\d{1,3}/) &&
|
|
|
|
line.indexOf("logged in as") >= 0) {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bans && pre === "***" && line.indexOf("banned") >= 0) {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (channelsettings && pre === "***") {
|
|
|
|
if (line.indexOf("Loading") >= 0 ||
|
|
|
|
line.indexOf("Loaded") >= 0 ||
|
|
|
|
line.indexOf("unloading") >= 0 ||
|
|
|
|
line.indexOf("rank") >= 0) {
|
|
|
|
include.push(line);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$("#chanlog_contents").text(include.join("\n"));
|
2013-11-16 02:23:57 +00:00
|
|
|
}
|
2013-12-15 03:59:47 +00:00
|
|
|
|
|
|
|
function makeModal() {
|
|
|
|
var wrap = $("<div/>").addClass("modal fade");
|
|
|
|
var dialog = $("<div/>").addClass("modal-dialog").appendTo(wrap);
|
|
|
|
var content = $("<div/>").addClass("modal-content").appendTo(dialog);
|
|
|
|
|
|
|
|
var head = $("<div/>").addClass("modal-header").appendTo(content);
|
|
|
|
$("<button/>").addClass("close")
|
|
|
|
.attr("data-dismiss", "modal")
|
|
|
|
.attr("data-hidden", "true")
|
|
|
|
.html("×")
|
|
|
|
.appendTo(head);
|
|
|
|
|
2014-01-19 02:18:00 +00:00
|
|
|
wrap.on("hidden.bs.modal", function () {
|
2013-12-15 03:59:47 +00:00
|
|
|
unhidePlayer();
|
|
|
|
wrap.remove();
|
|
|
|
});
|
|
|
|
return wrap;
|
|
|
|
}
|
2014-01-09 23:16:09 +00:00
|
|
|
|
|
|
|
function formatCSModList() {
|
|
|
|
var tbl = $("#cs-chanranks table");
|
|
|
|
tbl.find("tbody").remove();
|
|
|
|
var entries = tbl.data("entries") || [];
|
|
|
|
entries.sort(function(a, b) {
|
|
|
|
if (a.rank === b.rank) {
|
|
|
|
var x = a.name.toLowerCase();
|
|
|
|
var y = b.name.toLowerCase();
|
|
|
|
return y == x ? 0 : (x < y ? -1 : 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.rank - a.rank;
|
|
|
|
});
|
|
|
|
|
|
|
|
entries.forEach(function (entry) {
|
|
|
|
var tr = $("<tr/>").addClass("cs-chanrank-tr-" + entry.name);
|
|
|
|
var name = $("<td/>").text(entry.name).appendTo(tr);
|
|
|
|
name.addClass(getNameColor(entry.rank));
|
|
|
|
var rankwrap = $("<td/>");
|
|
|
|
var rank = $("<span/>").text(entry.rank).appendTo(rankwrap);
|
|
|
|
var dd = $("<div/>").addClass("btn-group");
|
|
|
|
var toggle = $("<button/>")
|
|
|
|
.addClass("btn btn-xs btn-default dropdown-toggle")
|
|
|
|
.attr("data-toggle", "dropdown")
|
|
|
|
.html("Edit <span class=caret></span>")
|
|
|
|
.appendTo(dd);
|
|
|
|
if (CLIENT.rank <= entry.rank) {
|
|
|
|
toggle.addClass("disabled");
|
|
|
|
}
|
|
|
|
|
|
|
|
var menu = $("<ul/>").addClass("dropdown-menu")
|
|
|
|
.attr("role", "menu")
|
|
|
|
.appendTo(dd);
|
|
|
|
|
|
|
|
var ranks = [
|
|
|
|
{ name: "Remove Moderator", rank: 1 },
|
|
|
|
{ name: "Moderator", rank: 2 },
|
|
|
|
{ name: "Channel Admin", rank: 3 },
|
|
|
|
{ name: "Channel Owner", rank: 4 }
|
|
|
|
];
|
|
|
|
|
|
|
|
ranks.forEach(function (r) {
|
|
|
|
var li = $("<li/>").appendTo(menu);
|
|
|
|
var a = $("<a/>")
|
|
|
|
.addClass(getNameColor(r.rank))
|
|
|
|
.attr("href", "javascript:void(0)")
|
|
|
|
.text(r.name)
|
|
|
|
.appendTo(li);
|
|
|
|
if (r.rank !== entry.rank) {
|
|
|
|
a.click(function () {
|
|
|
|
socket.emit("setChannelRank", {
|
|
|
|
user: entry.name,
|
|
|
|
rank: r.rank
|
|
|
|
});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
$("<span/>").addClass("glyphicon glyphicon-ok pull-right")
|
|
|
|
.appendTo(a);
|
|
|
|
li.addClass("disabled");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (r.rank > CLIENT.rank || (CLIENT.rank < 4 && r.rank === CLIENT.rank)) {
|
|
|
|
li.addClass("disabled");
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
dd.css("margin-right", "10px").prependTo(rankwrap);
|
|
|
|
rankwrap.appendTo(tr);
|
|
|
|
tr.appendTo(tbl);
|
|
|
|
});
|
|
|
|
}
|
2014-01-09 23:43:07 +00:00
|
|
|
|
|
|
|
function formatCSBanlist() {
|
|
|
|
var tbl = $("#cs-banlist table");
|
|
|
|
tbl.find("tbody").remove();
|
|
|
|
var entries = tbl.data("entries") || [];
|
|
|
|
var sparse = {};
|
|
|
|
for (var i = 0; i < entries.length; i++) {
|
|
|
|
if (!(entries[i].name in sparse)) {
|
|
|
|
sparse[entries[i].name] = [];
|
|
|
|
}
|
|
|
|
sparse[entries[i].name].push(entries[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
var flat = [];
|
|
|
|
for (var name in sparse) {
|
|
|
|
flat.push({
|
|
|
|
name: name,
|
|
|
|
bans: sparse[name]
|
|
|
|
});
|
|
|
|
}
|
|
|
|
flat.sort(function (a, b) {
|
|
|
|
var x = a.name.toLowerCase(),
|
|
|
|
y = b.name.toLowerCase();
|
|
|
|
return x === y ? 0 : (x > y ? 1 : -1);
|
|
|
|
});
|
|
|
|
|
|
|
|
var addBanRow = function (entry, after) {
|
|
|
|
var tr = $("<tr/>");
|
|
|
|
if (after) {
|
|
|
|
tr.insertAfter(after);
|
|
|
|
} else {
|
|
|
|
tr.appendTo(tbl);
|
|
|
|
}
|
|
|
|
var unban = $("<button/>").addClass("btn btn-xs btn-danger")
|
|
|
|
.appendTo($("<td/>").appendTo(tr));
|
2014-01-12 05:55:52 +00:00
|
|
|
unban.click(function () {
|
|
|
|
socket.emit("unban", {
|
|
|
|
id: entry.id,
|
|
|
|
name: entry.name
|
|
|
|
});
|
|
|
|
});
|
2014-01-09 23:43:07 +00:00
|
|
|
$("<span/>").addClass("glyphicon glyphicon-remove-circle").appendTo(unban);
|
|
|
|
$("<td/>").text(entry.ip).appendTo(tr);
|
|
|
|
$("<td/>").text(entry.name).appendTo(tr);
|
|
|
|
$("<td/>").text(entry.bannedby).appendTo(tr);
|
|
|
|
tr.popover({
|
|
|
|
title: "Ban Reason",
|
|
|
|
content: entry.reason,
|
|
|
|
placement: "left",
|
|
|
|
trigger: "hover"
|
|
|
|
});
|
|
|
|
return tr;
|
|
|
|
};
|
|
|
|
|
|
|
|
flat.forEach(function (person) {
|
|
|
|
var bans = person.bans;
|
|
|
|
var name = person.name;
|
|
|
|
var first = addBanRow(bans.shift());
|
|
|
|
|
|
|
|
if (bans.length > 0) {
|
|
|
|
var showmore = $("<button/>").addClass("btn btn-xs btn-default pull-right");
|
|
|
|
$("<span/>").addClass("glyphicon glyphicon-list").appendTo(showmore);
|
|
|
|
showmore.appendTo(first.find("td")[1]);
|
|
|
|
|
|
|
|
showmore.click(function () {
|
|
|
|
if (showmore.data("elems")) {
|
|
|
|
showmore.data("elems").forEach(function (e) {
|
|
|
|
e.remove();
|
|
|
|
});
|
|
|
|
showmore.data("elems", null);
|
|
|
|
} else {
|
|
|
|
var elems = [];
|
|
|
|
bans.forEach(function (b) {
|
|
|
|
elems.push(addBanRow(b, first));
|
|
|
|
});
|
|
|
|
showmore.data("elems", elems);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2014-01-16 17:53:34 +00:00
|
|
|
|
|
|
|
function formatCSChatFilterList() {
|
|
|
|
var tbl = $("#cs-chatfilters table");
|
|
|
|
tbl.find("tbody").remove();
|
|
|
|
tbl.find(".ui-sortable").remove();
|
|
|
|
var entries = tbl.data("entries") || [];
|
|
|
|
entries.forEach(function (f) {
|
|
|
|
var tr = $("<tr/>").appendTo(tbl);
|
2014-01-16 22:20:08 +00:00
|
|
|
var controlgroup = $("<div/>").addClass("btn-group")
|
|
|
|
.appendTo($("<td/>").appendTo(tr));
|
2014-01-16 17:53:34 +00:00
|
|
|
var control = $("<button/>").addClass("btn btn-xs btn-default")
|
|
|
|
.attr("title", "Edit this filter")
|
2014-01-16 22:20:08 +00:00
|
|
|
.appendTo(controlgroup);
|
2014-01-16 17:53:34 +00:00
|
|
|
$("<span/>").addClass("glyphicon glyphicon-list").appendTo(control);
|
2014-01-16 22:20:08 +00:00
|
|
|
var del = $("<button/>").addClass("btn btn-xs btn-danger")
|
|
|
|
.appendTo(controlgroup);
|
|
|
|
$("<span/>").addClass("glyphicon glyphicon-trash").appendTo(del);
|
|
|
|
del.click(function () {
|
|
|
|
socket.emit("removeFilter", f);
|
|
|
|
});
|
2014-01-16 17:53:34 +00:00
|
|
|
var name = $("<code/>").text(f.name).appendTo($("<td/>").appendTo(tr));
|
|
|
|
var activetd = $("<td/>").appendTo(tr);
|
|
|
|
var active = $("<input/>").attr("type", "checkbox")
|
|
|
|
.prop("checked", f.active)
|
|
|
|
.appendTo(activetd)
|
|
|
|
.change(function () {
|
|
|
|
f.active = $(this).prop("checked");
|
|
|
|
socket.emit("updateFilter", f);
|
|
|
|
});
|
|
|
|
|
2014-01-16 22:20:08 +00:00
|
|
|
var reset = function () {
|
|
|
|
control.data("editor") && control.data("editor").remove();
|
|
|
|
control.data("editor", null);
|
|
|
|
control.parent().find(".btn-success").remove();
|
|
|
|
var tbody = $(tbl.children()[1]);
|
|
|
|
if (tbody.find(".filter-edit-row").length === 0) {
|
|
|
|
tbody.sortable("enable");
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-01-16 17:53:34 +00:00
|
|
|
control.click(function () {
|
2014-01-16 22:20:08 +00:00
|
|
|
if (control.data("editor")) {
|
|
|
|
return reset();
|
|
|
|
}
|
|
|
|
$(tbl.children()[1]).sortable("disable");
|
|
|
|
var tr2 = $("<tr/>").insertAfter(tr).addClass("filter-edit-row");
|
2014-01-16 17:53:34 +00:00
|
|
|
var wrap = $("<td/>").attr("colspan", "3").appendTo(tr2);
|
|
|
|
var form = $("<form/>").addClass("form-inline").attr("role", "form")
|
|
|
|
.attr("action", "javascript:void(0)")
|
|
|
|
.appendTo(wrap);
|
|
|
|
var addTextbox = function (placeholder) {
|
|
|
|
var div = $("<div/>").addClass("form-group").appendTo(form)
|
|
|
|
.css("margin-right", "10px")
|
|
|
|
.css("max-width", "25%");
|
|
|
|
var input = $("<input/>").addClass("form-control")
|
|
|
|
.attr("type", "text")
|
|
|
|
.attr("placeholder", placeholder)
|
|
|
|
.attr("title", placeholder)
|
|
|
|
.appendTo(div);
|
|
|
|
return input;
|
|
|
|
};
|
|
|
|
|
|
|
|
var regex = addTextbox("Filter regex").val(f.source);
|
|
|
|
var flags = addTextbox("Regex flags").val(f.flags);
|
|
|
|
var replace = addTextbox("Replacement text").val(f.replace);
|
|
|
|
|
|
|
|
var checkwrap = $("<div/>").addClass("checkbox").appendTo(form);
|
|
|
|
var checklbl = $("<label/>").text("Filter Links").appendTo(checkwrap);
|
2014-01-16 22:20:08 +00:00
|
|
|
var filterlinks = $("<input/>").attr("type", "checkbox")
|
|
|
|
.prependTo(checklbl)
|
|
|
|
.prop("checked", f.filterlinks);
|
2014-01-16 17:53:34 +00:00
|
|
|
|
|
|
|
var save = $("<button/>").addClass("btn btn-xs btn-success")
|
2014-01-16 22:20:08 +00:00
|
|
|
.attr("title", "Save changes")
|
2014-01-16 17:53:34 +00:00
|
|
|
.insertAfter(control);
|
2014-01-16 22:20:08 +00:00
|
|
|
$("<span/>").addClass("glyphicon glyphicon-floppy-save").appendTo(save);
|
|
|
|
save.click(function () {
|
|
|
|
f.source = regex.val();
|
|
|
|
f.flags = flags.val();
|
|
|
|
f.replace = replace.val();
|
|
|
|
f.filterlinks = filterlinks.prop("checked");
|
|
|
|
try {
|
|
|
|
new RegExp(f.source, f.flags);
|
|
|
|
} catch (e) {
|
|
|
|
alert("Invalid regex: " + e);
|
|
|
|
}
|
|
|
|
|
|
|
|
socket.emit("updateFilter", f);
|
|
|
|
reset();
|
|
|
|
});
|
|
|
|
|
|
|
|
control.data("editor", tr2);
|
2014-01-16 17:53:34 +00:00
|
|
|
});
|
|
|
|
});
|
|
|
|
$(tbl.children()[1]).sortable({
|
|
|
|
start: function(ev, ui) {
|
|
|
|
FILTER_FROM = ui.item.prevAll().length;
|
|
|
|
},
|
|
|
|
update: function(ev, ui) {
|
|
|
|
FILTER_TO = ui.item.prevAll().length;
|
|
|
|
if(FILTER_TO != FILTER_FROM) {
|
|
|
|
socket.emit("moveFilter", {
|
|
|
|
from: FILTER_FROM,
|
|
|
|
to: FILTER_TO
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|