TradeList.java 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041
  1. /*
  2. * This program is free software: you can redistribute it and/or modify it under
  3. * the terms of the GNU General Public License as published by the Free Software
  4. * Foundation, either version 3 of the License, or (at your option) any later
  5. * version.
  6. *
  7. * This program is distributed in the hope that it will be useful, but WITHOUT
  8. * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
  9. * FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
  10. * details.
  11. *
  12. * You should have received a copy of the GNU General Public License along with
  13. * this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. package com.l2jserver.gameserver.model;
  16. import static com.l2jserver.gameserver.model.itemcontainer.PcInventory.MAX_ADENA;
  17. import java.util.List;
  18. import java.util.logging.Logger;
  19. import com.l2jserver.Config;
  20. import com.l2jserver.gameserver.datatables.ItemTable;
  21. import com.l2jserver.gameserver.model.actor.instance.L2PcInstance;
  22. import com.l2jserver.gameserver.model.itemcontainer.PcInventory;
  23. import com.l2jserver.gameserver.network.SystemMessageId;
  24. import com.l2jserver.gameserver.network.serverpackets.InventoryUpdate;
  25. import com.l2jserver.gameserver.network.serverpackets.ItemList;
  26. import com.l2jserver.gameserver.network.serverpackets.StatusUpdate;
  27. import com.l2jserver.gameserver.network.serverpackets.SystemMessage;
  28. import com.l2jserver.gameserver.templates.item.L2EtcItemType;
  29. import com.l2jserver.gameserver.templates.item.L2Item;
  30. import javolution.util.FastList;
  31. /**
  32. * @author Advi
  33. *
  34. */
  35. public class TradeList
  36. {
  37. public class TradeItem
  38. {
  39. private int _objectId;
  40. private final L2Item _item;
  41. private int _enchant;
  42. private int _type2;
  43. private long _count;
  44. private long _storeCount;
  45. private long _price;
  46. private final byte _elemAtkType;
  47. private final int _elemAtkPower;
  48. private int[] _elemDefAttr = {0, 0, 0, 0, 0, 0};
  49. public TradeItem(L2ItemInstance item, long count, long price)
  50. {
  51. _objectId = item.getObjectId();
  52. _item = item.getItem();
  53. _enchant = item.getEnchantLevel();
  54. _type2 = item.getCustomType2();
  55. _count = count;
  56. _price = price;
  57. _elemAtkType = item.getAttackElementType();
  58. _elemAtkPower = item.getAttackElementPower();
  59. for (byte i = 0; i < 6; i++)
  60. _elemDefAttr[i] = item.getElementDefAttr(i);
  61. }
  62. public TradeItem(L2Item item, long count, long price)
  63. {
  64. _objectId = 0;
  65. _item = item;
  66. _enchant = 0;
  67. _type2 = 0;
  68. _count = count;
  69. _storeCount = count;
  70. _price = price;
  71. _elemAtkType = Elementals.NONE;
  72. _elemAtkPower = 0;
  73. }
  74. public TradeItem(TradeItem item, long count, long price)
  75. {
  76. _objectId = item.getObjectId();
  77. _item = item.getItem();
  78. _enchant = item.getEnchant();
  79. _type2 = 0;
  80. _count = count;
  81. _storeCount = count;
  82. _price = price;
  83. _elemAtkType = item.getAttackElementType();
  84. _elemAtkPower = item.getAttackElementPower();
  85. for (byte i = 0; i < 6; i++)
  86. _elemDefAttr[i] = item.getElementDefAttr(i);
  87. }
  88. public void setObjectId(int objectId)
  89. {
  90. _objectId = objectId;
  91. }
  92. public int getObjectId()
  93. {
  94. return _objectId;
  95. }
  96. public L2Item getItem()
  97. {
  98. return _item;
  99. }
  100. public void setEnchant(int enchant)
  101. {
  102. _enchant = enchant;
  103. }
  104. public int getEnchant()
  105. {
  106. return _enchant;
  107. }
  108. public int getCustomType2()
  109. {
  110. return _type2;
  111. }
  112. public void setCount(long count)
  113. {
  114. _count = count;
  115. }
  116. public long getCount()
  117. {
  118. return _count;
  119. }
  120. public long getStoreCount()
  121. {
  122. return _storeCount;
  123. }
  124. public void setPrice(long price)
  125. {
  126. _price = price;
  127. }
  128. public long getPrice()
  129. {
  130. return _price;
  131. }
  132. public byte getAttackElementType()
  133. {
  134. return _elemAtkType;
  135. }
  136. public int getAttackElementPower()
  137. {
  138. return _elemAtkPower;
  139. }
  140. public int getElementDefAttr(byte i)
  141. {
  142. return _elemDefAttr[i];
  143. }
  144. }
  145. private static final Logger _log = Logger.getLogger(TradeList.class.getName());
  146. private final L2PcInstance _owner;
  147. private L2PcInstance _partner;
  148. private final List<TradeItem> _items;
  149. private String _title;
  150. private boolean _packaged;
  151. private boolean _confirmed = false;
  152. private boolean _locked = false;
  153. public TradeList(L2PcInstance owner)
  154. {
  155. _items = new FastList<TradeItem>();
  156. _owner = owner;
  157. }
  158. public L2PcInstance getOwner()
  159. {
  160. return _owner;
  161. }
  162. public void setPartner(L2PcInstance partner)
  163. {
  164. _partner = partner;
  165. }
  166. public L2PcInstance getPartner()
  167. {
  168. return _partner;
  169. }
  170. public void setTitle(String title)
  171. {
  172. _title = title;
  173. }
  174. public String getTitle()
  175. {
  176. return _title;
  177. }
  178. public boolean isLocked()
  179. {
  180. return _locked;
  181. }
  182. public boolean isConfirmed()
  183. {
  184. return _confirmed;
  185. }
  186. public boolean isPackaged()
  187. {
  188. return _packaged;
  189. }
  190. public void setPackaged(boolean value)
  191. {
  192. _packaged = value;
  193. }
  194. /**
  195. * Retrieves items from TradeList
  196. */
  197. public TradeItem[] getItems()
  198. {
  199. return _items.toArray(new TradeItem[_items.size()]);
  200. }
  201. /**
  202. * Returns the list of items in inventory available for transaction
  203. * @return L2ItemInstance : items in inventory
  204. */
  205. public TradeList.TradeItem[] getAvailableItems(PcInventory inventory)
  206. {
  207. final List<TradeList.TradeItem> list = new FastList<TradeList.TradeItem>();
  208. for (TradeList.TradeItem item : _items)
  209. {
  210. item = new TradeItem(item, item.getCount(), item.getPrice());
  211. inventory.adjustAvailableItem(item);
  212. list.add(item);
  213. }
  214. return list.toArray(new TradeList.TradeItem[list.size()]);
  215. }
  216. /**
  217. * Returns Item List size
  218. */
  219. public int getItemCount()
  220. {
  221. return _items.size();
  222. }
  223. /**
  224. * Adjust available item from Inventory by the one in this list
  225. * @param item : L2ItemInstance to be adjusted
  226. * @return TradeItem representing adjusted item
  227. */
  228. public TradeItem adjustAvailableItem(L2ItemInstance item)
  229. {
  230. if (item.isStackable())
  231. {
  232. for (TradeItem exclItem : _items)
  233. {
  234. if (exclItem.getItem().getItemId() == item.getItemId())
  235. {
  236. if (item.getCount() <= exclItem.getCount())
  237. return null;
  238. else return new TradeItem(item, item.getCount() - exclItem.getCount(), item.getReferencePrice());
  239. }
  240. }
  241. }
  242. return new TradeItem(item, item.getCount(), item.getReferencePrice());
  243. }
  244. /**
  245. * Adjust ItemRequest by corresponding item in this list using its <b>ObjectId</b>
  246. * @param item : ItemRequest to be adjusted
  247. */
  248. public void adjustItemRequest(ItemRequest item)
  249. {
  250. for (TradeItem filtItem : _items)
  251. {
  252. if (filtItem.getObjectId() == item.getObjectId())
  253. {
  254. if (filtItem.getCount() < item.getCount())
  255. item.setCount(filtItem.getCount());
  256. return;
  257. }
  258. }
  259. item.setCount(0);
  260. }
  261. /**
  262. * Add simplified item to TradeList
  263. * @param objectId : int
  264. * @param count : int
  265. * @return
  266. */
  267. public synchronized TradeItem addItem(int objectId, long count)
  268. {
  269. return addItem(objectId, count, 0);
  270. }
  271. /**
  272. * Add item to TradeList
  273. * @param objectId : int
  274. * @param count : long
  275. * @param price : long
  276. * @return
  277. */
  278. public synchronized TradeItem addItem(int objectId, long count, long price)
  279. {
  280. if (isLocked())
  281. {
  282. _log.warning(_owner.getName() + ": Attempt to modify locked TradeList!");
  283. return null;
  284. }
  285. L2Object o = L2World.getInstance().findObject(objectId);
  286. if (!(o instanceof L2ItemInstance))
  287. {
  288. _log.warning(_owner.getName() + ": Attempt to add invalid item to TradeList!");
  289. return null;
  290. }
  291. L2ItemInstance item = (L2ItemInstance)o;
  292. if (!(item.isTradeable() || (getOwner().isGM() && Config.GM_TRADE_RESTRICTED_ITEMS))
  293. || item.getItemType() == L2EtcItemType.QUEST)
  294. return null;
  295. if (count <= 0 || count > item.getCount())
  296. return null;
  297. if (!item.isStackable() && count > 1)
  298. {
  299. _log.warning(_owner.getName() + ": Attempt to add non-stackable item to TradeList with count > 1!");
  300. return null;
  301. }
  302. if ((PcInventory.MAX_ADENA / count) < price)
  303. {
  304. _log.warning(_owner.getName() + ": Attempt to overflow adena !");
  305. return null;
  306. }
  307. for (TradeItem checkitem : _items)
  308. {
  309. if (checkitem.getObjectId() == objectId)
  310. return null;
  311. }
  312. TradeItem titem = new TradeItem(item, count, price);
  313. _items.add(titem);
  314. // If Player has already confirmed this trade, invalidate the confirmation
  315. invalidateConfirmation();
  316. return titem;
  317. }
  318. /**
  319. * Add item to TradeList
  320. * @param objectId : int
  321. * @param count : long
  322. * @param price : long
  323. * @return
  324. */
  325. public synchronized TradeItem addItemByItemId(int itemId, long count, long price)
  326. {
  327. if (isLocked())
  328. {
  329. _log.warning(_owner.getName() + ": Attempt to modify locked TradeList!");
  330. return null;
  331. }
  332. L2Item item = ItemTable.getInstance().getTemplate(itemId);
  333. if (item == null)
  334. {
  335. _log.warning(_owner.getName() + ": Attempt to add invalid item to TradeList!");
  336. return null;
  337. }
  338. if (!item.isTradeable() || item.getItemType() == L2EtcItemType.QUEST)
  339. return null;
  340. if (!item.isStackable() && count > 1)
  341. {
  342. _log.warning(_owner.getName() + ": Attempt to add non-stackable item to TradeList with count > 1!");
  343. return null;
  344. }
  345. if ((PcInventory.MAX_ADENA / count) < price)
  346. {
  347. _log.warning(_owner.getName() + ": Attempt to overflow adena !");
  348. return null;
  349. }
  350. TradeItem titem = new TradeItem(item, count, price);
  351. _items.add(titem);
  352. // If Player has already confirmed this trade, invalidate the confirmation
  353. invalidateConfirmation();
  354. return titem;
  355. }
  356. /**
  357. * Remove item from TradeList
  358. * @param objectId : int
  359. * @param count : int
  360. * @return
  361. */
  362. public synchronized TradeItem removeItem(int objectId, int itemId, long count)
  363. {
  364. if (isLocked())
  365. {
  366. _log.warning(_owner.getName() + ": Attempt to modify locked TradeList!");
  367. return null;
  368. }
  369. for (TradeItem titem : _items)
  370. {
  371. if (titem.getObjectId() == objectId
  372. || titem.getItem().getItemId() == itemId)
  373. {
  374. // If Partner has already confirmed this trade, invalidate the confirmation
  375. if (_partner != null)
  376. {
  377. TradeList partnerList = _partner.getActiveTradeList();
  378. if (partnerList == null)
  379. {
  380. _log.warning(_partner.getName() + ": Trading partner (" + _partner.getName() + ") is invalid in this trade!");
  381. return null;
  382. }
  383. partnerList.invalidateConfirmation();
  384. }
  385. // Reduce item count or complete item
  386. if (count != -1 && titem.getCount() > count)
  387. titem.setCount(titem.getCount() - count);
  388. else _items.remove(titem);
  389. return titem;
  390. }
  391. }
  392. return null;
  393. }
  394. /**
  395. * Update items in TradeList according their quantity in owner inventory
  396. */
  397. public synchronized void updateItems()
  398. {
  399. for (TradeItem titem : _items)
  400. {
  401. L2ItemInstance item = _owner.getInventory().getItemByObjectId(titem.getObjectId());
  402. if (item == null || titem.getCount() < 1)
  403. removeItem(titem.getObjectId(), -1, -1);
  404. else if (item.getCount() < titem.getCount())
  405. titem.setCount(item.getCount());
  406. }
  407. }
  408. /**
  409. * Lockes TradeList, no further changes are allowed
  410. */
  411. public void lock()
  412. {
  413. _locked = true;
  414. }
  415. /**
  416. * Clears item list
  417. */
  418. public synchronized void clear()
  419. {
  420. _items.clear();
  421. _locked = false;
  422. }
  423. /**
  424. * Confirms TradeList
  425. * @return : boolean
  426. */
  427. public boolean confirm()
  428. {
  429. if (_confirmed)
  430. return true; // Already confirmed
  431. // If Partner has already confirmed this trade, proceed exchange
  432. if (_partner != null)
  433. {
  434. TradeList partnerList = _partner.getActiveTradeList();
  435. if (partnerList == null)
  436. {
  437. _log.warning(_partner.getName() + ": Trading partner (" + _partner.getName() + ") is invalid in this trade!");
  438. return false;
  439. }
  440. // Synchronization order to avoid deadlock
  441. TradeList sync1, sync2;
  442. if (getOwner().getObjectId() > partnerList.getOwner().getObjectId())
  443. {
  444. sync1 = partnerList; sync2 = this;
  445. }
  446. else
  447. {
  448. sync1 = this; sync2 = partnerList;
  449. }
  450. synchronized (sync1)
  451. {
  452. synchronized (sync2)
  453. {
  454. _confirmed = true;
  455. if (partnerList.isConfirmed())
  456. {
  457. partnerList.lock();
  458. lock();
  459. if (!partnerList.validate())
  460. return false;
  461. if (!validate())
  462. return false;
  463. doExchange(partnerList);
  464. }
  465. else _partner.onTradeConfirm(_owner);
  466. }
  467. }
  468. }
  469. else _confirmed = true;
  470. return _confirmed;
  471. }
  472. /**
  473. * Cancels TradeList confirmation
  474. */
  475. public void invalidateConfirmation()
  476. {
  477. _confirmed = false;
  478. }
  479. /**
  480. * Validates TradeList with owner inventory
  481. */
  482. private boolean validate()
  483. {
  484. // Check for Owner validity
  485. if (_owner == null || L2World.getInstance().findObject(_owner.getObjectId()) == null)
  486. {
  487. _log.warning("Invalid owner of TradeList");
  488. return false;
  489. }
  490. // Check for Item validity
  491. for (TradeItem titem : _items)
  492. {
  493. L2ItemInstance item = _owner.checkItemManipulation(titem.getObjectId(), titem.getCount(), "transfer");
  494. if (item == null || item.getCount() < 1)
  495. {
  496. _log.warning(_owner.getName() + ": Invalid Item in TradeList");
  497. return false;
  498. }
  499. }
  500. return true;
  501. }
  502. /**
  503. * Transfers all TradeItems from inventory to partner
  504. */
  505. private boolean TransferItems(L2PcInstance partner, InventoryUpdate ownerIU, InventoryUpdate partnerIU)
  506. {
  507. for (TradeItem titem : _items)
  508. {
  509. L2ItemInstance oldItem = _owner.getInventory().getItemByObjectId(titem.getObjectId());
  510. if (oldItem == null)
  511. return false;
  512. L2ItemInstance newItem = _owner.getInventory().transferItem("Trade", titem.getObjectId(), titem.getCount(), partner.getInventory(), _owner, _partner);
  513. if (newItem == null)
  514. return false;
  515. // Add changes to inventory update packets
  516. if (ownerIU != null)
  517. {
  518. if (oldItem.getCount() > 0 && oldItem != newItem)
  519. ownerIU.addModifiedItem(oldItem);
  520. else ownerIU.addRemovedItem(oldItem);
  521. }
  522. if (partnerIU != null)
  523. {
  524. if (newItem.getCount() > titem.getCount())
  525. partnerIU.addModifiedItem(newItem);
  526. else partnerIU.addNewItem(newItem);
  527. }
  528. }
  529. return true;
  530. }
  531. /**
  532. * Count items slots
  533. */
  534. public int countItemsSlots(L2PcInstance partner)
  535. {
  536. int slots = 0;
  537. for (TradeItem item : _items)
  538. {
  539. if (item == null)
  540. continue;
  541. L2Item template = ItemTable.getInstance().getTemplate(item.getItem().getItemId());
  542. if (template == null)
  543. continue;
  544. if (!template.isStackable())
  545. slots += item.getCount();
  546. else if (partner.getInventory().getItemByItemId(item.getItem().getItemId()) == null)
  547. slots++;
  548. }
  549. return slots;
  550. }
  551. /**
  552. * Calc weight of items in tradeList
  553. */
  554. public int calcItemsWeight()
  555. {
  556. int weight = 0;
  557. for (TradeItem item : _items)
  558. {
  559. if (item == null)
  560. continue;
  561. L2Item template = ItemTable.getInstance().getTemplate(item.getItem().getItemId());
  562. if (template == null)
  563. continue;
  564. weight += item.getCount() * template.getWeight();
  565. }
  566. return weight;
  567. }
  568. /**
  569. * Proceeds with trade
  570. */
  571. private void doExchange(TradeList partnerList)
  572. {
  573. boolean success = false;
  574. // check weight and slots
  575. if ((!getOwner().getInventory().validateWeight(partnerList.calcItemsWeight()))
  576. || !(partnerList.getOwner().getInventory().validateWeight(calcItemsWeight())))
  577. {
  578. partnerList.getOwner().sendPacket(new SystemMessage(SystemMessageId.WEIGHT_LIMIT_EXCEEDED));
  579. getOwner().sendPacket(new SystemMessage(SystemMessageId.WEIGHT_LIMIT_EXCEEDED));
  580. }
  581. else if ((!getOwner().getInventory().validateCapacity(partnerList.countItemsSlots(getOwner())))
  582. || (!partnerList.getOwner().getInventory().validateCapacity(countItemsSlots(partnerList.getOwner()))))
  583. {
  584. partnerList.getOwner().sendPacket(new SystemMessage(SystemMessageId.SLOTS_FULL));
  585. getOwner().sendPacket(new SystemMessage(SystemMessageId.SLOTS_FULL));
  586. }
  587. else
  588. {
  589. // Prepare inventory update packet
  590. InventoryUpdate ownerIU = Config.FORCE_INVENTORY_UPDATE ? null : new InventoryUpdate();
  591. InventoryUpdate partnerIU = Config.FORCE_INVENTORY_UPDATE ? null : new InventoryUpdate();
  592. // Transfer items
  593. partnerList.TransferItems(getOwner(), partnerIU, ownerIU);
  594. TransferItems(partnerList.getOwner(), ownerIU, partnerIU);
  595. // Send inventory update packet
  596. if (ownerIU != null)
  597. _owner.sendPacket(ownerIU);
  598. else _owner.sendPacket(new ItemList(_owner, false));
  599. if (partnerIU != null)
  600. _partner.sendPacket(partnerIU);
  601. else _partner.sendPacket(new ItemList(_partner, false));
  602. // Update current load as well
  603. StatusUpdate playerSU = new StatusUpdate(_owner.getObjectId());
  604. playerSU.addAttribute(StatusUpdate.CUR_LOAD, _owner.getCurrentLoad());
  605. _owner.sendPacket(playerSU);
  606. playerSU = new StatusUpdate(_partner.getObjectId());
  607. playerSU.addAttribute(StatusUpdate.CUR_LOAD, _partner.getCurrentLoad());
  608. _partner.sendPacket(playerSU);
  609. success = true;
  610. }
  611. // Finish the trade
  612. partnerList.getOwner().onTradeFinish(success);
  613. getOwner().onTradeFinish(success);
  614. }
  615. /**
  616. * Buy items from this PrivateStore list
  617. * @return : boolean true if success
  618. */
  619. public synchronized boolean privateStoreBuy(L2PcInstance player, ItemRequest[] items)
  620. {
  621. if (_locked)
  622. return false;
  623. if (!validate())
  624. {
  625. lock();
  626. return false;
  627. }
  628. int slots = 0;
  629. int weight = 0;
  630. long totalPrice = 0;
  631. final PcInventory ownerInventory = _owner.getInventory();
  632. final PcInventory playerInventory = player.getInventory();
  633. for (ItemRequest item : items)
  634. {
  635. boolean found = false;
  636. for (TradeItem ti : _items)
  637. {
  638. if (ti.getObjectId() == item.getObjectId())
  639. {
  640. if (ti.getPrice() == item.getPrice())
  641. {
  642. if (ti.getCount() < item.getCount())
  643. item.setCount(ti.getCount());
  644. found = true;
  645. }
  646. break;
  647. }
  648. }
  649. // item with this objectId and price not found in tradelist
  650. if (!found)
  651. {
  652. item.setCount(0);
  653. continue;
  654. }
  655. // check for overflow in the single item
  656. if ((MAX_ADENA / item.getCount()) < item.getPrice())
  657. {
  658. // private store attempting to overflow - disable it
  659. lock();
  660. return false;
  661. }
  662. totalPrice += item.getCount() * item.getPrice();
  663. // check for overflow of the total price
  664. if (MAX_ADENA < totalPrice
  665. || totalPrice < 0)
  666. {
  667. // private store attempting to overflow - disable it
  668. lock();
  669. return false;
  670. }
  671. // Check if requested item is available for manipulation
  672. L2ItemInstance oldItem = _owner.checkItemManipulation(item.getObjectId(), item.getCount(), "sell");
  673. if (oldItem == null || !oldItem.isTradeable())
  674. {
  675. // private store sell invalid item - disable it
  676. lock();
  677. return false;
  678. }
  679. L2Item template = ItemTable.getInstance().getTemplate(item.getItemId());
  680. if (template == null)
  681. continue;
  682. weight += item.getCount() * template.getWeight();
  683. if (!template.isStackable())
  684. slots += item.getCount();
  685. else if (playerInventory.getItemByItemId(item.getItemId()) == null)
  686. slots++;
  687. }
  688. if (totalPrice > playerInventory.getAdena())
  689. {
  690. player.sendPacket(new SystemMessage(SystemMessageId.YOU_NOT_ENOUGH_ADENA));
  691. return false;
  692. }
  693. if (!playerInventory.validateWeight(weight))
  694. {
  695. player.sendPacket(new SystemMessage(SystemMessageId.WEIGHT_LIMIT_EXCEEDED));
  696. return false;
  697. }
  698. if (!playerInventory.validateCapacity(slots))
  699. {
  700. player.sendPacket(new SystemMessage(SystemMessageId.SLOTS_FULL));
  701. return false;
  702. }
  703. // Prepare inventory update packets
  704. final InventoryUpdate ownerIU = new InventoryUpdate();
  705. final InventoryUpdate playerIU = new InventoryUpdate();
  706. final L2ItemInstance adenaItem = playerInventory.getAdenaInstance();
  707. playerInventory.reduceAdena("PrivateStore", totalPrice, player, _owner);
  708. playerIU.addItem(adenaItem);
  709. ownerInventory.addAdena("PrivateStore", totalPrice, _owner, player);
  710. ownerIU.addItem(ownerInventory.getAdenaInstance());
  711. boolean ok = true;
  712. // Transfer items
  713. for (ItemRequest item : items)
  714. {
  715. if (item.getCount() == 0)
  716. continue;
  717. // Check if requested item is available for manipulation
  718. L2ItemInstance oldItem = _owner.checkItemManipulation(item.getObjectId(), item.getCount(), "sell");
  719. if (oldItem == null)
  720. {
  721. // should not happens - validation already done
  722. lock();
  723. ok = false;
  724. break;
  725. }
  726. // Proceed with item transfer
  727. L2ItemInstance newItem = ownerInventory.transferItem("PrivateStore", item.getObjectId(), item.getCount(), playerInventory, _owner, player);
  728. if (newItem == null)
  729. {
  730. ok = false;
  731. break;
  732. }
  733. removeItem(item.getObjectId(), -1, item.getCount());
  734. // Add changes to inventory update packets
  735. if (oldItem.getCount() > 0 && oldItem != newItem)
  736. ownerIU.addModifiedItem(oldItem);
  737. else ownerIU.addRemovedItem(oldItem);
  738. if (newItem.getCount() > item.getCount())
  739. playerIU.addModifiedItem(newItem);
  740. else playerIU.addNewItem(newItem);
  741. // Send messages about the transaction to both players
  742. if (newItem.isStackable())
  743. {
  744. SystemMessage msg = new SystemMessage(SystemMessageId.C1_PURCHASED_S3_S2_S);
  745. msg.addString(player.getName());
  746. msg.addItemName(newItem);
  747. msg.addItemNumber(item.getCount());
  748. _owner.sendPacket(msg);
  749. msg = new SystemMessage(SystemMessageId.PURCHASED_S3_S2_S_FROM_C1);
  750. msg.addString(_owner.getName());
  751. msg.addItemName(newItem);
  752. msg.addItemNumber(item.getCount());
  753. player.sendPacket(msg);
  754. }
  755. else
  756. {
  757. SystemMessage msg = new SystemMessage(SystemMessageId.C1_PURCHASED_S2);
  758. msg.addString(player.getName());
  759. msg.addItemName(newItem);
  760. _owner.sendPacket(msg);
  761. msg = new SystemMessage(SystemMessageId.PURCHASED_S2_FROM_C1);
  762. msg.addString(_owner.getName());
  763. msg.addItemName(newItem);
  764. player.sendPacket(msg);
  765. }
  766. }
  767. // Send inventory update packet
  768. _owner.sendPacket(ownerIU);
  769. player.sendPacket(playerIU);
  770. return ok;
  771. }
  772. /**
  773. * Sell items to this PrivateStore list
  774. * @return : boolean true if success
  775. */
  776. public synchronized boolean privateStoreSell(L2PcInstance player, ItemRequest[] items)
  777. {
  778. if (_locked)
  779. return false;
  780. boolean ok = false;
  781. final PcInventory ownerInventory = _owner.getInventory();
  782. final PcInventory playerInventory = player.getInventory();
  783. // Prepare inventory update packet
  784. final InventoryUpdate ownerIU = new InventoryUpdate();
  785. final InventoryUpdate playerIU = new InventoryUpdate();
  786. long totalPrice = 0;
  787. for (ItemRequest item : items)
  788. {
  789. // searching item in tradelist using itemId
  790. boolean found = false;
  791. for (TradeItem ti : _items)
  792. {
  793. if (ti.getItem().getItemId() == item.getItemId())
  794. {
  795. // price should be the same
  796. if (ti.getPrice() == item.getPrice())
  797. {
  798. // if requesting more than available - decrease count
  799. if (ti.getCount() < item.getCount())
  800. item.setCount(ti.getCount());
  801. found = item.getCount() > 0;
  802. }
  803. break;
  804. }
  805. }
  806. // not found any item in the tradelist with same itemId and price
  807. // maybe another player already sold this item ?
  808. if (!found)
  809. continue;
  810. // check for overflow in the single item
  811. if ((MAX_ADENA / item.getCount()) < item.getPrice())
  812. {
  813. lock();
  814. break;
  815. }
  816. long _totalPrice = totalPrice + item.getCount() * item.getPrice();
  817. // check for overflow of the total price
  818. if (MAX_ADENA < _totalPrice
  819. || _totalPrice < 0)
  820. {
  821. lock();
  822. break;
  823. }
  824. if (ownerInventory.getAdena() < _totalPrice)
  825. continue;
  826. // Check if requested item is available for manipulation
  827. int objectId = item.getObjectId();
  828. L2ItemInstance oldItem = player.checkItemManipulation(objectId, item.getCount(), "sell");
  829. // private store - buy use same objectId for buying several non-stackable items
  830. if (oldItem == null)
  831. {
  832. // searching other items using same itemId
  833. oldItem = playerInventory.getItemByItemId(item.getItemId());
  834. if (oldItem == null)
  835. continue;
  836. objectId = oldItem.getObjectId();
  837. oldItem = player.checkItemManipulation(objectId, item.getCount(), "sell");
  838. if (oldItem == null)
  839. continue;
  840. }
  841. if (!oldItem.isTradeable())
  842. continue;
  843. // Proceed with item transfer
  844. L2ItemInstance newItem = playerInventory.transferItem("PrivateStore", objectId, item.getCount(), ownerInventory, player, _owner);
  845. if (newItem == null)
  846. continue;
  847. removeItem(-1, item.getItemId(), item.getCount());
  848. ok = true;
  849. // increase total price only after successful transaction
  850. totalPrice = _totalPrice;
  851. // Add changes to inventory update packets
  852. if (oldItem.getCount() > 0 && oldItem != newItem)
  853. playerIU.addModifiedItem(oldItem);
  854. else playerIU.addRemovedItem(oldItem);
  855. if (newItem.getCount() > item.getCount())
  856. ownerIU.addModifiedItem(newItem);
  857. else ownerIU.addNewItem(newItem);
  858. // Send messages about the transaction to both players
  859. if (newItem.isStackable())
  860. {
  861. SystemMessage msg = new SystemMessage(SystemMessageId.PURCHASED_S3_S2_S_FROM_C1);
  862. msg.addString(player.getName());
  863. msg.addItemName(newItem);
  864. msg.addItemNumber(item.getCount());
  865. _owner.sendPacket(msg);
  866. msg = new SystemMessage(SystemMessageId.C1_PURCHASED_S3_S2_S);
  867. msg.addString(_owner.getName());
  868. msg.addItemName(newItem);
  869. msg.addItemNumber(item.getCount());
  870. player.sendPacket(msg);
  871. }
  872. else
  873. {
  874. SystemMessage msg = new SystemMessage(SystemMessageId.PURCHASED_S2_FROM_C1);
  875. msg.addString(player.getName());
  876. msg.addItemName(newItem);
  877. _owner.sendPacket(msg);
  878. msg = new SystemMessage(SystemMessageId.C1_PURCHASED_S2);
  879. msg.addString(_owner.getName());
  880. msg.addItemName(newItem);
  881. player.sendPacket(msg);
  882. }
  883. }
  884. if (totalPrice > 0)
  885. {
  886. // Transfer adena
  887. if (totalPrice > ownerInventory.getAdena())
  888. // should not happens, just a precaution
  889. return false;
  890. final L2ItemInstance adenaItem = ownerInventory.getAdenaInstance();
  891. ownerInventory.reduceAdena("PrivateStore", totalPrice, _owner, player);
  892. ownerIU.addItem(adenaItem);
  893. playerInventory.addAdena("PrivateStore", totalPrice, player, _owner);
  894. playerIU.addItem(playerInventory.getAdenaInstance());
  895. }
  896. if (ok)
  897. {
  898. // Send inventory update packet
  899. _owner.sendPacket(ownerIU);
  900. player.sendPacket(playerIU);
  901. }
  902. return ok;
  903. }
  904. }