Friend Filter for SteamDB Instant Search

在 SteamDB 的即时搜索页面上直接查看好友是否拥有该游戏并进行过滤

// ==UserScript==
// @name        Friend Filter for SteamDB Instant Search
// @namespace   https://greasyfork.org/users/726
// @author      Deparsoul
// @description 在 SteamDB 的即时搜索页面上直接查看好友是否拥有该游戏并进行过滤
// @icon        https://blog.algolia.com/wp-content/themes/algolia/favicon.ico
// @include     https://steamdb.info/instantsearch*
// @version     20171005
// @require     https://ajax.aspnetcdn.com/ajax/jQuery/jquery-2.2.2.min.js
// @grant       GM_xmlhttpRequest
// @connect     steamcommunity.com
// ==/UserScript==

'use strict';

$('.container-search').prepend(
        '<div class="row-fluid"><div class="panel"><div class="panel-heading">点击选中你想要分析的好友<button id="friend_list_toggle">显示全部好友 / 仅显示选中好友</button></div><input type="text" placeholder="输入好友名称、昵称、分组等进行搜索" id="friend_filter" style="width:100%;"><ul id="friend_list"></ul><div style="clear:both;"></div></div></div>'+
        '<style>#friend_list{list-style-type:none;}#friend_list>li{display:block;float:left;width:200px;height:32px;margin-right:10px;margin-bottom:10px;cursor:pointer;opacity:.5;}.friend_avatar{margin-right:2px;vertical-align:top;}#friend_list>li:hover{background-color:#ddd;}#friend_list>li.friend_chosen{background-color:#ddd;opacity:1;}#friend_list>li>div{display:inline-block;margin-left:4px;}.friend_name, .friend_extra{font-size:14px;line-height:18px;display:inline-block;width:110px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;float:left;}.friend_extra{clear:left;font-size:10px;line-height:10px;}.friend_games{display:inline-block;float:right;padding:5px;width:40px;height:32px;text-align:center;}.friend_details>span{line-height:32px;color:white;padding:0px 4px;}.s-hit:hover{opacity:1!important;}.friend_options>p{margin-bottom:8px;}.friend_options label{font-weight:normal;}.friend_loading{background:url("") no-repeat 50% 50%}</style>'
);

$('.container > .row > .span4').prepend(
        '<div class="panel"><div class="panel-heading">好友过滤选项</div><div class="friend_options">'+
        '<p>每页最多显示 <select id="friend_hPP"><option value="20">20</option><option value="50">50</option><option value="100">100</option><option value="200">200</option><option value="500">500</option><option value="1000">1000</option></select> 条结果</p>'+
        '<p>显示 <select id="friend_show"><option value="5">至少 5 人拥有</option><option value="4">至少 4 人拥有</option><option value="3">至少 3 人拥有</option><option value="2">至少 2 人拥有</option><option value="1">至少 1 人拥有</option><option value="-1">所选好友都没有</option><option selected value="0">所有</option></select> 的游戏</p>'+
        '<p><label><input type="checkbox" id="friend_opacity" checked> 用透明度区分拥有游戏的好友人数</label></p>'+
        '<p><label><input type="checkbox" id="friend_store" checked> 点击搜索结果直接打开Steam商店页面</label></p>'+
        '</div></div>'
);

function add_player(data){
    var avatar = data.m_strAvatarHash;
    avatar = 'https://steamcdn-a.akamaihd.net/steamcommunity/public/images/avatars/'+avatar.substr(0,2)+'/'+avatar+'.jpg';
    $('#friend_list').append('<li data-id="'+data.m_unAccountID+'" data-steamid="'+data.m_ulSteamID+'"><img class="friend_avatar" title="'+data.m_strName+'" src="'+avatar+'"><div><span class="friend_name">'+data.m_strName+'</span><span class="friend_extra">'+(data.m_strNickname||'')+'</span></div><span class="friend_games"></span></li>');
}

GM_xmlhttpRequest({
    method: 'GET',
    url: 'https://steamcommunity.com/chat/',
    onload: function(response) {
        var html = response.responseText;
        var match = html.match(/WebAPI, (.*) /);
        if (match) {
            var argv = JSON.parse('['+match[1]+']');
            add_player(argv[0]);
            for(var i=0;i<argv[1].length;++i)
                add_player(argv[1][i]);
            argv[2].forEach(function(group){
                group.members.forEach(function(id){
                    var li = $('#friend_list>li[data-id="'+id+'"]');
                    var extra = li.find('.friend_extra');
                    var text = extra.text();
                    if(text)
                        text += ', '+group.name;
                    else
                        text = group.name;
                    extra.text(text);
                });
            });
        }else{
            $('#friend_list').html('<a target="_blank" href="https://steamcommunity.com/login">请先登录 Steam 社区</a>,然后刷新本页面');
        }
    }
});

var cache = {};
function load_cache(li){
    var steamid = li.data('steamid');
    if(cache.hasOwnProperty(steamid)){
        refresh_hits();
        return;
    }
    cache[steamid] = {};
    var games = li.find('.friend_games');
    games.addClass('friend_loading');
    GM_xmlhttpRequest({
        method: 'GET',
        url: 'https://steamcommunity.com/profiles/'+steamid+'/games/?tab=all',
        onload: function(response) {
            var count = 0;
            var html = response.responseText;
            var reg = /"appid":(\d+),"name"/g;
            var match = reg.exec(html);
            while (match != null) {
                var appid = match[1];
                cache[steamid][appid] = true;
                ++count;
                match = reg.exec(html);
            }
            games.removeClass('friend_loading').text(count);
            refresh_hits();
        }
    });
}

$('#friend_list').on('click', 'li', function(){
    var li = $(this);
    if(li.hasClass('friend_chosen')){
        li.removeClass('friend_chosen');
        refresh_hits();
    }else{
        li.addClass('friend_chosen');
        load_cache(li);
    }
    li.detach();
    var last_chosen = $('.friend_chosen:last');
    if(last_chosen.length)
        li.insertAfter(last_chosen);
    else
        $('#friend_list').prepend(li);
});

$('#friend_filter').keyup(function(){
    var input = $(this);
    var q = input.val().trim().toLowerCase();
    $('#friend_list>li').each(function(){
        var li = $(this);
        if(q && li.text().toLowerCase().indexOf(q)<0)
            li.hide();
        else
            li.show();
    });
});
$('#friend_list_toggle').click(function(){
    var hide = $('#friend_list>li:not(.friend_chosen):visible').length>0;
    $('#friend_filter').val('').keyup();
    if(hide)
        $('#friend_list>li:not(.friend_chosen)').hide();
});

var refreshing = false;
var timer = -1;
function refresh_hits(level){
    level = level || 0;

    var friends = $('.friend_chosen');
    $('#friend_show>option').each(function(){
        var option = $(this);
        var val = option.val();
        if(val>friends.length)
            option.hide();
        else
            option.show();
    });
    if($('#friend_show').val()>friends.length)
        $('#friend_show').val($('#friend_show>option:visible:first').val());
    var show = $('#friend_show').val();

    clearTimeout(timer);
    if(refreshing)
        return;
    console.log('refreshing');
    refreshing = true;

    $('.s-hit').each(function(){
        if(friends.length<1)
            return;

        var hit = $(this);

        hit.removeClass('s-hit--owned');

        var details = hit.find('.friend_avatars');
        if(level==0 || details.length==0){
            var appid = hit.attr('href').match(/\d+/)[0];
            if(details.length>0){
                details.html('');
            }else{
                hit.find('.s-hit--details').after('<div class="friend_details"><span>好友</span><span class="friend_avatars"></span></div>');
                details = hit.find('.friend_avatars');
            }
            friends.each(function(){
                var friend = $(this);
                if(cache[friend.data('steamid')][appid])
                details.append(friend.find('.friend_avatar').clone());
            });
        }else{
            if(level>1)
                return;
        }

        var count = hit.find('.friend_avatar').length;
        var hide = false;
        if(show==-1){
            if(count>0)
                hide = true;
        }else{
            if(count<friends.length && count<show)
                hide = true;
        }
        if(hide)
            hit.hide();
        else
            hit.show();
        var opacity = 1;
        if($('#friend_opacity').is(':checked'))
            opacity = .3+.7*(count/friends.length);
        hit.css('opacity', opacity);
    });
    refreshing = false;
}

$('#hits').bind("DOMSubtreeModified",function(){
    if(refreshing)
        return;
    clearTimeout(timer);
    timer = setTimeout(function(){
        refresh_hits(2);
    }, 500);
});

var hPP = location.href.match(/hPP=(\d+)/);
hPP = hPP ? hPP[1] : 20;
$('#friend_hPP').val(hPP).change(function(){
    if(!confirm('修改每页结果数量需要刷新网页,你需要重新选择好友,是否确定要修改?')){
        $(this).val(hPP);
        return false;
    }
    hPP = $(this).val();
    hPP = 'hPP='+hPP;
    var href = location.href.replace(/hPP=(\d+)/, hPP);
    if(href==location.href){
        if(href.match(/\?/))
            href += '&'+hPP;
        else
            href += '?'+hPP;
    }
    location.href = href;
});
$('#friend_show, #friend_opacity').change(function(){
    refresh_hits(1);
});
$('#hits').on('click', '.s-hit', function(){
    if($('#friend_store').is(':checked')){
        window.open('http://store.steampowered.com'+$(this).attr('href'), '_blank');
        return false;
    }else{
        return true;
    }
});