/*
* Copyright (c) 2006-2008, openmetaverse.org
* All rights reserved.
*
* - Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* - Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
* - Neither the name of the openmetaverse.org nor the names
* of its contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
using System;
using OpenMetaverse.Packets;
using OpenMetaverse.Interfaces;
using OpenMetaverse.Messages.Linden;
using System.Collections.Generic;
namespace OpenMetaverse
{
/// Describes tasks returned in LandStatReply
public class EstateTask
{
public Vector3 Position;
public float Score;
public float MonoScore;
public UUID TaskID;
public uint TaskLocalID;
public string TaskName;
public string OwnerName;
}
///
/// Estate level administration and utilities
///
public class EstateTools
{
private GridClient Client;
/// Textures for each of the four terrain height levels
public GroundTextureSettings GroundTextures;
/// Upper/lower texture boundaries for each corner of the sim
public GroundTextureHeightSettings GroundTextureLimits;
///
/// Constructor for EstateTools class
///
///
public EstateTools(GridClient client)
{
GroundTextures = new GroundTextureSettings();
GroundTextureLimits = new GroundTextureHeightSettings();
Client = client;
Client.Network.RegisterCallback(PacketType.LandStatReply, LandStatReplyHandler);
Client.Network.RegisterCallback(PacketType.EstateOwnerMessage, EstateOwnerMessageHandler);
Client.Network.RegisterCallback(PacketType.EstateCovenantReply, EstateCovenantReplyHandler);
Client.Network.RegisterEventCallback("LandStatReply", new Caps.EventQueueCallback(LandStatCapsReplyHandler));
}
#region Enums
/// Used in the ReportType field of a LandStatRequest
public enum LandStatReportType
{
TopScripts = 0,
TopColliders = 1
}
/// Used by EstateOwnerMessage packets
public enum EstateAccessDelta : uint
{
BanUser = 64,
BanUserAllEstates = 66,
UnbanUser = 128,
UnbanUserAllEstates = 130,
AddManager = 256,
AddManagerAllEstates = 257,
RemoveManager = 512,
RemoveManagerAllEstates = 513,
AddUserAsAllowed = 4,
AddAllowedAllEstates = 6,
RemoveUserAsAllowed = 8,
RemoveUserAllowedAllEstates = 10,
AddGroupAsAllowed = 16,
AddGroupAllowedAllEstates = 18,
RemoveGroupAsAllowed = 32,
RemoveGroupAllowedAllEstates = 34
}
/// Used by EstateOwnerMessage packets
public enum EstateAccessReplyDelta : uint
{
AllowedUsers = 17,
AllowedGroups = 18,
EstateBans = 20,
EstateManagers = 24
}
///
///
///
[Flags]
public enum EstateReturnFlags : uint
{
/// No flags set
None = 2,
/// Only return targets scripted objects
ReturnScripted = 6,
/// Only return targets objects if on others land
ReturnOnOthersLand = 3,
/// Returns target's scripted objects and objects on other parcels
ReturnScriptedAndOnOthers = 7
}
#endregion
#region Structs
/// Ground texture settings for each corner of the region
// TODO: maybe move this class to the Simulator object and implement it there too
public struct GroundTextureSettings
{
public UUID Low;
public UUID MidLow;
public UUID MidHigh;
public UUID High;
}
/// Used by GroundTextureHeightSettings
public struct GroundTextureHeight
{
public float Low;
public float High;
}
/// The high and low texture thresholds for each corner of the sim
public struct GroundTextureHeightSettings
{
public GroundTextureHeight SW;
public GroundTextureHeight NW;
public GroundTextureHeight SE;
public GroundTextureHeight NE;
}
#endregion
#region Event delegates, Raise Events
/// The event subscribers. null if no subcribers
private EventHandler m_TopCollidersReply;
/// Raises the TopCollidersReply event
/// A TopCollidersReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnTopCollidersReply(TopCollidersReplyEventArgs e)
{
EventHandler handler = m_TopCollidersReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_TopCollidersReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler TopCollidersReply
{
add { lock (m_TopCollidersReply_Lock) { m_TopCollidersReply += value; } }
remove { lock (m_TopCollidersReply_Lock) { m_TopCollidersReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_TopScriptsReply;
/// Raises the TopScriptsReply event
/// A TopScriptsReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnTopScriptsReply(TopScriptsReplyEventArgs e)
{
EventHandler handler = m_TopScriptsReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_TopScriptsReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler TopScriptsReply
{
add { lock (m_TopScriptsReply_Lock) { m_TopScriptsReply += value; } }
remove { lock (m_TopScriptsReply_Lock) { m_TopScriptsReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_EstateUsersReply;
/// Raises the EstateUsersReply event
/// A EstateUsersReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnEstateUsersReply(EstateUsersReplyEventArgs e)
{
EventHandler handler = m_EstateUsersReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_EstateUsersReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler EstateUsersReply
{
add { lock (m_EstateUsersReply_Lock) { m_EstateUsersReply += value; } }
remove { lock (m_EstateUsersReply_Lock) { m_EstateUsersReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_EstateGroupsReply;
/// Raises the EstateGroupsReply event
/// A EstateGroupsReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnEstateGroupsReply(EstateGroupsReplyEventArgs e)
{
EventHandler handler = m_EstateGroupsReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_EstateGroupsReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler EstateGroupsReply
{
add { lock (m_EstateGroupsReply_Lock) { m_EstateGroupsReply += value; } }
remove { lock (m_EstateGroupsReply_Lock) { m_EstateGroupsReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_EstateManagersReply;
/// Raises the EstateManagersReply event
/// A EstateManagersReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnEstateManagersReply(EstateManagersReplyEventArgs e)
{
EventHandler handler = m_EstateManagersReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_EstateManagersReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler EstateManagersReply
{
add { lock (m_EstateManagersReply_Lock) { m_EstateManagersReply += value; } }
remove { lock (m_EstateManagersReply_Lock) { m_EstateManagersReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_EstateBansReply;
/// Raises the EstateBansReply event
/// A EstateBansReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnEstateBansReply(EstateBansReplyEventArgs e)
{
EventHandler handler = m_EstateBansReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_EstateBansReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler EstateBansReply
{
add { lock (m_EstateBansReply_Lock) { m_EstateBansReply += value; } }
remove { lock (m_EstateBansReply_Lock) { m_EstateBansReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_EstateCovenantReply;
/// Raises the EstateCovenantReply event
/// A EstateCovenantReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnEstateCovenantReply(EstateCovenantReplyEventArgs e)
{
EventHandler handler = m_EstateCovenantReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_EstateCovenantReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler EstateCovenantReply
{
add { lock (m_EstateCovenantReply_Lock) { m_EstateCovenantReply += value; } }
remove { lock (m_EstateCovenantReply_Lock) { m_EstateCovenantReply -= value; } }
}
/// The event subscribers. null if no subcribers
private EventHandler m_EstateUpdateInfoReply;
/// Raises the EstateUpdateInfoReply event
/// A EstateUpdateInfoReplyEventArgs object containing the
/// data returned from the data server
protected virtual void OnEstateUpdateInfoReply(EstateUpdateInfoReplyEventArgs e)
{
EventHandler handler = m_EstateUpdateInfoReply;
if (handler != null)
handler(this, e);
}
/// Thread sync lock object
private readonly object m_EstateUpdateInfoReply_Lock = new object();
/// Raised when the data server responds to a request.
public event EventHandler EstateUpdateInfoReply
{
add { lock (m_EstateUpdateInfoReply_Lock) { m_EstateUpdateInfoReply += value; } }
remove { lock (m_EstateUpdateInfoReply_Lock) { m_EstateUpdateInfoReply -= value; } }
}
#endregion
#region Public Methods
///
/// Requests estate information such as top scripts and colliders
///
///
///
///
///
public void LandStatRequest(int parcelLocalID, LandStatReportType reportType, uint requestFlags, string filter)
{
LandStatRequestPacket p = new LandStatRequestPacket();
p.AgentData.AgentID = Client.Self.AgentID;
p.AgentData.SessionID = Client.Self.SessionID;
p.RequestData.Filter = Utils.StringToBytes(filter);
p.RequestData.ParcelLocalID = parcelLocalID;
p.RequestData.ReportType = (uint)reportType;
p.RequestData.RequestFlags = requestFlags;
Client.Network.SendPacket(p);
}
/// Requests estate settings, including estate manager and access/ban lists
public void RequestInfo()
{
EstateOwnerMessage("getinfo", "");
}
/// Requests the "Top Scripts" list for the current region
public void RequestTopScripts()
{
//EstateOwnerMessage("scripts", "");
LandStatRequest(0, LandStatReportType.TopScripts, 0, "");
}
/// Requests the "Top Colliders" list for the current region
public void RequestTopColliders()
{
//EstateOwnerMessage("colliders", "");
LandStatRequest(0, LandStatReportType.TopColliders, 0, "");
}
///
/// Set several estate specific configuration variables
///
/// The Height of the waterlevel over the entire estate. Defaults to 20
/// The maximum height change allowed above the baked terrain. Defaults to 4
/// The minimum height change allowed below the baked terrain. Defaults to -4
/// true to use
/// if True forces the sun position to the position in SunPosition
/// The current position of the sun on the estate, or when FixedSun is true the static position
/// the sun will remain. 6.0 = Sunrise, 30.0 = Sunset
public void SetTerrainVariables(float WaterHeight, float TerrainRaiseLimit,
float TerrainLowerLimit, bool UseEstateSun, bool FixedSun, float SunPosition)
{
List simVariables = new List();
simVariables.Add(WaterHeight.ToString(Utils.EnUsCulture));
simVariables.Add(TerrainRaiseLimit.ToString(Utils.EnUsCulture));
simVariables.Add(TerrainLowerLimit.ToString(Utils.EnUsCulture));
simVariables.Add(UseEstateSun ? "Y" : "N");
simVariables.Add(FixedSun ? "Y" : "N");
simVariables.Add(SunPosition.ToString(Utils.EnUsCulture));
simVariables.Add("Y"); //Not used?
simVariables.Add("N"); //Not used?
simVariables.Add("0.00"); //Also not used?
EstateOwnerMessage("setregionterrain", simVariables);
}
///
/// Request return of objects owned by specified avatar
///
/// The Agents owning the primitives to return
/// specify the coverage and type of objects to be included in the return
/// true to perform return on entire estate
public void SimWideReturn(UUID Target, EstateReturnFlags flag, bool EstateWide)
{
if (EstateWide)
{
List param = new List();
param.Add(flag.ToString());
param.Add(Target.ToString());
EstateOwnerMessage("estateobjectreturn", param);
}
else
{
SimWideDeletesPacket simDelete = new SimWideDeletesPacket();
simDelete.AgentData.AgentID = Client.Self.AgentID;
simDelete.AgentData.SessionID = Client.Self.SessionID;
simDelete.DataBlock.TargetID = Target;
simDelete.DataBlock.Flags = (uint)flag;
Client.Network.SendPacket(simDelete);
}
}
///
///
///
public void EstateOwnerMessage(string method, string param)
{
List listParams = new List();
listParams.Add(param);
EstateOwnerMessage(method, listParams);
}
///
/// Used for setting and retrieving various estate panel settings
///
/// EstateOwnerMessage Method field
/// List of parameters to include
public void EstateOwnerMessage(string method, List listParams)
{
EstateOwnerMessagePacket estate = new EstateOwnerMessagePacket();
estate.AgentData.AgentID = Client.Self.AgentID;
estate.AgentData.SessionID = Client.Self.SessionID;
estate.AgentData.TransactionID = UUID.Zero;
estate.MethodData.Invoice = UUID.Random();
estate.MethodData.Method = Utils.StringToBytes(method);
estate.ParamList = new EstateOwnerMessagePacket.ParamListBlock[listParams.Count];
for (int i = 0; i < listParams.Count; i++)
{
estate.ParamList[i] = new EstateOwnerMessagePacket.ParamListBlock();
estate.ParamList[i].Parameter = Utils.StringToBytes(listParams[i]);
}
Client.Network.SendPacket((Packet)estate);
}
///
/// Kick an avatar from an estate
///
/// Key of Agent to remove
public void KickUser(UUID userID)
{
EstateOwnerMessage("kickestate", userID.ToString());
}
///
/// Ban an avatar from an estate
/// Key of Agent to remove
/// Ban user from this estate and all others owned by the estate owner
public void BanUser(UUID userID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.BanUserAllEstates : (uint)EstateAccessDelta.BanUser;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(userID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
/// Unban an avatar from an estate
/// Key of Agent to remove
/// /// Unban user from this estate and all others owned by the estate owner
public void UnbanUser(UUID userID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.UnbanUserAllEstates : (uint)EstateAccessDelta.UnbanUser;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(userID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
///
/// Send a message dialog to everyone in an entire estate
///
/// Message to send all users in the estate
public void EstateMessage(string message)
{
List listParams = new List();
listParams.Add(Client.Self.FirstName + " " + Client.Self.LastName);
listParams.Add(message);
EstateOwnerMessage("instantmessage", listParams);
}
///
/// Send a message dialog to everyone in a simulator
///
/// Message to send all users in the simulator
public void SimulatorMessage(string message)
{
List listParams = new List();
listParams.Add("-1");
listParams.Add("-1");
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(Client.Self.FirstName + " " + Client.Self.LastName);
listParams.Add(message);
EstateOwnerMessage("simulatormessage", listParams);
}
///
/// Send an avatar back to their home location
///
/// Key of avatar to send home
public void TeleportHomeUser(UUID pest)
{
List listParams = new List();
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(pest.ToString());
EstateOwnerMessage("teleporthomeuser", listParams);
}
///
/// Begin the region restart process
///
public void RestartRegion()
{
EstateOwnerMessage("restart", "120");
}
///
/// Cancels a region restart
///
public void CancelRestart()
{
EstateOwnerMessage("restart", "-1");
}
/// Estate panel "Region" tab settings
public void SetRegionInfo(bool blockTerraform, bool blockFly, bool allowDamage, bool allowLandResell, bool restrictPushing, bool allowParcelJoinDivide, float agentLimit, float objectBonus, bool mature)
{
List listParams = new List();
if (blockTerraform) listParams.Add("Y"); else listParams.Add("N");
if (blockFly) listParams.Add("Y"); else listParams.Add("N");
if (allowDamage) listParams.Add("Y"); else listParams.Add("N");
if (allowLandResell) listParams.Add("Y"); else listParams.Add("N");
listParams.Add(agentLimit.ToString());
listParams.Add(objectBonus.ToString());
if (mature) listParams.Add("21"); else listParams.Add("13"); //FIXME - enumerate these settings
if (restrictPushing) listParams.Add("Y"); else listParams.Add("N");
if (allowParcelJoinDivide) listParams.Add("Y"); else listParams.Add("N");
EstateOwnerMessage("setregioninfo", listParams);
}
/// Estate panel "Debug" tab settings
public void SetRegionDebug(bool disableScripts, bool disableCollisions, bool disablePhysics)
{
List listParams = new List();
if (disableScripts) listParams.Add("Y"); else listParams.Add("N");
if (disableCollisions) listParams.Add("Y"); else listParams.Add("N");
if (disablePhysics) listParams.Add("Y"); else listParams.Add("N");
EstateOwnerMessage("setregiondebug", listParams);
}
/// Used for setting the region's terrain textures for its four height levels
///
///
///
///
public void SetRegionTerrain(UUID low, UUID midLow, UUID midHigh, UUID high)
{
List listParams = new List();
listParams.Add("0 " + low.ToString());
listParams.Add("1 " + midLow.ToString());
listParams.Add("2 " + midHigh.ToString());
listParams.Add("3 " + high.ToString());
EstateOwnerMessage("texturedetail", listParams);
EstateOwnerMessage("texturecommit", "");
}
/// Used for setting sim terrain texture heights
public void SetRegionTerrainHeights(float lowSW, float highSW, float lowNW, float highNW, float lowSE, float highSE, float lowNE, float highNE)
{
List listParams = new List();
listParams.Add("0 " + lowSW.ToString(Utils.EnUsCulture) + " " + highSW.ToString(Utils.EnUsCulture)); //SW low-high
listParams.Add("1 " + lowNW.ToString(Utils.EnUsCulture) + " " + highNW.ToString(Utils.EnUsCulture)); //NW low-high
listParams.Add("2 " + lowSE.ToString(Utils.EnUsCulture) + " " + highSE.ToString(Utils.EnUsCulture)); //SE low-high
listParams.Add("3 " + lowNE.ToString(Utils.EnUsCulture) + " " + highNE.ToString(Utils.EnUsCulture)); //NE low-high
EstateOwnerMessage("textureheights", listParams);
EstateOwnerMessage("texturecommit", "");
}
/// Requests the estate covenant
public void RequestCovenant()
{
EstateCovenantRequestPacket req = new EstateCovenantRequestPacket();
req.AgentData.AgentID = Client.Self.AgentID;
req.AgentData.SessionID = Client.Self.SessionID;
Client.Network.SendPacket(req);
}
///
/// Upload a terrain RAW file
///
/// A byte array containing the encoded terrain data
/// The name of the file being uploaded
/// The Id of the transfer request
public UUID UploadTerrain(byte[] fileData, string fileName)
{
AssetUpload upload = new AssetUpload();
upload.AssetData = fileData;
upload.AssetType = AssetType.Unknown;
upload.Size = fileData.Length;
upload.ID = UUID.Random();
// Tell the library we have a pending file to upload
Client.Assets.SetPendingAssetUploadData(upload);
// Create and populate a list with commands specific to uploading a raw terrain file
List paramList = new List();
paramList.Add("upload filename");
paramList.Add(fileName);
// Tell the simulator we have a new raw file to upload
Client.Estate.EstateOwnerMessage("terrain", paramList);
return upload.ID;
}
///
/// Teleports all users home in current Estate
///
public void TeleportHomeAllUsers()
{
List Params = new List();
Params.Add(Client.Self.AgentID.ToString());
EstateOwnerMessage("teleporthomeallusers", Params);
}
///
/// Remove estate manager
/// Key of Agent to Remove
/// removes manager to this estate and all others owned by the estate owner
public void RemoveEstateManager(UUID userID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.RemoveManagerAllEstates : (uint)EstateAccessDelta.RemoveManager;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(userID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
///
/// Add estate manager
/// Key of Agent to Add
/// Add agent as manager to this estate and all others owned by the estate owner
public void AddEstateManager(UUID userID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.AddManagerAllEstates : (uint)EstateAccessDelta.AddManager;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(userID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
///
/// Add's an agent to the estate Allowed list
/// Key of Agent to Add
/// Add agent as an allowed reisdent to All estates if true
public void AddAllowedUser(UUID userID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.AddAllowedAllEstates : (uint)EstateAccessDelta.AddUserAsAllowed;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(userID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
///
/// Removes an agent from the estate Allowed list
/// Key of Agent to Remove
/// Removes agent as an allowed reisdent from All estates if true
public void RemoveAllowedUser(UUID userID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.RemoveUserAllowedAllEstates : (uint)EstateAccessDelta.RemoveUserAsAllowed;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(userID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
///
///
/// Add's a group to the estate Allowed list
/// Key of Group to Add
/// Add Group as an allowed group to All estates if true
public void AddAllowedGroup(UUID groupID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.AddGroupAllowedAllEstates : (uint)EstateAccessDelta.AddAllowedAllEstates;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(groupID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
///
///
/// Removes a group from the estate Allowed list
/// Key of Group to Remove
/// Removes Group as an allowed Group from All estates if true
public void RemoveAllowedGroup(UUID groupID, bool allEstates)
{
List listParams = new List();
uint flag = allEstates ? (uint)EstateAccessDelta.RemoveGroupAllowedAllEstates : (uint)EstateAccessDelta.RemoveGroupAsAllowed;
listParams.Add(Client.Self.AgentID.ToString());
listParams.Add(flag.ToString());
listParams.Add(groupID.ToString());
EstateOwnerMessage("estateaccessdelta", listParams);
}
#endregion
#region Packet Handlers
/// Process an incoming packet and raise the appropriate events
/// The sender
/// The EventArgs object containing the packet data
protected void EstateCovenantReplyHandler(object sender, PacketReceivedEventArgs e)
{
EstateCovenantReplyPacket reply = (EstateCovenantReplyPacket)e.Packet;
OnEstateCovenantReply(new EstateCovenantReplyEventArgs(
reply.Data.CovenantID,
reply.Data.CovenantTimestamp,
Utils.BytesToString(reply.Data.EstateName),
reply.Data.EstateOwnerID));
}
/// Process an incoming packet and raise the appropriate events
/// The sender
/// The EventArgs object containing the packet data
protected void EstateOwnerMessageHandler(object sender, PacketReceivedEventArgs e)
{
EstateOwnerMessagePacket message = (EstateOwnerMessagePacket)e.Packet;
uint estateID;
string method = Utils.BytesToString(message.MethodData.Method);
//List parameters = new List();
if (method == "estateupdateinfo")
{
string estateName = Utils.BytesToString(message.ParamList[0].Parameter);
UUID estateOwner = new UUID(Utils.BytesToString(message.ParamList[1].Parameter));
estateID = Utils.BytesToUInt(message.ParamList[2].Parameter);
/*
foreach (EstateOwnerMessagePacket.ParamListBlock param in message.ParamList)
{
parameters.Add(Utils.BytesToString(param.Parameter));
}
*/
bool denyNoPaymentInfo;
if (Utils.BytesToUInt(message.ParamList[8].Parameter) == 0) denyNoPaymentInfo = true;
else denyNoPaymentInfo = false;
OnEstateUpdateInfoReply(new EstateUpdateInfoReplyEventArgs(estateName, estateOwner, estateID, denyNoPaymentInfo));
}
else if (method == "setaccess")
{
int count;
estateID = Utils.BytesToUInt(message.ParamList[0].Parameter);
if (message.ParamList.Length > 1)
{
//param comes in as a string for some reason
uint param;
if (!uint.TryParse(Utils.BytesToString(message.ParamList[1].Parameter), out param)) return;
EstateAccessReplyDelta accessType = (EstateAccessReplyDelta)param;
switch (accessType)
{
case EstateAccessReplyDelta.EstateManagers:
//if (OnGetEstateManagers != null)
{
if (message.ParamList.Length > 5)
{
if (!int.TryParse(Utils.BytesToString(message.ParamList[5].Parameter), out count)) return;
List managers = new List();
for (int i = 6; i < message.ParamList.Length; i++)
{
try
{
UUID managerID = new UUID(message.ParamList[i].Parameter, 0);
managers.Add(managerID);
}
catch (Exception ex) { Logger.Log(ex.Message, Helpers.LogLevel.Error, Client, ex); }
}
OnEstateManagersReply(new EstateManagersReplyEventArgs(estateID, count, managers));
}
}
break;
case EstateAccessReplyDelta.EstateBans:
//if (OnGetEstateBans != null)
{
if (message.ParamList.Length > 5)
{
if (!int.TryParse(Utils.BytesToString(message.ParamList[4].Parameter), out count)) return;
List bannedUsers = new List();
for (int i = 6; i < message.ParamList.Length; i++)
{
try
{
UUID bannedID = new UUID(message.ParamList[i].Parameter, 0);
bannedUsers.Add(bannedID);
}
catch (Exception ex) { Logger.Log(ex.Message, Helpers.LogLevel.Error, Client, ex); }
}
OnEstateBansReply(new EstateBansReplyEventArgs(estateID, count, bannedUsers));
}
}
break;
case EstateAccessReplyDelta.AllowedUsers:
//if (OnGetAllowedUsers != null)
{
if (message.ParamList.Length > 5)
{
if (!int.TryParse(Utils.BytesToString(message.ParamList[2].Parameter), out count)) return;
List allowedUsers = new List();
for (int i = 6; i < message.ParamList.Length; i++)
{
try
{
UUID allowedID = new UUID(message.ParamList[i].Parameter, 0);
allowedUsers.Add(allowedID);
}
catch (Exception ex) { Logger.Log(ex.Message, Helpers.LogLevel.Error, Client, ex); }
}
OnEstateUsersReply(new EstateUsersReplyEventArgs(estateID, count, allowedUsers));
}
}
break;
case EstateAccessReplyDelta.AllowedGroups:
//if (OnGetAllowedGroups != null)
{
if (message.ParamList.Length > 5)
{
if (!int.TryParse(Utils.BytesToString(message.ParamList[3].Parameter), out count)) return;
List allowedGroups = new List();
for (int i = 6; i < message.ParamList.Length; i++)
{
try
{
UUID groupID = new UUID(message.ParamList[i].Parameter, 0);
allowedGroups.Add(groupID);
}
catch (Exception ex) { Logger.Log(ex.Message, Helpers.LogLevel.Error, Client, ex); }
}
OnEstateGroupsReply(new EstateGroupsReplyEventArgs(estateID, count, allowedGroups));
}
}
break;
}
}
}
}
/// Process an incoming packet and raise the appropriate events
/// The sender
/// The EventArgs object containing the packet data
protected void LandStatReplyHandler(object sender, PacketReceivedEventArgs e)
{
//if (OnLandStatReply != null || OnGetTopScripts != null || OnGetTopColliders != null)
//if (OnGetTopScripts != null || OnGetTopColliders != null)
{
LandStatReplyPacket p = (LandStatReplyPacket)e.Packet;
Dictionary Tasks = new Dictionary();
foreach (LandStatReplyPacket.ReportDataBlock rep in p.ReportData)
{
EstateTask task = new EstateTask();
task.Position = new Vector3(rep.LocationX, rep.LocationY, rep.LocationZ);
task.Score = rep.Score;
task.TaskID = rep.TaskID;
task.TaskLocalID = rep.TaskLocalID;
task.TaskName = Utils.BytesToString(rep.TaskName);
task.OwnerName = Utils.BytesToString(rep.OwnerName);
Tasks.Add(task.TaskID, task);
}
LandStatReportType type = (LandStatReportType)p.RequestData.ReportType;
if (type == LandStatReportType.TopScripts)
{
OnTopScriptsReply(new TopScriptsReplyEventArgs((int)p.RequestData.TotalObjectCount, Tasks));
}
else if (type == LandStatReportType.TopColliders)
{
OnTopCollidersReply(new TopCollidersReplyEventArgs((int) p.RequestData.TotalObjectCount, Tasks));
}
/*
if (OnGetTopColliders != null)
{
//FIXME - System.UnhandledExceptionEventArgs
OnLandStatReply(
type,
p.RequestData.RequestFlags,
(int)p.RequestData.TotalObjectCount,
Tasks
);
}
*/
}
}
private void LandStatCapsReplyHandler(string capsKey, IMessage message, Simulator simulator)
{
LandStatReplyMessage m = (LandStatReplyMessage)message;
Dictionary Tasks = new Dictionary();
foreach (LandStatReplyMessage.ReportDataBlock rep in m.ReportDataBlocks)
{
EstateTask task = new EstateTask();
task.Position = rep.Location;
task.Score = rep.Score;
task.MonoScore = rep.MonoScore;
task.TaskID = rep.TaskID;
task.TaskLocalID = rep.TaskLocalID;
task.TaskName = rep.TaskName;
task.OwnerName = rep.OwnerName;
Tasks.Add(task.TaskID, task);
}
LandStatReportType type = (LandStatReportType)m.ReportType;
if (type == LandStatReportType.TopScripts)
{
OnTopScriptsReply(new TopScriptsReplyEventArgs((int)m.TotalObjectCount, Tasks));
}
else if (type == LandStatReportType.TopColliders)
{
OnTopCollidersReply(new TopCollidersReplyEventArgs((int)m.TotalObjectCount, Tasks));
}
}
#endregion
}
#region EstateTools EventArgs Classes
/// Raised on LandStatReply when the report type is for "top colliders"
public class TopCollidersReplyEventArgs : EventArgs
{
private readonly int m_objectCount;
private readonly Dictionary m_Tasks;
///
/// The number of returned items in LandStatReply
///
public int ObjectCount { get { return m_objectCount; } }
///
/// A Dictionary of Object UUIDs to tasks returned in LandStatReply
///
public Dictionary Tasks { get { return m_Tasks; } }
/// Construct a new instance of the TopCollidersReplyEventArgs class
/// The number of returned items in LandStatReply
/// Dictionary of Object UUIDs to tasks returned in LandStatReply
public TopCollidersReplyEventArgs(int objectCount, Dictionary tasks)
{
this.m_objectCount = objectCount;
this.m_Tasks = tasks;
}
}
/// Raised on LandStatReply when the report type is for "top Scripts"
public class TopScriptsReplyEventArgs : EventArgs
{
private readonly int m_objectCount;
private readonly Dictionary m_Tasks;
///
/// The number of scripts returned in LandStatReply
///
public int ObjectCount { get { return m_objectCount; } }
///
/// A Dictionary of Object UUIDs to tasks returned in LandStatReply
///
public Dictionary Tasks { get { return m_Tasks; } }
/// Construct a new instance of the TopScriptsReplyEventArgs class
/// The number of returned items in LandStatReply
/// Dictionary of Object UUIDs to tasks returned in LandStatReply
public TopScriptsReplyEventArgs(int objectCount, Dictionary tasks)
{
this.m_objectCount = objectCount;
this.m_Tasks = tasks;
}
}
/// Returned, along with other info, upon a successful .RequestInfo()
public class EstateBansReplyEventArgs : EventArgs
{
private readonly uint m_estateID;
private readonly int m_count;
private readonly List m_banned;
///
/// The identifier of the estate
///
public uint EstateID { get { return m_estateID; } }
///
/// The number of returned itmes
///
public int Count { get { return m_count; } }
///
/// List of UUIDs of Banned Users
///
public List Banned { get { return m_banned; } }
/// Construct a new instance of the EstateBansReplyEventArgs class
/// The estate's identifier on the grid
/// The number of returned items in LandStatReply
/// User UUIDs banned
public EstateBansReplyEventArgs(uint estateID, int count, List banned)
{
this.m_estateID = estateID;
this.m_count = count;
this.m_banned = banned;
}
}
/// Returned, along with other info, upon a successful .RequestInfo()
public class EstateUsersReplyEventArgs : EventArgs
{
private readonly uint m_estateID;
private readonly int m_count;
private readonly List m_allowedUsers;
///
/// The identifier of the estate
///
public uint EstateID { get { return m_estateID; } }
///
/// The number of returned items
///
public int Count { get { return m_count; } }
///
/// List of UUIDs of Allowed Users
///
public List AllowedUsers { get { return m_allowedUsers; } }
/// Construct a new instance of the EstateUsersReplyEventArgs class
/// The estate's identifier on the grid
/// The number of users
/// Allowed users UUIDs
public EstateUsersReplyEventArgs(uint estateID, int count, List allowedUsers)
{
this.m_estateID = estateID;
this.m_count = count;
this.m_allowedUsers = allowedUsers;
}
}
/// Returned, along with other info, upon a successful .RequestInfo()
public class EstateGroupsReplyEventArgs : EventArgs
{
private readonly uint m_estateID;
private readonly int m_count;
private readonly List m_allowedGroups;
///
/// The identifier of the estate
///
public uint EstateID { get { return m_estateID; } }
///
/// The number of returned items
///
public int Count { get { return m_count; } }
///
/// List of UUIDs of Allowed Groups
///
public List AllowedGroups { get { return m_allowedGroups; } }
/// Construct a new instance of the EstateGroupsReplyEventArgs class
/// The estate's identifier on the grid
/// The number of Groups
/// Allowed Groups UUIDs
public EstateGroupsReplyEventArgs(uint estateID, int count, List allowedGroups)
{
this.m_estateID = estateID;
this.m_count = count;
this.m_allowedGroups = allowedGroups;
}
}
/// Returned, along with other info, upon a successful .RequestInfo()
public class EstateManagersReplyEventArgs : EventArgs
{
private readonly uint m_estateID;
private readonly int m_count;
private readonly List m_Managers;
///
/// The identifier of the estate
///
public uint EstateID { get { return m_estateID; } }
///
/// The number of returned items
///
public int Count { get { return m_count; } }
///
/// List of UUIDs of the Estate's Managers
///
public List Managers { get { return m_Managers; } }
/// Construct a new instance of the EstateManagersReplyEventArgs class
/// The estate's identifier on the grid
/// The number of Managers
/// Managers UUIDs
public EstateManagersReplyEventArgs(uint estateID, int count, List managers)
{
this.m_estateID = estateID;
this.m_count = count;
this.m_Managers = managers;
}
}
/// Returned, along with other info, upon a successful .RequestInfo()
public class EstateCovenantReplyEventArgs : EventArgs
{
private readonly UUID m_covenantID;
private readonly long m_timestamp;
private readonly string m_estateName;
private readonly UUID m_estateOwnerID;
///
/// The Covenant
///
public UUID CovenantID { get { return m_covenantID; } }
///
/// The timestamp
///
public long Timestamp { get { return m_timestamp; } }
///
/// The Estate name
///
public String EstateName { get { return m_estateName; } }
///
/// The Estate Owner's ID (can be a GroupID)
///
public UUID EstateOwnerID { get { return m_estateOwnerID; } }
/// Construct a new instance of the EstateCovenantReplyEventArgs class
/// The Covenant ID
/// The timestamp
/// The estate's name
/// The Estate Owner's ID (can be a GroupID)
public EstateCovenantReplyEventArgs(UUID covenantID, long timestamp, string estateName, UUID estateOwnerID)
{
this.m_covenantID = covenantID;
this.m_timestamp = timestamp;
this.m_estateName = estateName;
this.m_estateOwnerID = estateOwnerID;
}
}
/// Returned, along with other info, upon a successful .RequestInfo()
public class EstateUpdateInfoReplyEventArgs : EventArgs
{
private readonly uint m_estateID;
private readonly bool m_denyNoPaymentInfo;
private readonly string m_estateName;
private readonly UUID m_estateOwner;
///
/// The estate's name
///
public String EstateName { get { return m_estateName; } }
///
/// The Estate Owner's ID (can be a GroupID)
///
public UUID EstateOwner { get { return m_estateOwner; } }
///
/// The identifier of the estate on the grid
///
public uint EstateID { get { return m_estateID; } }
///
public bool DenyNoPaymentInfo { get { return m_denyNoPaymentInfo; } }
/// Construct a new instance of the EstateUpdateInfoReplyEventArgs class
/// The estate's name
/// The Estate Owners ID (can be a GroupID)
/// The estate's identifier on the grid
///
public EstateUpdateInfoReplyEventArgs(string estateName, UUID estateOwner, uint estateID, bool denyNoPaymentInfo)
{
this.m_estateName = estateName;
this.m_estateOwner = estateOwner;
this.m_estateID = estateID;
this.m_denyNoPaymentInfo = denyNoPaymentInfo;
}
}
#endregion
}