Greasy Fork is available in English.

Deezer Analytics

Extended information for Deezer

// ==UserScript==
// @name         Deezer Analytics
// @namespace    com.deezer.analytics
// @version      1.19
// @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
// @grant        GM_getResourceText
// @require      https://ajax.googleapis.com/ajax/libs/jquery/2.1.4/jquery.min.js
// @resource     Chartist https://cdnjs.cloudflare.com/ajax/libs/chartist/0.9.7/chartist.min.css
// @require      https://cdnjs.cloudflare.com/ajax/libs/chartist/0.9.7/chartist.min.js
// ==/UserScript==

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

    var core = {
        variables: {
            ajaxQueue: [],
            ajaxIsRunning: false,
            infoBox: null,
            cache: {},
            callback: {
                change: [],
                timer: []
            },
            lastUrl: null,
            updateTokenInterval: null
        },
        init: function()
        {
            /*jshint multistr: true */
            GM_addStyle(" \
.deezer-icon-favorite { background-image: url(); background-repeat:no-repeat; background-size:10px 10px; background-position:5px center; } \
.deezer-icon-genre { background-image: url(); background-repeat:no-repeat; background-size:12px 12px; background-position:5px center; } \
.deezer-icon-global { background-image: url(); background-repeat:no-repeat; background-size:12px 12px; 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-track { 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; } \
\
");
            GM_addStyle(GM_getResourceText("Chartist"));

            $("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);

            settings.PROFILE_ID = x.USER.USER_ID;

            core.createToken();

            setTimeout(core.load, 2000);
        },
        load: function()
        {
            setInterval(core.updateToken, 3600000);
            setInterval(core.cache, 600000);
            setInterval(core.doAjax, 100);
            setInterval(core.changeEvent, 1000);
            setInterval(core.timerEvent, 5000);

            core.createInfo();
            core.cache();

            album.init();
            cover.init();
            hearthis.init();
            profile.init();
        },
        changeEvent: function()
        {
            if (core.variables.lastUrl != d.location.href)
            {
                setTimeout(function()
                           {
                    for (var i = 0; i < core.variables.callback.change.length; i++)
                        core.variables.callback.change[i].call(null);
                }, 2000);
            }
        },
        timerEvent: function()
        {
            for (var i = 0; i < core.variables.callback.timer.length; i++)
                core.variables.callback.timer[i].call(null);
        },
        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);
                core.updateToken();
            }
            else if (settings.ACCESS_TOKEN === null)
            {
                d.location.href = w.location.protocol + "//connect.deezer.com/oauth/auth.php?app_id=" + settings.APP_ID + "&redirect_uri=" + (w.location.protocol == "http:" ? "http://goo.gl/dD16FJ" : "https://goo.gl/PL2AFN") + "&perms=basic_access,listening_history&response_type=token";
            }
            else
            {
                core.updateToken();
            }
        },
        updateToken: function(fnFinish)
        {
            $("#UpdateToken").remove();

            var iframe = $('<iframe/>')
            .attr("id", "UpdateToken")
            .css({
                top: -10000,
                left: -10000,
                position: "absolute",
            })
            .attr("src", w.location.protocol + "//connect.deezer.com/oauth/auth.php?app_id=" + settings.APP_ID + "&redirect_uri=" + (w.location.protocol == "http:" ? "http://goo.gl/dD16FJ" : "https://goo.gl/PL2AFN") + "&perms=basic_access,listening_history&response_type=token")
            .appendTo("body");

            clearInterval(core.variables.updateTokenInterval);
            var attempts = 0;
            core.variables.updateTokenInterval = setInterval(function()
                                                             {
                var updateTokenDocument = utils.document($("#UpdateToken"));

                if (updateTokenDocument != null)
                {
                    if (attempts > 10)
                    {
                        $("#UpdateToken").remove();
                        w.localStorage.removeItem("deezer-access-token");
                        core.createToken();
                        return;
                    }

                    if (updateTokenDocument !== null && updateTokenDocument.location && updateTokenDocument.location.hash && updateTokenDocument.location.hash.length > 0 && updateTokenDocument.location.hash.match(/access_token=([^&]+)/i) !== null)
                    {
                        clearInterval(core.variables.updateTokenInterval);

                        var access_token = updateTokenDocument.location.hash.match(/access_token=([^&]+)/i)[1];
                        settings.ACCESS_TOKEN = access_token;
                        w.localStorage.setItem("deezer-access-token", access_token);
                        $("#UpdateToken").remove();

                        if (fnFinish)
                            fnFinish();
                    }
                    else if (utils.document($("#UpdateToken")) !== null && utils.document($("#UpdateToken")).querySelector("body").innerHTML.indexOf("ERR_") > -1)
                    {
                        $("#UpdateToken").remove();
                    }
                }
                else
                    $("#UpdateToken").remove();

                attempts++;
            }, 1000);
        },
        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.queueAjax("Albums...", w.location.protocol + "//api.deezer.com/user/" + settings.PROFILE_ID + "/albums", function(data) { core.variables.cache.albums = data; });
                core.queueAjax("Artists...", w.location.protocol + "//api.deezer.com/user/" + settings.PROFILE_ID + "/artists", function(data) { core.variables.cache.artists = data; });
                core.queueAjax("Followings...", w.location.protocol + "//api.deezer.com/user/" + settings.PROFILE_ID + "/followings", function(data) { core.variables.cache.followings = data; core.cacheFollowings(); });
                core.queueAjax("Favorites...", w.location.protocol + "//api.deezer.com/user/" + settings.PROFILE_ID + "/tracks", function(data) { core.variables.cache.favorites = data; });
                core.queueAjax("History...", w.location.protocol + "//api.deezer.com/user/" + settings.PROFILE_ID + "/history", function(data) { core.variables.cache.history = data; });
            }
        },
        cacheFollowings: function()
        {
            if (core.variables.cache.followings)
            {
                core.variables.cache.followingsDetail = {};

                var addToAjax = function(profileid, name)
                {
                    core.variables.cache.followingsDetail[profileid] = JSON.parse(w.localStorage.getItem("deezer-cache-followings-" + profileid)) || {};
                    if (!w.localStorage.getItem("deezer-cache-followings-" + profileid) || Math.round(new Date().getTime() / 1000) - (w.localStorage.getItem("deezer-cache-followings-" + profileid + "-timestamp") || Math.round(new Date(2000, 0, 1).getTime() / 1000)) > 86400)
                    {
                        core.queueAjax("Albums (" + name + ")...", w.location.protocol + "//api.deezer.com/user/" + profileid + "/albums", function(data) { core.variables.cache.followingsDetail[profileid].albums = data.map(function(a) { return a.id; }); });
                        core.queueAjax("Artists (" + name + ")...", w.location.protocol + "//api.deezer.com/user/" + profileid + "/artists", function(data) { core.variables.cache.followingsDetail[profileid].artists = data.map(function(a) { return a.id; }); });
                        core.queueAjax("Favorites (" + name + ")...", w.location.protocol + "//api.deezer.com/user/" + profileid + "/tracks", function(data)
                                       {
                            core.variables.cache.followingsDetail[profileid].favorites = data.map(function(a) { return { album: a.album.id, track: a.id }; });
                            w.localStorage.setItem("deezer-cache-followings-" + profileid, JSON.stringify(core.variables.cache.followingsDetail[profileid]));
                            w.localStorage.setItem("deezer-cache-followings-" + profileid + "-timestamp", Math.round(new Date().getTime() / 1000));
                        });
                    }
                };

                for (var i = 0; i < core.variables.cache.followings.length; i++)
                    addToAjax(core.variables.cache.followings[i].id, core.variables.cache.followings[i].name);
            }
        },
        queueAjax: function(info, url, fn, data)
        {
            core.variables.ajaxQueue.push({ info: info, url: url, fn: fn, data: data || [] });
        },
        doAjax: 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 = {
        variables: {
            albumScoreMyChart: null,
            albumScoreFriendsChart: null
        },
        init: function()
        {
            /*jshint multistr: true */
            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:#2e9ac2; 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; } \
div.album-headings { position:relative; } \
#DeezerAlbumScoreMyCanvas { position:absolute; top:70px; z-index:10; } \
#DeezerAlbumScoreMyCanvas .ct-series-a .ct-slice-pie { fill: #66bf31; } \
#DeezerAlbumScoreMyCanvas .ct-series-b .ct-slice-pie { fill: #eeeeee; } \
#DeezerAlbumScoreMyRating { position:absolute; top:125px; left:65px; font-weight:bold; z-index:11; } \
#DeezerAlbumScoreFriendsCanvas { position:absolute; top:70px; left:120px; z-index:10; } \
#DeezerAlbumScoreFriendsCanvas .ct-series-a .ct-slice-pie { fill: #2e9ac2; } \
#DeezerAlbumScoreFriendsCanvas .ct-series-b .ct-slice-pie { fill: #eeeeee; } \
#DeezerAlbumScoreFriendsRating { position:absolute; top:125px; left:185px; font-weight:bold; z-index:11; } \
");
            core.variables.callback.change.push(album.updateFriends);
            core.variables.callback.change.push(album.updateScore);
            core.variables.callback.change.push(album.updateTracks);
            core.variables.callback.timer.push(album.updateScore);
            core.variables.callback.timer.push(album.updateTracks);
        },
        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 = $("#page_naboo_album span.icon-love.active").length;
                var allTracks = $("#page_naboo_album span.icon-love").length;

                var rating = album.calculateRating(lovedTracks, allTracks);

                var albumScoreCanvas = $("#DeezerAlbumScoreMyCanvas");
                if (albumScoreCanvas.length === 0)
                {
                    albumScoreCanvas = $('<div id="DeezerAlbumScoreMyCanvas"></div>').appendTo("div.album-headings");
                    albumScoreRating = $('<div id="DeezerAlbumScoreMyRating"></div>').text(Math.round(rating * 100) + '%').appendTo("div.album-headings");
                    album.variables.albumScoreMyChart = new Chartist.Pie("#DeezerAlbumScoreMyCanvas", {
                        labels: [' ', ' '],
                        series: [Math.round(rating * 100), Math.round((1 - rating) * 100)]
                    }, {
                        width: '75px',
                        height: '75px'
                    });
                }
                else
                {
                    $("#DeezerAlbumScoreMyRating").text(Math.round(rating * 100) + '%');
                    album.variables.albumScoreMyChart.update({
                        labels: [' ', ' '],
                        series: [Math.round(rating * 100), Math.round((1 - rating) * 100)]
                    });
                }
            }
        },
        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 = d.location.href.match(/^.+\/album\/([\d]+)(\/.*)?$/)[1];
                var allTracks = $("#page_naboo_album span.icon-love").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; });
                        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; });
                            if (friendsFavorites.length > 0)
                            {
                                friendsRating += album.calculateRating(friendsFavorites.length, allTracks);
                                friendsRatingCount++;
                            }
                        }
                    }
                }

                var ratingFavorites = (friendsRatingCount > 0 ? friendsRating / friendsRatingCount : 0);
                var ratingAlbums = friendsTotal / core.variables.cache.followings.length;
                var rating = ratingFavorites + ratingAlbums > 1 ? 1 : ratingFavorites + ratingAlbums;

                var albumScoreCanvas = $("#DeezerAlbumScoreFriendsCanvas");
                if (albumScoreCanvas.length === 0)
                {
                    albumScoreCanvas = $('<div id="DeezerAlbumScoreFriendsCanvas"></div>').appendTo("div.album-headings");
                    albumScoreRating = $('<div id="DeezerAlbumScoreFriendsRating"></div>').text(Math.round(rating * 100) + '%').appendTo("div.album-headings");
                    album.variables.albumScoreFriendsChart = new Chartist.Pie("#DeezerAlbumScoreFriendsCanvas", {
                        labels: [' ', ' '],
                        series: [Math.round(rating * 100), Math.round((1 - rating) * 100)]
                    }, {
                        width: '75px',
                        height: '75px'
                    });
                }
                else
                {
                    $("#DeezerAlbumScoreFriendsRating").text(Math.round(rating * 100) + '%');
                    album.variables.albumScoreFriendsChart.update({
                        labels: [' ', ' '],
                        series: [Math.round(rating * 100), Math.round((1 - rating) * 100)]
                    });
                }
            }
        },
        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("data-key");

                    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.track == 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()
        {
            /*jshint multistr: true */
            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; } \
section.card-trending { display:none; } \
");
        }
    };

    var profile = {
        init: function()
        {
            /*jshint multistr: true */
            GM_addStyle(" \
#page_profile div.user-section { display:none; } \
");
        }
    };

    var cover = {
        init: function()
        {
            /*jshint multistr: true */
            GM_addStyle(" \
span.sent-to-mobile { display:none; } \
div.deezer-profile-album-cover { position:absolute; z-index:11; top:0; left:0; right:0; padding-left:0px; padding-right:0px; background-color:rgba(100, 180, 41, 0.9); } \
div.deezer-profile-album-cover * { font-size:9px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; letter-spacing:1px; } \
div.deezer-profile-album-info { height:20px; position:relative; } \
span.deezer-profile-album-globalcount { position:absolute; display:inline-block; left:0; width:40%; top:0; bottom:0; line-height:20px; padding-left:23px; box-sizing:border-box; overflow:hidden; white-space:nowrap; } \
span.deezer-profile-album-friendscount { position:absolute; display:inline-block; left:40%; width:30%; top:0; bottom:0; line-height:20px; padding-left:23px; box-sizing:border-box; overflow:hidden; white-space:nowrap; } \
span.deezer-profile-album-rating { position:absolute; display:inline-block; left:70%; width:30%; top:0; bottom:0; line-height:20px; padding-left:23px; box-sizing:border-box; overflow:hidden; white-space:nowrap; } \
div.deezer-profile-album-genre { height:20px; position:relative; } \
span.deezer-profile-album-genre-item { opacity:0; position:absolute; display:inline-block; line-height:20px; left:23px; top:0; bottom:0; right:0; font-size:11px; transition:opacity linear 500ms; } \
span.deezer-profile-album-genre-item-visible { opacity:1; } \
\
div.thumbnail .sticker { bottom:60px; } \
\
div.deezer-profile-fan-cover { position:absolute; z-index:11; height:45px; bottom:0; left:0; right:0; padding-left:0px; padding-right:0px; background-color:rgba(52, 147, 192, 0.9); } \
div.deezer-profile-fan-cover * { font-size:9px; color:#ffffff; font-family:'Open Sans',Arial,sans-serif; letter-spacing:1px; } \
div.deezer-profile-fan-album { position:absolute; top:0; left:5px; right:10px; display:inline-block; line-height:15px; padding-left:23px; } \
div.deezer-profile-fan-artist { position:absolute; top:15px; left:5px; right:10px; display:inline-block; line-height:15px; padding-left:23px; } \
div.deezer-profile-fan-favorites { position:absolute; top:30px; left:5px; right:10px; display:inline-block; line-height:15px; padding-left:23px; } \
");

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

            core.variables.callback.change.push(cover.album);
            core.variables.callback.change.push(cover.fans);

            setInterval(cover.switchAlbumGenre, 2000);
        },
        album: function()
        {
            if (core.variables.cache.albums && core.variables.cache.favorites && core.variables.cache.followings && core.variables.cache.followingsDetail)
            {
                var albumElements = $(".thumbnail-grid a[href^='/album/']").not(".deezer-ignore");

                var loadAlbum = function(id, name)
                {
                    var createElement = function(id)
                    {
                        var figure = $(".thumbnail-grid a[href='/album/" + id + "']").closest(".thumbnail-col").find("figure");
                        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++;
                                    }
                                }
                            }
                        }

                        var cover = $('<div/>')
                        .addClass("deezer-profile-album-cover")
                        .appendTo(figure);

                        if (albumItem.genres.data.length > 0)
                        {
                            var genres = $('<div/>')
                            .addClass("deezer-profile-album-genre deezer-icon-genre")
                            .appendTo(cover);

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

                        var coverInfo = $('<div/>')
                        .addClass("deezer-profile-album-info")
                        .appendTo(cover);

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


                    };

                    if (!core.variables.cache.albumsDetail[id])
                    {
                        core.queueAjax("Album (" + name + ")...", w.location.protocol + "//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++)
                {
                    albumElements.eq(i).addClass("deezer-ignore");

                    if (albumElements.eq(i).attr("href").match(/\/album\/([\d]+)/) != null)
                        loadAlbum(albumElements.eq(i).attr("href").match(/\/album\/([\d]+)/)[1], albumElements.eq(i).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 = $(".thumbnail-grid a[href^='/profile/']").not(".deezer-ignore");

                var loadFan = function(fanElement, id, name)
                {
                    var renderFan = function(id)
                    {
                        if (!core.variables.cache.fans[id].albums || !core.variables.cache.fans[id].artists || !core.variables.cache.fans[id].favorites)
                        {
                            fanElement.removeClass("deezer-ignore");
                            return;
                        }

                        var figure = fanElement.closest(".thumbnail-col").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;
                        var mineFavorites = core.variables.cache.favorites.map(function(a) { return a.id; });
                        var fanFavorites = core.variables.cache.fans[id].favorites.map(function(a) { return a.id; });
                        var intersectFavorites = mineFavorites.filter(function(a) { return fanFavorites.indexOf(a) > -1; }).length;

                        var cover = $('<div/>')
                        .addClass("deezer-profile-fan-cover")
                        .appendTo(figure);

                        $('<div/>')
                            .addClass("deezer-profile-fan-album deezer-icon-album")
                            .addClass(intersectAlbums === 0 ? "deezer-fan-inactive" : "")
                            .text((intersectAlbums > 0 ? Math.round(intersectAlbums / mineAlbums.length * 100) : 0) + "% (" + intersectAlbums + " of " + fanAlbums.length + ")")
                            .appendTo(cover);
                        $('<div/>')
                            .addClass("deezer-profile-fan-artist deezer-icon-artist")
                            .addClass(intersectArtists === 0 ? "deezer-fan-inactive" : "")
                            .text((intersectArtists > 0 ? Math.round(intersectArtists / mineArtists.length * 100) : 0) + "% (" + intersectArtists + " of " + fanArtists.length + ")")
                            .appendTo(cover);
                        $('<div/>')
                            .addClass("deezer-profile-fan-favorites deezer-icon-track")
                            .addClass(intersectFavorites === 0 ? "deezer-fan-inactive" : "")
                            .text((intersectFavorites > 0 ? Math.round(intersectFavorites / mineFavorites.length * 100) : 0) + "% (" + intersectFavorites + " of " + fanFavorites.length + ")")
                            .appendTo(cover);
                    };

                    if (!core.variables.cache.fans[id])
                    {
                        core.variables.cache.fans[id] = {
                            albums: [],
                            artists: []
                        };
                        core.queueAjax("Fan albums (" + name + ")...", w.location.protocol + "//api.deezer.com/user/" + id + "/albums", function(data)
                                       {
                            core.variables.cache.fans[id].albums = data || [];
                        });
                        core.queueAjax("Fan artists (" + name + ")...", w.location.protocol + "//api.deezer.com/user/" + id + "/artists", function(data)
                                       {
                            core.variables.cache.fans[id].artists = data || [];
                        });
                        core.queueAjax("Fan favorites (" + name + ")...", w.location.protocol + "//api.deezer.com/user/" + id + "/tracks", function(data)
                                       {
                            core.variables.cache.fans[id].favorites = data || [];
                            renderFan(id);
                        });
                    }
                    else
                        renderFan(id);
                };

                for (var i = 0; i < fans.length; i++)
                {
                    fans.eq(i).addClass("deezer-ignore");

                    if (fans.eq(i).attr("href").match(/\/profile\/([\d]+)/) != null)
                        loadFan(fans.eq(i), fans.eq(i).attr("href").match(/\/profile\/([\d]+)/)[1], fans.eq(i).text());
                }
            }
        }
    };

    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
            {
                var x = 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);