// ==UserScript==
// @name kroleg talkrooms
// @version 0.0.8
// @datecreated 2019-04-01
// @lastupdated 2021-08-08
// @author kroleg
// @grant none
// @match https://talkrooms.ru/*
// @match https://www.talkrooms.ru/*
// @updatedURL https://extension.kroleg.tk/chatmod/engines/talkrooms.ru.user.js
// @description Скрипт изменяет внешний вид чата и отключает плавный автоскролл
// @namespace https://greasyfork.org/users/386075
// ==/UserScript==
//if(window.location.href.match(/^http[s]?:\/\/(?:www.)*talkrooms.ru/i) && /Chrome|Firefox|Opera|Safari/i.test(navigator.userAgent)) {
(function() {
'use strict';
var
version = ' .:: ЧатМод Кролега v 0.0.8 ::.',
recipient = null,
messHist = {
historyMess: [''],
historyCount: 0,
historyMax: 50
},
ACCESS_OWNER = 80,
ACCESS_ADMIN = 70,
ACCESS_MODER = 50,
uLevel = {
0: 'Гость: пользователь без привелегий.',
20: 'Приглашённый: доступ к комнате только по приглашению.',
50: 'Модератор: следит за порядком и наказывает нарушителей.',
70: 'Админ: назначает модераторов и меняет название комнаты.'
},
uIgnored = {
0: 'снять бан',
15: 'на 15 минут',
120: 'на 2 чиса',
720: 'на 12 чисоф',
10080: 'на ниделю',
null: 'нафсикда'
},
Role,
Users = [],
Colors = [
['ffffff', 'c0c0c0'],
['cbcbcb', 'd3d3d3'],
['A6BACF', 'D1E0F0'],
['84C2FC', 'C4E1FC'],
['BE6DFD', 'ECD4FF'],
['F464F3', 'FFBAFF'],
['EA5B5B', 'FFD4D4'],
['d13363', 'ffc9da'],
['e36b00', 'ffc38d'],
['B6F01A', 'E6FFA3'],
['D5E800', 'F8FFAC'],
['F8FB00', 'FDFF8C'],
['FFC600', 'FBF0C8'],
['FFE245', 'FFF4BB'],
['BECA85', 'E9F3B9'],
['a6b53e', 'dcee62'],
['87E0B3', 'C4FFE1'],
['8CEE89', 'D7FFD5'],
['6AC053', 'DCF1D6'],
['B4A794', 'E4D5BE'],
['B69EC0', 'E7CBF2'],
['C696A5', 'E5D8DC'],
['bf985b', 'e9c48b'],
['c97a4d', 'f5dbbd'],
['2ebba5', '77f7e3'],
['3c9fe4', '86cdff'],
['8E98EF', 'D4D8FF'],
['708bff', 'b5c3ff']
];
function log(s) {
console.log(s);
}//func
function localSysMess(mess) {
var
chat = $('.talk-current');
chat.append('<div class="sys"><span>' + mess + '</span><span>' + ' (' + formatDate(new Date(), 'd mL в hh:ii') + ')</span></div>')
$('.talk-content').scrollTop($('.talk-content')[0].scrollHeight);
}//func
function getRandomInt(min, max) {
return Math.floor(Math.random() * (max - min + 1)) + min;
}//func
function reverse(s) {
return s.split('').reverse().join('');
}//
function formatDate(formatDate, formatString) {
var
month = ['января', 'февраля', 'марта', 'апреля', 'мая', 'июня', 'июля', 'августа', 'сентября', 'октября', 'ноября', 'декабря'],
z = '0',
yyyy = formatDate.getFullYear(),
yy = yyyy.toString().substring(2),
m = formatDate.getMonth() + 1,
mm = m < 10 ? z + m : m,
mL = month[m - 1],
d = formatDate.getDate(),
dd = d < 10 ? z + d : d,
h = formatDate.getHours(),
hh = h < 10 ? z + h : h,
i = formatDate.getMinutes(),
ii = i < 10 ? z + i : i,
s = formatDate.getSeconds(),
ss = s < 10 ? z + s : s,
ms = formatDate.getMilliseconds(),
tz = formatDate.toString().match(/([-\+][\d]+)\s/);
if(tz) tz = tz[1].replace(/[\d]{2}/, '$&:');
return formatString
.replace(/yyyy/i, yyyy)
.replace(/yy/i, yy)
.replace(/mL/i, mL)
.replace(/mm/i, mm)
.replace(/m/i, m)
.replace(/dd/i, dd)
.replace(/d/i, d)
.replace(/hh/i, hh)
.replace(/h/i, h)
.replace(/ii/i, ii)
.replace(/i/i, i)
.replace(/ss/i, ss)
.replace(/s/i, s)
.replace(/ms/i, ms)
.replace(/tz/i, tz);
}//func
function getUserObj(key, val) {
for(var k in Users)
if(Users[k][key] == val) return Users[k];
return false;
}//func
function removeUser(arr) {
var
i = 0,
remUser = [],
tArr = [];
for(var k in Users) {
remUser[i] = Users[k];
for(var n in arr)
if(Users[k].role_id == arr[n].role_id) {
tArr.push(Users[k]);
remUser.splice(i, 1);
break;
}//if
i++;
}//for
Users = tArr;
remUser = remUser.filter(Boolean);
return remUser;
}//func
function getRole(role_id) {
var
room = Rooms.selected;
return room.rolesOnline.get(role_id) || room.rolesWaiting.get(role_id);
}//func
function textNodeReplace(obj, txt) {
obj.contents().filter(function() { return this.nodeType == 3; })[0].textContent = txt;
}//func
function historyMessKbd(sw) {
var
input = $('#input');
if(sw < 0) {
if(messHist.historyCount > 0) input.val(messHist.historyMess[--messHist.historyCount]);
} else if(messHist.historyCount < messHist.historyMess.length) input.val(messHist.historyMess[++messHist.historyCount]);
}//func
function userOnlineModify() {
var
container = $('.room-users'),
template = $.template('#user-template');
function renderOwner() {
return '<em class="own">[own]</em>';
}
function renderAdmin() {
return '<em class="adm">[adm]</em>';
}
function renderModerator() {
return '<em class="mod">[mod]</em>';
}
function renderRole(data) {
var
role = getRole(data.role_id);
if(!data.userpicUrl) data.userpicUrl = Userpics.getUrl(data);
var
$role = template(data);
if(role.level == ACCESS_OWNER) $role.find('.nickname').parent().append(renderOwner()); else
if(role.level == ACCESS_ADMIN) $role.find('.nickname').parent().append(renderAdmin()); else
if(role.level == ACCESS_MODER) $role.find('.nickname').parent().append(renderModerator());
if(data.role_id === Rooms.selected.myRole.role_id) $role.addClass('me');
if(data.annoying) $role.addClass('annoying');
$role.attr('data-role', data.role_id);
$role.find('.nickname').css({color: '#' + Colors[role.nickColor || 0][0]});
$role.find('.userpic')
.unbind()
.mouseover(function(e) {
var
target = $(e.target).parent(),
role_id = target.data('role'),
role = getRole(role_id);
Profile.show(role, { nickname: role.nickname, top: event.pageY, left: event.pageX, target: target });
})
.mousedown(function() {
Profile.hide();
})
.mouseout(function() {
Profile.hide();
})
.click(function(e) {
var
uinfo = $('#uinfo'),
target = $(e.target).parent(),
role_id = target.data('role'),
role = Role = getRole(role_id),
iSrc = $(this).css('background-image'),
ulev = $('#ulevel input[type=radio]'),
uign = $('#uignore input[type=radio]'),
exp = role.expired == null && role.ignored != null ? 'null' : Math.round((new Date(role.expired) - new Date(role.ignored)) / 1000 / 60);
if(uinfo.data('role_id') == role_id) {
uinfo
.removeData()
.hide();
setTimeout(function() { Profile.hide(); }, 0);
return;
}
iSrc = /http/.test(iSrc)
? iSrc.replace(/url\("([^\n]+)userpics\/(\d+)\.png\?(\d+)"\)/, '$1photos/$2.jpg?$3')
: iSrc.replace(/url\("([^\n]+)"\)/, '$1');
$('.uadm > div:nth-child(2), #ulevel, #uignore').hide();
if(Rooms.selected.myRole.role_id != role.role_id && Rooms.selected.myRole.level > role.level) {
ulev.removeAttr('checked');
uign.removeAttr('checked');
if(Rooms.selected.myRole.level >= 50) {
if(Role.level >= 20) $('#uignore').show();
uign.removeAttr('checked');
checkedSet(uign, exp);
/*
$(uign).each(function(k, v) {
if($(v).val() == exp) {
$(v).prop({checked: true});
return false;
}
});
*/
}
if(Rooms.selected.myRole.level >= 70) {
if(Rooms.selected.myRole.level < 80) $('#ulevel > li').eq(4).remove();
if(Role.level >= 20) $('#ulevel').show();
checkedSet(ulev, role.level);
/*
$(ulev).each(function(k, v) {
if($(v).val() == role.level) {
$(v).prop({checked: true});
return false;
}
});
*/
}
$('.uadm').show();
} else $('.uadm').hide();
// textNodeReplace($('#uinfo h4'), role.status ? 'Информация :: Статус: ' + role.status : 'Информация');
textNodeReplace($('#uinfo h4'), role.nickname);
// $('#uinfo .usernick').html(role.nickname).css({color: '#' + Colors[role.nickColor || 0][0]});
$('#uinfo .cont img').attr('src', iSrc);
uinfo.data('role_id', role_id).show();
setTimeout(function() { Profile.hide(); }, 0);
});
return $role[0];
}
function Group(selector) {
this.elem = container.find(selector);
this.list = this.elem.find('.users-list');
this.amount = this.elem.find('.users-amount');
}
Group.prototype.show = function(roles) {
this.list.html('');
if(roles.length) {
this.amount.html(roles.length);
this.list.append(roles.map(renderRole));
this.elem.show();
} else this.elem.hide();
};
var
onlineGroup = new Group('.users-online'),
ignoredGroup = new Group('.users-ignored'),
waitingGroup = new Group('.users-requests');
function showOnline(room) {
var
useHidden = !room.myRole.isModerator,
useIgnored = !room.myRole.ignored,
online = [],
hidden = [],
ignored = [];
room.rolesOnline.items.forEach(function(role) {
if(useIgnored && role.ignored) {
role.annoying = false;
ignored.push(role);
} else if(useHidden && Me.isHidden(role)) {
role.annoying = true;
hidden.push(role);
} else {
role.annoying = false;
online.push(role);
}
});
online = online.concat(hidden);
onlineGroup.show(online);
ignoredGroup.show(room.myRole.isModerator ? ignored : []);
}
function showWaiting(room) {
waitingGroup.show(room.myRole.isModerator ? room.rolesWaiting.items : []);
}
function updated(room) {
Rooms.triggerSelected('selected.roles.updated', room);
}
Rooms.on('explore', function() {
container.hide();
});
Rooms.on('select', function() {
container.hide();
});
Rooms.on('selected.ready', function(room) {
showOnline(room);
showWaiting(room);
container.show();
$('#input').focus();
});
Rooms.on('selected.denied', function(room) {
container.hide();
});
Rooms.on('selected.roles.updated', showOnline);
Rooms.on('selected.waiting.updated', showWaiting);
// Update ignored and hidden groups
Rooms.on('my.rank.changed', showOnline);
//User login
Rooms.pipe('role.online', function(room, data) {
room.rolesOnline.add(data);
updated(room);
});
//User logout
Rooms.pipe('role.offline', function(room, data) {
room.rolesOnline.remove(data.role_id);
updated(room);
});
$('.room-users')[0].addEventListener('DOMSubtreeModified', function(e) {
var
tmp,
nickColor,
uObj,
uArr = Rooms.selected.rolesOnline.index;
if(Object.keys(uArr).length < Users.length) {
uObj = removeUser(uArr)[0];
return
// localSysMess('<span data-role="' + uObj.role_id + '" style="color: #' + Colors[uObj.nickColor || 0][0] + '" class="nickname">' + uObj.nickname + '</span> покидает чат');
}
for(var k in uArr) {
if(uArr[k].role_id && !getUserObj('role_id', uArr[k].role_id)) {
nickColor = getRandomInt(0, Colors.length - 1);
tmp = uArr[k];
tmp.nickColor = nickColor;
Users.push(tmp);
if(Rooms.selected.myRole.role_id == uArr[k].role_id) {
$('.reply-wrapper > .userpic')
.removeAttr('style')
.html(Rooms.selected.myRole.nickname + ':')
.css({color: '#' + Colors[nickColor][0]});
}
// localSysMess('<span data-role="' + uArr[k].role_id + '" style="color: #' + Colors[nickColor || 0][0] + '" class="nickname">' + uArr[k].nickname + '</span> входит в чат');
Users.sort();
}//if new user
}//for
}, false);
}//func
function messageSendModify() {
window.sendMess = function(type) {
var
k,
input = $('#input'),
txt = input.val().trim(),
nick = txt.substring(0, txt.indexOf(',')),
uObj = getUserObj('nickname', nick),
mentions = txt.split(','),
options = {
room_id: Room.data.room_id,
content: txt,
mentions: []
};
if(txt == '') return;
messHist.historyMess.push(txt);
if(messHist.historyMess.length > messHist.historyMax) messHist.historyMess.shift();
messHist.historyCount = messHist.historyMess.length;
if(type == 'sec') options.recipient_role_id = recipient ? recipient.role_id : uObj ? uObj.role_id : null;
else if(mentions.length)
for(k in mentions) {
if(!mentions[k]) continue;
nick = mentions[k].trim();
uObj = getUserObj('nickname', nick);
if(uObj) options.mentions.push(uObj.role_id);
}//for
Room.send(options);
setTimeout(function() {
$('.talk-content').scrollTop($('.talk-content')[0].scrollHeight + 1000);
}, 10);
recipient = null;
input.val('').focus();
$('.reply-form .cbutton').removeClass('button-down');
}//func
var
container = $('.room-users, .talk-current');
function getData(elem) {
var
room = Rooms.selected,
role_id = Number(elem.attr('data-role'));
recipient = { role_id: role_id };
return room.rolesOnline.get(role_id) || room.rolesWaiting.get(role_id);
}
container.on('click', '.user:not(.me) .nickname, .message .nickname, .sys .nickname', function(event) {
if(event.target.nodeName !== 'A') {
var
user = $(this).closest('.user'),
data = getData(user.length ? user : $(this));
if(data && data.come_in != null) {
Profile.show(data, {
target: user
});
}
// else if(data) Room.replyTo(data);
}
});
Room.replyPrivate = function(data) {
var
input = $('#input');
if(!data.nickname) return;
input.focus().val(data.nickname + ', ' + input.val());
recipient = data;
};
Room.replyTo = function(role) {
var
field = $('#input'),
raw = field.get(0),
pos = raw.selectionStart;
if(role && role.nickname) {
field.focus().val(role.nickname + ', ' + field.val());
if('setSelectionRange' in raw) {
pos = pos ? pos + role.nickname.length + 2 : raw.value.length;
raw.setSelectionRange(pos, pos);
}
} else field.focus();
};
}//func
function headerModify() {
var
title = $('.toolbar-title'),
tools = $('.toolbar-tools'),
date = $('.header-date');
function getTitle(room) {
if(room.state === 'lost') return 'Комната не найдена';
if(room.state === 'deleted') return '<span class="toolbar-deleted">' + room.data.topic + '</span>';
return room.data.topic;
}
function showTopic(room) {
title.html(room.data.topic);
}
function showTitle(text) {
toggleToolbar(false);
title.html(text + '<span>' + version + '</span>');
}
function toggleToolbar(ready) {
title.toggleClass('changing', !ready);
tools.toggleClass('hidden', !ready);
date.toggleClass('hidden', !ready);
}
Rooms.on('select', function(room) {
showTitle(getTitle(room));
toggleToolbar(room.state === 'ready');
});
Rooms.on('selected.ready', function(room) {
showTitle(getTitle(room));
toggleToolbar(true);
});
Rooms.on('selected.denied', function(room) {
showTitle(getTitle(room));
});
Rooms.on('selected.topic.updated', showTopic);
}//func
function fixCSS() {
var
// css = document.styleSheets[document.styleSheets.length - 2],
css = $('<style type="text/css"></style>'),
cssFont = $('<style type="text/css"></style>'),
rulesFont = ['@import url("https://fonts.googleapis.com/css2?family=PT+Sans+Narrow&display=swap")'],
rules = [
'body { font: 14px Arial; color: #000; }',
'a, .side-subscriptions a, .hall-shuffle .link, .hall a, .message a { color: #1270b2; }',
'select, textarea, input, a { outline: none !important; }',
'.message { line-height: 14px; }',
'.message a { color: #D5D6A5; }',
'.message a:hover { text-decoration: none; }',
'.message, .sys { margin: 0 0 1px 0; width: 100%; }',
'.speech { margin: 0px 0px 0px 1px; padding: 0 1px; }',
'.speech-author > .nickname { margin: 4px 0 0px 0; display: inline-block; color: #ab5f1e; }',
'.speech-author, .message { display: inline-block; }',
'.date { box-shadow: rgba(0, 0, 0, 0.5) 0px 1px 0px 0px, rgb(68, 68, 68) 0px 2px 0px 0px; height: 22px; border: 0; }',
'#main span.nickname { text-decoration: underline; cursor: pointer; }',
'#main span.nickname:hover { text-decoration: none; }',
'#main, #side, .talk-reply, .talk-overlay, .entry-text, .hall, .date-text { background: #333; color: #bbb; box-shadow: none; }',
'.date-text { top: 15px; padding: 0 4px 0 4px; color: #828282; font: normal 14px Arial; }',
'#main, .header-main { left: 0; right: 21%; position: absolute; margin: 0; background: #333; box-shadow: none; }',
'#header, .header-main, .header-toolbar { height: 48px; }',
// '.side-content, .talk-content { top: 48px; }',
'#side, .header-side { right: 0; width: 21%; position: absolute; }',
'#side { overflow-y: hidden; }',
'.header-side { background: #333; }',
'#header { box-shadow: none; }',
'.talk-reply { margin: 0 0 24px 0}',
'.header-title, .toolbar-title { padding: 9px 0; }',
'.reply-form { min-height: auto; padding: 0 0 12px 0; margin: 0; }',
'.reply-wrapper .userpic, #main .userpic { height: 22px; width: 226px !important; background-image: none !important; visibility: visible; left: 0; margin: -3px 3px 0 0; float: left; text-align: right; border: 0; }',
'.reply-wrapper { padding: 0px 0px 0px 0px; margin: 0px 0px 0 0px; }',
'#input, .talk-edit textarea, .textfield input { height: 16px !important; overflow-y: hidden; width: 64%; border-radius: 20px; margin: 12px 0 0 0px; box-shadow: 1px 1px 0px 0px rgba(255, 255, 255, 0.45), -1px -1px 0px 0px rgba(0, 0, 0, 0.2); background-color: #444; color: #ccc; border: 0; padding: 3px 4px 3px 10px; } ',
'.talk-edit textarea { width: auto ; padding: 3px; box-shadow: none; border-radius: 2px; word-wrap: break-word; }',
'.reply-send { float: right; border: 0; margin: 3px 183px 0 0; position: relative; width: auto; height: auto; top: 0; right: 0;}',
'.reply-field { float: left; }',
'#input { margin: 3px 0px 2px 232px; width: 420px; float: left; line-height: 16px; }',
'.with-my-name, .speech.personal { padding: 0px 0px 0px 0px; margin: 0px 0px 1px 0px; background-color: #3e3e3e; border: 0; border-radius: 0; }',
'.speech.personal { margin: 0 1px 1px 4px; }',
'.talk-content { bottom: 74px; }',
'.talk-edit { background: none; box-shadow: none; display: inline-block; }',
'.msg-edit { background-position: 6px 3px; }',
'.msg-time { display: none; left: -47px; top: 18px; color: #929292; font: normal 10px Tahoma; text-align: center; width: 21px; }',
'.with-my-name time { left: 29px; top: 48px; }',
'.speech > div:nth-child(2) > time { display: block; }',
'.header-title, .toolbar-title { font-size: 18px; color: #8cafff; }',
'.toolbar-title > span { color: #ff7600; display: inline-block; width: 100%; text-align: center; font: normal 20px "PT Sans Narrow"; position: absolute; left: -30px; }',
'.toolbar-tools { z-index: 1; }',
'.user > .nickname { color: #000000; cursor: pointer; }',
'.talk-archive, .talk-current { padding: 12px 0 0 0; }',
'.side-content { position: relative; overflow-y: auto; background: none; }',
'.user { border: 0; box-shadow: none; padding: 0; margin: 0 0 4px 0; line-height: 14px; white-space: nowrap; }',
'.user .nickname { color: #bbb; text-decoration: underline; margin: 0 0 0 22px; }',
'.user .nickname:hover { text-decoration: none; }',
'.user .userpic { width: 14px; height: 14px; border: none; background-size: 100%; border-radius: 50%; margin: 0px 0 0 0; }',
'.user em { vertical-align: 2px; margin: 0 0 0 2px; font: normal 11px Arial; color: #ffb817; }',
'.user em.own { color: #ff6060; }',
'.user em.adm { color: #54d89b; }',
'.user em.mod { color: #74c3f1; }',
'.user.me:after { top: 0; }',
'.users-list { margin: 0; }',
'.sys { color: #ffcc00; padding-left: 240px; }',
'.room-users { margin: 0 0 0 13px; }',
'.cbutton { text-decoration: none; position: relative; display: inline-block !important; cursor: pointer; background-color: #555; border-radius: 2px; color: #fff; width: 48px; height: 22px; margin: 1px 3px 0 0px; box-shadow: 0px 0px 1px 1px rgba(0, 0, 0, 0.26), 0px 1px 0px 0px rgba(255, 255, 255, 0.3) inset; font: normal 12px/13px "PT Sans Narrow"; }',
'.cbutt-active { color: #ffcc00; }',
'.cbutton span { display: inline-block; position: absolute; height: 15px; text-align: center; top: 0px; left: 0px; bottom: 0px; right: 0px; text-shadow: rgba(0, 0, 0, 0.4) -1px -1px 0px; font: 12px/13px "PT Sans Narrow"; margin: auto; }',
'.cbutton:active span, .button-down span { margin: 5px 0 0 0px; }',
'.cbutton:active, .button-down, #room-settings .alarm-off .button:active { margin: 0 3px 0 0px; background-color: #454545; box-shadow: 0px -1px 0px rgba(255, 255, 255, .2) inset, 0px 1px 0px rgba(0, 0, 0, 0.5) inset, 1px 0px 0px rgba(0, 0, 0, 0.5) inset, -1px 0px 0px rgba(0, 0, 0, 0.5) inset; }',
'#room-settings .submit { padding: inherit; display: block; }',
'#room-settings .submit .cbutton { margin: 0 20px 14px 0; padding: 0 1px; }',
'#room-settings .alarm-off .cbutton { width: 360px; margin: 1px 0; }',
'#room-settings .alarm-off .button { border: 0; background: none; background-image: none; padding: 0; font-size: inherit; line-height: inherit; box-shadow: 0px 0px 1px 1px rgba(0, 0, 0, 0.26), 0px 1px 0px 0px rgba(255, 255, 255, 0.3) inset;}',
'#room-settings .cbutton { background-color: #555 !important; }',
'#room-settings .alarm-off .button:active { margin: 1px 3px 1px 0px; }',
'.room-remove { background: none; border: 0; box-shadow: none; }',
'.room-remove .link { color: #ff4131; }',
'.alarm-active { padding: 3px 14px; border-radius: 2px; font-size: 14px; line-height: inherit; }',
'.alarm-cancel { top: -3px; }',
'.textfield input { padding: 10px 0 9px 13px; }',
'#go { background-color: #1A64B4; }',
'#pvt { background-color: #2C8046; }',
'.filter-my { background: #555; border-color: none; width: 81px; margin: 12px 0 0 0; }',
'.filter-my:active { margin: 13px 0 0 0; }',
'.filter-my-selected { background: #2a707b; }',
'.toolbar-settings { margin: 0px 4px 0 10px; }',
'.popup, .popup-scroll, .profile-userpic { margin: 0; padding: 0; }',
'.profile-section, .popup-content { border-radius: 0; background: none; }',
'@keyframes profile-zoom { from { opacity: 0; } to { opacity: 1; }}',
'.profile-close, .profile-edit-button, .profile-photo:after, .nickname-hint, .reply-warning { display: none !important; }',
'#profile { width: auto; max-width: 156px; max-height: 202px; }',
'.profile-role { max-width: 156px; max-height: 202px; }',
'#profile .popup-content { margin: 0; width: auto; max-width: 156px; max-height: 202px; transform-origin: none; animation: .1s ease-out profile-zoom; }',
'.profile-photo { max-width: 156px; max-height: 202px; margin: 0; border-color: rgba(255, 255, 255, 0.85) !important; padding: 0; border-radius: 2px; background-color: #333; border: 1px solid #555; }',
'.profile-photo-img { max-width: 152px; width: auto; max-height: 198px; }',
'.profile-actions, .profile-nickname, .profile-status, .updated-notice { display: none !important; }',
'.popup-scroll { overflow-y: hidden; }',
'#uinfo { z-index: 10000; color: #bbb; padding: 5px; width: 417px; height: 500px; font-size: 12px; background-color: #303030; box-shadow: rgba(0, 0, 0, 0.12) 0px 0px 31px 13px; border: 1px solid #555; display: none; border-radius: 4px; }',
'#uinfo h4 { margin: 0 0 5px 0; padding: 1px 3px 3px 6px; border-radius: 2px; background-color: #5272af; color: #fff; font: normal 13px/15px "PT Sans Narrow"; padding: 3px 0 3px 5px; }',
// '#uinfo .cont { position: relative; height: 440px; overflow-y: auto; }',
'#uinfo .cont { position: relative; height: 474px; overflow-y: auto; }',
'#uinfo .cont img { max-height: 440px; max-width: 413px; }',
'#uinfo .cont .uadm { margin: 440px 0 0 0; display; none; }',
'.uadm > div:first-child { text-align: center; padding: 3px; margin: 0 5px 4px 5px; background: #B13E3E; color: #fff; border-radius: 2px; cursor: pointer; }',
'.uadm ul, .uadm li { padding: 0; list-style-type: none; }',
'.uadm ul { margin: 0px 0 13px 5px; }',
'.uadm li { margin: 0px 7px 3px 7px; }',
'.uadm li:first-child { background: #626263; color: #fff; padding: 3px 9px; border-radius: 2px; margin: 0 5px 8px 0; }',
'.uadm label { cursor: pointer; position: relative; bottom: 2px; left: 5px; font: normal 13px Arial; }',
'.uadm label:hover { color: #eee; }',
'#uinvite li { margin: 0; }',
'#uinvite li:first-child { margin: 0px 5px 9px 0px; background: #98435e; text-align: center; }',
'#uinvite li > span { display: inline-block; padding: 0px; text-align: center; margin: 0; width: 49%; background: #866f5f; color: #fff; cursor: pointer; padding: 3px 0 4px 0; border-radius: 2px; }',
'#uinvite li > span:last-child { margin-left: 3px; background: #6983cc; }',
'#uinvite .esc { color: #5d3232; display: inline-block; margin: 0 0 0 5px; }',
'#uinvite .esc:hover { color: #d6bdb7; display; none; }',
'.button-block .cbutton { width: 67px; margin: 0 0 3px 0; }',
'.button-block { text-align: center; }',
'.center { margin: auto; position: absolute; left: 0; right: 0; top: 0; bottom: 0; }',
'h4 > span:nth-child(1) { background: url("") no-repeat; margin: 1px 5px 0 0; float: right; cursor: pointer; width: 12px; height: 12px; padding: 0; }',
'h4 > span:active { margin-top: 2px; }',
'.header-date:before, .talk-load { border-radius: 2px !important; background: #333333; }',
'.talk-load { background: #464646; box-shadow: 0 0 0 1px rgba(0,0,0,0.05), 0 1px 8px 0 rgba(0,0,0,0.20); border: none; }',
'.popup-content { background: #303030; color: #bbb; box-shadow: rgba(0, 0, 0, 0.12) 0px 0px 31px 13px; border: 1px solid #555; border-radius: 4px; }',
'#settings-subscription, .settings-section, #room-settings .submit { background: none; }',
'.settings-section { border: none !important; padding: 0 20px; }',
'#settings-subscription, .userpic-highlight { box-shadow: none !important; }',
'.popup-content { margin: 31px 23px 23px 23px; }',
'.toolbar-settings:hover { animation: anim-sett 4s 1; }',
'@-webkit-keyframes anim-sett { 0% { -webkit-transform: rotate(0deg); } 100% { -webkit-transform: rotate(360deg); } }',
'.talk-content { overflow-y: auto; }',
'::-webkit-scrollbar { width: 15px; height: 15px;}' ,
'::-webkit-scrollbar-track { background-color: #444; box-shadow: -1px 0px 0 0 rgba(0, 0, 0, .8), inset -1px 0px 0 0 rgba(255, 255, 255, .1); }',
'::-webkit-scrollbar-button, ::-webkit-scrollbar-thumb { background-color: #646464; box-shadow: inset 1px 1px 0 0 rgba(255, 255, 255, .15); border-bottom: 1px solid #333; }',
'::-webkit-scrollbar-button { box-shadow: inset 1px 1px 0 0 rgba(255, 255, 255, .1); }',
'::-webkit-scrollbar-thumb:hover { background-color: #747474; box-shadow: inset 1px 1px 0 0 rgba(255, 255, 255, .20); }',
'::-webkit-scrollbar-button:start:hover, ::-webkit-scrollbar-button:end:hover { background-color: #747474; }',
'::-webkit-scrollbar-button:start { background: #646464 url(https://storage.googleapis.com/google-code-archive-downloads/v2/code.google.com/vallalarblogs/ui-icons_222222_256x240.png) 0px -17px no-repeat; }',
'::-webkit-scrollbar-button:end { background: #646464 url(https://storage.googleapis.com/google-code-archive-downloads/v2/code.google.com/vallalarblogs/ui-icons_222222_256x240.png) -65px -16px no-repeat; }',
'::-webkit-scrollbar-thumb { background: #646464 url("") 5px 50% no-repeat; }',
'::-webkit-scrollbar-thumb:active { background: #646464 url("") 4px 49.5% no-repeat; background-color: #646464; box-shadow: inset 1px 1px 0 0 rgba(255, 255, 255, .15); }',
'::selection { background-color: #bf8ec5; color: #fff; }'
];
/*
$(rules).each(function(k, v) {
css[0].insertRule(v, css.cssRules.length);
});
*/
$('head').append(css.html(rules.join('\n')));
$('body').append(cssFont.html(rulesFont.join('\n')));
$('.side-content').height($(window).height() - $('.talk-reply').height() - 71);
}//func
function profileShow() {
$.popup = function(selector, show, hide) {
var self = $(selector);
var elem = self.find('.popup-content').get(0) || self.get(0);
function clickout(event) {
if (!event.button && elem !== event.target && !$.contains(elem, event.target)) self.hide();
}
function escape(event) {
if (event.which === 27 || event.keyCode === 27) self.hide();
}
show = show || self.show;
hide = hide || self.hide;
self.show = function() {
document.addEventListener('touchstart', clickout, true);
// document.addEventListener('mousedown', clickout, true);
document.addEventListener('keydown', escape, true);
show.apply(self, arguments);
};
self.hide = function() {
document.removeEventListener('touchstart', clickout, true);
// document.removeEventListener('mousedown', clickout, true);
document.removeEventListener('keydown', escape, true);
hide.apply(self, arguments);
};
return self;
};
var
$popup = $.popup('#profile'),
$content = $popup.find('.popup-content'),
$scroller = $popup.find('.popup-scroll'),
$sections = $popup.find('.profile-section');
function getCenter(target) {
var
rect = target.getBoundingClientRect();
return {
top: Math.round((rect.top + rect.bottom) / 2),
left: Math.round((rect.left + rect.right) / 2)
};
}
function preloadPhoto(role, context) {
if(role.photo) {
var
img = new Image();
img.src = '/photos/' + role.photo;
img.onload = function() {
Profile.trigger('ready', role, img);
Profile.fit();
$popup.css({
left: context.left - $('#profile').width() - 20,
top: context.top
});
};
} else {
Profile.trigger('ready', role);
Profile.fit();
$popup.css({
left: context.left - $('#profile').width() - 20,
top: context.top
});
}
}
Profile.fit = function() {
if(!this.context || !this.context.top) return;
var
wh = window.innerHeight,
ch = $content.height(),
top = Math.min(this.context.top - 55, wh - ch - 25);
if(top < 15) {
$popup.css('top', 15);
$scroller.height(wh - 20).scrollTop(15 - top);
} else {
$popup.css('top', top);
$scroller.height('');
}
};
Profile.show = function(role, context, edit) {
var
// pm = $popup.find('#profile-moderate'),
me = Rooms.selected.isMy(role);
$sections.hide();
Profile.role = role;
Profile.context = context;
// Profile.profileModerate = pm.clone(true);
Profile.trigger(edit ? 'edit' : 'show', role, me);
if(!context.nickname) context.nickname = role.nickname;
$popup.find('#profile-moderate').hide();
$popup.show();
// Profile.fit();
if(role.message_id) {
Rest.roles
.get(role.role_id)
.done(function(data) {
$.extend(role, data);
preloadPhoto(role, context);
})
} else preloadPhoto(role, context);
};
}//func
function fixSmoothScroll() {
var
content = Talk.content.get(0),
scroller = $({}),
interrupted;
function setPosition(value) {
content.scrollTop = Math.round(value);
}
function getDuration(a, b) {
// return 10 + Math.abs(a - b) * 1;
return 0;
}
function isNear(element) {
return element && element.getBoundingClientRect().top - content.getBoundingClientRect().bottom < 10;
}
function scrollEnd() {
Talk.content.dequeue();
}
Talk.scrollFurther = function(node) {
this.content.queue(function(next) {
var
cur = content.scrollTop,
pos = content.scrollHeight - content.offsetHeight - 1;
if(pos > cur && !interrupted && node.parentNode && isNear(node)) {
scroller[0].position = cur;
scroller
.animate({position: pos}, {
complete: scrollEnd,
duration: getDuration(cur, pos),
step: setPosition
});
setTimeout(function() {
$('.talk-content').scrollTop($('.talk-content')[0].scrollHeight + 1000);
}, 10);
} else next();
});
};
}//func
function messageModify() {
var
renderSpeech = $.template(
'<div><div class="speech">' +
'<span class="speech-author" data-role="{role_id}">' +
//'<div class="userpic" style="background-image: url({userpicUrl});"></div>' +
'<span class="nickname" style="color: #{nickColor};">{nickname}</span>' +
'</span>' +
'</div></div>'),
renderRecipient = $.template('#recipient-template'),
renderMessage = $.template(
'<div><span class="message" style="color: #{messColor};" data-id="{message_id}"><span data-role="{role_id}" href="#" style="color: #{nickColor};" class="nickname">{nickname}</span>: {content}' +
// '<time class="msg-time" datetime="{created}">{time}</time>' +
'<span class="msg-text"></span>' +
'</span></div>'),
edit = $('<span class="msg-edit" title="Редактировать сообщение"></span>')[0];
function Message(data) {
var
created = new Date(data.created),
message = renderMessage({
message_id: data.message_id,
created: data.created,
content: Talk.format(data.content) || '…',
time: created.toHumanTime(),
messColor: Colors[data.nickColor][1],
role_id: data.role_id,
nickname: data.nickname,
nickColor: Colors[data.nickColor][0]
});
if(data.isMy) {
// if (created.daysAgo() < 2) message.find('.msg-text').append(edit.cloneNode(true));
} else if(data.mentionsMe) message.addClass('with-my-name');
this.timestamp = created.getTime();
this.date = created.toDateString();
this.node = message[0];
this.data = data;
}
Message.prototype.appendTo = function(parent, previous) {
var
sys = $(parent).find('.sys'),
snode = sys.length ? sys.parent()[0] : previous.node ? previous.node.parentNode : false;
if(oneSpeech(previous, this))
snode.appendChild(this.node).parentNode.appendChild($('<br>')[0]); else {
if(this.date !== previous.date) parent.appendChild(Talk.getDate(this.date).node);
appendSpeech(this.data, parent).appendChild(this.node);
}
return this;
};
function oneSpeech(a, b) {
return a.date === b.date && b.timestamp - a.timestamp < 3600000 * 3 && oneContext(a.data, b.data);
}
function oneContext(a, b) {
return a.role_id === b.role_id && a.nickname === b.nickname && a.recipient_role_id == b.recipient_role_id;
}
function appendSpeech(data, parent) {
var
speech = renderSpeech({
role_id: data.role_id,
nickname: data.nickname,
nickColor: Colors[data.nickColor][0],
userpicUrl: Userpics.getUrl(data)
}),
node = speech[0],
recipient_id = data.recipient_role_id;
if(recipient_id) {
var
recipient = renderRecipient({
role_id: recipient_id,
nickname: recipient_id === Room.myRole.role_id ? 'я' : data.recipient_nickname
});
speech.find('.speech-author').append(recipient);
speech.addClass('personal');
}
parent.appendChild(node);
$(node).find('.speech-author').remove();
return node;
}
Talk.createMessage = function(data) {
var
room = Rooms.selected,
role = getRole(data.role_id);
// uObj = getUserObj('role_id', data.role_id);
data.isMy = room.isMy(data);
// log(role);
// log(data.role_id);
// data.nickColor = uObj.nickColor || 0;
// data.nickColor = 0;
data.nickColor = role && role.nickColor ? role.nickColor : 0;
if(!data.isMy && data.mentions) data.mentionsMe = room.mentionsMe(data.mentions);
return new Message(data);
};
}//func
function filterForMe() {
var
toolbar = $('.header-toolbar'),
control = $('.filter-my');
Rooms.on('selected.ready', function(room) {
control.toggleClass('filter-my-selected', room.forMeOnly === true);
});
control.on('click', function() {
if(toolbar.data('wasDragged')) return;
var
room = Rooms.selected;
room.forMeOnly = !room.forMeOnly;
Talk.forMeOnly = room.forMeOnly;
control.toggleClass('filter-my-selected', Talk.forMeOnly);
Talk.content.addClass('talk-loading');
Talk.loadRecent();
});
}//func
function Moderator() {
// [не пускать] [пригласить]
// [выгнать] [назначить]
var
notToLet = $('<span class="moder-rej">Не пускать</span>'),
banRelease = $('<span class="esc">[ отменить ]</span>'),
banish = $('<span class="banish">Выгнать</span>'),
$request = $('#uinvite');
notToLet.click(function(e) {
Profile.role = Role;
Profile.send({
come_in: false
});
});
banish.click(function(e) {
//выгнать
Profile.role = Role;
Profile.send({
level: Profile.role.user_id ? 10 : 0,
come_in: false
});
});
banRelease.click(function(e) {
e.stopPropagation();
//отменить изгнание
Profile.role = Role;
Profile.send({
level: Rooms.selected.data.level,
come_in: null
});
});
//пригласить
$request.find('.moder-inv').on('click', function() {
Profile.role = Role;
Profile.send({
level: Rooms.selected.data.level,
come_in: null
});
});
Profile.on('moderated', function(state) {
var
room = Rooms.selected,
myLevel = room.myRole.level;
// log(Role);
Profile.role = Role;
notToLet = notToLet.clone(true);
banRelease = banRelease.clone(true);
banish = banish.clone(true);
//Role.level = 20 - member of room. Role.level = 0 - not member
// log(state);
//log(Rooms.selected.myRole.level);
switch(state) {
case 'banished':
$request.find('.moder-rej').unbind('click').html('<span>Не может войти</span>').append(banRelease);
$('#ulevel, #uignore').slideUp();
break;
case 'guest':
// if(Role.level >= 20) $request.hide(); else $request.show();
$request.find('.moder-rej').unbind('click').html(banish);
if(Role) checkedSet($('#ulevel input[type=radio]'), Role.level);
if(myLevel >= 50) $('#uignore').slideDown(); else $('#uignore').slideUp();
if(myLevel >= 70) {
if(myLevel < 80) $('#ulevel > li').eq(4).remove();
$('#ulevel').slideDown();
} else $('#ulevel').slideUp();
break;
case 'request':
$request.find('.moder-rej').replaceWith(notToLet);
break;
}//sw
});
}//func
function Admin() {
var
topic = $('#edit-room-topic'),
hash = $('#edit-room-hash'),
watched = $('#edit-room-watched'),
levels = $('.room-levels input'),
submit = $('#room-settings .cbutton.submit');
function validateHash(value, full) {
if(/[^a-zA-Z\d\-+]/.test(value)) return false;
if(full && value.length < 3) return false;
if(value.length > 32) return false;
return true;
}
function filterChanged(current, data) {
var
value,
empty = true,
changed = {};
for(var key in data) {
value = data[key];
if(value !== current[key]) {
changed[key] = value;
empty = false;
}
}
if(!empty) return changed;
}
function updateRoom(room) {
var
data = filterChanged(room.data, {
topic: topic.val(),
hash: hash.val(),
//searchable: searchable.prop('checked') ? 1 : 0,
watched: watched.prop('checked') ? 1 : 0,
level: Number(levels.filter(':checked').attr('value') || room.data.level)
});
if(!data) {
Settings.hide();
} else if (data.topic === '') {
topic.val().focus();
} else if (data.hash && !validateHash(data.hash, true)) {
hash.focus();
} else {
submit.prop('disabled', true);
Rest.rooms
.update(room.data.hash, data)
.always(function() {
submit.prop('disabled', false);
})
.done(function() {
Settings.hide();
})
.fail(showError);
}
}
function showError(error) {
if(error.status === 409) {
var
context = hash.parent();
context.append('<p class="error">Увы, этот адрес уже занят, выберите другой</p>');
hash.one('input', function() {
context.find('.error').remove();
});
}
}
submit.on('click', function() {
if(!this.disabled) updateRoom(Rooms.selected);
});
}//func
function init() {
var
i,
j,
u,
div,
uinfo = $('<div id="uinfo"><h4>Информация<span title="Закрыть"></span></h4>' +
'<div class="cont">' +
'<img class="center" />' +
'<div class="uadm">' +
'<div onclick="admToggle()">Администрирование</div>' +
'<div>' +
'<ul id="uignore"></ul>' +
'<ul id="ulevel"></ul>' +
'<ul id="uinvite"></ul>' +
'</div></div></div></div>');
$('body').append(uinfo);
/*
$.getStylesheet = function(href) {
var
$d = $.Deferred(),
$link = $('<link/>', {
rel: 'stylesheet',
type: 'text/css',
href: href
}).appendTo('head');
$d.resolve($link);
return $d.promise();
};
*/
(function($){
$.fn.extend({
center: function (options) {
var
options = $.extend({
inside: window,
transition: 0,
minX: 0,
minY: 0,
withScrolling: true,
vertical: true,
horizontal: true
}, options);
return this.each(function() {
var
props = { position: 'absolute' };
if(options.vertical) {
var
top = ($(options.inside).height() - $(this).outerHeight()) / 2;
if(options.withScrolling) top += $(options.inside).scrollTop() || 0;
top = (top > options.minY ? top : options.minY);
$.extend(props, {top: top + 'px'});
}
if(options.horizontal) {
var
left = ($(options.inside).width() - $(this).outerWidth()) / 2;
if(options.withScrolling) left += $(options.inside).scrollLeft() || 0;
left = (left > options.minX ? left : options.minX);
$.extend(props, {left: left + 'px'});
}
if(options.transition > 0) $(this).animate(props, options.transition); else $(this).css(props);
return $(this);
});
}
});
})(jQuery);
$(window).bind('resize', function() {
$('#uinfo').center({transition: 100});
});
$.getScript('https://code.jquery.com/ui/1.12.1/jquery-ui.min.js')
.done(function() {
uinfo.draggable({ handle: 'h4' })
.find('h4').dblclick(function() {
uinfo.hide().removeData();
})
.find('span:first').click(function() {
uinfo.hide().removeData();
});
uinfo.center();
});//done
div = $('<ul><li>Назначить статус пользователю</li></ul>');
j = 0;
for(i in uLevel) {
j++;
(function(key, j) {
u = $('<li><input' + (j == 1 ? ' checked' : '' ) + ' id="ul' + j + '" type="radio" name="ulev" value="' + i + '" /><label for="ul' + j + '">' + uLevel[i] + '</label></li>');
div.append(u);
})(i, j);
}//for
$('#ulevel').append(div.children());
div = $('<ul><li>Забанить пользователя</li></ul>');
j = 0;
for(i in uIgnored) {
j++;
(function(i) {
u = $('<li><input' + (j == 1 ? ' checked' : '' ) + ' id="ui' + j + '" type="radio" name="uign" value="' + i + '" /><label for="ui' + j + '">' + uIgnored[i] + '</label></li>');
div.append(u);
})(i);
}//for
$('#uignore').append(div.children());
div = $('<ul><li>Модерирование</li><li><span class="moder-rej">Не пускать</span><span class="moder-inv">Пригласить</span></li></ul>');
$('#uinvite').append(div.children());
$('.uadm > div:nth-child(2)').append($('<div class="button-block"><a title="Применить изменения" class="cbutton" onclick="userStatusEdit()"><span>Применить</span></a><div>'));
$('.reply-send').html(
'<a title="Отправить сообщение" id="go" tabindex="2" class="cbutton" onclick="sendMess(\'nosec\');"><span>Тадам</span></a>' +
'<a title="Шепнуть приватно" id="pvt" tabindex="3" class="cbutton" onclick="sendMess(\'sec\');"><span>Приват</span></a>' +
'<a title="Стереть текст сообщения" tabindex="4" class="cbutton" onclick="$(\'#input\').val(\'\').focus();"><span>Clear</span></a>').unbind();
$('.reply-field textarea').replaceWith($('<input tabindex="1" id="input" class="" type="text" placeholder="текст сообщения" maxlength="" />'));
$('.toolbar-filter.filter-my').replaceWith($('<a id="withme" title="Сообщения: от меня и для меня" class="cbutton filter-my-selected filter-my"><span>Мне и от меня</span></a>'));
var
rcont = $('#room-settings .popup-content');
rcont.find('.alarm-off .button').replaceWith($('<a title="" class="button cbutton"><span>Объявить тревогу</span></a>'));
rcont.find('.submit button').replaceWith($('<a title="" class="cbutton submit"><span>Готово</span></a>'));
$(document)
.click(function() {
$('.reply-form .cbutton').removeClass('button-down');
})
.keyup(function(e) {
switch(e.keyCode) {
//Tab
case 9:
if(!$('.reply-form :focus').length) $('#go').focus();
$('.reply-form .cbutton').removeClass('button-down');
$('.reply-form :focus').addClass('button-down');
e.preventDefault();
break;
//Esc
case 27:
$('#input').val('');
$('.reply-form .cbutton').blur();
$('.reply-form .cbutton').removeClass('button-down');
input.focus();
e.preventDefault();
break;
case 13:
switch($('.reply-form :focus').attr('id')) {
case 'go': return sendMess('nosec');
case 'pvt': return sendMess('sec');
}//sw
e.preventDefault();
break;
}//sw
});
$('#input')
.keydown(function(event) {
if(/(10|13)/.test(event.keyCode)) sendMess('nosec');
})
.keyup(function(e) {
switch(e.keyCode) {
case 38:
historyMessKbd(-1);
e.preventDefault();
break;
case 40:
historyMessKbd(1);
e.preventDefault();
break;
}//sw
})
.on('wheel', function(e) {
historyMessKbd(e.originalEvent.deltaY);
});
window.userStatusEdit = function() {
var
lev = $('#ulevel :radio[name=ulev]').filter(':checked').val(),
exp = $('#uignore :radio[name=uign]').filter(':checked').val();
if(exp !== null) exp *= 60;
Profile.role = Role;
if(exp == 0) Profile.send({ignored: false}); else {
Profile.send({ignored: true});
setTimeout(function() {Profile.send({expired: exp}); }, 0);
}
if(Role.level != lev) setTimeout(function() {Profile.send({level: lev}); }, 0);
$('#uinfo').removeData().fadeOut(150);
}//func
window.admToggle = function() {
$('.uadm > div:nth-child(2)').slideToggle(200);
$('.cont').animate({ scrollTop: $('.cont')[0].scrollHeight }, 200);
}//func
window.checkedSet = function(arr, val) {
arr.each(function(k, v) {
if($(v).val() == val) {
$(v).prop({checked: true});
return false;
}
});
}//func
}//func
fixSmoothScroll();
fixCSS();
headerModify();
messageModify();
profileShow();
userOnlineModify();
messageSendModify();
init();
filterForMe();
Moderator();
Admin();
})();
//}//if match chat