phoenix-firestorm/indra/newview/llviewergesture.cpp

261 lines
6.2 KiB
C++

/**
* @file llviewergesture.cpp
* @brief LLViewerGesture class implementation
*
* Copyright (c) 2002-$CurrentYear$, Linden Research, Inc.
* $License$
*/
#include "llviewerprecompiledheaders.h"
#include "llviewergesture.h"
#include "audioengine.h"
#include "lldir.h"
#include "llviewerinventory.h"
#include "sound_ids.h" // for testing
#include "llchatbar.h"
#include "llkeyboard.h" // for key shortcuts for testing
#include "llinventorymodel.h"
#include "llvoavatar.h"
#include "llxfermanager.h"
#include "llviewermessage.h" // send_guid_sound_trigger
#include "llviewernetwork.h"
#include "llagent.h"
// Globals
LLViewerGestureList gGestureList;
const F32 LLViewerGesture::SOUND_VOLUME = 1.f;
LLViewerGesture::LLViewerGesture()
: LLGesture()
{ }
LLViewerGesture::LLViewerGesture(KEY key, MASK mask, const std::string &trigger,
const LLUUID &sound_item_id,
const std::string &animation,
const std::string &output_string)
: LLGesture(key, mask, trigger, sound_item_id, animation, output_string)
{
}
LLViewerGesture::LLViewerGesture(U8 **buffer, S32 max_size)
: LLGesture(buffer, max_size)
{
}
LLViewerGesture::LLViewerGesture(const LLViewerGesture &rhs)
: LLGesture((LLGesture)rhs)
{
}
BOOL LLViewerGesture::trigger(KEY key, MASK mask)
{
if (mKey == key && mMask == mask)
{
doTrigger( TRUE );
return TRUE;
}
else
{
return FALSE;
}
}
BOOL LLViewerGesture::trigger(const std::string &trigger_string)
{
// Assumes trigger_string is lowercase
if (mTriggerLower == trigger_string)
{
doTrigger( FALSE );
return TRUE;
}
else
{
return FALSE;
}
}
// private
void LLViewerGesture::doTrigger( BOOL send_chat )
{
if (mSoundItemID != LLUUID::null)
{
LLViewerInventoryItem *item;
item = gInventory.getItem(mSoundItemID);
if (item)
{
send_sound_trigger(item->getAssetUUID(), SOUND_VOLUME);
}
}
if (!mAnimation.empty())
{
// AFK animations trigger the special "away" state, which
// includes agent control settings. JC
if (mAnimation == "enter_away_from_keyboard_state" || mAnimation == "away")
{
gAgent.setAFK();
}
else
{
LLUUID anim_id = gAnimLibrary.stringToAnimState(mAnimation.c_str());
gAgent.sendAnimationRequest(anim_id, ANIM_REQUEST_START);
}
}
if ( send_chat && !mOutputString.empty())
{
// Don't play nodding animation, since that might not blend
// with the gesture animation.
gChatBar->sendChatFromViewer(mOutputString, CHAT_TYPE_NORMAL, FALSE);
}
}
LLViewerGestureList::LLViewerGestureList()
: LLGestureList()
{
mIsLoaded = FALSE;
}
void LLViewerGestureList::saveToServer()
{
U8 *buffer = new U8[getMaxSerialSize()];
U8 *end = serialize(buffer);
if (end - buffer > getMaxSerialSize())
{
llerrs << "Wrote off end of buffer, serial size computation is wrong" << llendl;
}
//U64 xfer_id = gXferManager->registerXfer(buffer, end - buffer);
// write to a file because mem<->mem xfer isn't implemented
LLUUID random_uuid;
char filename[LL_MAX_PATH]; /* Flawfinder: ignore */
random_uuid.generate();
random_uuid.toString(filename);
strcat(filename,".tmp"); /* Flawfinder: ignore */
char filename_and_path[LL_MAX_PATH]; /* Flawfinder: ignore */
snprintf(filename_and_path, LL_MAX_PATH, "%s%s%s", /* Flawfinder: ignore */
gDirUtilp->getTempDir().c_str(),
gDirUtilp->getDirDelimiter().c_str(),
filename);
FILE* fp = LLFile::fopen(filename_and_path, "wb"); /* Flawfinder: ignore */
if (fp)
{
fwrite(buffer, end - buffer, 1, fp);
fclose(fp);
gMessageSystem->newMessageFast(_PREHASH_GestureUpdate);
gMessageSystem->nextBlockFast(_PREHASH_AgentBlock);
gMessageSystem->addUUIDFast(_PREHASH_AgentID, gAgent.getID());
gMessageSystem->addStringFast(_PREHASH_Filename, filename);
gMessageSystem->addBOOLFast(_PREHASH_ToViewer, FALSE);
gMessageSystem->sendReliable(gUserServer);
}
delete[] buffer;
}
/*
void LLViewerGestureList::requestFromServer()
{
gMessageSystem->newMessageFast(_PREHASH_GestureRequest);
gMessageSystem->nextBlockFast(_PREHASH_AgentBlock);
gMessageSystem->addUUIDFast(_PREHASH_AgentID, agent_get_id());
gMessageSystem->addU8("Reset", 0);
gMessageSystem->sendReliable(gUserServer);
}
void LLViewerGestureList::requestResetFromServer( BOOL is_male )
{
gMessageSystem->newMessageFast(_PREHASH_GestureRequest);
gMessageSystem->nextBlockFast(_PREHASH_AgentBlock);
gMessageSystem->addUUIDFast(_PREHASH_AgentID, agent_get_id());
gMessageSystem->addU8("Reset", is_male ? 1 : 2);
gMessageSystem->sendReliable(gUserServer);
mIsLoaded = FALSE;
}
*/
// helper for deserialize that creates the right LLGesture subclass
LLGesture *LLViewerGestureList::create_gesture(U8 **buffer, S32 max_size)
{
return new LLViewerGesture(buffer, max_size);
}
// See if the prefix matches any gesture. If so, return TRUE
// and place the full text of the gesture trigger into
// output_str
BOOL LLViewerGestureList::matchPrefix(const std::string& in_str, std::string* out_str)
{
S32 in_len = in_str.length();
LLString in_str_lc = in_str;
LLString::toLower(in_str_lc);
for (S32 i = 0; i < count(); i++)
{
LLGesture* gesture = get(i);
const std::string &trigger = gesture->getTrigger();
if (in_len > (S32)trigger.length())
{
// too short, bail out
continue;
}
std::string trigger_trunc = utf8str_truncate(trigger, in_len);
LLString::toLower(trigger_trunc);
if (in_str_lc == trigger_trunc)
{
*out_str = trigger;
return TRUE;
}
}
return FALSE;
}
// static
void LLViewerGestureList::xferCallback(void *data, S32 size, void** /*user_data*/, S32 status)
{
if (LL_ERR_NOERR == status)
{
U8 *buffer = (U8 *)data;
U8 *end = gGestureList.deserialize(buffer, size);
if (end - buffer > size)
{
llerrs << "Read off of end of array, error in serialization" << llendl;
}
gGestureList.mIsLoaded = TRUE;
}
else
{
llwarns << "Unable to load gesture list!" << llendl;
}
}
// static
void LLViewerGestureList::processGestureUpdate(LLMessageSystem *msg, void** /*user_data*/)
{
char remote_filename[MAX_STRING]; /* Flawfinder: ignore */
msg->getStringFast(_PREHASH_AgentBlock, _PREHASH_Filename, MAX_STRING, remote_filename);
gXferManager->requestFile(remote_filename, LL_PATH_CACHE, msg->getSender(), TRUE, xferCallback, NULL,
LLXferManager::HIGH_PRIORITY);
}