diff options
Diffstat (limited to 'linden/indra/newview/llfloateractivespeakers.cpp')
-rw-r--r-- | linden/indra/newview/llfloateractivespeakers.cpp | 328 |
1 files changed, 198 insertions, 130 deletions
diff --git a/linden/indra/newview/llfloateractivespeakers.cpp b/linden/indra/newview/llfloateractivespeakers.cpp index 877ebca..8432d21 100644 --- a/linden/indra/newview/llfloateractivespeakers.cpp +++ b/linden/indra/newview/llfloateractivespeakers.cpp | |||
@@ -135,6 +135,16 @@ LLSD LLSpeakerVoiceModerationEvent::getValue() | |||
135 | return LLString("voice"); | 135 | return LLString("voice"); |
136 | } | 136 | } |
137 | 137 | ||
138 | LLSpeakerListChangeEvent::LLSpeakerListChangeEvent(LLSpeakerMgr* source, const LLUUID& speaker_id) | ||
139 | : LLEvent(source, "Speaker added/removed from speaker mgr"), | ||
140 | mSpeakerID(speaker_id) | ||
141 | { | ||
142 | } | ||
143 | |||
144 | LLSD LLSpeakerListChangeEvent::getValue() | ||
145 | { | ||
146 | return mSpeakerID; | ||
147 | } | ||
138 | 148 | ||
139 | // helper sort class | 149 | // helper sort class |
140 | struct LLSortRecentSpeakers | 150 | struct LLSortRecentSpeakers |
@@ -210,9 +220,9 @@ void* LLFloaterActiveSpeakers::createSpeakersPanel(void* data) | |||
210 | } | 220 | } |
211 | 221 | ||
212 | // | 222 | // |
213 | // LLPanelActiveSpeakers::LLSpeakerListener | 223 | // LLPanelActiveSpeakers::SpeakerMuteListener |
214 | // | 224 | // |
215 | bool LLPanelActiveSpeakers::LLSpeakerListener::handleEvent(LLPointer<LLEvent> event, const LLSD& userdata) | 225 | bool LLPanelActiveSpeakers::SpeakerMuteListener::handleEvent(LLPointer<LLEvent> event, const LLSD& userdata) |
216 | { | 226 | { |
217 | LLPointer<LLSpeaker> speakerp = (LLSpeaker*)event->getSource(); | 227 | LLPointer<LLSpeaker> speakerp = (LLSpeaker*)event->getSource(); |
218 | if (speakerp.isNull()) return false; | 228 | if (speakerp.isNull()) return false; |
@@ -231,6 +241,35 @@ bool LLPanelActiveSpeakers::LLSpeakerListener::handleEvent(LLPointer<LLEvent> ev | |||
231 | 241 | ||
232 | 242 | ||
233 | // | 243 | // |
244 | // LLPanelActiveSpeakers::SpeakerAddListener | ||
245 | // | ||
246 | bool LLPanelActiveSpeakers::SpeakerAddListener::handleEvent(LLPointer<LLEvent> event, const LLSD& userdata) | ||
247 | { | ||
248 | mPanel->addSpeaker(event->getValue().asUUID()); | ||
249 | return true; | ||
250 | } | ||
251 | |||
252 | |||
253 | // | ||
254 | // LLPanelActiveSpeakers::SpeakerRemoveListener | ||
255 | // | ||
256 | bool LLPanelActiveSpeakers::SpeakerRemoveListener::handleEvent(LLPointer<LLEvent> event, const LLSD& userdata) | ||
257 | { | ||
258 | mPanel->removeSpeaker(event->getValue().asUUID()); | ||
259 | return true; | ||
260 | } | ||
261 | |||
262 | // | ||
263 | // LLPanelActiveSpeakers::SpeakerClearListener | ||
264 | // | ||
265 | bool LLPanelActiveSpeakers::SpeakerClearListener::handleEvent(LLPointer<LLEvent> event, const LLSD& userdata) | ||
266 | { | ||
267 | mPanel->mSpeakerList->clearRows(); | ||
268 | return true; | ||
269 | } | ||
270 | |||
271 | |||
272 | // | ||
234 | // LLPanelActiveSpeakers | 273 | // LLPanelActiveSpeakers |
235 | // | 274 | // |
236 | LLPanelActiveSpeakers::LLPanelActiveSpeakers(LLSpeakerMgr* data_source, BOOL show_text_chatters) : | 275 | LLPanelActiveSpeakers::LLPanelActiveSpeakers(LLSpeakerMgr* data_source, BOOL show_text_chatters) : |
@@ -243,12 +282,14 @@ LLPanelActiveSpeakers::LLPanelActiveSpeakers(LLSpeakerMgr* data_source, BOOL sho | |||
243 | mSpeakerMgr(data_source) | 282 | mSpeakerMgr(data_source) |
244 | { | 283 | { |
245 | setMouseOpaque(FALSE); | 284 | setMouseOpaque(FALSE); |
246 | mSpeakerListener = new LLSpeakerListener(this); | 285 | mSpeakerMuteListener = new SpeakerMuteListener(this); |
247 | } | 286 | mSpeakerAddListener = new SpeakerAddListener(this); |
248 | 287 | mSpeakerRemoveListener = new SpeakerRemoveListener(this); | |
249 | LLPanelActiveSpeakers::~LLPanelActiveSpeakers() | 288 | mSpeakerClearListener = new SpeakerClearListener(this); |
250 | { | 289 | |
251 | 290 | mSpeakerMgr->addListener(mSpeakerAddListener, "add"); | |
291 | mSpeakerMgr->addListener(mSpeakerRemoveListener, "remove"); | ||
292 | mSpeakerMgr->addListener(mSpeakerClearListener, "clear"); | ||
252 | } | 293 | } |
253 | 294 | ||
254 | BOOL LLPanelActiveSpeakers::postBuild() | 295 | BOOL LLPanelActiveSpeakers::postBuild() |
@@ -283,6 +324,57 @@ BOOL LLPanelActiveSpeakers::postBuild() | |||
283 | return TRUE; | 324 | return TRUE; |
284 | } | 325 | } |
285 | 326 | ||
327 | void LLPanelActiveSpeakers::addSpeaker(const LLUUID& speaker_id) | ||
328 | { | ||
329 | if (mSpeakerList->getItemIndex(speaker_id) >= 0) | ||
330 | { | ||
331 | // already have this speaker | ||
332 | return; | ||
333 | } | ||
334 | |||
335 | LLPointer<LLSpeaker> speakerp = mSpeakerMgr->findSpeaker(speaker_id); | ||
336 | if (speakerp) | ||
337 | { | ||
338 | // since we are forced to sort by text, encode sort order as string | ||
339 | LLString speaking_order_sort_string = llformat("%010d", speakerp->mSortIndex); | ||
340 | |||
341 | LLSD row; | ||
342 | row["id"] = speaker_id; | ||
343 | |||
344 | LLSD& columns = row["columns"]; | ||
345 | |||
346 | columns[0]["column"] = "icon_speaking_status"; | ||
347 | columns[0]["type"] = "icon"; | ||
348 | columns[0]["value"] = gViewerArt.getString("icn_active-speakers-dot-lvl0.tga"); | ||
349 | |||
350 | LLString speaker_name; | ||
351 | if (speakerp->mDisplayName.empty()) | ||
352 | { | ||
353 | speaker_name = LLCacheName::getDefaultName(); | ||
354 | } | ||
355 | else | ||
356 | { | ||
357 | speaker_name = speakerp->mDisplayName; | ||
358 | } | ||
359 | columns[1]["column"] = "speaker_name"; | ||
360 | columns[1]["type"] = "text"; | ||
361 | columns[1]["value"] = speaker_name; | ||
362 | |||
363 | columns[2]["column"] = "speaking_status"; | ||
364 | columns[2]["type"] = "text"; | ||
365 | |||
366 | // print speaking ordinal in a text-sorting friendly manner | ||
367 | columns[2]["value"] = speaking_order_sort_string; | ||
368 | |||
369 | mSpeakerList->addElement(row); | ||
370 | } | ||
371 | } | ||
372 | |||
373 | void LLPanelActiveSpeakers::removeSpeaker(const LLUUID& speaker_id) | ||
374 | { | ||
375 | mSpeakerList->deleteSingleItem(mSpeakerList->getItemIndex(speaker_id)); | ||
376 | } | ||
377 | |||
286 | void LLPanelActiveSpeakers::handleSpeakerSelect() | 378 | void LLPanelActiveSpeakers::handleSpeakerSelect() |
287 | { | 379 | { |
288 | LLUUID speaker_id = mSpeakerList->getValue().asUUID(); | 380 | LLUUID speaker_id = mSpeakerList->getValue().asUUID(); |
@@ -296,8 +388,8 @@ void LLPanelActiveSpeakers::handleSpeakerSelect() | |||
296 | childSetValue("moderator_allow_voice", selected_speakerp ? !selected_speakerp->mModeratorMutedVoice : TRUE); | 388 | childSetValue("moderator_allow_voice", selected_speakerp ? !selected_speakerp->mModeratorMutedVoice : TRUE); |
297 | childSetValue("moderator_allow_text", selected_speakerp ? !selected_speakerp->mModeratorMutedText : TRUE); | 389 | childSetValue("moderator_allow_text", selected_speakerp ? !selected_speakerp->mModeratorMutedText : TRUE); |
298 | 390 | ||
299 | mSpeakerListener->clearDispatchers(); | 391 | mSpeakerMuteListener->clearDispatchers(); |
300 | selected_speakerp->addListener(mSpeakerListener); | 392 | selected_speakerp->addListener(mSpeakerMuteListener); |
301 | } | 393 | } |
302 | } | 394 | } |
303 | 395 | ||
@@ -307,159 +399,131 @@ void LLPanelActiveSpeakers::refreshSpeakers() | |||
307 | LLUUID selected_id = mSpeakerList->getSelectedValue().asUUID(); | 399 | LLUUID selected_id = mSpeakerList->getSelectedValue().asUUID(); |
308 | S32 scroll_pos = mSpeakerList->getScrollInterface()->getScrollPos(); | 400 | S32 scroll_pos = mSpeakerList->getScrollInterface()->getScrollPos(); |
309 | 401 | ||
310 | BOOL sort_ascending = mSpeakerList->getSortAscending(); | ||
311 | LLString sort_column = mSpeakerList->getSortColumnName(); | ||
312 | // TODO: put this in xml | ||
313 | // enforces default sort column of speaker status | ||
314 | if (sort_column.empty()) | ||
315 | { | ||
316 | sort_column = "speaking_status"; | ||
317 | } | ||
318 | |||
319 | mSpeakerMgr->update(); | 402 | mSpeakerMgr->update(); |
320 | 403 | ||
321 | // clear scrolling list widget of names | 404 | const LLString icon_image_0 = gViewerArt.getString("icn_active-speakers-dot-lvl0.tga"); |
322 | mSpeakerList->clearRows(); | 405 | const LLString icon_image_1 = gViewerArt.getString("icn_active-speakers-dot-lvl1.tga"); |
406 | const LLString icon_image_2 = gViewerArt.getString("icn_active-speakers-dot-lvl2.tga"); | ||
407 | |||
408 | |||
409 | std::vector<LLScrollListItem*> items = mSpeakerList->getAllData(); | ||
410 | |||
411 | LLUUID mute_icon_image = LLUUID(gViewerArt.getString("mute_icon.tga")); | ||
323 | 412 | ||
324 | LLSpeakerMgr::speaker_list_t speaker_list; | 413 | LLSpeakerMgr::speaker_list_t speaker_list; |
325 | mSpeakerMgr->getSpeakerList(&speaker_list, mShowTextChatters); | 414 | mSpeakerMgr->getSpeakerList(&speaker_list, mShowTextChatters); |
326 | for (LLSpeakerMgr::speaker_list_t::const_iterator speaker_it = speaker_list.begin(); speaker_it != speaker_list.end(); ++speaker_it) | 415 | for (std::vector<LLScrollListItem*>::iterator item_it = items.begin(); |
416 | item_it != items.end(); | ||
417 | ++item_it) | ||
327 | { | 418 | { |
328 | LLUUID speaker_id = (*speaker_it)->mID; | 419 | LLScrollListItem* itemp = (*item_it); |
329 | LLPointer<LLSpeaker> speakerp = (*speaker_it); | 420 | LLUUID speaker_id = itemp->getUUID(); |
421 | |||
422 | LLPointer<LLSpeaker> speakerp = mSpeakerMgr->findSpeaker(speaker_id); | ||
423 | if (!speakerp) | ||
424 | { | ||
425 | continue; | ||
426 | } | ||
330 | 427 | ||
331 | // since we are forced to sort by text, encode sort order as string | 428 | // since we are forced to sort by text, encode sort order as string |
332 | LLString speaking_order_sort_string = llformat("%010d", speakerp->mSortIndex); | 429 | LLString speaking_order_sort_string = llformat("%010d", speakerp->mSortIndex); |
333 | 430 | ||
334 | LLSD row; | 431 | LLScrollListCell* icon_cell = itemp->getColumn(0); |
335 | row["id"] = speaker_id; | 432 | if (icon_cell) |
433 | { | ||
336 | 434 | ||
337 | row["columns"][0]["column"] = "icon_speaking_status"; | 435 | LLString icon_image_id; |
338 | row["columns"][0]["type"] = "icon"; | ||
339 | LLString icon_image_id; | ||
340 | 436 | ||
341 | S32 icon_image_idx = llmin(2, llfloor((speakerp->mSpeechVolume / LLVoiceClient::OVERDRIVEN_POWER_LEVEL) * 3.f)); | 437 | S32 icon_image_idx = llmin(2, llfloor((speakerp->mSpeechVolume / LLVoiceClient::OVERDRIVEN_POWER_LEVEL) * 3.f)); |
342 | switch(icon_image_idx) | 438 | switch(icon_image_idx) |
343 | { | ||
344 | case 0: | ||
345 | icon_image_id = gViewerArt.getString("icn_active-speakers-dot-lvl0.tga"); | ||
346 | break; | ||
347 | case 1: | ||
348 | icon_image_id = gViewerArt.getString("icn_active-speakers-dot-lvl1.tga"); | ||
349 | break; | ||
350 | case 2: | ||
351 | icon_image_id = gViewerArt.getString("icn_active-speakers-dot-lvl2.tga"); | ||
352 | break; | ||
353 | } | ||
354 | //if (speakerp->mTyping) | ||
355 | //{ | ||
356 | // S32 typing_anim_idx = llround(mIconAnimationTimer.getElapsedTimeF32() * TYPING_ANIMATION_FPS) % 3; | ||
357 | // switch(typing_anim_idx) | ||
358 | // { | ||
359 | // case 0: | ||
360 | // row["columns"][0]["overlay"] = LLUUID(gViewerArt.getString("icn_active-speakers-typing1.tga")); | ||
361 | // break; | ||
362 | // case 1: | ||
363 | // row["columns"][0]["overlay"] = LLUUID(gViewerArt.getString("icn_active-speakers-typing2.tga")); | ||
364 | // break; | ||
365 | // case 2: | ||
366 | // row["columns"][0]["overlay"] = LLUUID(gViewerArt.getString("icn_active-speakers-typing3.tga")); | ||
367 | // break; | ||
368 | // default: | ||
369 | // break; | ||
370 | // } | ||
371 | //} | ||
372 | |||
373 | LLColor4 icon_color; | ||
374 | if (speakerp->mStatus == LLSpeaker::STATUS_MUTED) | ||
375 | { | ||
376 | row["columns"][0]["value"] = gViewerArt.getString("mute_icon.tga"); | ||
377 | if(speakerp->mModeratorMutedVoice) | ||
378 | { | 439 | { |
379 | icon_color.setVec(0.5f, 0.5f, 0.5f, 1.f); | 440 | case 0: |
441 | icon_image_id = icon_image_0; | ||
442 | break; | ||
443 | case 1: | ||
444 | icon_image_id = icon_image_1; | ||
445 | break; | ||
446 | case 2: | ||
447 | icon_image_id = icon_image_2; | ||
448 | break; | ||
449 | } | ||
450 | |||
451 | LLColor4 icon_color; | ||
452 | if (speakerp->mStatus == LLSpeaker::STATUS_MUTED) | ||
453 | { | ||
454 | icon_cell->setValue(mute_icon_image); | ||
455 | if(speakerp->mModeratorMutedVoice) | ||
456 | { | ||
457 | icon_color.setVec(0.5f, 0.5f, 0.5f, 1.f); | ||
458 | } | ||
459 | else | ||
460 | { | ||
461 | icon_color.setVec(1.f, 71.f / 255.f, 71.f / 255.f, 1.f); | ||
462 | } | ||
380 | } | 463 | } |
381 | else | 464 | else |
382 | { | 465 | { |
383 | icon_color.setVec(1.f, 71.f / 255.f, 71.f / 255.f, 1.f); | 466 | icon_cell->setValue(icon_image_id); |
467 | icon_color = speakerp->mDotColor; | ||
468 | |||
469 | if (speakerp->mStatus > LLSpeaker::STATUS_VOICE_ACTIVE) // if voice is disabled for this speaker | ||
470 | { | ||
471 | // non voice speakers have hidden icons, render as transparent | ||
472 | icon_color.setVec(0.f, 0.f, 0.f, 0.f); | ||
473 | } | ||
384 | } | 474 | } |
385 | } | ||
386 | else | ||
387 | { | ||
388 | row["columns"][0]["value"] = icon_image_id; | ||
389 | icon_color = speakerp->mDotColor; | ||
390 | 475 | ||
391 | if (speakerp->mStatus > LLSpeaker::STATUS_VOICE_ACTIVE) // if voice is disabled for this speaker | 476 | icon_cell->setColor(icon_color); |
477 | |||
478 | if (speakerp->mStatus > LLSpeaker::STATUS_VOICE_ACTIVE && speakerp->mStatus != LLSpeaker::STATUS_MUTED) // if voice is disabled for this speaker | ||
392 | { | 479 | { |
393 | // non voice speakers have hidden icons, render as transparent | 480 | // non voice speakers have hidden icons, render as transparent |
394 | icon_color.setVec(0.f, 0.f, 0.f, 0.f); | 481 | icon_cell->setColor(LLColor4::transparent); |
395 | } | 482 | } |
396 | } | 483 | } |
397 | 484 | ||
398 | row["columns"][0]["color"] = icon_color.getValue(); | 485 | // update name column |
399 | 486 | LLScrollListCell* name_cell = itemp->getColumn(1); | |
400 | if (speakerp->mStatus > LLSpeaker::STATUS_VOICE_ACTIVE && speakerp->mStatus != LLSpeaker::STATUS_MUTED) // if voice is disabled for this speaker | 487 | if (name_cell) |
401 | { | ||
402 | // non voice speakers have hidden icons, render as transparent | ||
403 | row["columns"][0]["color"] = LLColor4(0.f, 0.f, 0.f, 0.f).getValue(); | ||
404 | } | ||
405 | row["columns"][1]["column"] = "speaker_name"; | ||
406 | row["columns"][1]["type"] = "text"; | ||
407 | if (speakerp->mStatus == LLSpeaker::STATUS_NOT_IN_CHANNEL) | ||
408 | { | 488 | { |
409 | // draw inactive speakers in gray | 489 | if (speakerp->mStatus == LLSpeaker::STATUS_NOT_IN_CHANNEL) |
410 | row["columns"][1]["color"] = LLColor4::grey4.getValue(); | 490 | { |
411 | } | 491 | // draw inactive speakers in gray |
492 | name_cell->setColor(LLColor4::grey4); | ||
493 | } | ||
412 | 494 | ||
413 | LLString speaker_name; | 495 | LLString speaker_name; |
414 | if (speakerp->mDisplayName.empty()) | 496 | if (speakerp->mDisplayName.empty()) |
415 | { | 497 | { |
416 | speaker_name = LLCacheName::getDefaultName(); | 498 | speaker_name = LLCacheName::getDefaultName(); |
417 | } | 499 | } |
418 | else | 500 | else |
419 | { | 501 | { |
420 | speaker_name = speakerp->mDisplayName; | 502 | speaker_name = speakerp->mDisplayName; |
503 | } | ||
504 | |||
505 | if (speakerp->mIsModerator) | ||
506 | { | ||
507 | speaker_name += LLString(" ") + getFormattedUIString("moderator_label"); | ||
508 | } | ||
509 | |||
510 | name_cell->setValue(speaker_name); | ||
511 | ((LLScrollListText*)name_cell)->setFontStyle(speakerp->mIsModerator ? LLFontGL::BOLD : LLFontGL::NORMAL); | ||
421 | } | 512 | } |
422 | 513 | ||
423 | if (speakerp->mIsModerator) | 514 | // update speaking order column |
515 | LLScrollListCell* speaking_status_cell = itemp->getColumn(2); | ||
516 | if (speaking_status_cell) | ||
424 | { | 517 | { |
425 | speaker_name += LLString(" ") + getFormattedUIString("moderator_label"); | 518 | // print speaking ordinal in a text-sorting friendly manner |
519 | speaking_status_cell->setValue(speaking_order_sort_string); | ||
426 | } | 520 | } |
427 | row["columns"][1]["value"] = speaker_name; | ||
428 | row["columns"][1]["font-style"] = speakerp->mIsModerator ? "BOLD" : "NORMAL"; | ||
429 | |||
430 | row["columns"][2]["column"] = "speaking_status"; | ||
431 | row["columns"][2]["type"] = "text"; | ||
432 | |||
433 | // print speaking ordinal in a text-sorting friendly manner | ||
434 | row["columns"][2]["value"] = speaking_order_sort_string; | ||
435 | |||
436 | mSpeakerList->addElement(row); | ||
437 | } | 521 | } |
438 | 522 | ||
439 | //restore sort order, selection, etc | 523 | // we potentially modified the sort order by touching the list items |
440 | mSpeakerList->sortByColumn(sort_column, sort_ascending); | 524 | mSpeakerList->setSorted(FALSE); |
441 | |||
442 | // temporarily disable commit callback while restoring original selection | ||
443 | mSpeakerList->setCommitCallback(NULL); | ||
444 | |||
445 | // make sure something is selected | ||
446 | if (selected_id.isNull()) | ||
447 | { | ||
448 | mSpeakerList->selectFirstItem(); | ||
449 | handleSpeakerSelect(); | ||
450 | } | ||
451 | else | ||
452 | { | ||
453 | // reselect original speaker but don't call handleSpeakerSelect() | ||
454 | // as that would change the moderation mute checkboxes before they | ||
455 | // have had time to get confirmation from the server | ||
456 | mSpeakerList->selectByValue(selected_id); | ||
457 | } | ||
458 | |||
459 | mSpeakerList->setCommitCallback(onSelectSpeaker); | ||
460 | 525 | ||
461 | LLPointer<LLSpeaker> selected_speakerp = mSpeakerMgr->findSpeaker(selected_id); | 526 | LLPointer<LLSpeaker> selected_speakerp = mSpeakerMgr->findSpeaker(selected_id); |
462 | |||
463 | 527 | ||
464 | if (gMuteListp) | 528 | if (gMuteListp) |
465 | { | 529 | { |
@@ -847,6 +911,7 @@ LLPointer<LLSpeaker> LLSpeakerMgr::setSpeaker(const LLUUID& id, const LLString& | |||
847 | speakerp->mStatus = status; | 911 | speakerp->mStatus = status; |
848 | mSpeakers.insert(std::make_pair(speakerp->mID, speakerp)); | 912 | mSpeakers.insert(std::make_pair(speakerp->mID, speakerp)); |
849 | mSpeakersSorted.push_back(speakerp); | 913 | mSpeakersSorted.push_back(speakerp); |
914 | fireEvent(new LLSpeakerListChangeEvent(this, speakerp->mID), "add"); | ||
850 | } | 915 | } |
851 | else | 916 | else |
852 | { | 917 | { |
@@ -974,6 +1039,8 @@ void LLSpeakerMgr::update() | |||
974 | // remove speakers that have been gone too long | 1039 | // remove speakers that have been gone too long |
975 | if (speakerp->mStatus == LLSpeaker::STATUS_NOT_IN_CHANNEL && speakerp->mActivityTimer.hasExpired()) | 1040 | if (speakerp->mStatus == LLSpeaker::STATUS_NOT_IN_CHANNEL && speakerp->mActivityTimer.hasExpired()) |
976 | { | 1041 | { |
1042 | fireEvent(new LLSpeakerListChangeEvent(this, speakerp->mID), "remove"); | ||
1043 | |||
977 | mSpeakers.erase(speakerp->mID); | 1044 | mSpeakers.erase(speakerp->mID); |
978 | sorted_speaker_it = mSpeakersSorted.erase(sorted_speaker_it); | 1045 | sorted_speaker_it = mSpeakersSorted.erase(sorted_speaker_it); |
979 | } | 1046 | } |
@@ -1221,6 +1288,7 @@ void LLActiveSpeakerMgr::updateSpeakerList() | |||
1221 | // always populate from active voice channel | 1288 | // always populate from active voice channel |
1222 | if (LLVoiceChannel::getCurrentVoiceChannel() != mVoiceChannel) | 1289 | if (LLVoiceChannel::getCurrentVoiceChannel() != mVoiceChannel) |
1223 | { | 1290 | { |
1291 | fireEvent(new LLSpeakerListChangeEvent(this, LLUUID::null), "clear"); | ||
1224 | mSpeakers.clear(); | 1292 | mSpeakers.clear(); |
1225 | mSpeakersSorted.clear(); | 1293 | mSpeakersSorted.clear(); |
1226 | mVoiceChannel = LLVoiceChannel::getCurrentVoiceChannel(); | 1294 | mVoiceChannel = LLVoiceChannel::getCurrentVoiceChannel(); |