GW2WvW.net PvF Expansion Pack

Implements an "ignore user" function and other improvements on the gw2wvw.net forum.

当前为 2015-02-22 提交的版本,查看 最新版本

// ==UserScript==
// @name         GW2WvW.net PvF Expansion Pack
// @namespace    http://www.gw2wvw.net/
// @version      0.2
// @description  Implements an "ignore user" function and other improvements on the gw2wvw.net forum.
// @author       Anvil Pants
// @include      http://www.gw2wvw.net/*
// @grant        metadata
// @grant        GM_getValue
// @grant        GM_setValue
// @run-at       document-end
// @require      http://ajax.googleapis.com/ajax/libs/jquery/1.11.2/jquery.js
// ==/UserScript==

pvfExpansion = new Object();
pvfExpansion.var = new Object();
pvfExpansion.var.start = new Date().getTime();

pvfExpansion.getImage = function(img) {
    var images = [];
    images['plus'] = '';
    images['minus'] = '';
    return images[img];
}

// http://stackoverflow.com/a/21789375
pvfExpansion.heredoc = function(f) {
    return f.toString().match(/\/\*\s*([\s\S]*?)\s*\*\//m)[1].replace(/(\/\*[\s\S]*?\*) \//g, '$1/');
};

// http://stackoverflow.com/a/10192255
pvfExpansion.uniqueArrayEntries = function(array) {
    return $.grep(array, function(el, index) {
        return index === $.inArray(el, array);
    });
}

pvfExpansion.var.css = pvfExpansion.heredoc(function() {/*
body {
  background: #080808 !important;
  margin: 0 !important; 
  padding: 0 !important; 
}

#main, body.overlay, #page, #header, #page-wrapper, #content, #footer-wrapper {
  background-color: transparent !important;
}

#page-wrapper {
  width: auto !important;
  margin: 0 !important;
  padding: 0 3% 0 3% !important;
}

#main-wrapper {
  width: auto !important;
  margin: 0 !important;
  padding: 0 !important;
}

#main {
  margin: 10px 0 0 0 !important;
  padding: 0 !important;
}

.node h2 {
  border: none !important;
  margin: 0 0 1em 0 !important;
  padding: 0 !important;
}

.node,
.panel-pane,
.pane-content {
  background: none !important;
}

.node ul.rate-fivestar-processed {
  padding-left: 0 !important;
}

.node .field-type-taxonomy-term-reference .field-items .field-item:first-child {
  margin-left: 0 !important;
}

.view-header a:not(:hover),
.tabs li a:not(:hover),
.af-button-large:not(:hover),
ul.links a:not(:hover),
ul.pager a:not(:hover) {
  background: none !important;
}

table.sticky-header {
  background: #222 !important;
}

.messages.status {
  background: #511 !important;
  border-color: #922 !important;
  color: #ccc !important;
}

.view-content .views-row,
.view-content .views-table,
.view-content .views-table td,
.view-content .views-table th,
.view-content .views-table tr {
  background: none !important;
}

.view-content .views-row .content {
  clear: both !important;
}

.view-content .views-table{
  cell-spacing: 0 !important;
}

.view-content .views-table th,
.view-content .views-table tr:not(.views-row-last):not(:last-child) {
  border-bottom: 1px solid #555 !important;
}

.view-userposts li.views-row {
  background: none !important;
  list-style: none !important;
  margin: 1em 0 2em 0 !important;
}


/**
 * Forms
 * /

form fieldset {
  border: none !important;
}

input, select {
  -webkit-filter: invert(100%) grayscale(100%) !important;
  filter: invert(100%) grayscale(100%) !important;
}

.filter-wrapper {
  background-color: transparent !important;
}


/**
 * Avatars
 * /

.user-picture > a > img,
.author-pane-general > .picture > img {
  background: #000 !important;
  border: 2px solid #000 !important; 
  border-radius: 10px !important; 
  padding: 0 !important; 
  margin: 2px 10px !important; 
}


/**
 * Usernames
 * /

.author-name {
  width: 130px !important;
}
.username {
  line-height: 125% !important;
  word-break: break-all !important;
  word-break: break-word !important;
  -moz-hyphens: auto !important;
  -webkit-hyphens: auto !important;
  hyphens: auto !important;
}


/**
 * Front page
 * /

body.front .panel-pane,
body.front .pane-content,
body.front .views-row,
body.front table,
body.front table td {
  background: none !important;
  border-color: transparent !important;
}

/**
 * Front page videos
 * /

body.front table.cols-2 {
  border-collapse: separate !important;
  border-spacing: 10px !important;
}

body.front table.cols-2 td {
  position: relative !important;
  background: #000 !important;
  -webkit-border-radius: 15px !important;
  -moz-border-radius: 15px !important;
  border-radius: 15px !important;
  overflow-y: hidden !important;
  width: 50% !important;
  vertical-align: bottom !important;
}

body.front table.cols-2 td .views-field-title {
  position: absolute !important;
  z-index: 2 !important;
  top: 0 !important;
  left: 0 !important;
  right: 0 !important;
  background: linear-gradient(to bottom, rgba(0,0,0,1) 0%, rgba(0,0,0,0.9) 30%, rgba(0,0,0,0.7) 60%, rgba(0,0,0,0) 100%) !important;
  height: 100% !important;
  padding: 0 !important;
}

body.front table.cols-2 td:not(:hover) .views-field-title {
  visibility: hidden !important;
}

body.front table.cols-2 td .views-field-title a {
  display: block !important;
  height: 100% !important;
  padding: .5em .5em 1.5em .5em !important;
  margin: 0 !important;
  text-decoration: none !important;
}

body.front table.cols-2 td .views-field-name,
body.front table.cols-2 td .views-field-term-node-tid {
  position: relative !important;
  z-index: 3 !important;
}

body.front .views-field-field-youtube-video a {
  /**
   * force aspect ratio on youtube thumbnails
   * http://www.sitepoint.com/maintain-image-aspect-ratios-responsive-web-design/
   * /
  display: block !important;
  width: 100% !important;
  position: relative !important;
  height: 0 !important;
  padding: 56.25% 0 0 !important;
  margin: 0 0 1em 0 !important;
  overflow: hidden !important;
}

body.front .views-field-field-youtube-video img {
  position: absolute !important;
  display: block !important;
  max-width: 100% !important;
  max-height: 100% !important;
  left: 0 !important;
  right: 0 !important;
  top: 0 !important;
  bottom: 0 !important;
  margin: auto !important;
}


/**
 * Forum tables
 * /

.forum-table-wrap,
.forum-table tr,
.forum-table td {
  background: none !important;
}

.forum-table tr:not(.views-row-last):not(:last-child) {
  border-bottom: 1px solid #222 !important;
}

.forum-table tr:hover {
  background: #222 !important;
}

.forum-table-topics td.views-field-last-updated {
  width: auto !important;
}


/**
 * Forum posts
 * /

.quote-msg {
  background: none !important;
  border: 0 !important;
  border-left: 2px solid #555 !important;
  border-radius: 10px !important;
  color: rgb(102, 102, 102) !important;
  font-size: 11pt !important;
  padding: 5px 0 5px 20px !important;
  margin: 1.5em 0 !important;
}

.quote-author {
  display: block !important;
  margin: 0 !important;
  padding: 0 !important;
}

.author-pane-line {
  font-size: .9em !important;
}

.author-pane-line.author-name {
  font-size: 1.5em !important;
}

.author-signature, .post-edited {
  border: none !important; 
  color: #666 !important;
  font-size: .85em !important; 
  margin: 0 0 0 30px !important; 
  padding: 0 !important;
}

.author-signature > blockquote {
  border: none !important; 
  margin: 0 !important; 
  padding: 10px 0 !important;
}

.forum-post {
  background: none !important;
  margin: 0 0 30px 0 !important;
  padding: 10px 20px !important;
}

.forum-post-panel-main {
  border: 0 !important;
}

.forum-post-content {
  padding: .2em 0 1em 2em !important;
}

.forum-post-content, .author-signature {
  /**
   * Force line wrap
   * /
  word-break: break-all !important;
  word-break: break-word !important;
  -moz-hyphens: auto !important;
  -webkit-hyphens: auto !important;
  hyphens: auto !important;
}
                               
.forum-post-content p {
  margin: .75em 0 !important;
}

.forum-post-content > .field-type-text-with-summary {
  /**
   * float clearing caused excessive whitespace in first post of forum thread
   * /
  clear: none !important;
}

.forum-post-footer {
  border: 0 !important;
  background: none !important;
}

.forum-post-info {
  padding: .3em 0 !important;
}

.forum-post-title {
  font-style: italic !important;
  font-size: .9em !important;
  padding: 0 2em !important;
  color: #777 !important;
}

.forum-post-title:before {
  content: "Opening Post:" !important;
  font-style: italic !important;
  font-size: .9em !important;
  padding-right: 1em !important;
}

.forum-jump-links {
  display: none !important;
}

.forum-post-footer ul.links,
.forum-post-footer ul.links > li {
  padding-top: 0 !important;
  padding-right: 0 !important;
  padding-bottom: 0 !important;
  padding-left: 0 !important;
  margin-top: 0 !important;
  margin-right: 0 !important;
  margin-bottom: 0 !important;
  margin-left: 0 !important;
}

.forum-post-footer ul.links > li > a {
  padding-top: .3em !important;
  padding-right: .3em !important;
  padding-bottom: .3em !important;
  padding-left: .3em !important;
  margin-top: 0 !important;
  margin-right: 0 !important;
  margin-bottom: 0 !important;
  margin-left: .3em !important;
}

.forum-post-footer ul.links > li.last > a {
  margin-right: -.3em !important;
}

.forum-post-number {
  margin-right: 0 !important;
}


/**
 * Polls
 * /

.poll > .bar {
  background: none !important;
  margin: .5em 0 !important;
}
.poll > .bar > .foreground {
  background: #922 !important;
}

.poll .vote-form,
.poll .vote-form .choices {
  margin: 0 !important;
  text-align: left !important;
}


/**
 * Lightbox
 * /

#outerImageContainer, #imageDataContainer, #bottomNavClose {
  background-color: #000 !important;
}

/**
 * ckEditor
 * /

div.textarea-processed {
  -webkit-filter: invert(90%) brightness(0.8) grayscale(100%) !important;
  filter: invert(90%) brightness(0.8) grayscale(100%) !important;
}


/**
 * PvF Expansion Pack
 * /

.ignore-user {
  display: none !important;
}

.ignore-user-alert {
  background: rgba(10,10,10,0.7) !important;
  border-radius: 10px !important;
  color: #666 !important;
  margin: 1em 20px !important;
  padding: .5em !important;
  min-height: 2em !important;
}

.ignore-user-alert ~ .ignore-user-alert  {
  display: none !important;
}

#ignored-notice { 
  position: fixed !important; 
  z-index: 10000 !important;
  top: 0 !important; 
  left: 0 !important; 
  font-size: 10pt !important; 
  background: black !important; 
  border-bottom-left-radius: 7px !important; 
  border-bottom-right-radius: 7px !important; 
  max-height: 25% !important; 
  overflow-y: auto !important; 
  padding: .25em 1em !important; 
  list-style-type: none !important; 
} 
#ignored-notice:not(:hover) > li:not(#ignored-notice-heading) {
  display: none !important;
} 
#ignored-notice:not(:hover), #ignored-notice:not(:hover) * {
  opacity: 0.5 !important;
}

.quote-toggle-img {
  float: right !important;
}
.forum-post-content .quote-author:hover {
  cursor: pointer !important;
  -webkit-filter: invert(100%) !important;
  filter: invert(100%) !important;
}
*/});

pvfExpansion.css = function() {
    var css = document.createElement("style");
    css.media = 'all';
    css.type = 'text/css';
    css.innerHTML = pvfExpansion.var.css;
    document.querySelector('head').appendChild(css);
}



/**
 * Ignore list
 */

pvfExpansion.getIgnoreList = function() {
    var ignoreList = GM_getValue('gw2wvw_ignore');
    if (!ignoreList) return [];
    return JSON.parse(ignoreList);
}

pvfExpansion.addIgnoreList = function(asshole) {
    asshole = String(decodeURIComponent(asshole));
    var confirmed = confirm('Do you want to ignore "'+asshole+'"?');
    if (asshole !== 'null' && asshole !== 'undefined' && confirmed) {
        var ignoreList = GM_getValue('gw2wvw_ignore');
        if (!ignoreList) var ignoreList = [];
        else ignoreList = JSON.parse(ignoreList);
        if (ignoreList.indexOf(asshole) == -1) {
            ignoreList.push(asshole);
            GM_setValue('gw2wvw_ignore', JSON.stringify(ignoreList));
        }
    }
    return confirmed;
}

pvfExpansion.delIgnoreList = function(asshole) {
    asshole = String(decodeURIComponent(asshole));
    var confirmed = confirm('Do you want to stop ignoring "'+asshole+'"?');
    if (asshole !== 'null' && asshole !== 'undefined' && confirmed) {
        var ignoreList = GM_getValue('gw2wvw_ignore');
        if (!ignoreList) return;
        else ignoreList = JSON.parse(ignoreList);
        if (ignoreList.indexOf(asshole) !== -1) {
            ignoreList = $.grep(ignoreList, function(value) {
                return value != asshole;
            });
            GM_setValue('gw2wvw_ignore', JSON.stringify(ignoreList));
        }
    }
    return confirmed;
}

pvfExpansion.resetIgnoreList = function() {
    GM_setValue('gw2wvw_ignore', '');
}

pvfExpansion.fuckThoseGuys = function() {
    var hideNodes = [];
    var ignoreList = pvfExpansion.getIgnoreList();
    var ignoreUserAlert = null;
    var ignoredUsers = [];
    var showNodes = [];
    var searchNodes = null;
    
    // reset all ignored user alerts
    $('.ignore-user-alert').remove();
    
    /* Look for quoted posts to ignore. */
    $('blockquote > div:first-child > em, blockquote > p:first-child > em').each(function(){
        if (this.innerHTML !== 'undefined') {
            if ($.inArray(this.innerHTML, ignoreList) !== -1) {
                ignoredUsers.push(this.innerHTML);
                hideNodes.push($(this).parents('blockquote'));
            }
            else {
                showNodes.push($(this).parents('blockquote'));
            }
        }
    });
    
    /* Look for posts to ignore. */
    $('div.author-name > a.username').each(function(){
        if (this.innerHTML !== 'undefined') {
            if ($.inArray(this.innerHTML, ignoreList) !== -1) {
                ignoredUsers.push(this.innerHTML);
                hideNodes.push($(this).parents('div.forum-post'));
            }
            else {
                showNodes.push($(this).parents('div.forum-post'));
            }
        }
    });
    
    ignoredUsers = pvfExpansion.uniqueArrayEntries(ignoredUsers);
    
    for (var i in showNodes) {
        $(showNodes[i]).removeClass('ignore-user');
    }
    
    for (var i in hideNodes) {
        $(hideNodes[i]).addClass('ignore-user');
        ignoreUserAlert = document.createElement('div');
        ignoreUserAlert.setAttribute('class', 'ignore-user-alert');
        ignoreUserAlert.innerHTML = ' ';
        $(hideNodes[i]).after(ignoreUserAlert);
    }
    
    if ($(ignoredUsers).length > 0) {
        // create ignored notice block
        var ignoredNotice = document.createElement('ul');
        ignoredNotice.setAttribute('id', 'ignored-notice');
        var ignoredNoticeUser = document.createElement('li');
        
        ignoredNoticeUser.innerHTML = $(ignoredUsers).length+' user';
        if ($(ignoredUsers).length > 1) {
            ignoredNoticeUser.innerHTML = ignoredNoticeUser.innerHTML+'s';
        }
        ignoredNoticeUser.innerHTML = ignoredNoticeUser.innerHTML+' ignored on page:';
        
        ignoredNoticeUser.setAttribute('id', 'ignored-notice-heading');
        ignoredNotice.appendChild(ignoredNoticeUser);        
        for (var i in ignoredUsers) {
            var ignoredNoticeUser = document.createElement('li');
            var ignoreLink = document.createElement('a');
            ignoreLink.setAttribute('class', 'author-pane-line author-pane-link-line');
            ignoreLink.setAttribute('href', '');
            ignoreLink.innerHTML = ignoredUsers[i];
            $(ignoreLink).click({asshole: ignoredUsers[i]},function(e){
                e.preventDefault();
                // in case "works for me" turns to shit
                // http://stackoverflow.com/questions/24642541/gm-getvalue-is-not-defined-in-injected-code
                //console.log("GM_setValue: ", GM_setValue("extra_markdown", true) );
                if(pvfExpansion.delIgnoreList(e.data.asshole)) {
                    pvfExpansion.fuckThoseGuys();
                }
            });
            ignoredNoticeUser.appendChild(ignoreLink);
            ignoredNotice.appendChild(ignoredNoticeUser);
        }
        
        if (document.getElementById('ignored-notice') == null) {
            // add new ignored notice block
            document.getElementsByTagName('body')[0].appendChild(ignoredNotice);
        }
        else {
            $('#ignored-notice').replaceWith(ignoredNotice);
        }
        
    }
    else {
        $('#ignored-notice').remove();
    }
    
}


/**
 * Cleanup and enhancement
 */

pvfExpansion.addForumPostHyperinks = function() {
    var searchNodes = document.evaluate("//a[@class='username']", document, null, XPathResult.UNORDERED_NODE_ITERATOR_TYPE, null);
    var resultNodes = [];
    while (aResult = searchNodes.iterateNext()) {
        resultNodes.push(aResult);
    }
    for (var i in resultNodes) {
        if ($(resultNodes[i].parentNode.parentNode).hasClass('author-pane-general')) {
            
            var newContainer = document.createElement('div');
            newContainer.setAttribute('class', 'author-pane-section author-pane-contact');
            $(newContainer).insertAfter(resultNodes[i].parentNode.parentNode);
            
            var ignoreLink = document.createElement('a');
            ignoreLink.setAttribute('class', 'author-pane-line author-pane-link-line');
            ignoreLink.setAttribute('href', '');
            ignoreLink.innerHTML = 'Ignore';
            $(ignoreLink).click({asshole: resultNodes[i].innerHTML},function(e){
                e.preventDefault();
                // in case "works for me" turns to shit
                // http://stackoverflow.com/questions/24642541/gm-getvalue-is-not-defined-in-injected-code
                //console.log("GM_setValue: ", GM_setValue("extra_markdown", true) );
                if(pvfExpansion.addIgnoreList(e.data.asshole)) {
                    pvfExpansion.fuckThoseGuys();
                }
            });
            newContainer.appendChild(ignoreLink);
            
        }
    }
}

pvfExpansion.cleanerMarkup = function() {
    
    // forum posts: div.forum-post
    // user page user comments: div.view-userposts li.views-row
    // user page user content: div.view-usercontent div.node
    $('div.forum-post, div.view-userposts li.views-row, div.view-usercontent div.node').each(function(i){
        
        var postId = $(this).attr('id');
        if ((postId === 'undefined') || !postId) {
            postId = 'fake-post-id-'+i;
        }
        
        // the "snip" feature is replaced
        $(this).find('blockquote').show();
        $(this).find('div.quote-snip').remove();
        
        // text not wrapped in <p>
        $(this).find('div.field-item, blockquote, div.author-signature').contents()
        .filter(function() {
            // a non-empty text node
            return ((this.nodeType === 3) && (this.nodeValue) && (this.nodeValue.trim() !== ''));
        })
        .wrap('<p></p>').end();
        
        // pilcrow cancer
        $(this).find('br').remove();
        $(this).find('p').each(function(i){
            this.innerHTML = this.innerHTML.replace(/&nbsp;/ig,' ');
            if (this.innerHTML.trim() === '') this.remove();
        });
        $(this).find('blockquote').each(function(i){
            this.innerHTML = this.innerHTML.replace(/&nbsp;/ig,' ');
            if (this.innerHTML.trim() === '') this.remove();
        });
        
        // narrow context to post body
        $(this).find('div.field-name-body, div.field-name-comment-body, div.views-field-comment-body').each(function(i){
            
            // quote and attribution markup
            $(this).find('blockquote').each(function(i){
                
                var quoteId = postId+'-quote-'+i;
                
                // used p wrapper instead of div.quote-author, move attribution outside of and above p
                $(this).children('p:first-child').children('em:first-child').detach().insertBefore($(this).children('p:first-child'));
                
                // missing div.quote-author wrapper on attribution
                $(this).children('em:first-child').wrap('<div class="quote-author"></div>').after(' wrote:');
                
                // missing quote-msg class, inconsistent quote-nest-*, which triggers the poorly done "snip" feature
                $(this).removeClass().addClass('quote-msg').attr('id', quoteId);
                
                // remove unneeded "wrote:" text which was repositioned above
                $(this).children('p:nth-child(2)').each(function(i){
                    if (this.innerHTML && (this.innerHTML !== 'undefined')) {
                        this.innerHTML = this.innerHTML.replace(/^\s*wrote:\s*/i,'');
                        if (this.innerHTML.trim() === '') {
                            $(this).remove();
                        }
                    }
                });
                
                // missing attribution for quote
                if ($(this).children('.quote-author').length === 0) {
                    $(this).prepend('<div class="quote-author"><em class="placeholder">(unattributed)</em> wrote:</div>');
                }
                
                // content after quote-author wrapped in new collapsible container
                $(this).append('<div class="quote-container"></div>');
                $(this).children().not('.quote-container, .quote-author').detach().appendTo($(this).children('.quote-container'));
                
                // add comment toggle
                $(this).children('.quote-author').click({quoteId: quoteId},function(e){
                    $('#'+e.data.quoteId+' > .quote-container:visible').slideUp({
                        duration: 300,
                        start: function(){
                            $('#'+e.data.quoteId+' > .quote-author > .quote-toggle-img').attr('src', pvfExpansion.getImage('plus'));
                        }
                    });
                    $('#'+e.data.quoteId+' > .quote-container:hidden').slideDown({
                        duration: 300,
                        start: function(){
                            $('#'+e.data.quoteId+' > .quote-author > .quote-toggle-img').attr('src', pvfExpansion.getImage('minus'));
                        }
                    });
                })
                .append('<img class="quote-toggle-img" src="'+pvfExpansion.getImage('minus')+'">');
                
                // configure nested quote series
                // default: first is open, children and siblings are closed
                if (i > 0) {
                    $(this).children('.quote-container').hide();
                    $(this).find('.quote-author > .quote-toggle-img').attr('src', pvfExpansion.getImage('plus'));
                }
                else {
                    $(this).children('.quote-container').show();
                }
                
            });
        });
        
    });
    
}


/**
 * Start app
 */

$(document).ready(function() {
    
    pvfExpansion.css();
    pvfExpansion.cleanerMarkup();
    
    // paths where forum-style content appears
    if (document.location.href.match(/^http\:\/\/www\.gw2wvw\.net\/(?:comment\/|content\/|topic\/|user$|users\/)/i)) {
        // hide ignored user posts
        pvfExpansion.fuckThoseGuys();
        pvfExpansion.addForumPostHyperinks();
    }
    
    pvfExpansion.var.end = new Date().getTime();
    pvfExpansion.var.time = pvfExpansion.var.end - pvfExpansion.var.start;
    console.log('PvF Expansion execution time: ' + pvfExpansion.var.time);
});

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址