您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Скрипт изменяет внешний вид чата и отключает плавный автоскролл
// ==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