1060 lines
30 KiB
C++
1060 lines
30 KiB
C++
/**
|
|
* @file llfloaterreporter.cpp
|
|
* @brief Bug and abuse reports.
|
|
*
|
|
* $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$
|
|
*/
|
|
|
|
#include "llviewerprecompiledheaders.h"
|
|
|
|
// self include
|
|
#include "llfloaterreporter.h"
|
|
|
|
#include <sstream>
|
|
|
|
// linden library includes
|
|
#include "llassetstorage.h"
|
|
#include "llcachename.h"
|
|
#include "llfontgl.h"
|
|
#include "llgl.h" // for renderer
|
|
#include "llimagej2c.h"
|
|
#include "llinventory.h"
|
|
#include "llnotificationsutil.h"
|
|
#include "llstring.h"
|
|
#include "llsys.h"
|
|
#include "sgversion.h"
|
|
#include "message.h"
|
|
#include "v3math.h"
|
|
|
|
// viewer project includes
|
|
#include "llagent.h"
|
|
#include "llbutton.h"
|
|
#include "llcheckboxctrl.h"
|
|
#include "llinventorypanel.h"
|
|
#include "lllineeditor.h"
|
|
#include "lltexturectrl.h"
|
|
#include "llscrolllistctrl.h"
|
|
#include "llimview.h"
|
|
#include "lltextbox.h"
|
|
#include "lldispatcher.h"
|
|
#include "llviewertexteditor.h"
|
|
#include "llviewerobject.h"
|
|
#include "llviewerregion.h"
|
|
#include "llcombobox.h"
|
|
#include "lltooldraganddrop.h"
|
|
#include "lluiconstants.h"
|
|
#include "lluploaddialog.h"
|
|
#include "llcallingcard.h"
|
|
#include "llviewerobjectlist.h"
|
|
#include "lltoolobjpicker.h"
|
|
#include "lltoolmgr.h"
|
|
#include "llresourcedata.h" // for LLResourceData
|
|
#include "llviewerwindow.h"
|
|
#include "llviewertexturelist.h"
|
|
#include "llworldmap.h"
|
|
#include "llfloateravatarpicker.h"
|
|
#include "lldir.h"
|
|
#include "llselectmgr.h"
|
|
#include "llviewerbuild.h"
|
|
#include "lluictrlfactory.h"
|
|
#include "llviewernetwork.h"
|
|
|
|
#include "llassetuploadresponders.h"
|
|
|
|
#include "lltrans.h"
|
|
|
|
// <edit>
|
|
#include "llviewercontrol.h"
|
|
// </edit>
|
|
|
|
// [RLVa:KB]
|
|
#include "rlvhandler.h"
|
|
// [/RLVa:KB]
|
|
|
|
const U32 INCLUDE_SCREENSHOT = 0x01 << 0;
|
|
|
|
class AIHTTPTimeoutPolicy;
|
|
extern AIHTTPTimeoutPolicy userReportResponder_timeout;
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Globals
|
|
//-----------------------------------------------------------------------------
|
|
|
|
// this map keeps track of current reporter instances
|
|
// there can only be one instance of each reporter type
|
|
LLMap< EReportType, LLFloaterReporter* > gReporterInstances;
|
|
|
|
// keeps track of where email is going to - global to avoid a pile
|
|
// of static/non-static access outside my control
|
|
namespace {
|
|
static BOOL gDialogVisible = FALSE;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
// Member functions
|
|
//-----------------------------------------------------------------------------
|
|
LLFloaterReporter::LLFloaterReporter(
|
|
const std::string& name,
|
|
const LLRect& rect,
|
|
const std::string& title,
|
|
EReportType report_type)
|
|
:
|
|
LLFloater(name, rect, title),
|
|
mReportType(report_type),
|
|
mObjectID(),
|
|
mScreenID(),
|
|
mAbuserID(),
|
|
mDeselectOnClose( FALSE ),
|
|
mPicking( FALSE),
|
|
mPosition(),
|
|
mCopyrightWarningSeen( FALSE ),
|
|
mResourceDatap(new LLResourceData())
|
|
{
|
|
if (report_type == BUG_REPORT)
|
|
{
|
|
LLUICtrlFactory::getInstance()->buildFloater(this, "floater_report_bug.xml");
|
|
}
|
|
else
|
|
{
|
|
LLUICtrlFactory::getInstance()->buildFloater(this, "floater_report_abuse.xml");
|
|
}
|
|
|
|
childSetText("abuse_location_edit", gAgent.getSLURL() );
|
|
|
|
// [RLVa:KB] - Checked: 2009-07-08 (RLVa-1.0.0e) | Modified: RLVa-1.0.0a
|
|
if (rlv_handler_t::isEnabled())
|
|
{
|
|
// Can't filter these since they get sent as part of the report so just hide them instead
|
|
if (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWLOC))
|
|
{
|
|
childSetVisible("abuse_location_edit", false);
|
|
childSetVisible("pos_field", false);
|
|
}
|
|
if (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES))
|
|
{
|
|
childSetVisible("owner_name", false);
|
|
childSetVisible("abuser_name_edit", false);
|
|
}
|
|
}
|
|
// [/RLVa:KB]
|
|
|
|
LLButton* pick_btn = getChild<LLButton>("pick_btn");
|
|
if (pick_btn)
|
|
{
|
|
// XUI: Why aren't these in viewerart.ini?
|
|
pick_btn->setImages( std::string("UIImgFaceUUID"),
|
|
std::string("UIImgFaceSelectedUUID") );
|
|
childSetAction("pick_btn", onClickObjPicker, this);
|
|
}
|
|
|
|
if (report_type != BUG_REPORT)
|
|
{
|
|
// abuser name is selected from a list
|
|
LLLineEditor* le = getChild<LLLineEditor>("abuser_name_edit");
|
|
le->setEnabled( FALSE );
|
|
}
|
|
|
|
childSetAction("select_abuser", onClickSelectAbuser, this);
|
|
|
|
childSetAction("send_btn", onClickSend, this);
|
|
childSetAction("cancel_btn", onClickCancel, this);
|
|
|
|
enableControls(TRUE);
|
|
|
|
// convert the position to a string
|
|
LLVector3d pos = gAgent.getPositionGlobal();
|
|
LLViewerRegion *regionp = gAgent.getRegion();
|
|
if (regionp)
|
|
{
|
|
pos -= regionp->getOriginGlobal();
|
|
}
|
|
setPosBox(pos);
|
|
|
|
gReporterInstances.addData(report_type, this);
|
|
|
|
// Take a screenshot, but don't draw this floater.
|
|
setVisible(FALSE);
|
|
takeScreenshot();
|
|
setVisible(TRUE);
|
|
|
|
// Default text to be blank
|
|
childSetText("object_name", LLStringUtil::null);
|
|
childSetText("owner_name", LLStringUtil::null);
|
|
|
|
childSetFocus("summary_edit");
|
|
|
|
mDefaultSummary = childGetText("details_edit");
|
|
|
|
gDialogVisible = TRUE;
|
|
|
|
// only request details for abuse reports (not BUG reports)
|
|
if (report_type != BUG_REPORT)
|
|
{
|
|
// send a message and ask for information about this region -
|
|
// result comes back in processRegionInfo(..)
|
|
LLMessageSystem* msg = gMessageSystem;
|
|
msg->newMessage("RequestRegionInfo");
|
|
msg->nextBlock("AgentData");
|
|
msg->addUUID("AgentID", gAgent.getID());
|
|
msg->addUUID("SessionID", gAgent.getSessionID());
|
|
gAgent.sendReliableMessage();
|
|
};
|
|
}
|
|
|
|
// static
|
|
void LLFloaterReporter::processRegionInfo(LLMessageSystem* msg)
|
|
{
|
|
U32 region_flags;
|
|
msg->getU32("RegionInfo", "RegionFlags", region_flags);
|
|
|
|
if ( gDialogVisible )
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseEmailLL");
|
|
};
|
|
}
|
|
|
|
// virtual
|
|
LLFloaterReporter::~LLFloaterReporter()
|
|
{
|
|
gReporterInstances.removeData(mReportType);
|
|
// child views automatically deleted
|
|
mObjectID = LLUUID::null;
|
|
|
|
if (mPicking)
|
|
{
|
|
closePickTool(this);
|
|
}
|
|
|
|
mPosition.setVec(0.0f, 0.0f, 0.0f);
|
|
|
|
std::for_each(mMCDList.begin(), mMCDList.end(), DeletePointer() );
|
|
mMCDList.clear();
|
|
|
|
delete mResourceDatap;
|
|
gDialogVisible = FALSE;
|
|
}
|
|
|
|
// virtual
|
|
void LLFloaterReporter::draw()
|
|
{
|
|
childSetEnabled("screen_check", TRUE );
|
|
|
|
LLFloater::draw();
|
|
}
|
|
|
|
void LLFloaterReporter::enableControls(BOOL enable)
|
|
{
|
|
childSetEnabled("category_combo", enable);
|
|
childSetEnabled("screen_check", enable);
|
|
childDisable("screenshot");
|
|
childSetEnabled("pick_btn", enable);
|
|
childSetEnabled("summary_edit", enable);
|
|
childSetEnabled("details_edit", enable);
|
|
childSetEnabled("send_btn", enable);
|
|
childSetEnabled("cancel_btn", enable);
|
|
}
|
|
|
|
void LLFloaterReporter::getObjectInfo(const LLUUID& object_id)
|
|
{
|
|
// TODO --
|
|
// 1 need to send to correct simulator if object is not
|
|
// in same simulator as agent
|
|
// 2 display info in widget window that gives feedback that
|
|
// we have recorded the object info
|
|
// 3 can pick avatar ==> might want to indicate when a picked
|
|
// object is an avatar, attachment, or other category
|
|
|
|
mObjectID = object_id;
|
|
|
|
if (LLUUID::null != mObjectID)
|
|
{
|
|
// get object info for the user's benefit
|
|
LLViewerObject* objectp = NULL;
|
|
objectp = gObjectList.findObject( mObjectID );
|
|
if (objectp)
|
|
{
|
|
if ( objectp->isAttachment() )
|
|
{
|
|
objectp = (LLViewerObject*)objectp->getRoot();
|
|
}
|
|
|
|
// correct the region and position information
|
|
LLViewerRegion *regionp = objectp->getRegion();
|
|
if (regionp)
|
|
{
|
|
childSetText("sim_field", regionp->getName());
|
|
// [RLVa:KB] - Checked: 2009-07-04 (RLVa-1.0.0a)
|
|
if ( (rlv_handler_t::isEnabled()) && (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWLOC)) )
|
|
{
|
|
childSetText("sim_field", RlvStrings::getString(RLV_STRING_HIDDEN_REGION));
|
|
}
|
|
// [/RLVa:KB]
|
|
LLVector3d global_pos;
|
|
global_pos.setVec(objectp->getPositionRegion());
|
|
setPosBox(global_pos);
|
|
}
|
|
|
|
if (objectp->isAvatar())
|
|
{
|
|
// we have the information we need
|
|
std::string object_owner;
|
|
|
|
LLNameValue* firstname = objectp->getNVPair("FirstName");
|
|
LLNameValue* lastname = objectp->getNVPair("LastName");
|
|
if (firstname && lastname)
|
|
{
|
|
object_owner.append(firstname->getString());
|
|
object_owner.append(1, ' ');
|
|
object_owner.append(lastname->getString());
|
|
}
|
|
else
|
|
{
|
|
object_owner.append("Unknown");
|
|
}
|
|
childSetText("object_name", object_owner);
|
|
// [RLVa:KB] - Checked: 2009-07-08 (RLVa-1.0.0e) | Added: RVLa-1.0.0e
|
|
if (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES))
|
|
{
|
|
childSetVisible("object_name", false); // Hide the object name if the picked object represents an avataz
|
|
}
|
|
// [/RLVa:KB]
|
|
childSetText("owner_name", object_owner);
|
|
childSetText("abuser_name_edit", object_owner);
|
|
mAbuserID = object_id;
|
|
}
|
|
else
|
|
{
|
|
// we have to query the simulator for information
|
|
// about this object
|
|
LLMessageSystem* msg = gMessageSystem;
|
|
U32 request_flags = (mReportType == BUG_REPORT) ? BUG_REPORT_REQUEST : COMPLAINT_REPORT_REQUEST;
|
|
msg->newMessageFast(_PREHASH_RequestObjectPropertiesFamily);
|
|
msg->nextBlockFast(_PREHASH_AgentData);
|
|
msg->addUUIDFast(_PREHASH_AgentID, gAgent.getID());
|
|
msg->addUUIDFast(_PREHASH_SessionID, gAgent.getSessionID());
|
|
msg->nextBlockFast(_PREHASH_ObjectData);
|
|
msg->addU32Fast(_PREHASH_RequestFlags, request_flags );
|
|
msg->addUUIDFast(_PREHASH_ObjectID, mObjectID);
|
|
LLViewerRegion* regionp = objectp->getRegion();
|
|
msg->sendReliable( regionp->getHost() );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::onClickSelectAbuser(void *userdata)
|
|
{
|
|
LLFloaterReporter *self = (LLFloaterReporter *)userdata;
|
|
|
|
gFloaterView->getParentFloater(self)->addDependentFloater(LLFloaterAvatarPicker::show(boost::bind(&LLFloaterReporter::callbackAvatarID, self, _1, _2), FALSE, TRUE ));
|
|
}
|
|
|
|
void LLFloaterReporter::callbackAvatarID(const uuid_vec_t& ids, const std::vector<LLAvatarName>& names)
|
|
{
|
|
if (ids.empty() || names.empty()) return;
|
|
|
|
// this should never be called in a bug report but here for safety.
|
|
if ( mReportType != BUG_REPORT )
|
|
{
|
|
childSetText("abuser_name_edit", names[0].getCompleteName() );
|
|
|
|
mAbuserID = ids[0];
|
|
|
|
refresh();
|
|
};
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::onClickSend(void *userdata)
|
|
{
|
|
LLFloaterReporter *self = (LLFloaterReporter *)userdata;
|
|
|
|
if (self->mPicking)
|
|
{
|
|
closePickTool(self);
|
|
}
|
|
|
|
if(self->validateReport())
|
|
{
|
|
// only show copyright alert for abuse reports
|
|
if ( self->mReportType != BUG_REPORT )
|
|
{
|
|
const int IP_CONTENT_REMOVAL = 66;
|
|
const int IP_PERMISSONS_EXPLOIT = 37;
|
|
LLComboBox* combo = self->getChild<LLComboBox>( "category_combo");
|
|
int category_value = combo->getSelectedValue().asInteger();
|
|
|
|
if ( ! self->mCopyrightWarningSeen )
|
|
{
|
|
|
|
std::string details_lc = self->childGetText("details_edit");
|
|
LLStringUtil::toLower( details_lc );
|
|
std::string summary_lc = self->childGetText("summary_edit");
|
|
LLStringUtil::toLower( summary_lc );
|
|
if ( details_lc.find( "copyright" ) != std::string::npos ||
|
|
summary_lc.find( "copyright" ) != std::string::npos ||
|
|
category_value == IP_CONTENT_REMOVAL ||
|
|
category_value == IP_PERMISSONS_EXPLOIT)
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseContainsCopyright");
|
|
self->mCopyrightWarningSeen = TRUE;
|
|
return;
|
|
}
|
|
}
|
|
else if (category_value == IP_CONTENT_REMOVAL)
|
|
{
|
|
// IP_CONTENT_REMOVAL *always* shows the dialog -
|
|
// ergo you can never send that abuse report type.
|
|
LLNotificationsUtil::add("HelpReportAbuseContainsCopyright");
|
|
return;
|
|
}
|
|
}
|
|
|
|
LLUploadDialog::modalUploadDialog(LLTrans::getString("uploading_abuse_report"));
|
|
// *TODO don't upload image if checkbox isn't checked
|
|
std::string url = gAgent.getRegion()->getCapability("SendUserReport");
|
|
std::string sshot_url = gAgent.getRegion()->getCapability("SendUserReportWithScreenshot");
|
|
if(!url.empty() || !sshot_url.empty())
|
|
{
|
|
self->sendReportViaCaps(url, sshot_url, self->gatherReport());
|
|
self->close();
|
|
}
|
|
else
|
|
{
|
|
if(self->childGetValue("screen_check"))
|
|
{
|
|
self->childDisable("send_btn");
|
|
self->childDisable("cancel_btn");
|
|
// the callback from uploading the image calls sendReportViaLegacy()
|
|
self->uploadImage();
|
|
}
|
|
else
|
|
{
|
|
self->sendReportViaLegacy(self->gatherReport());
|
|
LLUploadDialog::modalUploadFinished();
|
|
self->close();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::onClickCancel(void *userdata)
|
|
{
|
|
LLFloaterReporter *self = (LLFloaterReporter *)userdata;
|
|
|
|
// reset flag in case the next report also contains this text
|
|
self->mCopyrightWarningSeen = FALSE;
|
|
|
|
if (self->mPicking)
|
|
{
|
|
closePickTool(self);
|
|
}
|
|
self->close();
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::onClickObjPicker(void *userdata)
|
|
{
|
|
LLFloaterReporter *self = (LLFloaterReporter *)userdata;
|
|
LLToolObjPicker::getInstance()->setExitCallback(LLFloaterReporter::closePickTool, self);
|
|
LLToolMgr::getInstance()->setTransientTool(LLToolObjPicker::getInstance());
|
|
self->mPicking = TRUE;
|
|
self->childSetText("object_name", LLStringUtil::null);
|
|
self->childSetText("owner_name", LLStringUtil::null);
|
|
LLButton* pick_btn = self->getChild<LLButton>("pick_btn");
|
|
if (pick_btn) pick_btn->setToggleState(TRUE);
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::closePickTool(void *userdata)
|
|
{
|
|
LLFloaterReporter *self = (LLFloaterReporter *)userdata;
|
|
|
|
LLUUID object_id = LLToolObjPicker::getInstance()->getObjectID();
|
|
self->getObjectInfo(object_id);
|
|
|
|
LLToolMgr::getInstance()->clearTransientTool();
|
|
self->mPicking = FALSE;
|
|
LLButton* pick_btn = self->getChild<LLButton>("pick_btn");
|
|
if (pick_btn) pick_btn->setToggleState(FALSE);
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::showFromMenu(EReportType report_type)
|
|
{
|
|
if (gReporterInstances.checkData(report_type))
|
|
{
|
|
// ...bring that window to front
|
|
LLFloaterReporter *f = gReporterInstances.getData(report_type);
|
|
f->open(); /* Flawfinder: ignore */
|
|
}
|
|
else
|
|
{
|
|
LLFloaterReporter *f;
|
|
if (BUG_REPORT == report_type)
|
|
{
|
|
f = LLFloaterReporter::createNewBugReporter();
|
|
}
|
|
else if (COMPLAINT_REPORT == report_type)
|
|
{
|
|
f = LLFloaterReporter::createNewAbuseReporter();
|
|
}
|
|
else
|
|
{
|
|
llwarns << "Unknown LLViewerReporter type : " << report_type << llendl;
|
|
return;
|
|
}
|
|
|
|
f->center();
|
|
|
|
if (report_type == BUG_REPORT)
|
|
{
|
|
LLNotificationsUtil::add("HelpReportBug");
|
|
}
|
|
else
|
|
{
|
|
// popup for abuse reports is triggered elsewhere
|
|
}
|
|
|
|
// grab the user's name
|
|
std::string fullname;
|
|
gAgent.buildFullname(fullname);
|
|
f->childSetText("reporter_field", fullname);
|
|
}
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::showFromObject(const LLUUID& object_id)
|
|
{
|
|
LLFloaterReporter* f = createNewAbuseReporter();
|
|
f->center();
|
|
f->setFocus(TRUE);
|
|
|
|
// grab the user's name
|
|
std::string fullname;
|
|
gAgent.buildFullname(fullname);
|
|
f->childSetText("reporter_field", fullname);
|
|
|
|
// Request info for this object
|
|
f->getObjectInfo(object_id);
|
|
|
|
// Need to deselect on close
|
|
f->mDeselectOnClose = TRUE;
|
|
|
|
f->open(); /* Flawfinder: ignore */
|
|
}
|
|
|
|
|
|
// static
|
|
LLFloaterReporter* LLFloaterReporter::getReporter(EReportType report_type)
|
|
{
|
|
LLFloaterReporter *self = NULL;
|
|
if (gReporterInstances.checkData(report_type))
|
|
{
|
|
// ...bring that window to front
|
|
self = gReporterInstances.getData(report_type);
|
|
}
|
|
return self;
|
|
}
|
|
|
|
LLFloaterReporter* LLFloaterReporter::createNewAbuseReporter()
|
|
{
|
|
return new LLFloaterReporter("complaint_reporter",
|
|
LLRect(),
|
|
"Report Abuse",
|
|
COMPLAINT_REPORT);
|
|
}
|
|
|
|
//static
|
|
LLFloaterReporter* LLFloaterReporter::createNewBugReporter()
|
|
{
|
|
return new LLFloaterReporter("bug_reporter",
|
|
LLRect(),
|
|
"Report Bug",
|
|
BUG_REPORT);
|
|
}
|
|
|
|
|
|
|
|
void LLFloaterReporter::setPickedObjectProperties(const std::string& object_name, const std::string& owner_name, const LLUUID owner_id)
|
|
{
|
|
childSetText("object_name", object_name);
|
|
// [RLVa:KB] - Checked: 2009-07-08 (RLVa-1.0.0e) | Added: RVLa-1.0.0e
|
|
if (gRlvHandler.hasBehaviour(RLV_BHVR_SHOWNAMES))
|
|
{
|
|
childSetVisible("object_name", true); // Show the object name if the picked object is actually an object
|
|
}
|
|
// [/RLVa:KB]
|
|
childSetText("owner_name", owner_name);
|
|
childSetText("abuser_name_edit", owner_name);
|
|
mAbuserID = owner_id;
|
|
}
|
|
|
|
|
|
bool LLFloaterReporter::validateReport()
|
|
{
|
|
// Ensure user selected a category from the list
|
|
LLSD category_sd = childGetValue("category_combo");
|
|
U8 category = (U8)category_sd.asInteger();
|
|
if(category >= 100) //This is here for reasons (like shenanigans)
|
|
{
|
|
LLNotificationsUtil::add("HelpReportNope");
|
|
return false;
|
|
}
|
|
if (category == 0)
|
|
{
|
|
if ( mReportType != BUG_REPORT )
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseSelectCategory");
|
|
}
|
|
else
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseAbuserNameEmpty");
|
|
}
|
|
return false;
|
|
}
|
|
|
|
if ( mReportType != BUG_REPORT )
|
|
{
|
|
if ( childGetText("abuser_name_edit").empty() )
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseAbuserLocationEmpty");
|
|
return false;
|
|
};
|
|
|
|
if ( childGetText("abuse_location_edit").empty() )
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseAbuserLocationEmpty");
|
|
return false;
|
|
};
|
|
};
|
|
|
|
if ( childGetText("summary_edit").empty() )
|
|
{
|
|
if ( mReportType != BUG_REPORT )
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseSummaryEmpty");
|
|
}
|
|
else
|
|
{
|
|
LLNotificationsUtil::add("HelpReportBugSummaryEmpty");
|
|
}
|
|
return false;
|
|
};
|
|
|
|
if ( childGetText("details_edit") == mDefaultSummary )
|
|
{
|
|
if ( mReportType != BUG_REPORT )
|
|
{
|
|
LLNotificationsUtil::add("HelpReportAbuseDetailsEmpty");
|
|
}
|
|
else
|
|
{
|
|
LLNotificationsUtil::add("HelpReportBugDetailsEmpty");
|
|
}
|
|
return false;
|
|
};
|
|
return true;
|
|
}
|
|
|
|
LLSD LLFloaterReporter::gatherReport()
|
|
{
|
|
LLViewerRegion *regionp = gAgent.getRegion();
|
|
if (!regionp) return LLSD(); // *TODO handle this failure case more gracefully
|
|
|
|
// reset flag in case the next report also contains this text
|
|
mCopyrightWarningSeen = FALSE;
|
|
|
|
std::ostringstream summary;
|
|
if (!LLViewerLogin::getInstance()->isInProductionGrid())
|
|
{
|
|
summary << "Preview ";
|
|
}
|
|
|
|
std::string category_name;
|
|
LLComboBox* combo = getChild<LLComboBox>( "category_combo");
|
|
if (combo)
|
|
{
|
|
category_name = combo->getSelectedItemLabel(); // want label, not value
|
|
}
|
|
|
|
#if LL_WINDOWS
|
|
const char* platform = "Win";
|
|
const char* short_platform = "O:W";
|
|
#elif LL_DARWIN
|
|
const char* platform = "Mac";
|
|
const char* short_platform = "O:M";
|
|
#elif LL_LINUX
|
|
const char* platform = "Lnx";
|
|
const char* short_platform = "O:L";
|
|
#elif LL_SOLARIS
|
|
const char* platform = "Sol";
|
|
const char* short_platform = "O:S";
|
|
#else
|
|
const char* platform = "???";
|
|
const char* short_platform = "O:?";
|
|
#endif
|
|
|
|
|
|
if ( mReportType == BUG_REPORT)
|
|
{
|
|
summary << short_platform << " V" << gVersionMajor << "."
|
|
<< gVersionMinor << "."
|
|
<< gVersionPatch << "."
|
|
<< gVersionBuild
|
|
<< " (" << regionp->getName() << ")"
|
|
<< "[" << category_name << "] "
|
|
<< "\"" << childGetValue("summary_edit").asString() << "\"";
|
|
}
|
|
else
|
|
{
|
|
summary << ""
|
|
<< " |" << regionp->getName() << "|" // region reporter is currently in.
|
|
<< " (" << childGetText("abuse_location_edit") << ")" // region abuse occured in (freeform text - no LLRegionPicker tool)
|
|
<< " [" << category_name << "] " // updated category
|
|
<< " {" << childGetText("abuser_name_edit") << "} " // name of abuse entered in report (chosen using LLAvatarPicker)
|
|
<< " \"" << childGetValue("summary_edit").asString() << "\""; // summary as entered
|
|
};
|
|
|
|
std::ostringstream details;
|
|
if (mReportType != BUG_REPORT)
|
|
{
|
|
details << "V" << gVersionMajor << "." // client version moved to body of email for abuse reports
|
|
<< gVersionMinor << "."
|
|
<< gVersionPatch << "."
|
|
<< gVersionBuild << std::endl << std::endl;
|
|
}
|
|
std::string object_name = childGetText("object_name");
|
|
std::string owner_name = childGetText("owner_name");
|
|
if (!object_name.empty() && !owner_name.empty())
|
|
{
|
|
details << "Object: " << object_name << "\n";
|
|
details << "Owner: " << owner_name << "\n";
|
|
}
|
|
|
|
if ( mReportType != BUG_REPORT )
|
|
{
|
|
details << "Abuser name: " << childGetText("abuser_name_edit") << " \n";
|
|
details << "Abuser location: " << childGetText("abuse_location_edit") << " \n";
|
|
};
|
|
|
|
details << childGetValue("details_edit").asString();
|
|
|
|
std::string version_string;
|
|
version_string = llformat(
|
|
"%d.%d.%d %s %s %s %s",
|
|
gVersionMajor,
|
|
gVersionMinor,
|
|
gVersionPatch,
|
|
platform,
|
|
gSysCPU.getFamily().c_str(),
|
|
gGLManager.mGLRenderer.c_str(),
|
|
gGLManager.mDriverVersionVendorString.c_str());
|
|
|
|
// only send a screenshot ID if we're asked to and the email is
|
|
// going to LL - Estate Owners cannot see the screenshot asset
|
|
LLUUID screenshot_id = LLUUID::null;
|
|
if (childGetValue("screen_check"))
|
|
{
|
|
screenshot_id = childGetValue("screenshot");
|
|
};
|
|
|
|
LLSD report = LLSD::emptyMap();
|
|
report["report-type"] = (U8) mReportType;
|
|
report["category"] = childGetValue("category_combo");
|
|
report["position"] = mPosition.getValue();
|
|
report["check-flags"] = (U8)0; // this is not used
|
|
report["screenshot-id"] = screenshot_id;
|
|
report["object-id"] = mObjectID;
|
|
report["abuser-id"] = mAbuserID;
|
|
report["abuse-region-name"] = "";
|
|
report["abuse-region-id"] = LLUUID::null;
|
|
report["summary"] = summary.str();
|
|
report["version-string"] = version_string;
|
|
report["details"] = details.str();
|
|
return report;
|
|
}
|
|
|
|
void LLFloaterReporter::sendReportViaLegacy(const LLSD & report)
|
|
{
|
|
LLViewerRegion *regionp = gAgent.getRegion();
|
|
if (!regionp) return;
|
|
LLMessageSystem *msg = gMessageSystem;
|
|
msg->newMessageFast(_PREHASH_UserReport);
|
|
msg->nextBlockFast(_PREHASH_AgentData);
|
|
msg->addUUIDFast(_PREHASH_AgentID, gAgent.getID());
|
|
msg->addUUIDFast(_PREHASH_SessionID, gAgent.getSessionID());
|
|
|
|
msg->nextBlockFast(_PREHASH_ReportData);
|
|
msg->addU8Fast(_PREHASH_ReportType, report["report-type"].asInteger());
|
|
msg->addU8(_PREHASH_Category, report["category"].asInteger());
|
|
msg->addVector3Fast(_PREHASH_Position, LLVector3(report["position"]));
|
|
msg->addU8Fast(_PREHASH_CheckFlags, report["check-flags"].asInteger());
|
|
msg->addUUIDFast(_PREHASH_ScreenshotID, report["screenshot-id"].asUUID());
|
|
msg->addUUIDFast(_PREHASH_ObjectID, report["object-id"].asUUID());
|
|
msg->addUUID("AbuserID", report["abuser-id"].asUUID());
|
|
msg->addString("AbuseRegionName", report["abuse-region-name"].asString());
|
|
msg->addUUID("AbuseRegionID", report["abuse-region-id"].asUUID());
|
|
|
|
msg->addStringFast(_PREHASH_Summary, report["summary"].asString());
|
|
msg->addString("VersionString", report["version-string"]);
|
|
msg->addStringFast(_PREHASH_Details, report["details"] );
|
|
|
|
msg->sendReliable(regionp->getHost());
|
|
}
|
|
|
|
class LLUserReportScreenshotResponder : public LLAssetUploadResponder
|
|
{
|
|
public:
|
|
LLUserReportScreenshotResponder(const LLSD & post_data,
|
|
const LLUUID & vfile_id,
|
|
LLAssetType::EType asset_type):
|
|
LLAssetUploadResponder(post_data, vfile_id, asset_type)
|
|
{
|
|
}
|
|
void uploadFailed(const LLSD& content)
|
|
{
|
|
// *TODO pop up a dialog so the user knows their report screenshot didn't make it
|
|
LLUploadDialog::modalUploadFinished();
|
|
}
|
|
void uploadComplete(const LLSD& content)
|
|
{
|
|
// we don't care about what the server returns from this post, just clean up the UI
|
|
LLUploadDialog::modalUploadFinished();
|
|
}
|
|
|
|
/*virtual*/ char const* getName(void) const { return "LLUserReportScreenshotResponder"; }
|
|
};
|
|
|
|
class LLUserReportResponder : public LLHTTPClient::ResponderWithResult
|
|
{
|
|
public:
|
|
LLUserReportResponder() { }
|
|
|
|
/*virtual*/ void error(U32 status, const std::string& reason)
|
|
{
|
|
// *TODO do some user messaging here
|
|
LLUploadDialog::modalUploadFinished();
|
|
}
|
|
/*virtual*/ void result(const LLSD& content)
|
|
{
|
|
// we don't care about what the server returns
|
|
LLUploadDialog::modalUploadFinished();
|
|
}
|
|
/*virtual*/ AIHTTPTimeoutPolicy const& getHTTPTimeoutPolicy(void) const { return userReportResponder_timeout; }
|
|
/*virtual*/ char const* getName(void) const { return "LLUserReportResponder"; }
|
|
};
|
|
|
|
void LLFloaterReporter::sendReportViaCaps(std::string url, std::string sshot_url, const LLSD& report)
|
|
{
|
|
if(childGetValue("screen_check").asBoolean() && !sshot_url.empty())
|
|
{
|
|
// try to upload screenshot
|
|
LLHTTPClient::post(sshot_url, report, new LLUserReportScreenshotResponder(report,
|
|
mResourceDatap->mAssetInfo.mUuid,
|
|
mResourceDatap->mAssetInfo.mType));
|
|
}
|
|
else
|
|
{
|
|
// screenshot not wanted or we don't have screenshot cap
|
|
LLHTTPClient::post(url, report, new LLUserReportResponder);
|
|
}
|
|
}
|
|
|
|
void LLFloaterReporter::takeScreenshot()
|
|
{
|
|
const S32 IMAGE_WIDTH = 1024;
|
|
const S32 IMAGE_HEIGHT = 768;
|
|
|
|
LLPointer<LLImageRaw> raw = new LLImageRaw;
|
|
// Warning: This crops left and right in case of wide-screen monitor:
|
|
if( !gViewerWindow->rawSnapshot(raw, IMAGE_WIDTH, IMAGE_HEIGHT, (F32)IMAGE_WIDTH / IMAGE_HEIGHT, TRUE, FALSE))
|
|
{
|
|
llwarns << "Unable to take screenshot" << llendl;
|
|
return;
|
|
}
|
|
LLPointer<LLImageJ2C> upload_data = LLViewerTextureList::convertToUploadFile(raw);
|
|
|
|
// create a resource data
|
|
mResourceDatap->mInventoryType = LLInventoryType::IT_NONE;
|
|
mResourceDatap->mNextOwnerPerm = 0; // not used
|
|
mResourceDatap->mExpectedUploadCost = 0; // we expect that abuse screenshots are free
|
|
mResourceDatap->mAssetInfo.mTransactionID.generate();
|
|
mResourceDatap->mAssetInfo.mUuid = mResourceDatap->mAssetInfo.mTransactionID.makeAssetID(gAgent.getSecureSessionID());
|
|
if (BUG_REPORT == mReportType)
|
|
{
|
|
mResourceDatap->mAssetInfo.mType = LLAssetType::AT_TEXTURE;
|
|
mResourceDatap->mPreferredLocation = LLFolderType::EType(-1);
|
|
}
|
|
else if (COMPLAINT_REPORT == mReportType)
|
|
{
|
|
mResourceDatap->mAssetInfo.mType = LLAssetType::AT_TEXTURE;
|
|
mResourceDatap->mPreferredLocation = LLFolderType::EType(-2);
|
|
}
|
|
else
|
|
{
|
|
llwarns << "Unknown LLFloaterReporter type" << llendl;
|
|
}
|
|
mResourceDatap->mAssetInfo.mCreatorID = gAgentID;
|
|
mResourceDatap->mAssetInfo.setName("screenshot_name");
|
|
mResourceDatap->mAssetInfo.setDescription("screenshot_descr");
|
|
|
|
// store in VFS
|
|
LLVFile::writeFile(upload_data->getData(),
|
|
upload_data->getDataSize(),
|
|
gVFS,
|
|
mResourceDatap->mAssetInfo.mUuid,
|
|
mResourceDatap->mAssetInfo.mType);
|
|
|
|
// store in the image list so it doesn't try to fetch from the server
|
|
LLPointer<LLViewerFetchedTexture> image_in_list =
|
|
LLViewerTextureManager::getFetchedTexture(mResourceDatap->mAssetInfo.mUuid, TRUE, LLGLTexture::BOOST_NONE, LLViewerTexture::FETCHED_TEXTURE);
|
|
image_in_list->createGLTexture(0, raw, 0, TRUE, LLViewerTexture::OTHER);
|
|
|
|
// the texture picker then uses that texture
|
|
LLTexturePicker* texture = getChild<LLTextureCtrl>("screenshot");
|
|
if (texture)
|
|
{
|
|
texture->setImageAssetID(mResourceDatap->mAssetInfo.mUuid);
|
|
texture->setDefaultImageAssetID(mResourceDatap->mAssetInfo.mUuid);
|
|
texture->setCaption(getString("Screenshot"));
|
|
}
|
|
|
|
}
|
|
|
|
void LLFloaterReporter::uploadImage()
|
|
{
|
|
llinfos << "*** Uploading: " << llendl;
|
|
llinfos << "Type: " << LLAssetType::lookup(mResourceDatap->mAssetInfo.mType) << llendl;
|
|
llinfos << "UUID: " << mResourceDatap->mAssetInfo.mUuid << llendl;
|
|
llinfos << "Name: " << mResourceDatap->mAssetInfo.getName() << llendl;
|
|
llinfos << "Desc: " << mResourceDatap->mAssetInfo.getDescription() << llendl;
|
|
|
|
gAssetStorage->storeAssetData(mResourceDatap->mAssetInfo.mTransactionID,
|
|
mResourceDatap->mAssetInfo.mType,
|
|
LLFloaterReporter::uploadDoneCallback,
|
|
(void*)mResourceDatap, TRUE);
|
|
}
|
|
|
|
|
|
// static
|
|
void LLFloaterReporter::uploadDoneCallback(const LLUUID &uuid, void *user_data, S32 result, LLExtStat ext_status) // StoreAssetData callback (fixed)
|
|
{
|
|
LLUploadDialog::modalUploadFinished();
|
|
|
|
LLResourceData* data = (LLResourceData*)user_data;
|
|
|
|
if(result < 0)
|
|
{
|
|
LLSD args;
|
|
args["REASON"] = std::string(LLAssetStorage::getErrorString(result));
|
|
LLNotificationsUtil::add("ErrorUploadingReportScreenshot", args);
|
|
|
|
std::string err_msg("There was a problem uploading a report screenshot");
|
|
err_msg += " due to the following reason: " + args["REASON"].asString();
|
|
llwarns << err_msg << llendl;
|
|
return;
|
|
}
|
|
|
|
EReportType report_type = UNKNOWN_REPORT;
|
|
if (data->mPreferredLocation == -1)
|
|
{
|
|
report_type = BUG_REPORT;
|
|
}
|
|
else if (data->mPreferredLocation == -2)
|
|
{
|
|
report_type = COMPLAINT_REPORT;
|
|
}
|
|
else
|
|
{
|
|
llwarns << "Unknown report type : " << data->mPreferredLocation << llendl;
|
|
}
|
|
|
|
LLFloaterReporter *self = getReporter(report_type);
|
|
if (self)
|
|
{
|
|
self->mScreenID = uuid;
|
|
llinfos << "Got screen shot " << uuid << llendl;
|
|
self->sendReportViaLegacy(self->gatherReport());
|
|
self->close();
|
|
}
|
|
}
|
|
|
|
|
|
void LLFloaterReporter::setPosBox(const LLVector3d &pos)
|
|
{
|
|
mPosition.setVec(pos);
|
|
std::string pos_string = llformat("{%.1f, %.1f, %.1f}",
|
|
mPosition.mV[VX],
|
|
mPosition.mV[VY],
|
|
mPosition.mV[VZ]);
|
|
childSetText("pos_field", pos_string);
|
|
}
|
|
|
|
void LLFloaterReporter::setDescription(const std::string& description, LLMeanCollisionData *mcd)
|
|
{
|
|
LLFloaterReporter *self = gReporterInstances[COMPLAINT_REPORT];
|
|
if (self)
|
|
{
|
|
self->childSetText("details_edit", description);
|
|
|
|
for_each(self->mMCDList.begin(), self->mMCDList.end(), DeletePointer());
|
|
self->mMCDList.clear();
|
|
if (mcd)
|
|
{
|
|
self->mMCDList.push_back(new LLMeanCollisionData(mcd));
|
|
}
|
|
}
|
|
}
|
|
|
|
void LLFloaterReporter::addDescription(const std::string& description, LLMeanCollisionData *mcd)
|
|
{
|
|
LLFloaterReporter *self = gReporterInstances[COMPLAINT_REPORT];
|
|
if (self)
|
|
{
|
|
LLTextEditor* text = self->getChild<LLTextEditor>("details_edit");
|
|
if (text)
|
|
{
|
|
text->insertText(description);
|
|
}
|
|
if (mcd)
|
|
{
|
|
self->mMCDList.push_back(new LLMeanCollisionData(mcd));
|
|
}
|
|
}
|
|
}
|