//  SuperTuxKart - a fun racing game with go-kart
//  Copyright (C) 2013-2015 Glenn De Jonghe
//
//  This program is free software; you can redistribute it and/or
//  modify it under the terms of the GNU General Public License
//  as published by the Free Software Foundation; either version 3
//  of the License, or (at your option) any later version.
//
//  This program is distributed in the hope that it will be useful,
//  but WITHOUT ANY WARRANTY; without even the implied warranty of
//  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
//  GNU General Public License for more details.
//
//  You should have received a copy of the GNU General Public License
//  along with this program; if not, write to the Free Software
//  Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.

#include "states_screens/online/online_profile_friends.hpp"

#include "guiengine/engine.hpp"
#include "guiengine/scalable_font.hpp"
#include "guiengine/screen.hpp"
#include "guiengine/widget.hpp"
#include "states_screens/dialogs/user_info_dialog.hpp"
#include "states_screens/online/online_user_search.hpp"
#include "states_screens/state_manager.hpp"
#include "utils/string_utils.hpp"
#include "utils/translation.hpp"

#include <IGUIButton.h>

using namespace GUIEngine;
using namespace irr::core;
using namespace irr::gui;
using namespace Online;

int OnlineProfileFriends::m_sort_column      = 0;
bool OnlineProfileFriends::m_sort_desc       = false;
bool OnlineProfileFriends::m_sort_default    = true;

// -----------------------------------------------------------------------------
/** Constructor for a display of all friends.
 */
OnlineProfileFriends::OnlineProfileFriends()
                    : OnlineProfileBase("online/profile_friends.stkgui")
{
}   // OnlineProfileFriends

// -----------------------------------------------------------------------------
/** Callback when the xml file was loaded.
 */
void OnlineProfileFriends::loadedFromFile()
{
    OnlineProfileBase::loadedFromFile();
    m_friends_list_widget = getWidget<ListWidget>("friends_list");
    assert(m_friends_list_widget != NULL);
    m_search_button_widget = getWidget<ButtonWidget>("search_button");
    assert(m_search_button_widget != NULL);
    m_search_box_widget = getWidget<TextBoxWidget>("search_box");
    assert(m_search_box_widget != NULL);
    m_friends_list_widget->setColumnListener(this);

}   // loadedFromFile

// ----------------------------------------------------------------------------
/** Callback before widgets are added. Clears all widgets.
*/
void OnlineProfileFriends::beforeAddingWidget()
{
    OnlineProfileBase::beforeAddingWidget();
    m_friends_list_widget->clearColumns();
    m_friends_list_widget->addColumn( _("Username"), 2 );
    if(m_visiting_profile->isCurrentUser())
    {
        m_friends_list_widget->addColumn( _("Since"), 1 );
        m_friends_list_widget->addColumn( _("Status"), 2 );
    }
}   // beforeAddingWidget

// -----------------------------------------------------------------------------
/** Called when entering this menu (before widgets are added).
 */
void OnlineProfileFriends::init()
{
    OnlineProfileBase::init();
    m_profile_tabs->select( m_friends_tab->m_properties[PROP_ID],
                            PLAYER_ID_GAME_MASTER );
    m_profile_tabs->setFocusForPlayer(PLAYER_ID_GAME_MASTER);
    assert(m_visiting_profile != NULL);
    m_visiting_profile->fetchFriends();
    m_waiting_for_friends = true;
    m_friends_list_widget->clear();
    m_friends_list_widget->addItem("loading",
                              StringUtils::loadingDots(_("Fetching friends")));
}   // init

// -----------------------------------------------------------------------------
/** Called when an event occurs (i.e. user clicks on something).
*/
void OnlineProfileFriends::eventCallback(Widget* widget,
                                         const std::string& name,
                                         const int playerID)
{
    OnlineProfileBase::eventCallback( widget, name, playerID);
    if (name == m_search_button_widget->m_properties[GUIEngine::PROP_ID])
    {
        OnlineUserSearch * instance = OnlineUserSearch::getInstance();
        instance->setSearchString(m_search_box_widget->getText().trim());
        StateManager::get()->replaceTopMostScreen(instance);
    }
    else if (name == m_friends_list_widget->m_properties[GUIEngine::PROP_ID])
    {
        std::string str_selection = m_friends_list_widget->getSelectionInternalName();
        int id;
        if (StringUtils::parseString(str_selection, &id))
        {
            new UserInfoDialog(id);
        }
    }
}   // eventCallback

// ----------------------------------------------------------------------------
void OnlineProfileFriends::onColumnClicked(int column_id, bool sort_desc, bool sort_default)
{
    m_sort_column = column_id;
    m_sort_desc = sort_desc;
    m_sort_default = sort_default;
    
    if (!m_waiting_for_friends)
    {
        displayResults();
    }
}   // onColumnClicked

// ----------------------------------------------------------------------------
bool OnlineProfileFriends::compareFriends(int f1, int f2)
{
    Online::OnlineProfile *p1 = ProfileManager::get()->getProfileByID(f1);
    Online::OnlineProfile *p2 = ProfileManager::get()->getProfileByID(f2);
    switch (m_sort_column)
    {
    case 0:    // sort by name
        return p1->getUserName().lower_ignore_case(p2->getUserName());
    case 1:   // sort by date
    {
        OnlineProfile::RelationInfo *r1 = p1->getRelationInfo();
        OnlineProfile::RelationInfo *r2 = p2->getRelationInfo();
        // While we are comparing dates that are strings, they are normalised
        // i.e. contain leading 0 etc, so a string compare works as expected.
        return r1->getDate().lower_ignore_case(r2->getDate());
    }
    case 2:   // sort by online status
    {
        OnlineProfile::RelationInfo *r1 = p1->getRelationInfo();
        OnlineProfile::RelationInfo *r2 = p2->getRelationInfo();
        
        if (r1->isPending() && r2->isPending())
        {
            // In case of same online status, sort by name
            if (r1->isAsker() == r2->isAsker())
            {
                return p1->getUserName().lower_ignore_case(p2->getUserName());
            }
            else
            {
                return r1->isAsker() > r2->isAsker();
            }
        }
        else if (r1->isPending() != r2->isPending())
        {
            return r1->isPending() > r2->isPending();
        }
        else
        {
            // In case of same online status, sort by name
            if (r1->isOnline() == r2->isOnline())
            {
                return p1->getUserName().lower_ignore_case(p2->getUserName());
            }
            else
            {
                return r1->isOnline() > r2->isOnline();
            }
        }
    }
    default:
        break;
    }   // switch

    return false;
}   // compareFriends

// ----------------------------------------------------------------------------
/** Displays the friends from a given profile.
 */
void OnlineProfileFriends::displayResults()
{
    m_friends_list_widget->clear();
    
    OnlineProfile::IDList friends = m_visiting_profile->getFriends();
    
    if (m_sort_desc && !m_sort_default)
    {
        std::sort(friends.rbegin(), friends.rend(), compareFriends);
    }
    else
    {
        std::sort(friends.begin(), friends.end(), compareFriends);
    }
                                     
    for (unsigned int i = 0; i < friends.size(); i++)
    {
        std::vector<ListWidget::ListCell> row;
        OnlineProfile* friend_profile =
            ProfileManager::get()->getProfileByID(friends[i]);

        // When looking at friends of a friend those profiles are not
        // guaranteed to be persistent, so they might not be found in cache.
        if (!friend_profile)
        {
            Log::warn("OnlineProfileFriends",
                      "Profile for %d not found - ignored.", friends[i]);
            continue;
        }
        row.push_back(ListWidget::ListCell(friend_profile->getUserName(),
                                           -1, 2)                         );
        if (m_visiting_profile->isCurrentUser())
        {
            OnlineProfile::RelationInfo * relation_info =
                                             friend_profile->getRelationInfo();
            row.push_back(ListWidget::ListCell(relation_info->getDate(),
                                               -1, 1, true)             );
            irr::core::stringw status("");
            if (relation_info->isPending())
            {
                status = (relation_info->isAsker() ? _("New Request")
                                                   : _("Pending")     );
            }
            else
                status = (relation_info->isOnline() ? _("Online")
                                                    : _("Offline") );
            row.push_back(ListWidget::ListCell(status, -1, 2, true));
        }
        m_friends_list_widget->addItem(StringUtils::toString(friends[i]), row);
    }
    m_waiting_for_friends = false;

}   // displayResults

// ----------------------------------------------------------------------------
/** Called each frame to check if results have arrived.
 *  \param delta Time step size.
 */
void OnlineProfileFriends::onUpdate(float delta)
{
    if(m_waiting_for_friends)
    {
        if (m_visiting_profile->hasFetchedFriends())
        {
            // When browsing other users friends there is no Since/Status 
            // columns, so force sorting by name
            if (!m_visiting_profile->isCurrentUser() && m_sort_column != 0)
            {
                m_sort_column = 0;
                m_sort_desc = false;
                m_sort_default = true;
            }
            
            displayResults();
        }
        else
        {
            m_friends_list_widget->renameItem("loading",
                              StringUtils::loadingDots(_("Fetching friends")));
        }
    }
}   // onUpdate
