// ==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) // @downloadURL https://update.greasyfork.icu/scripts/37992/MRAS%20Mobile%20Reactor%20advanced%20script.user.js // @updateURL https://update.greasyfork.icu/scripts/37992/MRAS%20Mobile%20Reactor%20advanced%20script.meta.js // ==/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 = '
' + '
user_id := ' + uWindow.user_id + '
' + '
token := ' + uWindow.token + '
' + '
favorite sync
'; makePopup({ html: html, attr: { id: 'user-info', 'class': 'popup-window', }, 'next': { html: '
<<
', 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 = '
'; 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);} })();