/**
 * @file llblocklist.cpp
 * @brief List of the blocked avatars and objects.
 *
 * $LicenseInfo:firstyear=2012&license=viewerlgpl$
 * Second Life Viewer Source Code
 * Copyright (C) 2012, Linden Research, Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation;
 * version 2.1 of the License only.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 * $/LicenseInfo$
 */

#include "llviewerprecompiledheaders.h"

#include "llblocklist.h"

#include "llavataractions.h"
#include "llblockedlistitem.h"
#include "llfloatersidepanelcontainer.h"
#include "llviewermenu.h"

static LLDefaultChildRegistry::Register<LLBlockList> r("block_list");

static const LLBlockListNameComparator 		NAME_COMPARATOR;
static const LLBlockListNameTypeComparator	NAME_TYPE_COMPARATOR;

LLBlockList::LLBlockList(const Params& p)
:	LLFlatListViewEx(p),
 	mDirty(true),
	mShouldAddAll(true),
	mActionType(NONE),
	mMuteListSize(0)
{

	LLMuteList::getInstance()->addObserver(this);
	mMuteListSize = LLMuteList::getInstance()->getMutes().size();

	// Set up context menu.
	LLUICtrl::CommitCallbackRegistry::ScopedRegistrar registrar;
	LLUICtrl::EnableCallbackRegistry::ScopedRegistrar enable_registrar;

	registrar.add		("Block.Action",	boost::bind(&LLBlockList::onCustomAction,	this, _2));
	enable_registrar.add("Block.Enable",	boost::bind(&LLBlockList::isActionEnabled,	this, _2));
	enable_registrar.add("Block.Check",     boost::bind(&LLBlockList::isMenuItemChecked, this, _2));
	enable_registrar.add("Block.Visible",   boost::bind(&LLBlockList::isMenuItemVisible, this, _2));
	
	LLToggleableMenu* context_menu = LLUICtrlFactory::getInstance()->createFromFile<LLToggleableMenu>(
									"menu_people_blocked_gear.xml",
									gMenuHolder,
									LLViewerMenuHolderGL::child_registry_t::instance());
	if(context_menu)
	{
		mContextMenu = context_menu->getHandle();
	}
}

LLBlockList::~LLBlockList()
{
	if (mContextMenu.get())
	{
		mContextMenu.get()->die();
	}

	LLMuteList::getInstance()->removeObserver(this);
}

void LLBlockList::createList()
{
	std::vector<LLMute> mutes = LLMuteList::instance().getMutes();
	std::vector<LLMute>::const_iterator mute_it = mutes.begin();

	for (; mute_it != mutes.end(); ++mute_it)
	{
		addNewItem(&*mute_it);
	}
}

BlockListActionType LLBlockList::getCurrentMuteListActionType()
{
	BlockListActionType type = NONE;
	U32 curSize = LLMuteList::getInstance()->getMutes().size();
	if( curSize > mMuteListSize)
		type = ADD;
	else if(curSize < mMuteListSize)
		type = REMOVE;

	return type;
}

void LLBlockList::onChangeDetailed(const LLMute &mute)
{
	mActionType = getCurrentMuteListActionType();

	mCurItemId = mute.mID;
	mCurItemName = mute.mName;
	mCurItemType = mute.mType;
	mCurItemFlags = mute.mFlags;

	refresh();
}

BOOL LLBlockList::handleRightMouseDown(S32 x, S32 y, MASK mask)
{
	BOOL handled = LLUICtrl::handleRightMouseDown(x, y, mask);

	LLToggleableMenu* context_menu = mContextMenu.get();
	if (context_menu && size())
	{
		context_menu->buildDrawLabels();
		context_menu->updateParent(LLMenuGL::sMenuContainer);
		LLMenuGL::showPopup(this, context_menu, x, y);
	}

	return handled;
}

void LLBlockList::removeListItem(const LLMute* mute)
{
	if (mute->mID.notNull())
	{
		removeItemByUUID(mute->mID);
	}
	else
	{
		removeItemByValue(mute->mName);
	}
}

void LLBlockList::hideListItem(LLBlockedListItem* item, bool show)
{
	item->setVisible(show);
}

void LLBlockList::setNameFilter(const std::string& filter)
{
	std::string filter_upper = filter;
	LLStringUtil::toUpper(filter_upper);
	if (mNameFilter != filter_upper)
	{
		mNameFilter = filter_upper;
		setDirty();
	}
}

void LLBlockList::sortByName()
{
	setComparator(&NAME_COMPARATOR);
	sort();
}

void LLBlockList::sortByType()
{
	setComparator(&NAME_TYPE_COMPARATOR);
	sort();
}

void LLBlockList::draw()
{
	if (mDirty)
	{
		refresh();
	}

	LLFlatListView::draw();
}

void LLBlockList::addNewItem(const LLMute* mute)
{
	LLBlockedListItem* item = new LLBlockedListItem(mute);
	if (!mNameFilter.empty())
	{
		item->highlightName(mNameFilter);
	}
	if (item->getUUID().notNull())
	{
		addItem(item, item->getUUID(), ADD_BOTTOM);
	}
	else
	{
		addItem(item, item->getName(), ADD_BOTTOM);
	}
}

void LLBlockList::refresh()
{
	bool have_filter = !mNameFilter.empty();

	// save selection to restore it after list rebuilt
	LLSD selected = getSelectedValue();
	LLSD next_selected;

	if(mShouldAddAll)	// creating list of blockers
	{
		clear();
		createList();
		mShouldAddAll = false;
	}
	else
	{
		// handle remove/add functionality
		LLMute mute(mCurItemId, mCurItemName, mCurItemType, mCurItemFlags);
		if(mActionType == ADD)
		{
			addNewItem(&mute);
		}
		else if(mActionType == REMOVE)
		{
			if ((mute.mID.notNull() && selected.isUUID() && selected.asUUID() == mute.mID)
				|| (mute.mID.isNull() && selected.isString() && selected.asString() == mute.mName))
			{
				// we are going to remove currently selected item, so select next item and save the selection to restore it
				if (!selectNextItemPair(false, true))
				{
					selectNextItemPair(true, true);
				}
				next_selected = getSelectedValue();
			}
			removeListItem(&mute);
		}
		mActionType = NONE;
	}

	// handle filter functionality
	if(have_filter || (!have_filter && !mPrevNameFilter.empty()))
	{
		// we should update visibility of our items if previous filter was not empty
		std::vector < LLPanel* > allItems;
		getItems(allItems);
		std::vector < LLPanel* >::iterator it = allItems.begin();

		for(; it != allItems.end() ; ++it)
		{
			LLBlockedListItem * curItem = dynamic_cast<LLBlockedListItem *> (*it);
			if(curItem)
	{
				hideListItem(curItem, findInsensitive(curItem->getName(), mNameFilter));
			}
		}
	}
	mPrevNameFilter = mNameFilter;

	if (selected.isDefined())
	{
		if (getItemPair(selected))
		{
			// restore previously selected item
			selectItemPair(getItemPair(selected), true);
		}
		else if (next_selected.isDefined() && getItemPair(next_selected))
		{
			// previously selected item was removed, so select next item
			selectItemPair(getItemPair(next_selected), true);
		}
	}
	mMuteListSize = LLMuteList::getInstance()->getMutes().size();

	// Sort the list.
	sort();

	setDirty(false);
}

bool LLBlockList::findInsensitive(std::string haystack, const std::string& needle_upper)
{
    LLStringUtil::toUpper(haystack);
    return haystack.find(needle_upper) != std::string::npos;
}

LLBlockedListItem* LLBlockList::getBlockedItem() const
{
	LLPanel* panel = LLFlatListView::getSelectedItem();
	LLBlockedListItem* item = dynamic_cast<LLBlockedListItem*>(panel);
	return item;
}

bool LLBlockList::isActionEnabled(const LLSD& userdata)
{
	bool action_enabled = true;

	const std::string command_name = userdata.asString();

	if ("profile_item" == command_name 
		|| "block_voice" == command_name
		|| "block_text" == command_name
		|| "block_particles" == command_name
		|| "block_obj_sounds" == command_name)
	{
		LLBlockedListItem* item = getBlockedItem();
		action_enabled = item && (LLMute::AGENT == item->getType());
	}

	if ("unblock_item" == command_name)
	{
		action_enabled = getSelectedItem() != NULL;
	}

	return action_enabled;
}

void LLBlockList::onCustomAction(const LLSD& userdata)
{
	if (!isActionEnabled(userdata))
	{
		return;
	}

	LLBlockedListItem* item = getBlockedItem();
	const std::string command_name = userdata.asString();

	if ("unblock_item" == command_name)
	{
		LLMute mute(item->getUUID(), item->getName());
		LLMuteList::getInstance()->remove(mute);
	}
	else if ("profile_item" == command_name)
	{
		switch(item->getType())
		{

		case LLMute::AGENT:
			LLAvatarActions::showProfile(item->getUUID());
			break;

		default:
			break;
		}
	}
	else if ("block_voice" == command_name)
	{
		toggleMute(LLMute::flagVoiceChat);
	}
	else if ("block_text" == command_name)
	{
		toggleMute(LLMute::flagTextChat);
	}
	else if ("block_particles" == command_name)
	{
		toggleMute(LLMute::flagParticles);
	}
	else if ("block_obj_sounds" == command_name)
	{
		toggleMute(LLMute::flagObjectSounds);
	}
}

bool LLBlockList::isMenuItemChecked(const LLSD& userdata)
{
	LLBlockedListItem* item = getBlockedItem();
	if (!item)
	{
		return false;
	}

	const std::string command_name = userdata.asString();

	if ("block_voice" == command_name)
	{
		return LLMuteList::getInstance()->isMuted(item->getUUID(), LLMute::flagVoiceChat);
	}
	else if ("block_text" == command_name)
	{
		return LLMuteList::getInstance()->isMuted(item->getUUID(), LLMute::flagTextChat);
	}
	else if ("block_particles" == command_name)
	{
		return LLMuteList::getInstance()->isMuted(item->getUUID(), LLMute::flagParticles);
	}
	else if ("block_obj_sounds" == command_name)
	{
		return LLMuteList::getInstance()->isMuted(item->getUUID(), LLMute::flagObjectSounds);
	}

	return false;
}

bool LLBlockList::isMenuItemVisible(const LLSD& userdata)
{
	LLBlockedListItem* item = getBlockedItem();
	const std::string command_name = userdata.asString();

	if ("block_voice" == command_name
		|| "block_text" == command_name
		|| "block_particles" == command_name
		|| "block_obj_sounds" == command_name)
	{
		return item && (LLMute::AGENT == item->getType());
	}

	return false;
}

void LLBlockList::toggleMute(U32 flags)
{
	LLBlockedListItem* item = getBlockedItem();
	LLMute mute(item->getUUID(), item->getName(), item->getType());

	if (!LLMuteList::getInstance()->isMuted(item->getUUID(), flags))
	{
		LLMuteList::getInstance()->add(mute, flags);
	}
	else
	{
		LLMuteList::getInstance()->remove(mute, flags);
	}
}

bool LLBlockListItemComparator::compare(const LLPanel* item1, const LLPanel* item2) const
{
	const LLBlockedListItem* blocked_item1 = dynamic_cast<const LLBlockedListItem*>(item1);
	const LLBlockedListItem* blocked_item2 = dynamic_cast<const LLBlockedListItem*>(item2);

	if (!blocked_item1 || !blocked_item2)
	{
		LL_ERRS() << "blocked_item1 and blocked_item2 cannot be null" << LL_ENDL;
		return true;
	}

	return doCompare(blocked_item1, blocked_item2);
}

bool LLBlockListNameComparator::doCompare(const LLBlockedListItem* blocked_item1, const LLBlockedListItem* blocked_item2) const
{
	std::string name1 = blocked_item1->getName();
	std::string name2 = blocked_item2->getName();

	LLStringUtil::toUpper(name1);
	LLStringUtil::toUpper(name2);

	return name1 < name2;
}

bool LLBlockListNameTypeComparator::doCompare(const LLBlockedListItem* blocked_item1, const LLBlockedListItem* blocked_item2) const
{
	LLMute::EType type1 = blocked_item1->getType();
	LLMute::EType type2 = blocked_item2->getType();

	// if mute type is LLMute::BY_NAME or LLMute::OBJECT it means that this mute is an object
	bool both_mutes_are_objects = (LLMute::OBJECT == type1 || LLMute::BY_NAME == type1) && (LLMute::OBJECT == type2 || LLMute::BY_NAME == type2);

	// mute types may be different, but since both LLMute::BY_NAME and LLMute::OBJECT types represent objects
	// it's needed to perform additional checking of both_mutes_are_objects variable
	if (type1 != type2 && !both_mutes_are_objects)
	{
		// objects in block list go first, so return true if mute type is not an avatar
		return LLMute::AGENT != type1;
	}

	return NAME_COMPARATOR.compare(blocked_item1, blocked_item2);
}