Fixed issues relating to stock price movements

This commit is contained in:
danielyxie 2019-04-27 19:20:51 -07:00 committed by danielyxie
parent 67632ced09
commit 87b4698d5b
2 changed files with 132 additions and 56 deletions

@ -1,5 +1,11 @@
import { Order } from "./Order"; import { Order } from "./Order";
import { Stock } from "./Stock"; import { Stock } from "./Stock";
import {
getBuyTransactionCost,
getSellTransactionGain,
processBuyTransactionPriceMovement,
processSellTransactionPriceMovement
} from "./StockMarketHelpers";
import { import {
getStockMarket4SDataCost, getStockMarket4SDataCost,
getStockMarket4STixApiCost getStockMarket4STixApiCost
@ -226,8 +232,8 @@ export function buyStock(stock, shares, workerScript=null) {
} }
// Does player have enough money? // Does player have enough money?
const totalPrice = stock.price * shares; const totalPrice = getBuyTransactionCost(stock, shares, PositionTypes.Long);
if (Player.money.lt(totalPrice + CONSTANTS.StockMarketCommission)) { if (Player.money.lt(totalPrice)) {
if (tixApi) { if (tixApi) {
workerScript.log(`ERROR: buyStock() failed because you do not have enough money to purchase this potiion. You need ${numeralWrapper.formatMoney(totalPrice + CONSTANTS.StockMarketCommission)}`); workerScript.log(`ERROR: buyStock() failed because you do not have enough money to purchase this potiion. You need ${numeralWrapper.formatMoney(totalPrice + CONSTANTS.StockMarketCommission)}`);
} else { } else {
@ -249,25 +255,19 @@ export function buyStock(stock, shares, workerScript=null) {
} }
const origTotal = stock.playerShares * stock.playerAvgPx; const origTotal = stock.playerShares * stock.playerAvgPx;
Player.loseMoney(totalPrice + CONSTANTS.StockMarketCommission); Player.loseMoney(totalPrice);
const newTotal = origTotal + totalPrice; const newTotal = origTotal + totalPrice;
stock.playerShares = Math.round(stock.playerShares + shares); stock.playerShares = Math.round(stock.playerShares + shares);
stock.playerAvgPx = newTotal / stock.playerShares; stock.playerAvgPx = newTotal / stock.playerShares;
processBuyTransactionPriceMovement(stock, shares, PositionTypes.Long);
displayStockMarketContent(); displayStockMarketContent();
const resultTxt = `Bought ${numeralWrapper.format(shares, '0,0')} shares of ${stock.symbol} for ${numeralWrapper.formatMoney(totalPrice)}. ` +
`Paid ${numeralWrapper.formatMoney(CONSTANTS.StockMarketCommission)} in commission fees.`
if (tixApi) { if (tixApi) {
if (workerScript.shouldLog("buyStock")) { if (workerScript.shouldLog("buyStock")) { workerScript.log(resultTxt); }
workerScript.log(
"Bought " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. Paid " +
numeralWrapper.format(CONSTANTS.StockMarketCommission, '($0.000a)') + " in commission fees."
);
}
} else { } else {
dialogBoxCreate( dialogBoxCreate(resultTxt);
"Bought " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. Paid " +
numeralWrapper.format(CONSTANTS.StockMarketCommission, '($0.000a)') + " in commission fees."
);
} }
return true; return true;
@ -297,8 +297,8 @@ export function sellStock(stock, shares, workerScript=null) {
if (shares > stock.playerShares) {shares = stock.playerShares;} if (shares > stock.playerShares) {shares = stock.playerShares;}
if (shares === 0) {return false;} if (shares === 0) {return false;}
const gains = stock.price * shares - CONSTANTS.StockMarketCommission; const gains = getSellTransactionGain(stock, shares, PositionTypes.Long);
let netProfit = ((stock.price - stock.playerAvgPx) * shares) - CONSTANTS.StockMarketCommission; let netProfit = gains - (stock.playerAvgPx * shares);
if (isNaN(netProfit)) { netProfit = 0; } if (isNaN(netProfit)) { netProfit = 0; }
Player.gainMoney(gains); Player.gainMoney(gains);
Player.recordMoneySource(netProfit, "stock"); Player.recordMoneySource(netProfit, "stock");
@ -311,21 +311,15 @@ export function sellStock(stock, shares, workerScript=null) {
if (stock.playerShares === 0) { if (stock.playerShares === 0) {
stock.playerAvgPx = 0; stock.playerAvgPx = 0;
} }
processSellTransactionPriceMovement(stock, shares, PositionTypes.Long);
displayStockMarketContent(); displayStockMarketContent();
const resultTxt = `Sold ${numeralWrapper.format(shares, '0,0')} shares of ${stock.symbol}. ` +
`After commissions, you gained a total of ${numeralWrapper.formatMoney(gains)}.`;
if (tixApi) { if (tixApi) {
if (workerScript.shouldLog("sellStock")) { if (workerScript.shouldLog("sellStock")) { workerScript.log(resultTxt); }
workerScript.log(
"Sold " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. After commissions, you gained " +
"a total of " + numeralWrapper.format(gains, '($0.000a)') + "."
);
}
} else { } else {
dialogBoxCreate( dialogBoxCreate(resultTxt);
"Sold " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. After commissions, you gained " +
"a total of " + numeralWrapper.format(gains, '($0.000a)') + "."
);
} }
return true; return true;
@ -355,7 +349,7 @@ export function shortStock(stock, shares, workerScript=null) {
} }
// Does the player have enough money? // Does the player have enough money?
const totalPrice = stock.price * shares; const totalPrice = getBuyTransactionCost(stock, shares, PositionTypes.Short);
if (Player.money.lt(totalPrice + CONSTANTS.StockMarketCommission)) { if (Player.money.lt(totalPrice + CONSTANTS.StockMarketCommission)) {
if (tixApi) { if (tixApi) {
workerScript.log("ERROR: shortStock() failed because you do not have enough " + workerScript.log("ERROR: shortStock() failed because you do not have enough " +
@ -385,21 +379,15 @@ export function shortStock(stock, shares, workerScript=null) {
const newTotal = origTotal + totalPrice; const newTotal = origTotal + totalPrice;
stock.playerShortShares = Math.round(stock.playerShortShares + shares); stock.playerShortShares = Math.round(stock.playerShortShares + shares);
stock.playerAvgShortPx = newTotal / stock.playerShortShares; stock.playerAvgShortPx = newTotal / stock.playerShortShares;
processBuyTransactionPriceMovement(stock, shares, PositionTypes.Short);
displayStockMarketContent(); displayStockMarketContent();
const resultTxt = `Bought a short position of ${numeralWrapper.format(shares, '0,0')} shares of ${stock.symbol} ` +
`for ${numeralWrapper.formatMoney(totalPrice)}. Paid ${numeralWrapper.formatMoney(CONSTANTS.StockMarketCommission)} ` +
`in commission fees.`;
if (tixApi) { if (tixApi) {
if (workerScript.disableLogs.ALL == null && workerScript.disableLogs.shortStock == null) { if (workerScript.shouldLog("shortStock")) { workerScript.log(resultTxt); }
workerScript.log(
"Bought a short position of " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. Paid " +
numeralWrapper.format(CONSTANTS.StockMarketCommission, '($0.000a)') + " in commission fees."
);
}
} else { } else {
dialogBoxCreate( dialogBoxCreate(resultTxt);
"Bought a short position of " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. Paid " +
numeralWrapper.format(CONSTANTS.StockMarketCommission, '($0.000a)') + " in commission fees."
);
} }
return true; return true;
@ -430,9 +418,19 @@ export function sellShort(stock, shares, workerScript=null) {
if (shares === 0) {return false;} if (shares === 0) {return false;}
const origCost = shares * stock.playerAvgShortPx; const origCost = shares * stock.playerAvgShortPx;
let profit = ((stock.playerAvgShortPx - stock.price) * shares) - CONSTANTS.StockMarketCommission; const totalGain = getSellTransactionGain(stock, shares, PositionTypes.Short);
if (totalGain == null || isNaN(totalGain) || origCost == null) {
if (tixApi) {
workerScript.log(`Failed to sell short position in a stock. This is probably either due to invalid arguments, or a bug`);
} else {
dialogBoxCreate(`Failed to sell short position in a stock. This is probably either due to invalid arguments, or a bug`);
}
return false;
}
let profit = totalGain - origCost;
if (isNaN(profit)) { profit = 0; } if (isNaN(profit)) { profit = 0; }
Player.gainMoney(origCost + profit); Player.gainMoney(totalGain);
Player.recordMoneySource(profit, "stock"); Player.recordMoneySource(profit, "stock");
if (tixApi) { if (tixApi) {
workerScript.scriptRef.onlineMoneyMade += profit; workerScript.scriptRef.onlineMoneyMade += profit;
@ -443,21 +441,14 @@ export function sellShort(stock, shares, workerScript=null) {
if (stock.playerShortShares === 0) { if (stock.playerShortShares === 0) {
stock.playerAvgShortPx = 0; stock.playerAvgShortPx = 0;
} }
processSellTransactionPriceMovement(stock, shares, PositionTypes.Short);
displayStockMarketContent(); displayStockMarketContent();
const resultTxt = `Sold your short position of ${numeralWrapper.format(shares, '0,0')} shares of ${stock.symbol}. ` +
`After commissions, you gained a total of ${numeralWrapper.formatMoney(totalGain)}`;
if (tixApi) { if (tixApi) {
if (workerScript.shouldLog("sellShort")) { if (workerScript.shouldLog("sellShort")) { workerScript.log(resultTxt); }
workerScript.log(
"Sold your short position of " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. After commissions, you gained " +
"a total of " + numeralWrapper.format(origCost + profit, '($0.000a)') + "."
);
}
} else { } else {
dialogBoxCreate( dialogBoxCreate(resultTxt);
"Sold your short position of " + numeralWrapper.format(shares, '0,0') + " shares of " + stock.symbol + " at " +
numeralWrapper.format(stock.price, '($0.000a)') + " per share. After commissions, you gained " +
"a total of " + numeralWrapper.format(origCost + profit, '($0.000a)') + "."
);
} }
return true; return true;

@ -51,6 +51,49 @@ export function getBuyTransactionCost(stock: Stock, shares: number, posType: Pos
return totalCost + CONSTANTS.StockMarketCommission; return totalCost + CONSTANTS.StockMarketCommission;
} }
/**
* Processes a buy transaction's resulting price movement.
* @param {Stock} stock - Stock being purchased
* @param {number} shares - Number of shares being transacted
* @param {PositionTypes} posType - Long or short position
*/
export function processBuyTransactionPriceMovement(stock: Stock, shares: number, posType: PositionTypes): void {
if (isNaN(shares) || shares <= 0 || !(stock instanceof Stock)) { return; }
// Cap the 'shares' arg at the stock's maximum shares. This'll prevent
// hanging in the case when a really big number is passed in
shares = Math.min(shares, stock.maxShares);
const isLong = (posType === PositionTypes.Long);
// If the number of shares doesn't trigger a price movement, just return
if (shares <= stock.shareTxUntilMovement) {
stock.shareTxUntilMovement -= shares;
return;
}
// Calculate how many iterations of price changes we need to account for
let remainingShares = shares - stock.shareTxUntilMovement;
let numIterations = 1 + Math.ceil(remainingShares / stock.shareTxForMovement);
// The initial cost calculation takes care of the first "iteration"
let currPrice = isLong ? stock.getAskPrice() : stock.getBidPrice();
for (let i = 1; i < numIterations; ++i) {
const amt = Math.min(stock.shareTxForMovement, remainingShares);
remainingShares -= amt;
// Price movement
if (isLong) {
currPrice *= (1 + (stock.priceMovementPerc / 100));
} else {
currPrice *= (1 - (stock.priceMovementPerc / 100));
}
}
stock.price = currPrice;
stock.shareTxUntilMovement = stock.shareTxForMovement - ((shares - stock.shareTxUntilMovement) % stock.shareTxForMovement);
}
/** /**
* Calculate the TOTAL amount of money gained from a sale (NOT net profit). This accounts * Calculate the TOTAL amount of money gained from a sale (NOT net profit). This accounts
* for spread, price movements, and commission. * for spread, price movements, and commission.
@ -115,3 +158,45 @@ export function getSellTransactionGain(stock: Stock, shares: number, posType: Po
return totalGain - CONSTANTS.StockMarketCommission; return totalGain - CONSTANTS.StockMarketCommission;
} }
/**
* Processes a sell transaction's resulting price movement
* @param {Stock} stock - Stock being sold
* @param {number} shares - Number of sharse being transacted
* @param {PositionTypes} posType - Long or short position
*/
export function processSellTransactionPriceMovement(stock: Stock, shares: number, posType: PositionTypes): void {
if (isNaN(shares) || shares <= 0 || !(stock instanceof Stock)) { return; }
// Cap the 'shares' arg at the stock's maximum shares. This'll prevent
// hanging in the case when a really big number is passed in
shares = Math.min(shares, stock.maxShares);
const isLong = (posType === PositionTypes.Long);
if (shares <= stock.shareTxUntilMovement) {
stock.shareTxUntilMovement -= shares;
return;
}
// Calculate how many iterations of price changes we need to accoutn for
let remainingShares = shares - stock.shareTxUntilMovement;
let numIterations = 1 + Math.ceil(remainingShares / stock.shareTxForMovement);
// The initial cost calculation takes care of the first "iteration"
let currPrice = isLong ? stock.getBidPrice() : stock.getAskPrice();
for (let i = 1; i < numIterations; ++i) {
const amt = Math.min(stock.shareTxForMovement, remainingShares);
remainingShares -= amt;
// Price movement
if (isLong) {
currPrice *= (1 - (stock.priceMovementPerc / 100));
} else {
currPrice *= (1 + (stock.priceMovementPerc / 100));
}
}
stock.price = currPrice;
stock.shareTxUntilMovement = stock.shareTxForMovement - ((shares - stock.shareTxUntilMovement) % stock.shareTxForMovement);
}