// ==UserScript==
// @name MRAS Mobile Reactor advanced script
// @description Представься, мразь! (с)
// @author Rus-Ivan
// @namespace m.joyreactor.cc
// @version 1.2.2
// @include *://m.joyreactor.cc/*
// @require https://greasemonkey.github.io/gm4-polyfill/gm4-polyfill.js
// @grant GM_xmlhttpRequest
// @grant GM_getValue
// @grant GM_setValue
// @grant GM.xmlHttpRequest
// @grant GM.getValue
// @grant GM.setValue
// @grant unsafeWindow
// @connect m.joyreactor.cc
// @connect joyreactor.cc
// @connect reactor.cc
// @icon http://joyreactor.cc/favicon.ico
// @run-at document-end
// ==/UserScript==
/*
========================
Для чего?
Облегчает просмотр постов на m.joyreactor.cc
Часто сижу на m.joyreactor.cc и приходится
1. открывать пост, чтобы
+ посмотреть гифку
+ почитать комментарии
2. открывать пост на основном хосте, чтобы
+ добавить в избранное
+ оценить комментарий
========================
Что делает этот скрипт?
1. при нажатии на кнопку/ссылку "Комментарии":
+ открывает комментарии
+ делает картинки/гифки/гиф-видео кликабельными (прим., чтобы загрузить гифки/гиф-видео, кликните на постер)
+ показывает панель управления гиф-видео
2. есть кнопка "Добавить в избранное"
3. возможность оценивать комментарии
========================
Дополнительные плюшки:
1. убирает редиректы с ссылок
2. сохраняет избранные посты в базе данных
========================
P.S. Этот скрипт я писал чисто для личного пользования,
если людям он покажется полезным, то возможно добавлю в него
какие-нибудь дополнительные плюшки
*/
function consoleLog(){window['console']['log'].apply(this, arguments);}
consoleLog("====== start " + GM.info.script.name + " v" + GM.info.script.version + "\n" + GM.info.script.description);
var DEBUG = false;
var FAVKEY = 'favorite-key';
(async function(){
'use strict';
try{
var uWindow = unsafeWindow;
consoleLog("window: ", uWindow);
var userFavorite = await GM.getValue( FAVKEY, "[]" );
userFavorite = JSON.parse(userFavorite);
syncFavorite();
removeRedirect();
makeVotableComments();
activateComments();
activateFavorite();
addNewStyle();
consoleLog("======== end " + GM.info.script.name + " v" + GM.info.script.version);
function syncFavorite()
{
var html = '<div class="" style="margin: 5px;">' +
'<div id="user_id">user_id := ' + uWindow.user_id + '</div>' +
'<div id="token">token := ' + uWindow.token + '</div></div>';
makePopup({
html: html,
attr: {
id: 'user-info',
style: 'position: fixed; top: 10px; right: 10px; background-color: #272727; color: #d0d0d0;' +
'text-align: center; z-index: 10; cursor: pointer; border-radius: 5px; border-color: #d0d0d0;' +
'border-style: solid; border-width: thin;',
},
'event': {
type: 'click',
handler: function(event){$('#' + this.id).style.display = 'none';},
},
});
}
function removeRedirect( doc )
{
var links = $$('a', doc), link;
consoleLog("links.length := ", links.length);
for( var i = 0, len = links.length; i < len; ++i )
{
link = links[i];
if( link['hostname'].indexOf('reactor') != -1 && link['pathname'].indexOf('redirect') != -1 )
link.href = decodeURIComponent(link.search.slice(5));
}
}
function makeVotableComments( doc )
{
var commentList = $$('.comment_rating'),
voteHTML = '<div class="vote-plus" title="голосовать за"></div><div class="vote-minus" title="голосовать против"></div>';
consoleLog("commentList.length := ", commentList.length);
for( var i = 0, len = commentList.length, commentRating; i < len; ++i )
{
commentRating = commentList[i];
if( !commentRating.querySelector('.vote-plus') )
commentRating.innerHTML += voteHTML;
if( !commentRating.classList.contains('comment-vote-active') )
{
commentRating.addEventListener('click', handleCommentVoteEvent, false);
commentRating.classList.add('comment-vote-active');
}
}
}
function handleCommentVoteEvent(event)
{
var t = event.target, act;
if( t.classList.contains('vote-plus') )
act = 'plus';
else if( t.classList.contains('vote-minus') )
act = 'minus';
if( !act )
return;
var commentId = t.parentNode.getAttribute('comment_id'),
data = 'token=' + uWindow.token;
GM.xmlHttpRequest({
url: 'http://joyreactor.cc/comment_vote/add/' + commentId + '/' + act + '?' + data,
method: 'GET',
context: {'commentId': commentId, 'act': act},
headers: {
'X-Requested-With': 'XMLHttpRequest',
'Referer': 'http://joyreactor.cc' + window.location.pathname,
},
onload: function(xhr){
try{
var cntx = xhr.context,
regEx = /\-?\d+(\.\d+)?/,
commentR = $('[comment_id="' + cntx.commentId + '"]'),
html = commentR.innerHTML,
voteChangeAct = (cntx.act == 'plus' ? 'minus': 'plus');
commentR.innerHTML = html.replace( regEx, xhr.responseText.match(regEx)[0] );
if( !$('.vote-change', commentR) )
$('.vote-' + voteChangeAct, commentR).classList.add('vote-change');
if( DEBUG )
{
consoleLog("comment vote [" + act + "]");
consoleLog("xhr.status: ", xhr.status, xhr.statusText);
consoleLog("xhr.response: ", xhr.response);
}
}catch(e){console.error(e);}
},
});
}
function activateFavorite()
{
// начало создания кнопки "Добавить в избранное"
var postList = $$('.postContainer');
consoleLog("postList.length := ", postList.length);
for( var i = 0, len = postList.length; i < len; ++i )
makeFavorite( postList[i] );
}
function makeFavorite( postContainer )
{
var postId, postRating, fav;
postRating = $('.post_rating', postContainer);
if( !postRating )
return;
postId = postContainer.id.match(/\d+/)[0];
fav = document.createElement('div');
fav.setAttribute('class', 'favorite_link');
fav.setAttribute('data-post-id', postId);
fav.setAttribute('title', 'Добавить в избранное');
postRating.parentNode.insertBefore( fav, postRating );
fav.addEventListener('click', handleFavoriteEvent, false);
consoleLog("post_rating: ", postRating, postRating.innerHTML.toString().trim());
if( isFavorite(postId) )
fav.classList.add('favorite');
}
function handleFavoriteEvent(event)
{
var t = this,
token = uWindow.token,
data = 'token=' + token + '&rand=' + Math.floor(1e4*Math.random()),
postId = t.getAttribute('data-post-id'),
act = 'create', url;
if( t.classList.contains('favorite') )
act = 'delete';
url = 'http://joyreactor.cc/favorite/' + act + '/' + postId + '?' + data;
// отправка xml-http запроса на создание/удаление [из] избранного
GM.xmlHttpRequest({
url: url,
method: 'GET',
context: {'t': t, 'token': token, 'data': data, 'act': act, 'postId': postId, 'url': url},
headers: {
'X-Requested-With': 'XMLHttpRequest',
'Referer': 'http://joyreactor.cc/post/' + postId,
},
onload: function(xhr){
var cntx = xhr.context;
if( cntx.t.classList.contains('favorite') )
{
cntx.t.classList.remove('favorite');
cntx.t.setAttribute('title', 'Добавить в избранное');
removeFromFavoriteStorage(cntx.postId);
}else{
cntx.t.classList.add('favorite');
cntx.t.setAttribute('title', 'Удалить из избранного');
addToFavoriteStorage(cntx.postId);
}
saveFavoriteStorage();
if( DEBUG )
{
consoleLog("-------------");
consoleLog("favorite[" + cntx.act + "] post-id: ", cntx.postId);
consoleLog("xhr.status : ", xhr.status, xhr.statusText);
consoleLog("xhr.response: ", xhr.response);
consoleLog("token: ", cntx.token);
consoleLog("data : ", cntx.data);
consoleLog("url : ", cntx.url);
}
},
});
}
function activateComments()
{
// начало создания кнопки "Комментарии"
var links = $$('.article .comments > a');
for( var i = 0, len = links.length; i < len; ++i )
links[i].addEventListener('click', handleCommentListEvent, false);
}
function handleCommentListEvent(event)
{
try{
// если зажать Ctrl, то ссылка откроется как обычно (в новом окне)
if( event.ctrlKey )
return;
// простой click на ссылку
event.preventDefault();
var t = event.target, ufoot, commentList;
if( t.tagName !== 'A' )
{
console.error("[handleCommentListEvent] invalid link: ", t);
return;
}
ufoot = t.parentNode.parentNode;
commentList = $('.comment_list_post', ufoot);
if( commentList && t.classList.contains('comment-list-opened') )
{
// если комментарии были открыты - то скрыть их
t.classList.remove('comment-list-opened');
commentList.style.display = 'none';
return;
}
else if( commentList )
{
// если комментарии были скрыты (см. выше), то показать их
commentList.style.display = 'block';
t.classList.add('comment-list-opened');
}
// загружает комментарии, делает картинки кликабельными
openCommentList( t );
}catch(e){console.error(e);}
}
function openCommentList( link )
{
if( !link || link.tagName !== 'A' )
{
console.error("[openCommentList] invalid link: ", link);
return;
}
GM.xmlHttpRequest({
url: link.href,
method: 'GET',
context: {
'link': link,
},
onload: setCommentList,
});
}
function setCommentList( xhr )
{
try{
if( xhr.status != 200 )
{
console.error("[setComments] xhr.status: ", xhr.status, xhr.statusText );
return;
}
var doc = document.implementation.createHTMLDocument("");
doc.documentElement.innerHTML = xhr.response;
removeRedirect(doc);
var xhrCommentList = $('.comment_list_post', doc),
ufoot = xhr.context.link.parentNode.parentNode,
commentList = $('.comment_list_post', ufoot);
var xhrImageList = $$('.image', doc),
imageList = $$('.image', ufoot.parentNode);
// создает комментарии
if( commentList )
commentList.innerHTML = xhrCommentList.innerHTML;
else
ufoot.appendChild(xhrCommentList);
makeVotableComments();
xhr.context.link.classList.add('comment-list-opened');
// создает кликабельную картинку в ленте на m.joyreactor.cc
// если это гифка/гиф-видео, то для ее загрузки нужно кликнуть на постер
for( var i = 0, xhrImg, img; i < imageList.length && i < xhrImageList.length; ++i )
{
xhrImg = xhrImageList[i];
img = imageList[i];
if( $('iframe', xhrImg) )
continue;
else if( $('.video_gif_holder', xhrImg) )
{
// код для гифок/видео-гифок
$('a.attribute_preview', img).addEventListener('click',
makeGifImageHandler(xhrImg, img), false);
}else
img.innerHTML = xhrImg.innerHTML;
}
}catch(err){console.error(err);}
}
function makeGifImageHandler( elm, img )
{
var video = $('video', elm);
if( video )
$('img', img).src = decodeURIComponent(video.getAttribute('poster'));
function handler(event)
{
try{
if( event.ctrlKey )
return;
event.preventDefault();
if( video )
video.setAttribute('controls', '');
this.parentNode.innerHTML = elm.innerHTML;
}catch(e){console.error(e);}
}
return handler;
}
async function saveFavoriteStorage()
{
GM.setValue( FAVKEY, JSON.stringify(userFavorite) );
}
function addToFavoriteStorage( postId )
{
if( userFavorite.indexOf(postId) == -1 )
userFavorite.push(postId);
}
function removeFromFavoriteStorage( postId )
{
var idx = userFavorite.indexOf(postId);
if( idx != -1 )
userFavorite.splice(idx, 1);
}
function isFavorite( postId )
{
return userFavorite.indexOf(postId) != -1;
}
function addStyle( cssClass, id )
{
var style = document.createElement('style');
style.type = 'text/css';
style.innerHTML = cssClass;
if( id !== undefined )
style.setAttribute('id', id);
var head = document.head || $('head');
return head.appendChild(style);
}
function addNewStyle()
{
addGifStyle('video-gif-css');
addFavoriteStyle('favorite-css');
addCommentVoteStyle('comment-css');
}
function addGifStyle( id )
{
addStyle(`
.video_gif_source:hover {
opacity: 1;
}
.video_gif_source[href*=".gif"] {
display: none !important;
}
.video_gif_source {
display: inline-block !important;
opacity: 0.6;
background: rgba(204, 204, 204, 0.6);
}
`, id );
}
function addFavoriteStyle( id )
{
var starN = "";
var starF = "";
addStyle(`
.favorite_link {
background: url(${starN});
cursor: pointer;
display: inline-block;
width: 32px;
height: 32px;
vertical-align: middle;
margin: 12px 16px 0 0;
float: right;
}
.post_rating {
margin-right: 16px;
}
.favorite_link:hover {
background: url(${starF});
transform: scale(1.4);
}
div.favorite {
background: url(${starF});
}
div.favorite:hover {
background: url(${starN});
transform: scale(1.4);
}
`, id );
}
function addCommentVoteStyle( id )
{
addStyle(`
span.comment_rating div.vote-plus {
background-position: 0 -120px;
}
span.comment_rating div.vote-minus {
background-position: -40px -120px;
}
.comment_rating div.vote-plus,
.comment_rating div.vote-minus {
width: 30px;
height: 30px;
line-height: 30px;
margin: 0 0 0 3px;
background: url(http://img0.joyreactor.cc/images/icon_smiles.png) no-repeat 0 -120px;
cursor: pointer;
display: inline-block;
vertical-align: middle;
/*transform: scale(0.75);*/
}
div.vote-change {
opacity: 0.5;
}
`, id);
}
function $( str, doc )
{
doc = doc || document;
return doc.querySelector(str);
}
function $$( str, doc )
{
doc = doc || document;
return doc.querySelectorAll(str);
}
function makePopup( prop )
{
var wnd = $('#' + prop.attr.id);
if( wnd )
{
wnd.style.display = 'block';
return wnd;
}
wnd = document.createElement('div');
for( var key in prop.attr )
wnd.setAttribute( key, prop.attr[key] );
wnd.innerHTML = prop.html || '';
$('body').appendChild(wnd);
if( prop.event )
wnd.addEventListener(prop.event.type, prop.event.handler, false);
return wnd;
}
function hide( str, doc )
{
var el = $(str, doc);
if( el )
el.style.display = 'none';
}
function show( str, doc )
{
var el = $(str, doc);
if( el )
el.style.display = 'block';
}
}catch(e){console.error(e);}
})();