llparcel.cpp
上传用户:king477883
上传日期:2021-03-01
资源大小:9553k
文件大小:35k
源码类别:

游戏引擎

开发平台:

C++ Builder

  1. /** 
  2.  * @file llparcel.cpp
  3.  * @brief A land parcel.
  4.  *
  5.  * $LicenseInfo:firstyear=2002&license=viewergpl$
  6.  * 
  7.  * Copyright (c) 2002-2010, Linden Research, Inc.
  8.  * 
  9.  * Second Life Viewer Source Code
  10.  * The source code in this file ("Source Code") is provided by Linden Lab
  11.  * to you under the terms of the GNU General Public License, version 2.0
  12.  * ("GPL"), unless you have obtained a separate licensing agreement
  13.  * ("Other License"), formally executed by you and Linden Lab.  Terms of
  14.  * the GPL can be found in doc/GPL-license.txt in this distribution, or
  15.  * online at http://secondlifegrid.net/programs/open_source/licensing/gplv2
  16.  * 
  17.  * There are special exceptions to the terms and conditions of the GPL as
  18.  * it is applied to this Source Code. View the full text of the exception
  19.  * in the file doc/FLOSS-exception.txt in this software distribution, or
  20.  * online at
  21.  * http://secondlifegrid.net/programs/open_source/licensing/flossexception
  22.  * 
  23.  * By copying, modifying or distributing this software, you acknowledge
  24.  * that you have read and understood your obligations described above,
  25.  * and agree to abide by those obligations.
  26.  * 
  27.  * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO
  28.  * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY,
  29.  * COMPLETENESS OR PERFORMANCE.
  30.  * $/LicenseInfo$
  31.  */
  32. #include "linden_common.h"
  33. #include "indra_constants.h"
  34. #include <iostream>
  35. #include "llparcel.h"
  36. #include "llstreamtools.h"
  37. #include "llmath.h"
  38. #include "llsd.h"
  39. #include "llsdutil.h"
  40. #include "lltransactiontypes.h"
  41. #include "lltransactionflags.h"
  42. #include "llsdutil_math.h"
  43. #include "message.h"
  44. #include "u64.h"
  45. static const F32 SOME_BIG_NUMBER = 1000.0f;
  46. static const F32 SOME_BIG_NEG_NUMBER = -1000.0f;
  47. static const std::string PARCEL_OWNERSHIP_STATUS_STRING[LLParcel::OS_COUNT+1] =
  48. {
  49.     "leased",
  50.     "lease_pending",
  51.     "abandoned",
  52.     "none"
  53. };
  54. // NOTE: Adding parcel categories also requires updating:
  55. // * floater_directory.xml category combobox
  56. // * floater_about_land.xml category combobox
  57. // * Web site "create event" tools
  58. // DO NOT DELETE ITEMS FROM THIS LIST WITHOUT DEEPLY UNDERSTANDING WHAT YOU'RE DOING.
  59. //
  60. static const std::string PARCEL_CATEGORY_STRING[LLParcel::C_COUNT] =
  61. {
  62.     "none",
  63.     "linden",
  64.     "adult",
  65.     "arts",
  66.     "store", // "business" legacy name
  67.     "educational",
  68.     "game",  // "gaming" legacy name
  69.     "gather", // "hangout" legacy name
  70.     "newcomer",
  71.     "park",
  72.     "home",  // "residential" legacy name
  73.     "shopping",
  74.     "stage",
  75.     "other",
  76. };
  77. static const std::string PARCEL_CATEGORY_UI_STRING[LLParcel::C_COUNT + 1] =
  78. {
  79.     "None",
  80.     "Linden Location",
  81.     "Adult",
  82.     "Arts and Culture",
  83.     "Business",
  84.     "Educational",
  85.     "Gaming",
  86.     "Hangout",
  87.     "Newcomer Friendly",
  88.     "Parks and Nature",
  89.     "Residential",
  90.     "Shopping",
  91.     "Stage",
  92.     "Other",
  93.     "Any",  // valid string for parcel searches
  94. };
  95. static const std::string PARCEL_ACTION_STRING[LLParcel::A_COUNT + 1] =
  96. {
  97.     "create",
  98.     "release",
  99.     "absorb",
  100.     "absorbed",
  101.     "divide",
  102.     "division",
  103.     "acquire",
  104.     "relinquish",
  105.     "confirm",
  106.     "unknown"
  107. };
  108. //const char* revert_action_to_string(LLParcel::ESaleTimerExpireAction action);
  109. //LLParcel::ESaleTimerExpireAction revert_string_to_action(const char* s);
  110. const std::string& category_to_ui_string(LLParcel::ECategory category);
  111. LLParcel::ECategory category_ui_string_to_category(const std::string& s);
  112. LLParcel::LLParcel()
  113. {
  114.     init(LLUUID::null, TRUE, FALSE, FALSE, 0, 0, 0, 0, 0, 1.f, 0);
  115. }
  116. LLParcel::LLParcel(const LLUUID &owner_id,
  117.                    BOOL modify, BOOL terraform, BOOL damage,
  118.                    time_t claim_date, S32 claim_price_per_meter,
  119.                    S32 rent_price_per_meter, S32 area, S32 sim_object_limit, F32 parcel_object_bonus,
  120.                    BOOL is_group_owned)
  121. {
  122.     init( owner_id, modify, terraform, damage, claim_date,
  123.           claim_price_per_meter, rent_price_per_meter, area, sim_object_limit, parcel_object_bonus,
  124.           is_group_owned);
  125. }
  126. // virtual
  127. LLParcel::~LLParcel()
  128. {
  129.     // user list cleaned up by LLDynamicArray destructor.
  130. }
  131. void LLParcel::init(const LLUUID &owner_id,
  132.                     BOOL modify, BOOL terraform, BOOL damage,
  133.                     time_t claim_date, S32 claim_price_per_meter,
  134.                     S32 rent_price_per_meter, S32 area, S32 sim_object_limit, F32 parcel_object_bonus,
  135.                     BOOL is_group_owned)
  136. {
  137. mID.setNull();
  138. mOwnerID = owner_id;
  139. mGroupOwned = is_group_owned;
  140. mClaimDate = claim_date;
  141. mClaimPricePerMeter = claim_price_per_meter;
  142. mRentPricePerMeter = rent_price_per_meter;
  143. mArea = area;
  144. mDiscountRate = 1.0f;
  145. mDrawDistance = 512.f;
  146. mUserLookAt.setVec(0.0f, 0.f, 0.f);
  147. // Default to using the parcel's landing point, if any.
  148. mLandingType = L_LANDING_POINT;
  149. // *FIX: if owner_id != null, should be owned or sale pending,
  150. // investigate init callers.
  151. mStatus = OS_NONE;
  152. mCategory = C_NONE;
  153. mAuthBuyerID.setNull();
  154. //mBuyerID.setNull();
  155. //mJoinNeighbors = 0x0;
  156. mSaleTimerExpires.setTimerExpirySec(0);
  157. mSaleTimerExpires.stop();
  158. mGraceExtension = 0;
  159. //mExpireAction = STEA_REVERT;
  160. //mRecordTransaction = FALSE;
  161. mAuctionID = 0;
  162. mInEscrow = false;
  163. mParcelFlags = PF_DEFAULT;
  164. setParcelFlag(PF_CREATE_OBJECTS,  modify);
  165. setParcelFlag(PF_ALLOW_TERRAFORM, terraform);
  166. setParcelFlag(PF_ALLOW_DAMAGE,    damage);
  167. mSalePrice = 10000;
  168. setName(LLStringUtil::null);
  169. setDesc(LLStringUtil::null);
  170. setMusicURL(LLStringUtil::null);
  171. setMediaURL(LLStringUtil::null);
  172. setMediaDesc(LLStringUtil::null);
  173. setMediaType(LLStringUtil::null);
  174. mMediaID.setNull();
  175. mMediaAutoScale = 0;
  176. mMediaLoop = TRUE;
  177. mObscureMedia = 1;
  178. mObscureMusic = 1;
  179. mMediaWidth = 0;
  180. mMediaHeight = 0;
  181. setMediaCurrentURL(LLStringUtil::null);
  182. mMediaURLFilterEnable = FALSE;
  183. mMediaURLFilterList = LLSD::emptyArray();
  184. mMediaAllowNavigate = TRUE;
  185. mMediaURLTimeout = 0.0f;
  186. mMediaPreventCameraZoom = FALSE;
  187. mGroupID.setNull();
  188. mPassPrice = PARCEL_PASS_PRICE_DEFAULT;
  189. mPassHours = PARCEL_PASS_HOURS_DEFAULT;
  190. mAABBMin.setVec(SOME_BIG_NUMBER, SOME_BIG_NUMBER, SOME_BIG_NUMBER);
  191. mAABBMax.setVec(SOME_BIG_NEG_NUMBER, SOME_BIG_NEG_NUMBER, SOME_BIG_NEG_NUMBER);
  192. mLocalID = 0;
  193. //mSimWidePrimCorrection = 0;
  194. setMaxPrimCapacity((S32)(sim_object_limit * area / (F32)(REGION_WIDTH_METERS * REGION_WIDTH_METERS)));
  195. setSimWideMaxPrimCapacity(0);
  196. setSimWidePrimCount(0);
  197. setOwnerPrimCount(0);
  198. setGroupPrimCount(0);
  199. setOtherPrimCount(0);
  200. setSelectedPrimCount(0);
  201. setTempPrimCount(0);
  202. setCleanOtherTime(0);
  203.     setRegionPushOverride(FALSE);
  204.     setRegionDenyAnonymousOverride(FALSE);
  205.     setRegionDenyAgeUnverifiedOverride(FALSE);
  206. setParcelPrimBonus(parcel_object_bonus);
  207. setPreviousOwnerID(LLUUID::null);
  208. setPreviouslyGroupOwned(FALSE);
  209. }
  210. void LLParcel::overrideOwner(const LLUUID& owner_id, BOOL is_group_owned)
  211. {
  212.     // Override with system permission (LLUUID::null)
  213.     // Overridden parcels have no group
  214.     mOwnerID = owner_id;
  215.     mGroupOwned = is_group_owned;
  216.     if(mGroupOwned)
  217.     {
  218.         mGroupID = mOwnerID;
  219.     }
  220.     else
  221.     {
  222.         mGroupID.setNull();
  223.     }
  224.     mInEscrow = false;
  225. }
  226. void LLParcel::overrideParcelFlags(U32 flags)
  227. {
  228.     mParcelFlags = flags;
  229. }
  230. void LLParcel::setName(const std::string& name)
  231. {
  232.     // The escaping here must match the escaping in the database
  233.     // abstraction layer.
  234.     mName = name;
  235.     LLStringFn::replace_nonprintable_in_ascii(mName, LL_UNKNOWN_CHAR);
  236. }
  237. void LLParcel::setDesc(const std::string& desc)
  238. {
  239.     // The escaping here must match the escaping in the database
  240.     // abstraction layer.
  241.     mDesc = desc;
  242.     mDesc = rawstr_to_utf8(mDesc);
  243. }
  244. void LLParcel::setMusicURL(const std::string& url)
  245. {
  246.     mMusicURL = url;
  247.     // The escaping here must match the escaping in the database
  248.     // abstraction layer.
  249.     // This should really filter the url in some way. Other than
  250.     // simply requiring non-printable.
  251.     LLStringFn::replace_nonprintable_in_ascii(mMusicURL, LL_UNKNOWN_CHAR);
  252. }
  253. void LLParcel::setMediaURL(const std::string& url)
  254. {
  255.     mMediaURL = url;
  256.     // The escaping here must match the escaping in the database
  257.     // abstraction layer if it's ever added.
  258.     // This should really filter the url in some way. Other than
  259.     // simply requiring non-printable.
  260.     LLStringFn::replace_nonprintable_in_ascii(mMediaURL, LL_UNKNOWN_CHAR);
  261. }
  262. void LLParcel::setMediaDesc(const std::string& desc)
  263. {
  264. // The escaping here must match the escaping in the database
  265. // abstraction layer.
  266. mMediaDesc = desc;
  267. mMediaDesc = rawstr_to_utf8(mMediaDesc);
  268. }
  269. void LLParcel::setMediaType(const std::string& type)
  270. {
  271. // The escaping here must match the escaping in the database
  272. // abstraction layer.
  273. mMediaType = type;
  274. mMediaType = rawstr_to_utf8(mMediaType);
  275. // This code attempts to preserve legacy movie functioning
  276. if(mMediaType.empty() && ! mMediaURL.empty())
  277. {
  278. setMediaType(std::string("video/vnd.secondlife.qt.legacy"));
  279. }
  280. }
  281. void LLParcel::setMediaWidth(S32 width)
  282. {
  283. mMediaWidth = width;
  284. }
  285. void LLParcel::setMediaHeight(S32 height)
  286. {
  287. mMediaHeight = height;
  288. }
  289. void LLParcel::setMediaCurrentURL(const std::string& url)
  290. {
  291.     mMediaCurrentURL = url;
  292.     // The escaping here must match the escaping in the database
  293.     // abstraction layer if it's ever added.
  294.     // This should really filter the url in some way. Other than
  295.     // simply requiring non-printable.
  296.     LLStringFn::replace_nonprintable_in_ascii(mMediaCurrentURL, LL_UNKNOWN_CHAR);
  297. }
  298. void LLParcel::setMediaURLResetTimer(F32 time)
  299. {
  300. mMediaResetTimer.start();
  301. mMediaResetTimer.setTimerExpirySec(time);
  302. }
  303. void LLParcel::setMediaURLFilterList(LLSD list)
  304. {
  305. // sanity check LLSD
  306. // must be array of strings
  307. if (!list.isArray())
  308. {
  309. return;
  310. }
  311. for (S32 i = 0; i < list.size(); i++)
  312. {
  313. if (!list[i].isString())
  314. return;
  315. }
  316. // can't be too big
  317. const S32 MAX_SIZE = 50;
  318. if (list.size() > MAX_SIZE)
  319. {
  320. LLSD new_list = LLSD::emptyArray();
  321. for (S32 i = 0; i < llmin(list.size(), MAX_SIZE); i++)
  322. {
  323. new_list.append(list[i]);
  324. }
  325. list = new_list;
  326. }
  327. mMediaURLFilterList = list;
  328. }
  329. // virtual
  330. void LLParcel::setLocalID(S32 local_id)
  331. {
  332.     mLocalID = local_id;
  333. }
  334. void LLParcel::setAllParcelFlags(U32 flags)
  335. {
  336.     mParcelFlags = flags;
  337. }
  338. void LLParcel::setParcelFlag(U32 flag, BOOL b)
  339. {
  340.     if (b)
  341.     {
  342.         mParcelFlags |= flag;
  343.     }
  344.     else
  345.     {
  346.         mParcelFlags &= ~flag;
  347.     }
  348. }
  349. BOOL LLParcel::allowModifyBy(const LLUUID &agent_id, const LLUUID &group_id) const
  350. {
  351.     if (agent_id == LLUUID::null)
  352.     {
  353.         // system always can enter
  354.         return TRUE;
  355.     }
  356.     else if (isPublic())
  357.     {
  358.         return TRUE;
  359.     }
  360.     else if (agent_id == mOwnerID)
  361.     {
  362.         // owner can always perform operations
  363.         return TRUE;
  364.     }
  365.     else if (mParcelFlags & PF_CREATE_OBJECTS)
  366.     {
  367.         return TRUE;
  368.     }
  369.     else if ((mParcelFlags & PF_CREATE_GROUP_OBJECTS)
  370.              && group_id.notNull() )
  371.     {
  372.         return (getGroupID() == group_id);
  373.     }
  374.     
  375.     return FALSE;
  376. }
  377. BOOL LLParcel::allowTerraformBy(const LLUUID &agent_id) const
  378. {
  379.     if (agent_id == LLUUID::null)
  380.     {
  381.         // system always can enter
  382.         return TRUE;
  383.     }
  384.     else if(OS_LEASED == mStatus)
  385.     {
  386.         if(agent_id == mOwnerID)
  387.         {
  388.             // owner can modify leased land
  389.             return TRUE;
  390.         }
  391.         else
  392.         {
  393.             // otherwise check other people
  394.             return mParcelFlags & PF_ALLOW_TERRAFORM;
  395.         }
  396.     }
  397.     else
  398.     {
  399.         return FALSE;
  400.     }
  401. }
  402. bool LLParcel::isAgentBlockedFromParcel(LLParcel* parcelp,
  403.                                         const LLUUID& agent_id,
  404.                                         const std::vector<LLUUID>& group_ids,
  405.                                         const BOOL is_agent_identified,
  406.                                         const BOOL is_agent_transacted,
  407.                                         const BOOL is_agent_ageverified)
  408. {
  409.     S32 current_group_access = parcelp->blockAccess(agent_id, LLUUID::null, is_agent_identified, is_agent_transacted, is_agent_ageverified);
  410.     S32 count;
  411.     bool is_allowed = (current_group_access == BA_ALLOWED) ? true: false;
  412.     LLUUID group_id;
  413.     
  414.     count = group_ids.size();
  415.     for (int i = 0; i < count && !is_allowed; i++)
  416.     {
  417.         group_id = group_ids[i];
  418.         current_group_access = parcelp->blockAccess(agent_id, group_id, is_agent_identified, is_agent_transacted, is_agent_ageverified);
  419.         
  420.         if (current_group_access == BA_ALLOWED) is_allowed = true;
  421.     }
  422.     
  423.     return !is_allowed;
  424. }
  425. BOOL LLParcel::isAgentBanned(const LLUUID& agent_id) const
  426. {
  427. // Test ban list
  428. if (mBanList.find(agent_id) != mBanList.end())
  429. {
  430. return TRUE;
  431. }
  432.     
  433.     return FALSE;
  434. }
  435. S32 LLParcel::blockAccess(const LLUUID& agent_id, const LLUUID& group_id,
  436.                           const BOOL is_agent_identified,
  437.                           const BOOL is_agent_transacted,
  438.                           const BOOL is_agent_ageverified) const
  439. {
  440.     // Test ban list
  441.     if (isAgentBanned(agent_id))
  442.     {
  443.         return BA_BANNED;
  444.     }
  445.     
  446.     // Always allow owner on (unless he banned himself, useful for
  447.     // testing). We will also allow estate owners/managers in if they 
  448.     // are not explicitly banned.
  449.     if (agent_id == mOwnerID)
  450.     {
  451.         return BA_ALLOWED;
  452.     }
  453.     
  454.     // Special case when using pass list where group access is being restricted but not 
  455.     // using access list.  In this case group members are allowed only if they buy a pass.
  456.     // We return BA_NOT_IN_LIST if not in list
  457.     BOOL passWithGroup = getParcelFlag(PF_USE_PASS_LIST) && !getParcelFlag(PF_USE_ACCESS_LIST) 
  458.     && getParcelFlag(PF_USE_ACCESS_GROUP) && !mGroupID.isNull() && group_id == mGroupID;
  459.     
  460.     
  461.     // Test group list
  462.     if (getParcelFlag(PF_USE_ACCESS_GROUP)
  463.         && !mGroupID.isNull()
  464.         && group_id == mGroupID
  465.         && !passWithGroup)
  466.     {
  467.         return BA_ALLOWED;
  468.     }
  469.     
  470.     // Test access list
  471.     if (getParcelFlag(PF_USE_ACCESS_LIST) || passWithGroup )
  472.     {
  473.         if (mAccessList.find(agent_id) != mAccessList.end())
  474.         {
  475.             return BA_ALLOWED;
  476.         }
  477.         
  478.         return BA_NOT_ON_LIST; 
  479.     }
  480.     
  481.     // If we're not doing any other limitations, all users
  482.     // can enter, unless
  483.     if (  !getParcelFlag(PF_USE_ACCESS_GROUP)
  484.                  && !getParcelFlag(PF_USE_ACCESS_LIST))
  485.     { 
  486.         //If the land is group owned, and you are in the group, bypass these checks
  487.         if(getIsGroupOwned() && group_id == mGroupID)
  488.         {
  489.             return BA_ALLOWED;
  490.         }
  491.         
  492.         // Test for "payment" access levels
  493.         // Anonymous - No Payment Info on File
  494.         if(getParcelFlag(PF_DENY_ANONYMOUS) && !is_agent_identified && !is_agent_transacted)
  495.         {
  496.             return BA_NO_ACCESS_LEVEL;
  497.         }
  498.         // AgeUnverified - Not Age Verified
  499.         if(getParcelFlag(PF_DENY_AGEUNVERIFIED) && !is_agent_ageverified)
  500.         {
  501. return BA_NOT_AGE_VERIFIED;
  502.         }
  503.     
  504.         return BA_ALLOWED;
  505.     }
  506.     
  507.     return BA_NOT_IN_GROUP;
  508.     
  509. }
  510. void LLParcel::setArea(S32 area, S32 sim_object_limit)
  511. {
  512.     mArea = area;
  513.     setMaxPrimCapacity((S32)(sim_object_limit * area / (F32)(REGION_WIDTH_METERS * REGION_WIDTH_METERS)));
  514. }
  515. void LLParcel::setDiscountRate(F32 rate)
  516. {
  517.     // this is to make sure that the rate is at least sane - this is
  518.     // not intended to enforce economy rules. It only enfoces that the
  519.     // rate is a scaler between 0 and 1.
  520.     mDiscountRate = llclampf(rate);
  521. }
  522. //-----------------------------------------------------------
  523. // File input and output
  524. //-----------------------------------------------------------
  525. BOOL LLParcel::importAccessEntry(std::istream& input_stream, LLAccessEntry* entry)
  526. {
  527.     skip_to_end_of_next_keyword("{", input_stream);
  528.     while (input_stream.good())
  529.     {
  530.         skip_comments_and_emptyspace(input_stream);
  531.         std::string line, keyword, value;
  532.         get_line(line, input_stream, MAX_STRING);
  533.         get_keyword_and_value(keyword, value, line);
  534.         
  535.         if ("}" == keyword)
  536.         {
  537.             break;
  538.         }
  539.         else if ("id" == keyword)
  540.         {
  541.             entry->mID.set( value );
  542.         }
  543.         else if ("name" == keyword)
  544.         {
  545.             // deprecated
  546.         }
  547.         else if ("time" == keyword)
  548.         {
  549.             S32 when;
  550.             LLStringUtil::convertToS32(value, when);
  551.             entry->mTime = when;
  552.         }
  553.         else if ("flags" == keyword)
  554.         {
  555.             U32 setting;
  556.             LLStringUtil::convertToU32(value, setting);
  557.             entry->mFlags = setting;
  558.         }
  559.         else
  560.         {
  561.             llwarns << "Unknown keyword in parcel access entry section: <" 
  562.             << keyword << ">" << llendl;
  563.         }
  564.     }
  565.     return input_stream.good();
  566. }
  567. BOOL LLParcel::importMediaURLFilter(std::istream& input_stream, std::string& url)
  568. {
  569. skip_to_end_of_next_keyword("{", input_stream);
  570. while(input_stream.good())
  571. {
  572. skip_comments_and_emptyspace(input_stream);
  573. std::string line, keyword, value;
  574. get_line(line, input_stream, MAX_STRING);
  575. get_keyword_and_value(keyword, value, line);
  576. if ("}" == keyword)
  577. {
  578. break;
  579. }
  580. else if ("url" == keyword)
  581. {
  582. url = value;
  583. }
  584. else
  585. {
  586. llwarns << "Unknown keyword in parcel media url filter section: <"
  587. << keyword << ">" << llendl;
  588. }
  589. }
  590. return input_stream.good();
  591. }
  592. // Assumes we are in a block "ParcelData"
  593. void LLParcel::packMessage(LLMessageSystem* msg)
  594. {
  595.     msg->addU32Fast( _PREHASH_ParcelFlags, getParcelFlags() );
  596.     msg->addS32Fast( _PREHASH_SalePrice, getSalePrice() );
  597.     msg->addStringFast( _PREHASH_Name,  getName() );
  598.     msg->addStringFast( _PREHASH_Desc,  getDesc() );
  599.     msg->addStringFast( _PREHASH_MusicURL,  getMusicURL() );
  600.     msg->addStringFast( _PREHASH_MediaURL,  getMediaURL() );
  601.     msg->addU8 ( "MediaAutoScale", getMediaAutoScale () );
  602.     msg->addUUIDFast( _PREHASH_MediaID,  getMediaID() );
  603.     msg->addUUIDFast( _PREHASH_GroupID,  getGroupID() );
  604.     msg->addS32Fast( _PREHASH_PassPrice, mPassPrice );
  605.     msg->addF32Fast( _PREHASH_PassHours, mPassHours );
  606.     msg->addU8Fast(  _PREHASH_Category,  (U8)mCategory);
  607.     msg->addUUIDFast( _PREHASH_AuthBuyerID, mAuthBuyerID);
  608.     msg->addUUIDFast( _PREHASH_SnapshotID, mSnapshotID);
  609.     msg->addVector3Fast(_PREHASH_UserLocation, mUserLocation);
  610.     msg->addVector3Fast(_PREHASH_UserLookAt, mUserLookAt);
  611.     msg->addU8Fast(  _PREHASH_LandingType, (U8)mLandingType);
  612. }
  613. // Assumes we are in a block "ParcelData"
  614. void LLParcel::packMessage(LLSD& msg)
  615. {
  616. // used in the viewer, the sim uses it's own packer
  617. msg["local_id"] = getLocalID();
  618. msg["parcel_flags"] = ll_sd_from_U32(getParcelFlags());
  619. msg["sale_price"] = getSalePrice();
  620. msg["name"] = getName();
  621. msg["description"] = getDesc();
  622. msg["music_url"] = getMusicURL();
  623. msg["media_url"] = getMediaURL();
  624. msg["media_desc"] = getMediaDesc();
  625. msg["media_type"] = getMediaType();
  626. msg["media_width"] = getMediaWidth();
  627. msg["media_height"] = getMediaHeight();
  628. msg["auto_scale"] = getMediaAutoScale();
  629. msg["media_loop"] = getMediaLoop();
  630. msg["media_current_url"] = getMediaCurrentURL();
  631. msg["obscure_media"] = getObscureMedia();
  632. msg["obscure_music"] = getObscureMusic();
  633. msg["media_id"] = getMediaID();
  634. msg["media_allow_navigate"] = getMediaAllowNavigate();
  635. msg["media_prevent_camera_zoom"] = getMediaPreventCameraZoom();
  636. msg["media_url_timeout"] = getMediaURLTimeout();
  637. msg["media_url_filter_enable"] = getMediaURLFilterEnable();
  638. msg["media_url_filter_list"] = getMediaURLFilterList();
  639. msg["group_id"] = getGroupID();
  640. msg["pass_price"] = mPassPrice;
  641. msg["pass_hours"] = mPassHours;
  642. msg["category"] = (U8)mCategory;
  643. msg["auth_buyer_id"] = mAuthBuyerID;
  644. msg["snapshot_id"] = mSnapshotID;
  645. msg["user_location"] = ll_sd_from_vector3(mUserLocation);
  646. msg["user_look_at"] = ll_sd_from_vector3(mUserLookAt);
  647. msg["landing_type"] = (U8)mLandingType;
  648. }
  649. void LLParcel::unpackMessage(LLMessageSystem* msg)
  650. {
  651. std::string buffer;
  652.     msg->getU32Fast( _PREHASH_ParcelData,_PREHASH_ParcelFlags, mParcelFlags );
  653.     msg->getS32Fast( _PREHASH_ParcelData,_PREHASH_SalePrice, mSalePrice );
  654.     msg->getStringFast( _PREHASH_ParcelData,_PREHASH_Name, buffer );
  655.     setName(buffer);
  656.     msg->getStringFast( _PREHASH_ParcelData,_PREHASH_Desc, buffer );
  657.     setDesc(buffer);
  658.     msg->getStringFast( _PREHASH_ParcelData,_PREHASH_MusicURL, buffer );
  659.     setMusicURL(buffer);
  660.     msg->getStringFast( _PREHASH_ParcelData,_PREHASH_MediaURL, buffer );
  661.     setMediaURL(buffer);
  662.     
  663.     // non-optimized version
  664.     msg->getU8 ( "ParcelData", "MediaAutoScale", mMediaAutoScale );
  665.     
  666.     msg->getUUIDFast( _PREHASH_ParcelData,_PREHASH_MediaID, mMediaID );
  667.     msg->getUUIDFast( _PREHASH_ParcelData,_PREHASH_GroupID, mGroupID );
  668.     msg->getS32Fast( _PREHASH_ParcelData,_PREHASH_PassPrice, mPassPrice );
  669.     msg->getF32Fast( _PREHASH_ParcelData,_PREHASH_PassHours, mPassHours );
  670.     U8 category;
  671.     msg->getU8Fast(  _PREHASH_ParcelData,_PREHASH_Category, category);
  672.     mCategory = (ECategory)category;
  673.     msg->getUUIDFast( _PREHASH_ParcelData,_PREHASH_AuthBuyerID, mAuthBuyerID);
  674.     msg->getUUIDFast( _PREHASH_ParcelData,_PREHASH_SnapshotID, mSnapshotID);
  675.     msg->getVector3Fast(_PREHASH_ParcelData,_PREHASH_UserLocation, mUserLocation);
  676.     msg->getVector3Fast(_PREHASH_ParcelData,_PREHASH_UserLookAt, mUserLookAt);
  677.     U8 landing_type;
  678.     msg->getU8Fast(  _PREHASH_ParcelData,_PREHASH_LandingType, landing_type);
  679.     mLandingType = (ELandingType)landing_type;
  680. // New Media Data
  681. // Note: the message has been converted to TCP
  682. if(msg->has("MediaData"))
  683. {
  684. msg->getString("MediaData", "MediaDesc", buffer);
  685. setMediaDesc(buffer);
  686. msg->getString("MediaData", "MediaType", buffer);
  687. setMediaType(buffer);
  688. msg->getS32("MediaData", "MediaWidth", mMediaWidth);
  689. msg->getS32("MediaData", "MediaHeight", mMediaHeight);
  690. msg->getU8 ( "MediaData", "MediaLoop", mMediaLoop );
  691. msg->getU8 ( "MediaData", "ObscureMedia", mObscureMedia );
  692. msg->getU8 ( "MediaData", "ObscureMusic", mObscureMusic );
  693. }
  694. else
  695. {
  696. setMediaType(std::string("video/vnd.secondlife.qt.legacy"));
  697. setMediaDesc(std::string("No Description available without Server Upgrade"));
  698. mMediaLoop = true;
  699. mObscureMedia = true;
  700. mObscureMusic = true;
  701. }
  702. if(msg->getNumberOfBlocks("MediaLinkSharing") > 0)
  703. {
  704. msg->getString("MediaLinkSharing", "MediaCurrentURL", buffer);
  705. setMediaCurrentURL(buffer);
  706. msg->getU8 ( "MediaLinkSharing", "MediaAllowNavigate", mMediaAllowNavigate );
  707. msg->getU8 ( "MediaLinkSharing", "MediaURLFilterEnable", mMediaURLFilterEnable );
  708. msg->getU8 ( "MediaLinkSharing", "MediaPreventCameraZoom", mMediaPreventCameraZoom );
  709. msg->getF32( "MediaLinkSharing", "MediaURLTimeout", mMediaURLTimeout);
  710. }
  711. else
  712. {
  713. setMediaCurrentURL(LLStringUtil::null);
  714. }
  715. }
  716. void LLParcel::packAccessEntries(LLMessageSystem* msg,
  717.  const std::map<LLUUID,LLAccessEntry>& list)
  718. {
  719.     access_map_const_iterator cit = list.begin();
  720.     access_map_const_iterator end = list.end();
  721.     
  722.     if (cit == end)
  723.     {
  724.         msg->nextBlockFast(_PREHASH_List);
  725.         msg->addUUIDFast(_PREHASH_ID, LLUUID::null );
  726.         msg->addS32Fast(_PREHASH_Time, 0 );
  727.         msg->addU32Fast(_PREHASH_Flags, 0 );
  728.         return;
  729.     }
  730.     
  731.     for ( ; cit != end; ++cit)
  732.     {
  733.         const LLAccessEntry& entry = (*cit).second;
  734.         
  735.         msg->nextBlockFast(_PREHASH_List);
  736.         msg->addUUIDFast(_PREHASH_ID,  entry.mID );
  737.         msg->addS32Fast(_PREHASH_Time,  entry.mTime );
  738.         msg->addU32Fast(_PREHASH_Flags, entry.mFlags );
  739.     }
  740. }
  741. void LLParcel::unpackAccessEntries(LLMessageSystem* msg,
  742.                                    std::map<LLUUID,LLAccessEntry>* list)
  743. {
  744.     LLUUID id;
  745.     S32 time;
  746.     U32 flags;
  747.     
  748.     S32 i;
  749.     S32 count = msg->getNumberOfBlocksFast(_PREHASH_List);
  750.     for (i = 0; i < count; i++)
  751.     {
  752.         msg->getUUIDFast(_PREHASH_List, _PREHASH_ID, id, i);
  753.         msg->getS32Fast(  _PREHASH_List, _PREHASH_Time, time, i);
  754.         msg->getU32Fast(  _PREHASH_List, _PREHASH_Flags, flags, i);
  755.         
  756.         if (id.notNull())
  757.         {
  758.             LLAccessEntry entry;
  759.             entry.mID = id;
  760.             entry.mTime = time;
  761.             entry.mFlags = flags;
  762.             
  763.             (*list)[entry.mID] = entry;
  764.         }
  765.     }
  766. }
  767. void LLParcel::expirePasses(S32 now)
  768. {
  769.     access_map_iterator itor = mAccessList.begin();
  770.     while (itor != mAccessList.end())
  771.     {
  772.         const LLAccessEntry& entry = (*itor).second;
  773.         
  774.         if (entry.mTime != 0 && entry.mTime < now)
  775.         {
  776.             mAccessList.erase(itor++);
  777.         }
  778.         else
  779.         {
  780.             ++itor;
  781.         }
  782.     }
  783. }
  784. bool LLParcel::operator==(const LLParcel &rhs) const
  785. {
  786.     if (mOwnerID != rhs.mOwnerID)
  787.         return FALSE;
  788.     
  789.     if (mParcelFlags != rhs.mParcelFlags)
  790.         return FALSE;
  791.     
  792.     if (mClaimDate != rhs.mClaimDate)
  793.         return FALSE;
  794.     
  795.     if (mClaimPricePerMeter != rhs.mClaimPricePerMeter)
  796.         return FALSE;
  797.     
  798.     if (mRentPricePerMeter != rhs.mRentPricePerMeter)
  799.         return FALSE;
  800.     
  801.     return TRUE;
  802. }
  803. // Calculate rent
  804. S32 LLParcel::getTotalRent() const
  805. {
  806.     return (S32)floor(0.5f + (F32)mArea * (F32)mRentPricePerMeter * (1.0f - mDiscountRate));
  807. }
  808. F32 LLParcel::getAdjustedRentPerMeter() const
  809. {
  810.     return ((F32)mRentPricePerMeter * (1.0f - mDiscountRate));
  811. }
  812. LLVector3 LLParcel::getCenterpoint() const
  813. {
  814.     LLVector3 rv;
  815.     rv.mV[VX] = (getAABBMin().mV[VX] + getAABBMax().mV[VX]) * 0.5f;
  816.     rv.mV[VY] = (getAABBMin().mV[VY] + getAABBMax().mV[VY]) * 0.5f;
  817.     rv.mV[VZ] = 0.0f;
  818.     return rv;
  819. }
  820. void LLParcel::extendAABB(const LLVector3& box_min, const LLVector3& box_max)
  821. {
  822.     // Patch up min corner of AABB
  823.     S32 i;
  824.     for (i=0; i<3; i++)
  825.     {
  826.         if (box_min.mV[i] < mAABBMin.mV[i])
  827.         {
  828.             mAABBMin.mV[i] = box_min.mV[i];
  829.         }
  830.     }
  831.     
  832.     // Patch up max corner of AABB
  833.     for (i=0; i<3; i++)
  834.     {
  835.         if (box_max.mV[i] > mAABBMax.mV[i])
  836.         {
  837.             mAABBMax.mV[i] = box_max.mV[i];
  838.         }
  839.     }
  840. }
  841. BOOL LLParcel::addToAccessList(const LLUUID& agent_id, S32 time)
  842. {
  843. if (mAccessList.size() >= (U32) PARCEL_MAX_ACCESS_LIST)
  844. {
  845. return FALSE;
  846. }
  847. if (agent_id == getOwnerID())
  848. {
  849. // Can't add owner to these lists
  850. return FALSE;
  851. }
  852. access_map_iterator itor = mAccessList.begin();
  853. while (itor != mAccessList.end())
  854. {
  855. const LLAccessEntry& entry = (*itor).second;
  856. if (entry.mID == agent_id)
  857. {
  858. if (time == 0 || (entry.mTime != 0 && entry.mTime < time))
  859. {
  860. mAccessList.erase(itor++);
  861. }
  862. else
  863. {
  864. // existing one expires later
  865. return FALSE;
  866. }
  867. }
  868. else
  869. {
  870. ++itor;
  871. }
  872. }
  873.     
  874.     removeFromBanList(agent_id);
  875.     
  876.     LLAccessEntry new_entry;
  877.     new_entry.mID  = agent_id;
  878.     new_entry.mTime  = time;
  879.     new_entry.mFlags = 0x0;
  880.     mAccessList[new_entry.mID] = new_entry;
  881.     return TRUE;
  882. }
  883. BOOL LLParcel::addToBanList(const LLUUID& agent_id, S32 time)
  884. {
  885. if (mBanList.size() >= (U32) PARCEL_MAX_ACCESS_LIST)
  886. {
  887. // Not using ban list, so not a rational thing to do
  888. return FALSE;
  889. }
  890. if (agent_id == getOwnerID())
  891. {
  892. // Can't add owner to these lists
  893. return FALSE;
  894. }
  895.     
  896.     access_map_iterator itor = mBanList.begin();
  897.     while (itor != mBanList.end())
  898.     {
  899.         const LLAccessEntry& entry = (*itor).second;
  900.         if (entry.mID == agent_id)
  901.         {
  902.             if (time == 0 || (entry.mTime != 0 && entry.mTime < time))
  903.             {
  904.                 mBanList.erase(itor++);
  905.             }
  906.             else
  907.             {
  908.                 // existing one expires later
  909.                 return FALSE;
  910.             }
  911.         }
  912.         else
  913.         {
  914.             ++itor;
  915.         }
  916.     }
  917.     
  918.     removeFromAccessList(agent_id);
  919.     
  920.     LLAccessEntry new_entry;
  921.     new_entry.mID  = agent_id;
  922.     new_entry.mTime  = time;
  923.     new_entry.mFlags = 0x0;
  924.     mBanList[new_entry.mID] = new_entry;
  925.     return TRUE;
  926. }
  927. BOOL remove_from_access_array(std::map<LLUUID,LLAccessEntry>* list,
  928.                               const LLUUID& agent_id)
  929. {
  930.     BOOL removed = FALSE;
  931.     access_map_iterator itor = list->begin();
  932.     while (itor != list->end())
  933.     {
  934.         const LLAccessEntry& entry = (*itor).second;
  935.         if (entry.mID == agent_id)
  936.         {
  937.             list->erase(itor++);
  938.             removed = TRUE;
  939.         }
  940.         else
  941.         {
  942.             ++itor;
  943.         }
  944.     }
  945.     return removed;
  946. }
  947. BOOL LLParcel::removeFromAccessList(const LLUUID& agent_id)
  948. {
  949.     return remove_from_access_array(&mAccessList, agent_id);
  950. }
  951. BOOL LLParcel::removeFromBanList(const LLUUID& agent_id)
  952. {
  953.     return remove_from_access_array(&mBanList, agent_id);
  954. }
  955. // static
  956. const std::string& LLParcel::getOwnershipStatusString(EOwnershipStatus status)
  957. {
  958.     return ownership_status_to_string(status);
  959. }
  960. // static
  961. const std::string& LLParcel::getCategoryString(ECategory category)
  962. {
  963.     return category_to_string(category);
  964. }
  965. // static
  966. const std::string& LLParcel::getCategoryUIString(ECategory category)
  967. {
  968.     return category_to_ui_string(category);
  969. }
  970. // static
  971. LLParcel::ECategory LLParcel::getCategoryFromString(const std::string& string)
  972. {
  973.     return category_string_to_category(string);
  974. }
  975. // static
  976. LLParcel::ECategory LLParcel::getCategoryFromUIString(const std::string& string)
  977. {
  978.     return category_ui_string_to_category(string);
  979. }
  980. // static
  981. const std::string& LLParcel::getActionString(LLParcel::EAction action)
  982. {
  983.     S32 index = 0;
  984.     if((action >= 0) && (action < LLParcel::A_COUNT))
  985.     {
  986.         index = action;
  987.     }
  988.     else
  989.     {
  990.         index = A_COUNT;
  991.     }
  992.     return PARCEL_ACTION_STRING[index];
  993. }
  994. BOOL LLParcel::isSaleTimerExpired(const U64& time)
  995. {
  996.     if (mSaleTimerExpires.getStarted() == FALSE)
  997.     {
  998.         return FALSE;
  999.     }
  1000.     BOOL expired = mSaleTimerExpires.checkExpirationAndReset(0.0);
  1001.     if (expired)
  1002.     {
  1003.         mSaleTimerExpires.stop();
  1004.     }
  1005.     return expired;
  1006. }
  1007. BOOL LLParcel::isMediaResetTimerExpired(const U64& time)
  1008. {
  1009.     if (mMediaResetTimer.getStarted() == FALSE)
  1010.     {
  1011.         return FALSE;
  1012.     }
  1013.     BOOL expired = mMediaResetTimer.checkExpirationAndReset(0.0);
  1014.     if (expired)
  1015.     {
  1016.         mMediaResetTimer.stop();
  1017.     }
  1018.     return expired;
  1019. }
  1020. void LLParcel::startSale(const LLUUID& buyer_id, BOOL is_buyer_group)
  1021. {
  1022. // TODO -- this and all Sale related methods need to move out of the LLParcel 
  1023. // base class and into server-side-only LLSimParcel class
  1024. setPreviousOwnerID(mOwnerID);
  1025. setPreviouslyGroupOwned(mGroupOwned);
  1026. mOwnerID = buyer_id;
  1027. mGroupOwned = is_buyer_group;
  1028. if(mGroupOwned)
  1029. {
  1030. mGroupID = mOwnerID;
  1031. }
  1032. else
  1033. {
  1034. mGroupID.setNull();
  1035. }
  1036. mSaleTimerExpires.start();
  1037. mSaleTimerExpires.setTimerExpirySec(DEFAULT_USEC_SALE_TIMEOUT / SEC_TO_MICROSEC);
  1038. mStatus = OS_LEASE_PENDING;
  1039. mClaimDate = time(NULL);
  1040. setAuctionID(0);
  1041. // clear the autoreturn whenever land changes hands
  1042. setCleanOtherTime(0);
  1043. }
  1044. void LLParcel::expireSale(
  1045. U32& type,
  1046. U8& flags,
  1047. LLUUID& from_id,
  1048. LLUUID& to_id)
  1049. {
  1050.     mSaleTimerExpires.setTimerExpirySec(0.0);
  1051.     mSaleTimerExpires.stop();
  1052.     setPreviousOwnerID(LLUUID::null);
  1053.     setPreviouslyGroupOwned(FALSE);
  1054.     setSellWithObjects(FALSE);
  1055.     type = TRANS_LAND_RELEASE;
  1056.     mStatus = OS_NONE;
  1057.     flags = pack_transaction_flags(mGroupOwned, FALSE);
  1058.     mAuthBuyerID.setNull();
  1059.     from_id = mOwnerID;
  1060.     mOwnerID.setNull();
  1061.     to_id.setNull();
  1062. }
  1063. void LLParcel::completeSale(
  1064. U32& type,
  1065. U8& flags,
  1066. LLUUID& to_id)
  1067. {
  1068. mSaleTimerExpires.setTimerExpirySec(0.0);
  1069. mSaleTimerExpires.stop();
  1070. mStatus = OS_LEASED;
  1071. type = TRANS_LAND_SALE;
  1072. flags = pack_transaction_flags(mGroupOwned, mGroupOwned);
  1073. to_id = mOwnerID;
  1074. mAuthBuyerID.setNull();
  1075. // Purchased parcels are assumed to no longer be for sale.
  1076. // Otherwise someone can snipe the sale.
  1077. setForSale(FALSE);
  1078. setAuctionID(0);
  1079. // Turn off show directory, since it's a recurring fee that
  1080. // the buyer may not want.
  1081. setParcelFlag(PF_SHOW_DIRECTORY, FALSE);
  1082. //should be cleared on sale.
  1083. mAccessList.clear();
  1084. mBanList.clear();
  1085. }
  1086. void LLParcel::clearSale()
  1087. {
  1088. mSaleTimerExpires.setTimerExpirySec(0.0);
  1089. mSaleTimerExpires.stop();
  1090. if(isPublic())
  1091. {
  1092. mStatus = OS_NONE;
  1093. }
  1094. else
  1095. {
  1096. mStatus = OS_LEASED;
  1097. }
  1098. mAuthBuyerID.setNull();
  1099. setForSale(FALSE);
  1100. setAuctionID(0);
  1101. setPreviousOwnerID(LLUUID::null);
  1102. setPreviouslyGroupOwned(FALSE);
  1103. setSellWithObjects(FALSE);
  1104. }
  1105. BOOL LLParcel::isPublic() const
  1106. {
  1107.     return (mOwnerID.isNull());
  1108. }
  1109. BOOL LLParcel::isBuyerAuthorized(const LLUUID& buyer_id) const
  1110. {
  1111.     if(mAuthBuyerID.isNull())
  1112.     {
  1113.         return TRUE;
  1114.     }
  1115.     return (mAuthBuyerID == buyer_id);
  1116. }
  1117. void LLParcel::clearParcel()
  1118. {
  1119. overrideParcelFlags(PF_DEFAULT);
  1120. setName(LLStringUtil::null);
  1121. setDesc(LLStringUtil::null);
  1122. setMediaURL(LLStringUtil::null);
  1123. setMediaType(LLStringUtil::null);
  1124. setMediaID(LLUUID::null);
  1125.     setMediaDesc(LLStringUtil::null);
  1126. setMediaAutoScale(0);
  1127. setMediaLoop(TRUE);
  1128. mObscureMedia = 1;
  1129. mObscureMusic = 1;
  1130. mMediaWidth = 0;
  1131. mMediaHeight = 0;
  1132. setMediaCurrentURL(LLStringUtil::null);
  1133. setMediaURLFilterList(LLSD::emptyArray());
  1134. setMediaURLFilterEnable(FALSE);
  1135. setMediaAllowNavigate(TRUE);
  1136. setMediaPreventCameraZoom(FALSE);
  1137. setMediaURLTimeout(0.0f);
  1138. setMusicURL(LLStringUtil::null);
  1139. setInEscrow(FALSE);
  1140. setAuthorizedBuyerID(LLUUID::null);
  1141. setCategory(C_NONE);
  1142. setSnapshotID(LLUUID::null);
  1143. setUserLocation(LLVector3::zero);
  1144. setUserLookAt(LLVector3::x_axis);
  1145. setLandingType(L_LANDING_POINT);
  1146. setAuctionID(0);
  1147. setGroupID(LLUUID::null);
  1148. setPassPrice(0);
  1149. setPassHours(0.f);
  1150. mAccessList.clear();
  1151. mBanList.clear();
  1152. //mRenterList.reset();
  1153. }
  1154. void LLParcel::dump()
  1155. {
  1156.     llinfos << "parcel " << mLocalID << " area " << mArea << llendl;
  1157.     llinfos << "  name <" << mName << ">" << llendl;
  1158.     llinfos << "  desc <" << mDesc << ">" << llendl;
  1159. }
  1160. const std::string& ownership_status_to_string(LLParcel::EOwnershipStatus status)
  1161. {
  1162.     if(status >= 0 && status < LLParcel::OS_COUNT)
  1163.     {
  1164.         return PARCEL_OWNERSHIP_STATUS_STRING[status];
  1165.     }
  1166.     return PARCEL_OWNERSHIP_STATUS_STRING[LLParcel::OS_COUNT];
  1167. }
  1168. LLParcel::EOwnershipStatus ownership_string_to_status(const std::string& s)
  1169. {
  1170.     for(S32 i = 0; i < LLParcel::OS_COUNT; ++i)
  1171.     {
  1172.         if(s == PARCEL_OWNERSHIP_STATUS_STRING[i])
  1173.         {
  1174.             return (LLParcel::EOwnershipStatus)i;
  1175.         }
  1176.     }
  1177.     return LLParcel::OS_NONE;
  1178. }
  1179. //const char* revert_action_to_string(LLParcel::ESaleTimerExpireAction action)
  1180. //{
  1181. // S32 index = 0;
  1182. // if(action >= 0 && action < LLParcel::STEA_COUNT)
  1183. // {
  1184. //  index = action;
  1185. // }
  1186. // return PARCEL_SALE_TIMER_ACTION[index];
  1187. //}
  1188.     
  1189. //LLParcel::ESaleTimerExpireAction revert_string_to_action(const char* s)
  1190. //{
  1191. // for(S32 i = 0; i < LLParcel::STEA_COUNT; ++i)
  1192. // {
  1193. //  if(0 == strcmp(s, PARCEL_SALE_TIMER_ACTION[i]))
  1194. //  {
  1195. //  return (LLParcel::ESaleTimerExpireAction)i;
  1196. //  }
  1197. // }
  1198. // return LLParcel::STEA_REVERT;
  1199. //}
  1200.     
  1201. const std::string& category_to_string(LLParcel::ECategory category)
  1202. {
  1203.     S32 index = 0;
  1204.     if((category >= 0) && (category < LLParcel::C_COUNT))
  1205.     {
  1206.         index = category;
  1207.     }
  1208.     return PARCEL_CATEGORY_STRING[index];
  1209. }
  1210. const std::string& category_to_ui_string(LLParcel::ECategory category)
  1211. {
  1212.     S32 index = 0;
  1213.     if((category >= 0) && (category < LLParcel::C_COUNT))
  1214.     {
  1215.         index = category;
  1216.     }
  1217.     else
  1218.     {
  1219.         // C_ANY = -1 , but the "Any" string is at the end of the list
  1220.         index = ((S32) LLParcel::C_COUNT);
  1221.     }
  1222.     return PARCEL_CATEGORY_UI_STRING[index];
  1223. }
  1224. LLParcel::ECategory category_string_to_category(const std::string& s)
  1225. {
  1226.     for(S32 i = 0; i < LLParcel::C_COUNT; ++i)
  1227.     {
  1228.         if(s == PARCEL_CATEGORY_STRING[i])
  1229.         {
  1230.             return (LLParcel::ECategory)i;
  1231.         }
  1232.     }
  1233.     llwarns << "Parcel category outside of possibilities " << s << llendl;
  1234.     return LLParcel::C_NONE;
  1235. }
  1236. LLParcel::ECategory category_ui_string_to_category(const std::string& s)
  1237. {
  1238.     for(S32 i = 0; i < LLParcel::C_COUNT; ++i)
  1239.     {
  1240.         if(s == PARCEL_CATEGORY_UI_STRING[i])
  1241.         {
  1242.             return (LLParcel::ECategory)i;
  1243.         }
  1244.     }
  1245.     // "Any" is a valid category for searches, and
  1246.     // is a distinct option from "None" and "Other"
  1247.     return LLParcel::C_ANY;
  1248. }