package com.planet_ink.coffee_mud.MOBS; import com.planet_ink.coffee_mud.core.interfaces.*; import com.planet_ink.coffee_mud.core.*; import com.planet_ink.coffee_mud.core.collections.*; import com.planet_ink.coffee_mud.Abilities.interfaces.*; import com.planet_ink.coffee_mud.Areas.interfaces.*; import com.planet_ink.coffee_mud.Behaviors.interfaces.*; import com.planet_ink.coffee_mud.CharClasses.interfaces.*; import com.planet_ink.coffee_mud.Commands.interfaces.*; import com.planet_ink.coffee_mud.Common.interfaces.*; import com.planet_ink.coffee_mud.Common.interfaces.TimeClock.TimePeriod; import com.planet_ink.coffee_mud.Exits.interfaces.*; import com.planet_ink.coffee_mud.Items.interfaces.*; import com.planet_ink.coffee_mud.Libraries.interfaces.XMLLibrary; import com.planet_ink.coffee_mud.Locales.interfaces.*; import com.planet_ink.coffee_mud.MOBS.interfaces.*; import com.planet_ink.coffee_mud.Races.interfaces.*; import java.util.*; /* Copyright 2001-2019 Bo Zimmerman Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ public class StdShopKeeper extends StdMOB implements ShopKeeper { @Override public String ID() { return "StdShopKeeper"; } protected CoffeeShop shop = ((CoffeeShop) CMClass.getCommon("DefaultCoffeeShop")).build(this); protected long whatIsSoldMask = 0; protected int invResetRate = 0; protected int invResetTickDown = 0; protected long budgetRemaining = Long.MAX_VALUE / 2; protected long budgetMax = Long.MAX_VALUE / 2; protected int budgetTickDown = 2; protected double[] devalueRate = null; protected String[] pricingAdjustments = new String[0]; protected String itemZapperMask = ""; protected Pair<Long,TimeClock.TimePeriod> budget = null; public StdShopKeeper() { super(); username = "a shopkeeper"; setDescription("He\\`s pleased to be of assistance."); setDisplayText("A shopkeeper is waiting to serve you."); CMLib.factions().setAlignment(this, Faction.Align.GOOD); setMoney(0); basePhyStats.setWeight(150); setWimpHitPoint(0); baseCharStats().setStat(CharStats.STAT_INTELLIGENCE, 16); baseCharStats().setStat(CharStats.STAT_CHARISMA, 25); basePhyStats().setArmor(0); baseState.setHitPoints(1000); recoverMaxState(); resetToMaxState(); recoverPhyStats(); recoverCharStats(); } @Override public boolean isSold(final int mask) { if (mask == 0) return whatIsSoldMask == 0; if ((whatIsSoldMask & 255) == mask) return true; return CMath.bset(whatIsSoldMask >> 8, CMath.pow(2, mask - 1)); } @Override public void addSoldType(final int mask) { if (mask == 0) whatIsSoldMask = 0; else { if ((whatIsSoldMask > 0) && (whatIsSoldMask < 256)) whatIsSoldMask = (CMath.pow(2, whatIsSoldMask - 1) << 8); for (int c = 0; c < ShopKeeper.DEAL_CONFLICTS.length; c++) { for (int c1 = 0; c1 < ShopKeeper.DEAL_CONFLICTS[c].length; c1++) { if (ShopKeeper.DEAL_CONFLICTS[c][c1] == mask) { for (c1 = 0; c1 < ShopKeeper.DEAL_CONFLICTS[c].length; c1++) { if ((ShopKeeper.DEAL_CONFLICTS[c][c1] != mask) && (isSold(ShopKeeper.DEAL_CONFLICTS[c][c1]))) addSoldType(-ShopKeeper.DEAL_CONFLICTS[c][c1]); } break; } } } if (mask < 0) whatIsSoldMask = CMath.unsetb(whatIsSoldMask, (CMath.pow(2, (-mask) - 1) << 8)); else whatIsSoldMask |= (CMath.pow(2, mask - 1) << 8); } } @Override public long getWhatIsSoldMask() { return whatIsSoldMask; } @Override public void setWhatIsSoldMask(final long newSellCode) { whatIsSoldMask = newSellCode; } @Override protected void cloneFix(final MOB E) { super.cloneFix(E); if (E instanceof StdShopKeeper) shop = ((CoffeeShop) ((StdShopKeeper) E).shop.copyOf()).build(this); } @Override public CoffeeShop getShop() { return shop; } @Override public void destroy() { super.destroy(); getShop().destroyStoreInventory(); } @Override public String storeKeeperString() { return CMLib.coffeeShops().storeKeeperString(getShop(), this); } @Override public boolean doISellThis(final Environmental thisThang) { return CMLib.coffeeShops().doISellThis(thisThang, this); } protected Area getStartArea() { Area A = CMLib.map().getStartArea(this); if (A == null) CMLib.map().areaLocation(this); if (A == null) A = CMLib.map().areas().nextElement(); return A; } protected void doInventoryReset() { invResetTickDown = finalInvResetRate(); // we should now be at a // positive number. if (invResetTickDown <= 0) invResetTickDown = Ability.TICKS_FOREVER; else { final CoffeeShop shop=(this instanceof Librarian)?((Librarian)this).getBaseLibrary():this.getShop(); shop.emptyAllShelves(); if (miscText != null) { String shoptext; if (CMProps.getBoolVar(CMProps.Bool.MOBCOMPRESS) && (miscText instanceof byte[])) shoptext = CMLib.coffeeMaker().getGenMOBTextUnpacked(this, CMLib.encoder().decompressString((byte[]) miscText)); else shoptext = CMLib.coffeeMaker().getGenMOBTextUnpacked(this, CMStrings.bytesToStr(miscText)); final List<XMLLibrary.XMLTag> xml = CMLib.xml().parseAllXML(shoptext); if (xml != null) { CMLib.coffeeMaker().populateShops(this, xml); recoverPhyStats(); recoverCharStats(); } } } } @Override public boolean tick(final Tickable ticking, final int tickID) { if (!super.tick(ticking, tickID)) return false; if ((tickID == Tickable.TICKID_MOB) && (isGeneric())) { if ((--invResetTickDown) <= 0) { doInventoryReset(); } if ((--budgetTickDown) <= 0) { budgetTickDown = 100; budgetRemaining = Long.MAX_VALUE / 2; final Pair<Long,TimeClock.TimePeriod> budget = finalBudget(); if(budget != null) { budgetRemaining = budget.first.longValue(); budgetTickDown = 100; final Room R=location(); final TimeClock C=((R != null) && (R.getArea() != null)) ? R.getArea().getTimeObj() : null; final int ticksPerDay = CMProps.getIntVar(CMProps.Int.TICKSPERMUDDAY); int hoursInDay = 1; int daysInMonth = 1; int monthsInYear = 1; int daysInWeek = 1; if(C!=null) { if(C.getHoursInDay()>0) hoursInDay=C.getHoursInDay(); if(C.getDaysInMonth()>0) daysInMonth=C.getDaysInMonth(); if(C.getMonthsInYear()>0) monthsInYear=C.getMonthsInYear(); if(C.getDaysInWeek()>0) daysInWeek=C.getDaysInWeek(); } switch(budget.second) { case ALLTIME: budgetTickDown = 100; break; case DAY: budgetTickDown = ticksPerDay; break; case HOUR: budgetTickDown = ticksPerDay / hoursInDay; break; case MONTH: budgetTickDown = ticksPerDay * daysInMonth; break; case SEASON: budgetTickDown = ticksPerDay * (daysInMonth * monthsInYear / 4); break; case WEEK: budgetTickDown = ticksPerDay * daysInWeek; break; case YEAR: budgetTickDown = ticksPerDay * (daysInMonth * monthsInYear); break; } } budgetMax = budgetRemaining; } } return true; } @Override public boolean okMessage(final Environmental myHost, final CMMsg msg) { if (msg.amITarget(this)) { switch (msg.targetMinor()) { case CMMsg.TYP_VALUE: case CMMsg.TYP_SELL: { if (!CMLib.coffeeShops().ignoreIfNecessary(msg.source(), finalIgnoreMask(), this)) return false; if (CMLib.coffeeShops().standardSellEvaluation(this, msg.source(), msg.tool(), this, budgetRemaining, budgetMax, msg.targetMinor() == CMMsg.TYP_SELL)) return super.okMessage(myHost, msg); return false; } case CMMsg.TYP_BID: { CMLib.commands().postSay(this, msg.source(), L("I'm afraid my prices are firm."), false, false); return false; } case CMMsg.TYP_BUY: case CMMsg.TYP_VIEW: { if (!CMLib.coffeeShops().ignoreIfNecessary(msg.source(), finalIgnoreMask(), this)) return false; if ((msg.targetMinor() == CMMsg.TYP_BUY) && (msg.tool() != null) && (!msg.tool().okMessage(myHost, msg))) return false; if (CMLib.coffeeShops().standardBuyEvaluation(this, msg.source(), msg.tool(), this, msg.targetMinor() == CMMsg.TYP_BUY)) return super.okMessage(myHost, msg); return false; } case CMMsg.TYP_LIST: { if (!CMLib.coffeeShops().ignoreIfNecessary(msg.source(), finalIgnoreMask(), this)) return false; return super.okMessage(myHost, msg); } default: break; } } return super.okMessage(myHost, msg); } @Override public void executeMsg(final Environmental myHost, final CMMsg msg) { if (msg.amITarget(this)) { final MOB mob = msg.source(); switch (msg.targetMinor()) { case CMMsg.TYP_GIVE: if (CMLib.flags().isAliveAwakeMobileUnbound(mob, true)) { if ((msg.tool() != null) && ((CMSecurity.isAllowed(msg.source(), location(), CMSecurity.SecFlag.ORDER) || (CMLib.law().doesHavePriviledgesHere(msg.source(), getStartRoom())) || (CMSecurity.isAllowed(msg.source(), location(), CMSecurity.SecFlag.CMDMOBS) && (isMonster())) || (CMSecurity.isAllowed(msg.source(), location(), CMSecurity.SecFlag.CMDROOMS) && (isMonster())))) && ((doISellThis(msg.tool())) || (isSold(DEAL_INVENTORYONLY)))) { CMLib.commands().postSay(this, msg.source(), L("Yes, I will now sell @x1.", msg.tool().name()), false, false); getShop().addStoreInventory(msg.tool(), 1, -1); if (isGeneric()) text(); return; } } super.executeMsg(myHost, msg); break; case CMMsg.TYP_VALUE: { super.executeMsg(myHost, msg); if (CMLib.flags().isAliveAwakeMobileUnbound(mob, true)) { final double pawningPrice = CMLib.coffeeShops().pawningPrice(this, mob, msg.tool(), this, getShop()).absoluteGoldPrice; final String currencyShort = CMLib.beanCounter().nameCurrencyShort(this, pawningPrice); CMLib.commands().postSay(this, mob, L("I'll give you @x1 for @x2.", currencyShort, msg.tool().name()), true, false); } break; } case CMMsg.TYP_SELL: // sell TO -- this is a shopkeeper purchasing from a player { super.executeMsg(myHost, msg); if (CMLib.flags().isAliveAwakeMobileUnbound(mob, true)) { final double paid = CMLib.coffeeShops().transactPawn(this, msg.source(), this, msg.tool()); if (paid > Double.MIN_VALUE) { budgetRemaining = budgetRemaining - Math.round(paid); if (mySession != null) mySession.stdPrintln(msg.source(), msg.target(), msg.tool(), msg.targetMessage()); if (!CMath.bset(msg.targetMajor(), CMMsg.MASK_OPTIMIZE)) mob.location().recoverRoomStats(); if (isGeneric()) text(); } } break; } case CMMsg.TYP_VIEW: { super.executeMsg(myHost, msg); if (CMLib.flags().isAliveAwakeMobileUnbound(mob, true)) { if ((msg.tool() != null) && (getShop().doIHaveThisInStock("$" + msg.tool().Name() + "$", mob))) { final String prefix = L("Interested in @x1? Here is some information for you: ",msg.tool().Name()); final String viewDesc = prefix + CMLib.coffeeShops().getViewDescription(msg.source(), msg.tool()); CMLib.commands().postSay(this, msg.source(), viewDesc, true, false); } } break; } case CMMsg.TYP_BUY: // buy-from -- this is a player buying from a // shopkeeper { super.executeMsg(myHost, msg); if (CMLib.flags().isAliveAwakeMobileUnbound(mob, true)) { final MOB mobFor = CMLib.coffeeShops().parseBuyingFor(msg.source(), msg.targetMessage()); if ((msg.tool() != null) && (getShop().doIHaveThisInStock("$" + msg.tool().Name() + "$", mobFor)) && (location() != null)) { final Environmental item = getShop().getStock("$" + msg.tool().Name() + "$", mobFor); if (item != null) CMLib.coffeeShops().transactMoneyOnly(this, msg.source(), this, item, !isMonster()); final List<Environmental> products = getShop().removeSellableProduct("$" + msg.tool().Name() + "$", mobFor); if (products.size() == 0) break; final Environmental product = products.get(0); if (product instanceof Item) { if (!CMLib.coffeeShops().purchaseItems((Item) product, products, this, mobFor)) return; } else if (product instanceof MOB) { if (CMLib.coffeeShops().purchaseMOB((MOB) product, this, this, mobFor)) { msg.modify(msg.source(), msg.target(), product, msg.sourceCode(), msg.sourceMessage(), msg.targetCode(), msg.targetMessage(), msg.othersCode(), msg.othersMessage()); product.executeMsg(myHost, msg); } } else if (product instanceof Ability) CMLib.coffeeShops().purchaseAbility((Ability) product, this, this, mobFor); if (mySession != null) mySession.stdPrintln(msg.source(), msg.target(), msg.tool(), msg.targetMessage()); if (!CMath.bset(msg.targetMajor(), CMMsg.MASK_OPTIMIZE)) mob.location().recoverRoomStats(); } } break; } case CMMsg.TYP_LIST: { super.executeMsg(myHost, msg); if (CMLib.flags().isAliveAwakeMobileUnbound(mob, true)) { final String forMask = CMLib.coffeeShops().getListForMask(msg.targetMessage()); List<Environmental> inventory = new XVector<Environmental>(getShop().getStoreInventory()); inventory = CMLib.coffeeShops().addRealEstateTitles(inventory, mob, getShop(), getStartRoom()); final int limit = CMParms.getParmInt(finalPrejudiceFactors(), "LIMIT", 0); final String s = CMLib.coffeeShops().getListInventory(this, mob, inventory, limit, this, forMask); if (s.length() > 0) mob.tell(s); } break; } default: super.executeMsg(myHost, msg); break; } } else super.executeMsg(myHost, msg); } @Override public String finalPrejudiceFactors() { if (prejudiceFactors().length() > 0) return prejudiceFactors(); return getStartArea().finalPrejudiceFactors(); } @Override public String prejudiceFactors() { return CMStrings.bytesToStr(miscText); } @Override public void setPrejudiceFactors(final String factors) { miscText = factors; } @Override public String finalIgnoreMask() { if (ignoreMask().length() > 0) return ignoreMask(); return getStartArea().finalIgnoreMask(); } @Override public String ignoreMask() { return ""; } @Override public void setIgnoreMask(final String factors) { } @Override public String[] finalItemPricingAdjustments() { if ((itemPricingAdjustments() != null) && (itemPricingAdjustments().length > 0)) return itemPricingAdjustments(); return getStartArea().finalItemPricingAdjustments(); } @Override public String[] itemPricingAdjustments() { return pricingAdjustments; } @Override public void setItemPricingAdjustments(final String[] factors) { pricingAdjustments = factors; } @Override public Pair<Long, TimePeriod> finalBudget() { if (budget != null) return budget; return getStartArea().finalBudget(); } @Override public String budget() { return budget == null ? "" : (budget.first + " " + budget.second.name()); } @Override public void setBudget(final String factors) { budget = CMLib.coffeeShops().parseBudget(factors); budgetTickDown = 0; } @Override public double[] finalDevalueRate() { if (devalueRate != null) return devalueRate; return getStartArea().finalDevalueRate(); } @Override public String devalueRate() { return (devalueRate == null) ? "" : (devalueRate[0] + " " + devalueRate[1]); } @Override public void setDevalueRate(final String factors) { devalueRate = CMLib.coffeeShops().parseDevalueRate(factors); } @Override public int finalInvResetRate() { if (invResetRate() != 0) return invResetRate(); return getStartArea().finalInvResetRate(); } @Override public int invResetRate() { return invResetRate; } @Override public void setInvResetRate(final int ticks) { invResetRate = ticks; invResetTickDown = 0; } @Override public void setWhatIsSoldZappermask(final String newSellMask) { itemZapperMask = newSellMask; } @Override public String getWhatIsSoldZappermask() { return itemZapperMask; } }