Deezer Analytics

Extended information for Deezer

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

// ==UserScript==
// @name         Deezer Analytics
// @namespace    com.deezer.analytics
// @version      1.0
// @description  Extended information for Deezer
// @author       panthera.p
// @match        http://www.deezer.com/*
// @match        https://www.deezer.com/*
// @grant        unsafeWindow
// @grant        GM_addStyle
// @grant        GM_xmlhttpRequest
// @require      https://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.min.js
// ==/UserScript==

(function(w, d, $, x)
 {
     var settings = {
         APP_ID: null,
         ACCESS_TOKEN: null,
         PROFILE_ID: null,
         REFRESH_INTERVAL: 3000,
         REFRESH_INTERVAL_BIG: 9000
     };

     var core = {
         variables: {
             ajaxQueue: [],
             ajaxIsRunning: false,
             infoBox: null,
             cache: {}
         },
         init: function()
         {
             GM_addStyle(" \
.deezer-icon-favorite { background-image: url(''); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-global { background-image: url(''); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-social { background-image: url(''); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-play { background-image: url(''); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-album { background-image: url(''); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-artist { background-image: url(''); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-social-big { background-image: url(''); background-repeat:no-repeat; background-size:20px 20px; background-position:5px center; } \
.deezer-icon-favorite-big { background-image: url(''); background-repeat:no-repeat; background-size:20px 20px; background-position:5px center; } \
\
");          if (d.location.protocol == "http:")
                 d.location.href = d.location.href.replace("http:", "https:");
             
             $("html").addClass("deezer");
             settings.APP_ID = w.localStorage.getItem("deezer-appid") || w.prompt("Enter you \"Application ID\" from \"http://developers.deezer.com/myapps\". When creting an application, write \"www.deezer.com\" as \"Application domain\".");
             if (settings.APP_ID == null)
                 return;
             else
                 w.localStorage.setItem("deezer-appid", settings.APP_ID);

             core.createToken();
             setInterval(core.updateToken, 60000);
             setInterval(core.cache, 600000);
             setTimeout(core.load, 2000);
             setInterval(core.getDataCallAjax, 250);
         },
         load: function()
         {
             core.setDefaultProfile();
             core.createInfo();
             core.cache();

             album.init();
             cover.init();
             hearthis.init();
             profile.init();
         },
         createToken: function()
         {
             settings.ACCESS_TOKEN = w.localStorage.getItem("deezer-access-token") || null;

             if (d.location.hash && d.location.hash.indexOf("access_token") > -1)
             {
                 var access_token = d.location.hash.match(/access_token=([^&]+)/i)[1];
                 settings.ACCESS_TOKEN = access_token;
                 w.localStorage.setItem("deezer-access-token", access_token);
             }
             else if (settings.ACCESS_TOKEN == null)
                 d.location.href = "https://connect.deezer.com/oauth/auth.php?app_id=" + settings.APP_ID + "&redirect_uri=" + w.location.protocol + "//" + w.location.host + w.location.pathname + "&perms=basic_access,listening_history&response_type=token";
         },
         updateToken: function(fnFinish)
         {
             var iframe = $('<iframe/>')
             .css({
                 top: -10000,
                 left: -10000,
                 position: "absolute",
             })
             .attr("src", "https://connect.deezer.com/oauth/auth.php?app_id=" + settings.APP_ID + "&redirect_uri=" + w.location.protocol + "//" + w.location.host + w.location.pathname + "&perms=basic_access,listening_history&response_type=token")
             .appendTo("body");

             setTimeout(function()
                        {
                            var access_token = utils.document(iframe).location.hash.match(/access_token=([^&]+)/i)[1];
                            settings.ACCESS_TOKEN = access_token;
                            w.localStorage.setItem("deezer-access-token", access_token);
                            iframe.remove();

                            if (fnFinish)
                                fnFinish();
                        }, 10000);
         },
         setDefaultProfile: function()
         {
             settings.PROFILE_ID = x.USER.USER_ID;
         },
         createInfo: function()
         {
             core.variables.infoBox = $('<div/>')
             .addClass("nav-link")
             .attr("id", "DeezerInfo")
             .text("Loading...")
             .appendTo("#menu_navigation > div");
         },
         cache: function()
         {
             if (settings.PROFILE_ID != null && settings.ACCESS_TOKEN != null)
             {
                 core.getData("Loading albums...", "https://api.deezer.com/user/" + settings.PROFILE_ID + "/albums", function(data) { core.variables.cache.albums = data; });
                 core.getData("Loading artists...", "https://api.deezer.com/user/" + settings.PROFILE_ID + "/artists", function(data) { core.variables.cache.artists = data; });
                 core.getData("Loading followings...", "https://api.deezer.com/user/" + settings.PROFILE_ID + "/followings", function(data) { core.variables.cache.followings = data; core.cacheFollowings(); });
                 core.getData("Loading favorites...", "https://api.deezer.com/user/" + settings.PROFILE_ID + "/tracks", function(data) { core.variables.cache.favorites = data; });
                 core.getData("Loading history...", "https://api.deezer.com/user/" + settings.PROFILE_ID + "/history", function(data) { core.variables.cache.history = data; });
             }
         },
         cacheFollowings: function()
         {
             core.variables.cache.followingsDetail = {};

             var addToAjax = function(profileid, name)
             {
                 core.variables.cache.followingsDetail[profileid] = {};
                 core.getData("Loading albums (" + name + ")...", "https://api.deezer.com/user/" + profileid + "/albums", function(data) { core.variables.cache.followingsDetail[profileid].albums = data; });
                 core.getData("Loading artists (" + name + ")...", "https://api.deezer.com/user/" + profileid + "/artists", function(data) { core.variables.cache.followingsDetail[profileid].artists = data; });
                 core.getData("Loading favorites (" + name + ")...", "https://api.deezer.com/user/" + profileid + "/tracks", function(data) { core.variables.cache.followingsDetail[profileid].favorites = data; });
             };

             for (var i = 0; i < core.variables.cache.followings.length; i++)
                 addToAjax(core.variables.cache.followings[i].id, core.variables.cache.followings[i].name);
         },
         getData: function(info, url, fn, data)
         {
             core.variables.ajaxQueue.push({ info: info, url: url, fn: fn, data: data || [] });
         },
         getDataCallAjax: function()
         {
             if (core.variables.ajaxQueue.length > 0 && !core.variables.ajaxIsRunning)
             {
                 core.variables.ajaxIsRunning = true;
                 var ajaxItem = core.variables.ajaxQueue.splice(0, 1)[0];
                 core.variables.infoBox.text(ajaxItem.info + " (+" + core.variables.ajaxQueue.length + ")");

                 GM_xmlhttpRequest({
                     url: ajaxItem.url + "?limit=10000&access_token=" + settings.ACCESS_TOKEN,
                     onload: function(data)
                     {
                         if (data.responseText.indexOf("OAuthException") == -1)
                         {
                             var json = JSON.parse(data.responseText);
                             if (json.data)
                             {
                                 ajaxItem.data = ajaxItem.data.concat(json.data);
                                 if (json.next)
                                     core.variables.ajaxQueue.push({ info: ajaxItem.info, url: json.next, fn: ajaxItem.fn, data: ajaxItem.data });
                                 else
                                     ajaxItem.fn.call(null, ajaxItem.data);
                             }
                             else
                                 ajaxItem.fn.call(null, json);

                             if (core.variables.ajaxQueue.length == 0)
                                 core.variables.infoBox.text("...");

                             core.variables.ajaxIsRunning = false;
                         }
                         else if (data.responseText.indexOf("OAuthException") > -1)
                         {
                             ajaxItem.fn.call(null, null);
                             if (data.responseText.indexOf("private") == -1)
                             {
                                 setTimeout(function()
                                            {
                                                core.variables.ajaxQueue.push(ajaxItem);
                                                core.variables.ajaxIsRunning = false;
                                            }, 10000);
                             }
                             else
                             {
                                 core.variables.ajaxIsRunning = false;
                             }
                         }
                     },
                     onerror: function(error)
                     {
                         alert(error);
                     }
                 });
             }
         }
     };

     var album = {
         init: function()
         {
             GM_addStyle(" \
#AlbumRating { position:absolute; left:270px; top:120px; } \
#AlbumFriendsRating { position:absolute; left:530px; top:120px; } \
#AlbumFriendsCount { position:absolute; left:790px; top:120px; width:50px; height:40px; line-height:40px; text-align:center; padding-left:30px; font-weight:bold; color:#333333; font-size:16px; } \
div.deezer-albumrating { width:220px; height:40px; line-height:40px; text-align:center; padding-left:30px; } \
span.deezer-albumrating-counter { font-weight:bold; color:#333333; font-size:16px; } \
div.deezer-albumrating-meter { position:absolute; bottom:0; left:0; right:0; height:3px; background-color:#aaaaaa; } \
#AlbumRating div.deezer-albumrating-meter-fill { position:absolute; top:0; left:0; height:3px; background-color:#67a91f; } \
#AlbumFriendsRating div.deezer-albumrating-meter-fill { position:absolute; top:0; left:0; height:3px; background-color:#ede045; } \
span.deezer-album-track-rating { background-color:#67a91f; padding:2px 5px 2px 20px; color:#ffffff; margin-right:10px; border-radius:2px; font-weight:bold; } \
span.deezer-album-track-rating.deezer-album-track-inactive { background-color:#666666; } \
span.deezer-album-track-friendscount { background-color:#bcaf12; padding:2px 5px 2px 20px; color:#ffffff; margin-right:10px; border-radius:2px; font-weight:bold; } \
span.deezer-album-track-friendscount.deezer-album-track-inactive { background-color:#666666; } \
");
             setInterval(album.updateFriends, settings.REFRESH_INTERVAL_BIG);
             setInterval(album.updateScore, settings.REFRESH_INTERVAL);
             setInterval(album.updateTracks, settings.REFRESH_INTERVAL_BIG);
         },
         calculateRating: function(tracksNumber, totalTracks)
         {
             var rating = Math.pow(tracksNumber, 1 + Math.pow(totalTracks, -0.7)) / totalTracks;
             return rating > 1.0 ? 1.0 : rating;
         },
         updateScore: function()
         {
             if ($("#page_naboo_album").length == 1)
             {
                 var lovedTracks = $("a[data-target='lovetrack'].selected").length;
                 var allTracks = $("a[data-target='lovetrack']").length;

                 var albumRatingBox = $("#AlbumRating");
                 if (albumRatingBox.length == 0)
                 {
                     albumRatingBox = $('<div/>')
                     .attr("id", "AlbumRating")
                     .addClass("deezer-albumrating deezer-icon-favorite-big")
                     .appendTo("#naboo_catalog_heading");

                     var counter = $('<span/>')
                     .addClass("deezer-albumrating-counter")
                     .appendTo(albumRatingBox);

                     var meter = $('<div/>')
                     .addClass("deezer-albumrating-meter")
                     .appendTo(albumRatingBox);
                     $('<div/>')
                     .addClass("deezer-albumrating-meter-fill")
                     .appendTo(meter);
                 }

                 var rating = album.calculateRating(lovedTracks, allTracks);
                 if (rating == 0)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("TBD");
                 else if (rating > 0 && rating < 0.20)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("POOR (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.20 && rating < 0.4)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("AVERAGE (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.4 && rating < 0.6)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("GOOD (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.6 && rating < 0.8)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("EXCELLENT (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.8)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("OVERWHELMING (" + Math.round(rating * 100) + ")");
                 albumRatingBox.find("div.deezer-albumrating-meter-fill").width(rating * albumRatingBox.outerWidth());
             }
         },
         updateFriends: function()
         {
             if ($("#page_naboo_album").length == 1 && core.variables.cache.albums && core.variables.cache.favorites && core.variables.cache.followings && core.variables.cache.followingsDetail)
             {
                 var id = $("figure[data-id]").data("id");
                 var allTracks = $("a[data-target='lovetrack']").length;
                 var friendsTotal = 0;
                 var friendsRating = 0.0;
                 var friendsRatingCount = 0;
                 for (var j = 0; j < core.variables.cache.followings.length; j++)
                 {
                     if (core.variables.cache.followingsDetail[core.variables.cache.followings[j].id] && core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].albums)
                     {
                         var friendsAlbum = core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].albums.filter(function(a) { return a.id == id; });
                         friendsTotal += friendsAlbum.length;

                         if (core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].favorites)
                         {
                             var friendsFavorites = core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].favorites.filter(function(a) { return a.album.id == id; });
                             if (friendsFavorites.length > 0)
                             {
                                 friendsRating += album.calculateRating(friendsFavorites.length, allTracks);
                                 friendsRatingCount++;
                             }
                         }
                     }
                 }

                 var albumRatingBox = $("#AlbumFriendsRating");
                 var albumCountBox = $("#AlbumFriendsCount");
                 if (albumRatingBox.length == 0)
                 {
                     albumRatingBox = $('<div/>')
                     .attr("id", "AlbumFriendsRating")
                     .addClass("deezer-albumrating deezer-icon-social-big")
                     .appendTo("#naboo_catalog_heading");

                     var counter = $('<span/>')
                     .addClass("deezer-albumrating-counter")
                     .appendTo(albumRatingBox);

                     var meter = $('<div/>')
                     .addClass("deezer-albumrating-meter")
                     .appendTo(albumRatingBox);
                     $('<div/>')
                     .addClass("deezer-albumrating-meter-fill")
                     .appendTo(meter);

                     albumCountBox = $('<div/>')
                     .attr("id", "AlbumFriendsCount")
                     .addClass("deezer-icon-social-big")
                     .appendTo("#naboo_catalog_heading");

                     $('<span/>')
                     .addClass("deezer-albumrating-counter")
                     .appendTo(albumCountBox);

                     $('<div/>')
                     .addClass("deezer-albumrating-meter")
                     .appendTo(albumCountBox);
                 }

                 var rating = (friendsRatingCount > 0 ? friendsRating / friendsRatingCount : 0);
                 if (rating == 0)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("TBD");
                 else if (rating > 0 && rating < 0.20)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("POOR (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.20 && rating < 0.4)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("AVERAGE (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.4 && rating < 0.6)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("GOOD (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.6 && rating < 0.8)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("EXCELLENT (" + Math.round(rating * 100) + ")");
                 else if (rating >= 0.8)
                     albumRatingBox.find("span.deezer-albumrating-counter").text("OVERWHELMING (" + Math.round(rating * 100) + ")");
                 albumRatingBox.find("div.deezer-albumrating-meter-fill").width(rating * albumRatingBox.outerWidth());

                 albumCountBox.find("span.deezer-albumrating-counter").text(friendsTotal);
             }
         },
         updateTracks: function()
         {
             if ($("#page_naboo_album").length == 1 && core.variables.cache.history)
             {
                 var tracks = $("table.datagrid-table tbody tr.song");
                 for (var i = 0; i < tracks.length; i++)
                 {
                     var track = tracks.eq(i);
                     var trackid = track.attr("id").match(/naboo_datagrid_track_([\d]+)/)[1];

                     var ratingElement = track.find("span.deezer-album-track-rating");
                     var friendsCountElement = track.find("span.deezer-album-track-friendscount");
                     if (ratingElement.length == 0)
                     {
                         friendsCountElement = $('<span/>')
                         .addClass("deezer-album-track-friendscount deezer-icon-favorite")
                         .prependTo(track.find("td.track div.wrapper"));
                         ratingElement = $('<span/>')
                         .addClass("deezer-album-track-rating deezer-icon-play")
                         .prependTo(track.find("td.track div.wrapper"));
                     }

                     var friendsTotal = 0;
                     for (var j = 0; j < core.variables.cache.followings.length; j++)
                     {
                         if (core.variables.cache.followingsDetail[core.variables.cache.followings[j].id] && core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].favorites)
                         {
                             var friendsTracks = core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].favorites.filter(function(a) { return a.id == trackid; });
                             friendsTotal += friendsTracks.length;
                         }
                     }

                     var trackCount = core.variables.cache.history.filter(function(a) { return a.id == trackid; }).length;
                     ratingElement.text(trackCount);
                     if (trackCount == 0)
                         ratingElement.addClass("deezer-album-track-inactive");
                     else
                         ratingElement.removeClass("deezer-album-track-inactive");
                     friendsCountElement.text(friendsTotal);
                     if (friendsTotal == 0)
                         friendsCountElement.addClass("deezer-album-track-inactive");
                     else
                         friendsCountElement.removeClass("deezer-album-track-inactive");
                 }
             }
         }
     };

     var hearthis = {
         init: function()
         {
             GM_addStyle(" \
section.card-edito-track { display:none; } \
section.card-radio { display:none; } \
section.card-algo-playlist { display:none; } \
section.card-edito-playlist { display:none; } \
section.card-auto-promo { display:none; } \
section.card-sponsor { display:none; } \
");
         }
     };

     var profile = {
         init: function()
         {
             GM_addStyle(" \
#profile_page_tab section[data-target='playlists'] { display:none; } \
#profile_page_tab section[data-target='albums'] { display:none; } \
#profile_page_tab section[data-target='artists'] { display:none; } \
#profile_page_tab section[data-target='following'] { display:none; } \
#profile_page_tab section[data-target='follower'] { display:none; } \
");
         }
     };

     var cover = {
         init: function()
         {
             GM_addStyle(" \
span.sent-to-mobile { display:none; } \
span.deezer-profile-album-rating { position:absolute; right:10px; bottom:10px; background-color:#67a91f; height:14px; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-profile-album-rating.deezer-profile-album-inactive { background-color:#666666; } \
span.deezer-profile-album-friendscount { position:absolute; right:10px; bottom:60px; background-color:#46a2ec; height:14px; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-profile-album-friendscount.deezer-profile-album-inactive { background-color:#666666; } \
span.deezer-profile-album-friendsrating { position:absolute; right:10px; bottom:35px; background-color:#bcaf12; height:14px; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-profile-album-friendsrating.deezer-profile-album-inactive { background-color:#666666; } \
span.deezer-profile-album-globalcount { position:absolute; right:10px; bottom:85px; background-color:#46a2ec; height:14px; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-profile-album-genre { position:absolute; right:10px; bottom:191px; background-color:#ab1d81; width:164px; height:14px; overflow:hidden; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-profile-album-genre-item { opacity:0; position:absolute; right:10px; top:3; transition:opacity linear 500ms; } \
span.deezer-profile-album-genre-item-visible { opacity:1; } \
\
span.deezer-fan-album { position:absolute; right:10px; bottom:10px; background-color:#67a91f; height:14px; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-fan-album.deezer-fan-inactive { background-color:#666666; } \
span.deezer-fan-artist { position:absolute; right:10px; bottom:35px; background-color:#67a91f; height:14px; font-size:12px; padding:2px 10px 2px 20px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; font-weight:bold; line-height:14px; text-align:right; vertical-align:top; letter-spacing:1px; } \
span.deezer-fan-artist.deezer-fan-inactive { background-color:#666666; } \
");

             if (!core.variables.cache.fans)
                 core.variables.cache.fans = {};
             if (!core.variables.cache.albumsDetail)
                 core.variables.cache.albumsDetail = {};

             setInterval(cover.album, settings.REFRESH_INTERVAL);
             setInterval(cover.fans, settings.REFRESH_INTERVAL);
             setInterval(cover.switchAlbumGenre, settings.REFRESH_INTERVAL / 2);
         },
         album: function()
         {
             if (core.variables.cache.albums && core.variables.cache.favorites && core.variables.cache.followings && core.variables.cache.followingsDetail)
             {
                 var albumElements = $("li[id^='album_']:not(.deezer-profile-album-hasrating), #naboo_tops_container li:not(.deezer-profile-album-hasrating), #discography_artist li:not(.deezer-profile-album-hasrating)");

                 var loadAlbum = function(id, name)
                 {
                     var createElement = function(id)
                     {
                         var figure = $("figure[data-id='" + id + "'], figure[data-reactid*='" + id + "']");
                         var albumItem = core.variables.cache.albumsDetail[id];
                         if (!albumItem) return;
                         var favoriteItems = core.variables.cache.favorites.filter(function(a) { return a.album.id == id; });
                         var rating = album.calculateRating(favoriteItems.length, albumItem.nb_tracks);

                         var friendsTotal = 0;
                         var friendsRating = 0.0;
                         var friendsRatingCount = 0;
                         for (var j = 0; j < core.variables.cache.followings.length; j++)
                         {
                             if (core.variables.cache.followingsDetail[core.variables.cache.followings[j].id] && core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].albums)
                             {
                                 var friendsAlbum = core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].albums.filter(function(a) { return a.id == id; });
                                 friendsTotal += friendsAlbum.length;

                                 if (core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].favorites)
                                 {
                                     var friendsFavorites = core.variables.cache.followingsDetail[core.variables.cache.followings[j].id].favorites.filter(function(a) { return a.album.id == id; });
                                     if (friendsFavorites.length > 0)
                                     {
                                         friendsRating += album.calculateRating(friendsFavorites.length, albumItem.nb_tracks);
                                         friendsRatingCount++;
                                     }
                                 }
                             }
                         }

                         $('<span/>')
                         .addClass("deezer-profile-album-friendscount deezer-icon-social")
                         .addClass(friendsTotal == 0 ? "deezer-profile-album-inactive" : "")
                         .text(friendsTotal)
                         .appendTo(figure);
                         $('<span/>')
                         .addClass("deezer-profile-album-friendsrating deezer-icon-social")
                         .addClass(friendsRatingCount == 0 ? "deezer-profile-album-inactive" : "")
                         .text((friendsRatingCount > 0 ? Math.round(friendsRating / friendsRatingCount * 100) : 0) + "%")
                         .appendTo(figure);
                         $('<span/>')
                         .addClass("deezer-profile-album-rating deezer-icon-favorite")
                         .addClass(rating == 0 ? "deezer-profile-album-inactive" : "")
                         .text(Math.round(rating * 100) + "%")
                         .appendTo(figure);
                         $('<span/>')
                         .addClass("deezer-profile-album-globalcount deezer-icon-global")
                         .text(utils.numberWithCommas(albumItem.fans))
                         .appendTo(figure);
                         var genres = $('<span/>')
                         .addClass("deezer-profile-album-genre")
                         .appendTo(figure);

                         for (var j = 0; j < albumItem.genres.data.length; j++)
                         {
                             $('<span/>')
                             .addClass("deezer-profile-album-genre-item")
                             .addClass(j == 0 ? "deezer-profile-album-genre-item-visible" : "")
                             .text(albumItem.genres.data[j].name)
                             .appendTo(genres);
                         }
                     };

                     if (!core.variables.cache.albumsDetail[id])
                     {
                         core.getData("Loading album (" + name + ")...", "https://api.deezer.com/album/" + id, function(data)
                                      {
                                          core.variables.cache.albumsDetail[id] = data;
                                          createElement(id);
                                      });
                     }
                     else
                         createElement(id);
                 };
                 for (var i = 0; i < albumElements.length; i++)
                 {
                     var li = albumElements.eq(i);
                     li.addClass("deezer-profile-album-hasrating");

                     var figure = li.find("figure");
                     if (albumElements.eq(i).find("figure").data("id"))
                         loadAlbum(albumElements.eq(i).find("figure").data("id"), albumElements.eq(i).find("dt a").text());
                     else if (albumElements.eq(i).find("figure").data("reactid") && (d.location.href.indexOf("charts/news/genre") > -1 || d.location.href.indexOf("charts/album/genre") > -1 || d.location.href.indexOf("artist") > -1))
                         loadAlbum(albumElements.eq(i).find("figure").data("reactid").match(/[$]([\d]+)/)[1], albumElements.eq(i).find("h4 a").text());
                 }
             }
         },
         switchAlbumGenre: function()
         {
             var genres = $("span.deezer-profile-album-genre-item-visible");
             for (var i = 0; i < genres.length; i++)
             {
                 var genre = genres.eq(i);
                 genre.removeClass("deezer-profile-album-genre-item-visible");
                 if (genre.next("span.deezer-profile-album-genre-item").length == 0)
                     genre.parent().children().eq(0).addClass("deezer-profile-album-genre-item-visible");
                 else
                     genre.next("span.deezer-profile-album-genre-item").addClass("deezer-profile-album-genre-item-visible");
             }
         },
         fans: function()
         {
             if (core.variables.cache.albums && core.variables.cache.artists && core.variables.cache.fans)
             {
                 var fans = $("li[id^='user_']:not(.deezer-fan-hasinfo), #naboo_fan_list li:not(.deezer-fan-hasinfo)");

                 var loadFan = function(fanElement, id)
                 {
                     fanElement.addClass("deezer-fan-hasinfo");

                     var renderFan = function(id)
                     {
                         if (!core.variables.cache.fans[id].albums || !core.variables.cache.fans[id].artists)
                         {
                             fanElement.removeClass("deezer-fan-hasinfo");
                             return;
                         }
                         
                         var figure = fanElement.find("figure");
                         var mineAlbums = core.variables.cache.albums.map(function(a) { return a.id; });
                         var fanAlbums = core.variables.cache.fans[id].albums.map(function(a) { return a.id; });
                         var intersectAlbums = mineAlbums.filter(function(a) { return fanAlbums.indexOf(a) > -1; }).length;
                         var mineArtists = core.variables.cache.artists.map(function(a) { return a.id; });
                         var fanArtists = core.variables.cache.fans[id].artists.map(function(a) { return a.id; });
                         var intersectArtists = mineArtists.filter(function(a) { return fanArtists.indexOf(a) > -1; }).length;

                         $('<span/>')
                         .addClass("deezer-fan-album deezer-icon-album")
                         .addClass(intersectAlbums == 0 ? "deezer-fan-inactive" : "")
                         .text((intersectAlbums > 0 ? Math.round(intersectAlbums / mineAlbums.length * 100) : 0) + "%")
                         .appendTo(figure);
                         $('<span/>')
                         .addClass("deezer-fan-artist deezer-icon-artist")
                         .addClass(intersectArtists == 0 ? "deezer-fan-inactive" : "")
                         .text((intersectArtists > 0 ? Math.round(intersectArtists / mineArtists.length * 100) : 0) + "%")
                         .appendTo(figure);
                     };

                     if (!core.variables.cache.fans[id])
                     {
                         core.variables.cache.fans[id] = {
                             albums: [],
                             artists: []
                         };
                         core.getData("Loading fan albums (" + id + ")...", "https://api.deezer.com/user/" + id + "/albums", function(data)
                                      {
                                          core.variables.cache.fans[id].albums = data || [];
                                      });
                         core.getData("Loading fan artists (" + id + ")...", "https://api.deezer.com/user/" + id + "/artists", function(data)
                                                       {
                                                           core.variables.cache.fans[id].artists = data || [];
                                                           renderFan(id);
                                                       });
                     }
                     else
                         renderFan(id);
                 };

                 for (var i = 0; i < fans.length; i++)
                 {
                     var fanElement = fans.eq(i);
                     if (fanElement.find("figure").data("id") && (d.location.href.indexOf("following") > -1 || d.location.href.indexOf("followers") > -1))
                         loadFan(fanElement, fanElement.find("figure").data("id"));
                     else if (fans.eq(i).data("reactid") && d.location.href.indexOf("album") > -1)
                         loadFan(fanElement, fans.eq(i).data("reactid").match(/[$]([\d]+)/)[1]);
                 }
             }
         }
     };

     var utils = {
         numberWithCommas: function (number)
         {
             return number.toString().replace(/\B(?=(\d{3})+(?!\d))/g, ",");
         },
         isCrossDomain: function(iframeElement)
         {
             // check if passed argument is String or Iframe element
             iframeElement = typeof iframeElement === "string" ? form.find(id) : iframeElement;
             iframeElement = iframeElement instanceof $ ? iframeElement.get(0) : (iframeElement[0] || iframeElement);

             try
             {
                 iframeElement.contentWindow.document;
                 return false;
             }
             catch (e)
             {
                 return true;
             }
         },
         document: function (iframeElement)
         {
             if (iframeElement != null)
             {
                 try
                 {
                     iframeElement = iframeElement instanceof $ ? iframeElement.get(0) : (iframeElement[0] || iframeElement);
                     return utils.isCrossDomain(iframeElement) ? null : (iframeElement.contentDocument || iframeElement.contentWindow.document);
                 }
                 catch (e)
                 {
                     return null;
                 }
             }
             else
                 return null;
         },
         window: function (iframeElement)
         {
             if (iframeElement != null)
             {
                 try
                 {
                     iframeElement = iframeElement instanceof $ ? iframeElement.get(0) : (iframeElement[0] || iframeElement);
                     return utils.isCrossDomain(iframeElement) ? null : iframeElement.contentWindow;
                 }
                 catch (e)
                 {
                     return null;
                 }
             }
             else
                 return null;
         }
     };

     if (d.location.href.indexOf("appcache") == -1)
     {
         core.init();
     }
 })(window, document, jQuery, unsafeWindow);