// ==UserScript==
// @name MRAS Mobile Reactor advanced script
// @description Представься, мразь! (с)
// @author Rus-Ivan
// @namespace m.joyreactor.cc
// @version 1.3.1
// @include *://m.joyreactor.cc/*
// @include *://m.reactor.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
// @license MIT
// @copyright 2018, Rus-Ivan (https://github.com/Rus-Ivan)
// ==/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);
var syncInProgress = false;
startSync();
removeRedirect();
makeVotableComments();
activateComments();
activateFavorite();
addNewStyle();
consoleLog("======== end " + GM.info.script.name + " v" + GM.info.script.version);
function startSync()
{
var html = '<div class="user-info-content" style="margin: 5px;">' +
'<div id="user_id">user_id := ' + uWindow.user_id + '</div>' +
'<div id="token">token := ' + uWindow.token + '</div>' +
'<div id="sync_fav">favorite sync</div></div>';
makePopup({
html: html,
attr: {
id: 'user-info',
'class': 'popup-window',
},
'next': {
html: '<div><span style="padding: 0 2px;"><<</span></div>',
attr: {
id: 'user-info-button',
'class': 'popup-window',
},
'event': {
type: 'click',
handler: function(event){
this.style.display = 'none';
},
},
'next': {
attr: {
id: 'user-info',
},
},
},
'event': {
type: 'click',
handler: function(event){
var t = event.target,
that = this;
that.style.zIndex = 15;
setTimeout(function(){that.style.display = 'none';}, 200);
if(t.id == 'sync_fav' )
syncFavorite();
},
},
});
syncFavorite();
}
async function syncFavorite()
{
try{
if( syncInProgress )
return;
var user_name = uWindow.user_name || getUserName();
consoleLog("user_name: ", user_name);
if( !user_name )
return;
var url = 'http://joyreactor.cc/user/' + user_name + '/favorite',
el = $('#sync_fav');
var lastPage = await GM.getValue('sync-favorite', -1);
lastPage = parseInt(lastPage, 10);
if( lastPage > 0 )
url += '/' + lastPage;
el.setAttribute('data-last-page', lastPage);
el.setAttribute('title', url);
el.setAttribute('data-user-favorite', url);
el.setAttribute('data-user-name', user_name);
el.innerHTML = 'favorite sync: ' + user_name + ' [' + lastPage + ']';
uWindow.user_name = user_name;
var syncComplete = await GM.getValue('sync-complete', false);
el.addEventListener('click', function(event){
if( event.ctrlKey )
window.open(this.getAttribute('data-user-favorite'));
}, false);
if( syncComplete )
{
el.parentNode.click();
return;
}
syncInProgress = true;
GM.xmlHttpRequest({
url: url,
method: 'GET',
Referer: 'http://joyreactor.cc',
context: {'url': url, count: 1, maxCount: 20},
onload: ajaxFavorite,
});
}catch(e){console.error(e);}
}
async function ajaxFavorite(xhr)
{
try{
if( xhr.status != 200 )
{
console.error("Error: xhr.status = ", xhr.status, xhr.statusText);
console.error("xhr: ", xhr);
return;
}
var doc = document.implementation.createHTMLDocument(""),
cntx = xhr.context, res;
doc.documentElement.innerHTML = xhr.response;
res = addPostsToFavorite(doc);
saveFavoriteStorage();
var page = getLocation(cntx.url, 'pathname').match(/\/[^\/]*$/)[0],
next_url = null;
page = /\d+/.test(page) ? parseInt(page.match(/\d+/)[0], 10) : -1;
consoleLog("--------------");
consoleLog("favorite page: ", page);
consoleLog("iter: ", cntx.count, "/", cntx.maxCount);
consoleLog("added: ", res);
var sync_fav = $('#sync_fav'),
usr_name = sync_fav.getAttribute('data-user-name');
sync_fav.innerHTML = 'favorite sync: ' + usr_name + ' [' + page + ']';
if( page > 0 )
await GM.setValue('sync-favorite', page);
next_url = $('.next', doc);
if( (res == 0 && page == 1) || cntx.count >= cntx.maxCount || !next_url )
{
if( page == 1 && res == 0 )
{
sync_fav.click();
GM.setValue('sync-complete', true);
}
syncInProgress = false;
return;
}
next_url = 'http://joyreactor.cc' + next_url.pathname;
setTimeout(function(){
GM.xmlHttpRequest({
url: next_url,
method: 'GET',
Referer: 'http://joyreactor.cc',
context: {url: next_url, count: cntx.count + 1, maxCount: cntx.maxCount},
onload: ajaxFavorite,
});
}, 2000 + getRandom(500, 1000) + getRandom(500, 900));
}catch(e){console.error(e);}
}
function getRandom( min, max )
{
return Math.floor(Math.random() * (max - min) + min);
}
function addPostsToFavorite(doc)
{
doc = doc || document;
var posts = $$('.postContainer', doc), count = 0;
for( var i = 0, post_id; i < posts.length; ++i )
{
post_id = posts[i].id.slice(13);
if( !isFavorite(post_id) )
{
++count;
addToFavoriteStorage(post_id);
}
}
return count;
}
function getUserName()
{
var spans = $$('span');
for( var i = 0, len = spans.length, span; i < len; ++i )
{
span = spans[i];
if( span.innerHTML.toString().indexOf('Привет') != -1 )
return span.innerHTML.replace(/Привет\,/i, '').trim().replace(/\s/g, '+');
}
return null;
}
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');
//addDarkStyle();
//addDarkPopupStyle();
addLightPopupStyle();
}
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 addDarkStyle( id )
{
addStyle(`
a ,
div.uhead_nick a {
color: #b1cbf7 !important;
}
a:hover ,
div.uhead_nick a:hover {
color: #d1ebf7 !important;
}
.post_content span {
color: #d2d2d2 !important;*/
/*background-color: #626f61 !important;*/
/*color: #f6f7b9 !important;*/
}
.submenuitem a, .article .comments a {
color: #FAF68C !important;
/*text-shadow: 0 1px 1px #c04e03 !important;*/
}
.taglist a {
color: #d2d2d2 !important;
}
.m_pagination a {
color: #f8c010 !important;
}
.m_pagination a:hover {
color: #F44336 !important;
}
`, id);
}
function addDarkPopupStyle( id )
{
addStyle(`
.popup-window {
position: fixed;
top: 10px;
right: 20px;
background-color: #272727;
color: #d0d0d0;
text-align: center;
z-index: 10;
font-size: 14px;
cursor: pointer;
border-radius: 5px;
border-color: #d0d0d0;
border-style: solid;
border-width: thin;
}
`, id);
}
function addLightPopupStyle( id )
{
addStyle(`
.popup-window {
position: fixed;
top: 10px;
right: 20px;
background-color: #d78917;
color: #faf68c;
text-shadow: 0 1px 1px #c04e03;
text-align: center;
font-size: 14px;
z-index: 10;
/*cursor: pointer;*/
border-radius: 5px;
border-color: #c04e03;
border-style: solid;
border-width: thin;
}
.user-info-content {
margin: 5px;
}
#sync_fav ,
#user-info-button {
cursor: pointer;
}
#sync_fav:hover {
color: #ffffd6;
}
`, id);
}
function $( str, doc )
{
doc = doc || document;
return doc.querySelector(str);
}
function $$( str, doc )
{
doc = doc || document;
return doc.querySelectorAll(str);
}
function makePopup( prop )
{
if( !prop )
return null;
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);
var f = prop.event,
n = prop.next,
callback = function(event){
f.handler.call(this, event);
makePopup(n);
};
if( f )
wnd.addEventListener(f.type, callback, 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';
}
function getLocation( href, prop )
{
if( !href )
return null;
var a = document.createElement('a');
a.href = href;
return a[prop];
}
}catch(e){console.error(e);}
})();