/** * @file llfloaterchat.cpp * @brief LLFloaterChat class implementation * * $LicenseInfo:firstyear=2002&license=viewergpl$ * * Copyright (c) 2002-2009, Linden Research, Inc. * * Second Life Viewer Source Code * The source code in this file ("Source Code") is provided by Linden Lab * to you under the terms of the GNU General Public License, version 2.0 * ("GPL"), unless you have obtained a separate licensing agreement * ("Other License"), formally executed by you and Linden Lab. Terms of * the GPL can be found in doc/GPL-license.txt in this distribution, or * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2 * * There are special exceptions to the terms and conditions of the GPL as * it is applied to this Source Code. View the full text of the exception * in the file doc/FLOSS-exception.txt in this software distribution, or * online at * http://secondlifegrid.net/programs/open_source/licensing/flossexception * * By copying, modifying or distributing this software, you acknowledge * that you have read and understood your obligations described above, * and agree to abide by those obligations. * * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, * COMPLETENESS OR PERFORMANCE. * $/LicenseInfo$ */ /** * Actually the "Chat History" floater. * Should be llfloaterchathistory, not llfloaterchat. */ #include "llviewerprecompiledheaders.h" #include "llfloaterchat.h" // project include #include "llagent.h" #include "llbutton.h" #include "llcheckboxctrl.h" #include "llcombobox.h" #include "llconsole.h" #include "llfloateractivespeakers.h" #include "llfloaterchatterbox.h" #include "llfloatermute.h" #include "llfloaterreg.h" #include "llfloaterscriptdebug.h" #include "llkeyboard.h" //#include "lllineeditor.h" #include "llmutelist.h" //#include "llresizehandle.h" #include "llchatbar.h" #include "llrecentpeople.h" #include "llstatusbar.h" #include "llviewertexteditor.h" #include "llviewergesture.h" // for triggering gestures #include "llviewermessage.h" #include "llviewerwindow.h" #include "llviewercontrol.h" #include "lluictrlfactory.h" #include "llchatbar.h" #include "lllogchat.h" #include "lltexteditor.h" #include "lltextparser.h" #include "llfloaterhtml.h" #include "llweb.h" #include "llstylemap.h" // linden library includes #include "audioengine.h" #include "llchat.h" #include "llfontgl.h" #include "llrect.h" #include "llerror.h" #include "llstring.h" #include "llwindow.h" #include "message.h" // // Constants // const F32 INSTANT_MSG_SIZE = 8.0f; const F32 CHAT_MSG_SIZE = 8.0f; // // Global statics // LLColor4 get_text_color(const LLChat& chat); // // Member Functions // LLFloaterChat::LLFloaterChat(const LLSD& seed) : LLFloater(), mPanel(NULL) { mFactoryMap["chat_panel"] = LLCallbackMap(createChatPanel, NULL); mFactoryMap["active_speakers_panel"] = LLCallbackMap(createSpeakersPanel, NULL); //Called from floater reg: LLUICtrlFactory::getInstance()->buildFloater(this,"floater_chat_history.xml"); } LLFloaterChat::~LLFloaterChat() { // Children all cleaned up by default view destructor. } void LLFloaterChat::setVisible(BOOL visible) { LLFloater::setVisible( visible ); } void LLFloaterChat::draw() { // enable say and shout only when text available childSetValue("toggle_active_speakers_btn", childIsVisible("active_speakers_panel")); LLChatBar* chat_barp = findChild("chat_panel", TRUE); if (chat_barp) { chat_barp->refresh(); } mPanel->refreshSpeakers(); LLFloater::draw(); } BOOL LLFloaterChat::postBuild() { mPanel = (LLPanelActiveSpeakers*)getChild("active_speakers_panel"); LLChatBar* chat_barp = findChild("chat_panel", TRUE); if (chat_barp) { chat_barp->setGestureCombo(getChild( "Gesture")); } childSetCommitCallback("show mutes",onClickToggleShowMute,this); //show mutes childSetVisible("Chat History Editor with mute",FALSE); childSetAction("toggle_active_speakers_btn", onClickToggleActiveSpeakers, this); return TRUE; } // public virtual void LLFloaterChat::onClose(bool app_quitting) { if (getHost()) { getHost()->setVisible(FALSE); } else { setVisible(FALSE); } } void LLFloaterChat::onVisibilityChange(BOOL new_visibility) { // Hide the chat overlay when our history is visible. updateConsoleVisibility(); LLFloater::onVisibilityChange(new_visibility); } void LLFloaterChat::setMinimized(BOOL minimized) { LLFloater::setMinimized(minimized); updateConsoleVisibility(); } void LLFloaterChat::updateConsoleVisibility() { // determine whether we should show console due to not being visible gConsole->setVisible( !isInVisibleChain() // are we not in part of UI being drawn? || isMinimized() // are we minimized? || (getHost() && getHost()->isMinimized() )); // are we hosted in a minimized floater? } void add_timestamped_line(LLViewerTextEditor* edit, LLChat chat, const LLColor4& color) { std::string line = chat.mText; bool prepend_newline = true; if (gSavedSettings.getBOOL("ChatShowTimestamps")) { edit->appendTime(prepend_newline); prepend_newline = false; } // If the msg is from an agent (not yourself though), // extract out the sender name and replace it with the hotlinked name. if (chat.mSourceType == CHAT_SOURCE_AGENT && chat.mFromID != LLUUID::null) { chat.mURL = llformat("secondlife:///app/agent/%s/about",chat.mFromID.asString().c_str()); } // If the chat line has an associated url, link it up to the name. if (!chat.mURL.empty() && (line.length() > chat.mFromName.length() && line.find(chat.mFromName,0) == 0)) { std::string start_line = line.substr(0, chat.mFromName.length() + 1); line = line.substr(chat.mFromName.length() + 1); const LLStyleSP &sourceStyle = LLStyleMap::instance().lookup(chat.mFromID,chat.mURL); edit->appendStyledText(start_line, false, prepend_newline, sourceStyle); prepend_newline = false; } edit->appendColoredText(line, false, prepend_newline, color); } void log_chat_text(const LLChat& chat) { std::string histstr; if (gSavedPerAccountSettings.getBOOL("LogChatTimestamp")) histstr = LLLogChat::timestamp(gSavedPerAccountSettings.getBOOL("LogTimestampDate")) + chat.mText; else histstr = chat.mText; LLLogChat::saveHistory(std::string("chat"),histstr); } // static void LLFloaterChat::addChatHistory(const LLChat& chat, bool log_to_file) { if ( gSavedPerAccountSettings.getBOOL("LogChat") && log_to_file) { log_chat_text(chat); } LLColor4 color = get_text_color(chat); if (!log_to_file) color = LLColor4::grey; //Recap from log file. if (chat.mChatType == CHAT_TYPE_DEBUG_MSG) { LLFloaterScriptDebug::addScriptLine(chat.mText, chat.mFromName, color, chat.mFromID); if (!gSavedSettings.getBOOL("ScriptErrorsAsChat")) { return; } } // could flash the chat button in the status bar here. JC LLFloaterChat* chat_floater = LLFloaterChat::getInstance(); LLViewerTextEditor* history_editor = chat_floater->getChild("Chat History Editor"); LLViewerTextEditor* history_editor_with_mute = chat_floater->getChild("Chat History Editor with mute"); history_editor->setParseHTML(TRUE); history_editor_with_mute->setParseHTML(TRUE); history_editor->setParseHighlights(TRUE); history_editor_with_mute->setParseHighlights(TRUE); if (!chat.mMuted) { add_timestamped_line(history_editor, chat, color); add_timestamped_line(history_editor_with_mute, chat, color); } else { // desaturate muted chat LLColor4 muted_color = lerp(color, LLColor4::grey, 0.5f); add_timestamped_line(history_editor_with_mute, chat, color); } // add objects as transient speakers that can be muted if (chat.mSourceType == CHAT_SOURCE_OBJECT) { chat_floater->mPanel->setSpeaker(chat.mFromID, chat.mFromName, LLSpeaker::STATUS_NOT_IN_CHANNEL, LLSpeaker::SPEAKER_OBJECT); } // start tab flashing on incoming text from other users (ignoring system text, etc) if (!chat_floater->isInVisibleChain() && chat.mSourceType == CHAT_SOURCE_AGENT) { LLFloaterChatterBox::getInstance()->setFloaterFlashing(chat_floater, TRUE); } } // static void LLFloaterChat::setHistoryCursorAndScrollToEnd() { LLViewerTextEditor* history_editor = LLFloaterChat::getInstance()->getChild("Chat History Editor"); LLViewerTextEditor* history_editor_with_mute = LLFloaterChat::getInstance()->getChild("Chat History Editor with mute"); if (history_editor) { history_editor->setCursorAndScrollToEnd(); } if (history_editor_with_mute) { history_editor_with_mute->setCursorAndScrollToEnd(); } } //static void LLFloaterChat::onClickMute(void *data) { LLFloaterChat* self = (LLFloaterChat*)data; LLComboBox* chatter_combo = self->getChild("chatter combobox"); const std::string& name = chatter_combo->getSimple(); LLUUID id = chatter_combo->getCurrentID(); if (name.empty()) return; LLMute mute(id); mute.setFromDisplayName(name); LLMuteList::getInstance()->add(mute); LLFloaterReg::showInstance("mute"); } //static void LLFloaterChat::onClickToggleShowMute(LLUICtrl* caller, void *data) { LLFloaterChat* floater = (LLFloaterChat*)data; //LLCheckBoxCtrl* BOOL show_mute = floater->getChild("show mutes")->get(); LLViewerTextEditor* history_editor = floater->getChild("Chat History Editor"); LLViewerTextEditor* history_editor_with_mute = floater->getChild("Chat History Editor with mute"); if (!history_editor || !history_editor_with_mute) return; //BOOL show_mute = floater->mShowMuteCheckBox->get(); if (show_mute) { history_editor->setVisible(FALSE); history_editor_with_mute->setVisible(TRUE); history_editor_with_mute->setCursorAndScrollToEnd(); } else { history_editor->setVisible(TRUE); history_editor_with_mute->setVisible(FALSE); history_editor->setCursorAndScrollToEnd(); } } // Put a line of chat in all the right places void LLFloaterChat::addChat(const LLChat& chat, BOOL from_instant_message, BOOL local_agent) { LLColor4 text_color = get_text_color(chat); BOOL invisible_script_debug_chat = chat.mChatType == CHAT_TYPE_DEBUG_MSG && !gSavedSettings.getBOOL("ScriptErrorsAsChat"); if (!invisible_script_debug_chat && !chat.mMuted && gConsole && !local_agent) { F32 size = CHAT_MSG_SIZE; if (chat.mSourceType == CHAT_SOURCE_SYSTEM) { text_color = gSavedSkinSettings.getColor("SystemChatColor"); } else if(from_instant_message) { text_color = gSavedSkinSettings.getColor("IMChatColor"); size = INSTANT_MSG_SIZE; } // We display anything if it's not an IM. If it's an IM, check pref... if ( !from_instant_message || gSavedSettings.getBOOL("IMInChatConsole") ) { gConsole->addLine(chat.mText, size, text_color); } } if(from_instant_message && gSavedPerAccountSettings.getBOOL("LogChatIM")) log_chat_text(chat); if(from_instant_message && gSavedSettings.getBOOL("IMInChatHistory")) addChatHistory(chat,false); triggerAlerts(chat.mText); // Add the sender to the list of people with which we've recently interacted. if(chat.mSourceType == CHAT_SOURCE_AGENT && chat.mFromID.notNull()) LLRecentPeople::instance().add(chat.mFromID); if(!from_instant_message) addChatHistory(chat); } // Moved from lltextparser.cpp to break llui/llaudio library dependency. //static void LLFloaterChat::triggerAlerts(const std::string& text) { LLTextParser* parser = LLTextParser::getInstance(); // bool spoken=FALSE; for (S32 i=0;imHighlights.size();i++) { LLSD& highlight = parser->mHighlights[i]; if (parser->findPattern(text,highlight) >= 0 ) { if(gAudiop) { if ((std::string)highlight["sound_lluuid"] != LLUUID::null.asString()) { gAudiop->triggerSound(highlight["sound_lluuid"].asUUID(), gAgent.getID(), 1.f, LLAudioEngine::AUDIO_TYPE_UI, gAgent.getPositionGlobal() ); } /* if (!spoken) { LLTextToSpeech* text_to_speech = NULL; text_to_speech = LLTextToSpeech::getInstance(); spoken = text_to_speech->speak((LLString)highlight["voice"],text); } */ } if (highlight["flash"]) { LLWindow* viewer_window = gViewerWindow->getWindow(); if (viewer_window && viewer_window->getMinimized()) { viewer_window->flashIcon(5.f); } } } } } LLColor4 get_text_color(const LLChat& chat) { LLColor4 text_color; if(chat.mMuted) { text_color.setVec(0.8f, 0.8f, 0.8f, 1.f); } else { switch(chat.mSourceType) { case CHAT_SOURCE_SYSTEM: text_color = gSavedSkinSettings.getColor4("SystemChatColor"); break; case CHAT_SOURCE_AGENT: if (chat.mFromID.isNull()) { text_color = gSavedSkinSettings.getColor4("SystemChatColor"); } else { if(gAgent.getID() == chat.mFromID) { text_color = gSavedSkinSettings.getColor4("UserChatColor"); } else { text_color = gSavedSkinSettings.getColor4("AgentChatColor"); } } break; case CHAT_SOURCE_OBJECT: if (chat.mChatType == CHAT_TYPE_DEBUG_MSG) { text_color = gSavedSkinSettings.getColor4("ScriptErrorColor"); } else if ( chat.mChatType == CHAT_TYPE_OWNER ) { text_color = gSavedSkinSettings.getColor4("llOwnerSayChatColor"); } else { text_color = gSavedSkinSettings.getColor4("ObjectChatColor"); } break; default: text_color.setToWhite(); } if (!chat.mPosAgent.isExactlyZero()) { LLVector3 pos_agent = gAgent.getPositionAgent(); F32 distance = dist_vec(pos_agent, chat.mPosAgent); if (distance > gAgent.getNearChatRadius()) { // diminish far-off chat text_color.mV[VALPHA] = 0.8f; } } } return text_color; } //static void LLFloaterChat::loadHistory() { LLLogChat::loadHistory(std::string("chat"), &chatFromLogFile, (void *)LLFloaterChat::getInstance()); } //static void LLFloaterChat::chatFromLogFile(LLLogChat::ELogLineType type , std::string line, void* userdata) { switch (type) { case LLLogChat::LOG_EMPTY: case LLLogChat::LOG_END: // *TODO: nice message from XML file here break; case LLLogChat::LOG_LINE: { LLChat chat; chat.mText = line; get_text_color(chat); addChatHistory(chat, FALSE); } break; default: // nothing break; } } //static void* LLFloaterChat::createSpeakersPanel(void* data) { return new LLPanelActiveSpeakers(LLLocalSpeakerMgr::getInstance(), TRUE); } //static void* LLFloaterChat::createChatPanel(void* data) { LLChatBar* chatp = new LLChatBar(); return chatp; } // static void LLFloaterChat::onClickToggleActiveSpeakers(void* userdata) { LLFloaterChat* self = (LLFloaterChat*)userdata; self->childSetVisible("active_speakers_panel", !self->childIsVisible("active_speakers_panel")); } //static LLFloaterChat* LLFloaterChat::getInstance() { LLFloater* inst = LLFloaterReg::getInstance("chat", LLSD()) ; return dynamic_cast(inst); }