2018-10-25 01:19:48 +11:00
/*
* Irresistible Gaming (c) 2018
2018-12-12 19:21:57 +11:00
* Developed by Lorenc
2018-11-01 17:48:25 +11:00
* Module: cnr\features\stocks\stocks.pwn
* Purpose: stock market system for players
2018-10-25 01:19:48 +11:00
*/
/* ** Includes ** */
#include < YSI\y_hooks >
/* ** Definitions ** */
2018-11-07 12:07:12 +11:00
#define MAX_STOCKS ( 12 )
2018-10-25 08:32:18 +11:00
2018-11-01 17:48:25 +11:00
#define STOCK_REPORTING_PERIOD ( 86400 ) // 1 day
2018-10-25 08:32:18 +11:00
#define STOCK_REPORTING_PERIODS ( 30 ) // last 30 periods (days)
#define DIALOG_STOCK_MARKET 8923
2018-10-31 18:19:18 +11:00
#define DIALOG_PLAYER_STOCKS 8924
2018-10-26 11:44:45 +11:00
#define DIALOG_STOCK_MARKET_BUY 8925
2018-10-31 18:19:18 +11:00
#define DIALOG_STOCK_MARKET_SELL 8926
2018-11-06 15:24:24 +11:00
#define DIALOG_STOCK_MARKET_OPTIONS 8927
#define DIALOG_STOCK_MARKET_HOLDERS 8928
2018-11-07 12:07:12 +11:00
#define DIALOG_STOCK_MARKET_INFO 8929
#define DIALOG_STOCK_POPTIONS 8930
2018-11-07 19:24:00 +11:00
#define DIALOG_STOCK_MARKET_DONATE 8931
2018-10-26 11:44:45 +11:00
#define STOCK_MM_USER_ID ( 0 )
2018-10-25 01:19:48 +11:00
2018-11-02 09:57:08 +11:00
/* ** Constants ** */
static const Float: STOCK_MARKET_TRADING_FEE = 0.01; // trading fee (buy/sell) percentage as decimal
2018-11-07 19:24:00 +11:00
static const Float: STOCK_MARKET_PRICE_FLOOR = 1.0; // the smallest price a stock can ever be
2018-11-02 09:57:08 +11:00
static const Float: STOCK_DEFAULT_START_POOL = 0.0; // the default amount that the pool is set to upon a new report
2018-11-07 19:24:00 +11:00
static const Float: STOCK_DEFAULT_START_DONATIONS = 0.0; // the default amount that the donations pool starts at (useless)
static const Float: STOCK_DEFAULT_START_PRICE = 0.0; // the default starting price for a new report (useless)
2018-11-02 09:57:08 +11:00
2019-02-24 19:55:14 +11:00
static const Float: STOCK_BANKRUPTCY_PERCENT = -80.0; // the % that makes a company go bankrupt
static const Float: STOCK_BANKRUPTCY_LOSS_PERCENT = 20.0; // the % that is lost by the user when a stock drops
// static const Float: STOCK_BANKRUPTCY_MIN_PERCENT = -5.0; // the % that should make a company go bankrupt 'again' after it is $1
2018-10-25 01:19:48 +11:00
/* ** Variables ** */
2018-10-25 08:32:18 +11:00
enum E_STOCK_MARKET_DATA
{
2018-11-07 18:10:36 +11:00
E_NAME[ 64 ], E_SYMBOL[ 4 ], E_DESCRIPTION[ 64 ],
Float: E_MAX_SHARES, Float: E_POOL_FACTOR, Float: E_PRICE_FACTOR,
2018-11-07 12:07:12 +11:00
Float: E_AVAILABLE_SHARES,
2018-10-25 08:32:18 +11:00
// market maker
2018-11-07 18:10:36 +11:00
Float: E_IPO_SHARES, Float: E_IPO_PRICE, Float: E_MAX_PRICE
2018-10-25 08:32:18 +11:00
};
enum E_STOCK_MARKET_PRICE_DATA
{
2018-11-07 19:24:00 +11:00
E_SQL_ID, Float: E_PRICE, Float: E_POOL,
Float: E_DONATIONS
2018-10-25 08:32:18 +11:00
};
2018-10-28 22:34:48 +11:00
enum
{
2018-11-01 21:55:21 +11:00
E_STOCK_MINING_COMPANY,
2018-11-03 16:06:58 +11:00
E_STOCK_AMMUNATION,
E_STOCK_VEHICLE_DEALERSHIP,
E_STOCK_SUPA_SAVE,
E_STOCK_TRUCKING_COMPANY,
2018-11-03 16:21:56 +11:00
E_STOCK_CLUCKIN_BELL,
2018-11-05 10:23:38 +11:00
E_STOCK_PAWN_STORE,
E_STOCK_CASINO,
2018-11-07 19:51:46 +11:00
E_STOCK_GOVERNMENT,
2019-02-17 18:58:21 +11:00
E_STOCK_AVIATION,
E_STOCK_BATTLE_ROYAL_CENTER
2018-10-28 22:34:48 +11:00
};
2018-10-25 08:32:18 +11:00
static stock
g_stockMarketData [ MAX_STOCKS ] [ E_STOCK_MARKET_DATA ],
2018-10-31 09:42:10 +11:00
g_stockMarketReportData [ MAX_STOCKS ] [ STOCK_REPORTING_PERIODS ] [ E_STOCK_MARKET_PRICE_DATA ],
2018-10-31 18:19:18 +11:00
Iterator: stockmarkets < MAX_STOCKS >,
2018-11-07 12:07:12 +11:00
Float: p_PlayerShares [ MAX_PLAYERS ] [ MAX_STOCKS ],
bool: p_PlayerHasShare [ MAX_PLAYERS ] [ MAX_STOCKS char ]
2018-10-25 08:32:18 +11:00
;
2018-10-25 01:19:48 +11:00
/* ** Hooks ** */
2018-10-25 08:32:18 +11:00
hook OnScriptInit( )
{
2018-10-26 04:58:12 +11:00
// server variables
AddServerVariable( "stock_report_time", "0", GLOBAL_VARTYPE_INT );
2019-02-24 19:55:14 +11:00
2018-10-31 19:01:35 +11:00
AddServerVariable( "stock_trading_fees", "0.0", GLOBAL_VARTYPE_FLOAT );
2018-10-26 04:58:12 +11:00
2018-11-07 12:07:12 +11:00
// ID NAME SYMBOL MAX SHARES IPO_PRICE MAX_PRICE POOL_FACTOR PRICE_FACTOR DESCRIPTION
2018-11-07 19:24:00 +11:00
CreateStockMarket( E_STOCK_MINING_COMPANY, "The Mining Company", "MC", 100000.0, 25.0, 500.0, 100000.0, 10.0, "Exporting mined ores" );
2018-11-11 12:29:41 +11:00
CreateStockMarket( E_STOCK_AMMUNATION, "Ammu-Nation", "A", 100000.0, 25.0, 500.0, 100000.0, 10.0, "Purchases at Ammu-Nation/Weapon Dealers/Facilities" );
CreateStockMarket( E_STOCK_VEHICLE_DEALERSHIP, "Vehicle Dealership", "VD", 100000.0, 25.0, 1000.0, 100000.0, 20.0, "Car jacker exports, vehicle and component sales" );
CreateStockMarket( E_STOCK_SUPA_SAVE, "Supa-Save", "SS", 100000.0, 25.0, 500.0, 100000.0, 10.0, "Purchases at Supa-Save and 24/7" );
CreateStockMarket( E_STOCK_TRUCKING_COMPANY, "The Trucking Company", "TC", 100000.0, 50.0, 500.0, 100000.0, 20.0, "Completed trucking missions" );
CreateStockMarket( E_STOCK_CLUCKIN_BELL, "Cluckin' Bell", "CB", 100000.0, 50.0, 500.0, 100000.0, 20.0, "Exporting meth bags" );
CreateStockMarket( E_STOCK_PAWN_STORE, "Pawn Store", "PS", 100000.0, 50.0, 500.0, 100000.0, 20.0, "Exported stolen furniture and toy sales" );
CreateStockMarket( E_STOCK_CASINO, "Casino", "CAS", 100000.0, 990.0, 7500.0, 100000.0, 150.0, "Money lost by players gambling" );
CreateStockMarket( E_STOCK_GOVERNMENT, "Government", "GOV", 100000.0, 750.0, 7500.0, 100000.0, 150.0, "Fireman and LEO activities" );
CreateStockMarket( E_STOCK_AVIATION, "Elitas Travel", "ET", 100000.0, 50.0, 500.0, 100000.0, 20.0, "Completed pilot missions and intercity travel" );
2019-02-17 18:58:21 +11:00
CreateStockMarket( E_STOCK_BATTLE_ROYAL_CENTER, "Battle Royale Center", "BRC", 100000.0, 50.0, 500.0, 100000.0, 20.0, "Battle Royale minigame fees" );
2018-11-11 12:58:57 +11:00
2018-12-23 09:56:48 +11:00
// force inactive share holders to sell their shares on startup
2018-11-11 12:58:57 +11:00
mysql_tquery( dbHandle, sprintf( "SELECT so.* FROM `STOCK_OWNERS` so JOIN `USERS` u ON so.`USER_ID`=u.`ID` WHERE UNIX_TIMESTAMP()-u.`LASTLOGGED` > 604800 AND so.`USER_ID` != %d", STOCK_MM_USER_ID ), "StockMarket_ForceShareSale", "" );
return 1;
}
2018-12-23 09:56:48 +11:00
hook OnServerGameDayEnd( )
2018-11-11 12:58:57 +11:00
{
2018-12-23 09:56:48 +11:00
// force inactive share holders to sell their shares every game day
mysql_tquery( dbHandle, sprintf( "SELECT so.* FROM `STOCK_OWNERS` so JOIN `USERS` u ON so.`USER_ID`=u.`ID` WHERE UNIX_TIMESTAMP()-u.`LASTLOGGED` > 604800 AND so.`USER_ID` != %d", STOCK_MM_USER_ID ), "StockMarket_ForceShareSale", "" );
2018-10-26 04:58:12 +11:00
return 1;
}
hook OnServerUpdate( )
{
new current_time = GetServerTime( );
new last_reporting = GetServerVariableInt( "stock_report_time" );
// check if its reporting time
if ( current_time > last_reporting )
{
// reporting period
UpdateServerVariableInt( "stock_report_time", current_time + STOCK_REPORTING_PERIOD );
// create a new reporting period for every stock there
foreach ( new s : stockmarkets )
{
2018-10-31 09:42:10 +11:00
StockMarket_ReleaseDividends( s );
2018-10-26 04:58:12 +11:00
}
print( "Successfully created new reporting period for all online companies" );
}
2018-10-25 08:32:18 +11:00
return 1;
}
2018-11-02 09:57:08 +11:00
hook OnPlayerDisconnect( playerid, reason )
{
2018-11-07 12:07:12 +11:00
for ( new i = 0; i < MAX_STOCKS; i ++ ) {
2018-11-02 09:57:08 +11:00
p_PlayerShares[ playerid ] [ i ] = 0.0;
2018-11-07 12:07:12 +11:00
p_PlayerHasShare[ playerid ] { i } = false;
2018-11-02 09:57:08 +11:00
}
return 1;
}
2018-10-26 11:44:45 +11:00
hook OnDialogResponse( playerid, dialogid, response, listitem, inputtext[ ] )
{
if ( dialogid == DIALOG_STOCK_MARKET && response )
{
new
x = 0;
foreach ( new s : stockmarkets )
{
if ( x == listitem )
{
2018-11-06 15:24:24 +11:00
ShowPlayerStockMarketOptions( playerid, s );
2018-10-26 11:44:45 +11:00
SetPVarInt( playerid, "stockmarket_selection", s );
2018-10-31 18:19:18 +11:00
break;
}
x ++;
}
return 1;
}
2018-11-07 12:07:12 +11:00
else if ( ( dialogid == DIALOG_STOCK_MARKET_HOLDERS || dialogid == DIALOG_STOCK_MARKET_INFO ) && ! response )
2018-11-06 15:24:24 +11:00
{
new
stockid = GetPVarInt( playerid, "stockmarket_selection" );
if ( ! Iter_Contains( stockmarkets, stockid ) ) {
return SendError( playerid, "There was an error with the stock you were seeing, please try again." );
}
return ShowPlayerStockMarketOptions( playerid, stockid );
}
else if ( dialogid == DIALOG_STOCK_MARKET_OPTIONS )
{
if ( ! response ) {
return ShowPlayerStockMarket( playerid );
}
new
stockid = GetPVarInt( playerid, "stockmarket_selection" );
if ( ! Iter_Contains( stockmarkets, stockid ) ) {
return SendError( playerid, "There was an error with the stock you were seeing, please try again." );
}
switch ( listitem )
{
case 0: StockMarket_ShowBuySlip( playerid, stockid );
2018-11-07 19:24:00 +11:00
case 1: StockMarket_ShowDonationSlip( playerid, stockid );
case 2: mysql_tquery( dbHandle, sprintf( "SELECT s.*, u.`NAME`, u.`ONLINE` FROM `STOCK_OWNERS` s LEFT JOIN `USERS` u ON s.`USER_ID` = u.`ID` WHERE s.`STOCK_ID`=%d ORDER BY s.`SHARES` DESC", stockid ), "StockMarket_ShowShareholders", "dd", playerid, stockid );
case 3:
2018-11-07 12:07:12 +11:00
{
new
2018-11-07 12:21:47 +11:00
Float: market_cap = g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ] * g_stockMarketData[ stockid ] [ E_MAX_SHARES ] / 1000000.0;
2018-11-07 12:07:12 +11:00
format(
szLargeString, sizeof ( szLargeString ),
2018-11-11 12:29:41 +11:00
""COL_GREY"Stock Name\t%s\n"COL_GREY"Stock Symbol\t%s\n"COL_GREY"Current Price\t%s\n"COL_GREY"Max Shares\t%s\n"COL_GREY"Market Capitalization\t%sM\n"COL_GREY"Earnings (24H)\t%s\n"COL_GREY"Donations (24H)\t%s\n"COL_GREY"Affected By\t%s",
2018-11-07 12:07:12 +11:00
g_stockMarketData[ stockid ] [ E_NAME ], g_stockMarketData[ stockid ] [ E_SYMBOL ],
cash_format( g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ], .decimals = 2 ),
number_format( g_stockMarketData[ stockid ] [ E_MAX_SHARES ], .decimals = 0 ),
cash_format( market_cap, .decimals = 2 ),
2018-11-11 12:29:41 +11:00
cash_format( g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ] - g_stockMarketReportData[ stockid ] [ 0 ] [ E_DONATIONS ], .decimals = 0 ),
cash_format( g_stockMarketReportData[ stockid ] [ 0 ] [ E_DONATIONS ], .decimals = 0 ),
2018-11-07 12:07:12 +11:00
g_stockMarketData[ stockid ] [ E_DESCRIPTION ]
);
ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET_INFO, DIALOG_STYLE_TABLIST, ""COL_WHITE"Stock Market", szLargeString, "Close", "Back" );
}
2018-11-06 15:24:24 +11:00
}
return 1;
}
2018-11-07 19:24:00 +11:00
else if ( dialogid == DIALOG_STOCK_MARKET_DONATE )
{
new
stockid = GetPVarInt( playerid, "stockmarket_selection" );
if ( ! Iter_Contains( stockmarkets, stockid ) ) {
return SendError( playerid, "There was an error with the stock you were seeing, please try again." );
}
if ( ! response ) {
return ShowPlayerStockMarketOptions( playerid, stockid );
}
new
donation_amount;
if ( sscanf( inputtext, "d", donation_amount ) ) SendError( playerid, "You have not specified a valid donation amount." );
else if ( ! ( 100 <= donation_amount <= 10000000 ) ) SendError( playerid, "Please specify an amount between $100 and $10,000,000 to donate." );
else if ( donation_amount > GetPlayerCash( playerid ) ) SendError( playerid, "You do not have this much money on you." );
else
{
new
2018-12-04 00:17:48 +11:00
final_donation_amount = floatround( float( donation_amount ) * 0.5 );
2018-11-07 19:24:00 +11:00
// contribute to earnings
StockMarket_UpdateEarnings( stockid, final_donation_amount, .donation_amount = final_donation_amount );
// reduce player balance and alert
GivePlayerCash( playerid, -donation_amount );
SendServerMessage( playerid, "The shareholders of "COL_GREY"%s"COL_WHITE" appreciate your donation of "COL_GOLD"%s"COL_WHITE"!", g_stockMarketData[ stockid ] [ E_NAME ], cash_format( donation_amount ) );
return 1;
}
return StockMarket_ShowDonationSlip( playerid, stockid );
}
2018-10-31 18:19:18 +11:00
else if ( dialogid == DIALOG_PLAYER_STOCKS && response )
{
new
x = 0;
2018-11-07 12:07:12 +11:00
foreach ( new stockid : stockmarkets ) if ( p_PlayerHasShare[ playerid ] { stockid } )
2018-10-31 18:19:18 +11:00
{
if ( x == listitem )
{
2018-11-07 12:07:12 +11:00
ShowPlayerShareOptions( playerid, stockid );
2018-10-31 18:19:18 +11:00
SetPVarInt( playerid, "stockmarket_selling_stock", stockid );
break;
2018-10-26 11:44:45 +11:00
}
x ++;
}
return 1;
}
2018-11-07 12:07:12 +11:00
else if ( dialogid == DIALOG_STOCK_POPTIONS )
2018-10-31 18:19:18 +11:00
{
if ( ! response ) {
2018-11-07 12:07:12 +11:00
return ShowPlayerShares( playerid );
2018-10-31 18:19:18 +11:00
}
new
stockid = GetPVarInt( playerid, "stockmarket_selling_stock" );
if ( ! Iter_Contains( stockmarkets, stockid ) ) {
return SendError( playerid, "There was an error processing your sell order, please try again." );
}
2018-11-07 12:07:12 +11:00
switch ( listitem )
{
case 0: StockMarket_ShowSellSlip( playerid, stockid );
case 1: mysql_tquery( dbHandle, sprintf( "SELECT * FROM `STOCK_SELL_ORDERS` WHERE `USER_ID` = %d AND `STOCK_ID` = %d", GetPlayerAccountID( playerid ), stockid ), "StockMarket_OnCancelOrder", "d", playerid );
}
return 1;
}
else if ( dialogid == DIALOG_STOCK_MARKET_SELL )
{
new
stockid = GetPVarInt( playerid, "stockmarket_selling_stock" );
if ( ! Iter_Contains( stockmarkets, stockid ) ) {
return SendError( playerid, "There was an error processing your sell order, please try again." );
}
if ( ! response ) {
return ShowPlayerShareOptions( playerid, stockid );
}
2018-10-31 18:19:18 +11:00
new
input_shares;
if ( sscanf( inputtext, "d", input_shares ) ) SendError( playerid, "You must use a valid value." );
else if ( input_shares > floatround( p_PlayerShares[ playerid ] [ stockid ], floatround_floor ) ) SendError( playerid, "You do not have this many shares available to sell." );
2018-11-01 17:48:25 +11:00
else if ( input_shares < 1 ) SendError( playerid, "The minimum number of shares you can sell is 1." );
2018-10-31 18:19:18 +11:00
else
{
new
Float: shares = float( input_shares );
if ( ( p_PlayerShares[ playerid ] [ stockid ] -= shares ) < 0.1 ) {
mysql_single_query( sprintf( "DELETE FROM `STOCK_OWNERS` WHERE `USER_ID`=%d AND `STOCK_ID`=%d", GetPlayerAccountID( playerid ), stockid ) );
} else {
StockMarket_GiveShares( stockid, GetPlayerAccountID( playerid ), -shares );
}
StockMarket_UpdateSellOrder( stockid, GetPlayerAccountID( playerid ), shares );
2018-11-07 12:07:12 +11:00
SendServerMessage( playerid, "You have placed a sell order for %s shares at %s each. "COL_ORANGE"Use /shares to cancel sell orders", number_format( shares, .decimals = 2 ), cash_format( g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ], .decimals = 2 ) );
2018-10-31 18:19:18 +11:00
return 1;
}
return StockMarket_ShowSellSlip( playerid, stockid );
}
2018-10-26 11:44:45 +11:00
else if ( dialogid == DIALOG_STOCK_MARKET_BUY )
{
new
stockid = GetPVarInt( playerid, "stockmarket_selection" );
if ( response )
{
new
2018-10-31 18:19:18 +11:00
shares;
2018-10-26 11:44:45 +11:00
2018-10-31 18:19:18 +11:00
if ( sscanf( inputtext, "d", shares ) ) SendError( playerid, "You must use a valid value." );
2018-11-03 16:06:58 +11:00
else if ( shares < 1 ) SendError( playerid, "The minimum number of shares you can buy is 1." );
2018-10-26 11:44:45 +11:00
else
{
2018-10-31 18:19:18 +11:00
mysql_tquery( dbHandle, sprintf( "SELECT * FROM `STOCK_SELL_ORDERS` WHERE `STOCK_ID`=%d ORDER BY `LIST_DATE` ASC", stockid ), "StockMarket_OnPurchaseOrder", "ddf", playerid, stockid, float( shares ) );
2018-10-26 11:44:45 +11:00
return 1;
}
return StockMarket_ShowBuySlip( playerid, stockid );
}
else
{
2018-11-06 15:24:24 +11:00
return ShowPlayerStockMarket( playerid );
2018-10-26 11:44:45 +11:00
}
}
return 1;
}
2018-10-25 08:32:18 +11:00
/* ** SQL Thread ** */
thread Stock_UpdateReportingPeriods( stockid )
{
new
rows = cache_get_row_count( );
if ( rows )
{
for ( new row = 0; row < rows; row ++ )
{
2018-10-31 09:42:10 +11:00
g_stockMarketReportData[ stockid ] [ row ] [ E_SQL_ID ] = cache_get_field_content_int( row, "ID" );
g_stockMarketReportData[ stockid ] [ row ] [ E_POOL ] = cache_get_field_content_float( row, "POOL" );
2018-11-07 19:24:00 +11:00
g_stockMarketReportData[ stockid ] [ row ] [ E_DONATIONS ] = cache_get_field_content_float( row, "DONATIONS" );
2018-10-31 18:19:18 +11:00
g_stockMarketReportData[ stockid ] [ row ] [ E_PRICE ] = cache_get_field_content_float( row, "PRICE" );
2018-10-25 08:32:18 +11:00
}
}
2018-10-26 11:44:45 +11:00
else // no historical reporting data, restock the market maker
{
2018-11-07 19:51:46 +11:00
// create 3 reports for the company using the IPO price ... this way the price is not $0
2018-11-02 09:57:08 +11:00
for ( new i = 0; i < 3; i ++ ) {
2018-11-07 19:51:46 +11:00
StockMarket_ReleaseDividends( stockid, .is_ipo = true );
2018-10-31 19:04:59 +11:00
}
2018-10-26 11:44:45 +11:00
2018-10-31 18:19:18 +11:00
// put market maker shares on the market
StockMarket_UpdateSellOrder( stockid, STOCK_MM_USER_ID, g_stockMarketData[ stockid ] [ E_IPO_SHARES ] );
2018-10-26 11:44:45 +11:00
}
2018-10-25 08:32:18 +11:00
return 1;
}
2018-11-07 19:51:46 +11:00
thread StockMarket_InsertReport( stockid, Float: default_start_pool, Float: default_start_price, Float: default_donation_pool, bool: is_ipo )
2018-10-26 04:58:12 +11:00
{
2019-02-24 19:55:14 +11:00
new Float: before_price = g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ];
2018-11-06 14:02:11 +11:00
// set the new price of the company [TODO: use parabola for factor difficulty?]
2018-11-07 19:24:00 +11:00
new Float: new_price = ( g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ] / g_stockMarketData[ stockid ] [ E_POOL_FACTOR ] ) * g_stockMarketData[ stockid ] [ E_PRICE_FACTOR ] + STOCK_MARKET_PRICE_FLOOR;
2018-11-02 09:57:08 +11:00
2018-11-07 18:10:36 +11:00
// reduce price of shares depending on shares available from the start (200K max shares from IPO 100k means 50% reduction)
new_price *= floatpower( 0.5, g_stockMarketData[ stockid ] [ E_MAX_SHARES ] / g_stockMarketData[ stockid ] [ E_IPO_SHARES ] - 1.0 );
// check if price exceeds maximum price
2018-11-02 09:57:08 +11:00
if ( new_price > g_stockMarketData[ stockid ] [ E_MAX_PRICE ] ) { // dont want wild market caps
new_price = g_stockMarketData[ stockid ] [ E_MAX_PRICE ];
}
2018-11-07 18:10:36 +11:00
// force a minimum of $1 per share
2018-11-07 19:24:00 +11:00
if ( new_price < STOCK_MARKET_PRICE_FLOOR ) {
new_price = STOCK_MARKET_PRICE_FLOOR;
2018-11-02 09:57:08 +11:00
}
2018-11-07 19:51:46 +11:00
// check if its an ipo... if it is then set to ipo price
if ( is_ipo ) {
new_price = g_stockMarketData[ stockid ] [ E_IPO_PRICE ];
}
2019-02-24 19:55:14 +11:00
// bankrupt motherf**kers
new Float: price_change = ( ( new_price / before_price ) - 1.0 ) * 100.0;
// stock is above the floor
if ( price_change <= STOCK_BANKRUPTCY_PERCENT )
{
format( szLargeString, sizeof( szLargeString ),
"SELECT USER_ID, SUM(HELD_SHARES) AS HELD_SHARES, SUM(OWNED_SHARES) AS OWNED_SHARES " \
"FROM ( " \
" (SELECT STOCK_ID, USER_ID, SHARES AS HELD_SHARES, 0 AS OWNED_SHARES FROM STOCK_SELL_ORDERS t1 WHERE STOCK_ID=%d) " \
" UNION ALL " \
" (SELECT STOCK_ID, USER_ID, 0 AS HELD_SHARES, SHARES AS OWNED_SHARES FROM STOCK_OWNERS t2 WHERE STOCK_ID=%d) " \
") t_union " \
"GROUP BY USER_ID",
stockid, stockid
);
mysql_tquery( dbHandle, szLargeString, "StockMarket_PanicSell", "d", stockid );
print(szLargeString);
}
// full bankruptcy if a stock is $1 (UNTESTED!)
// else if ( price_change <= STOCK_BANKRUPTCY_MIN_PERCENT && new_price == STOCK_MARKET_PRICE_FLOOR )
// {
// // purge all user association to the stock
// mysql_single_query( sprintf( "DELETE FROM `STOCK_SELL_ORDERS` WHERE `STOCK_ID`=%d", stockid ) );
// mysql_single_query( sprintf( "DELETE FROM `STOCK_OWNERS` WHERE `STOCK_ID`=%d", stockid ) );
// // allow market maker to sell the shares again at IPO price
// StockMarket_UpdateSellOrder( stockid, STOCK_MM_USER_ID, g_stockMarketData[ stockid ] [ E_IPO_SHARES ] );
// new_price = g_stockMarketData[ stockid ] [ E_IPO_PRICE ];
// // inform everyone of the sale
// SendClientMessageToAllFormatted( -1, ""COL_GREY"[STOCKS]"COL_WHITE" %s has went bankrupt, all shareholders have suffered a loss!", g_stockMarketData[ stockid ] [ E_NAME ] );
// printf( "[STOCK MAJOR BANKRUPTCY] %s(%d) has went majorly bankrupt.", g_stockMarketData[ stockid ] [ E_NAME ], stockid );
// }
2018-11-07 18:10:36 +11:00
// set the new price of the asset
2018-11-02 09:57:08 +11:00
g_stockMarketReportData[ stockid ] [ 0 ] [ E_PRICE ] = new_price;
mysql_single_query( sprintf( "UPDATE `STOCK_REPORTS` SET `PRICE` = %f WHERE `ID` = %d", g_stockMarketReportData[ stockid ] [ 0 ] [ E_PRICE ], g_stockMarketReportData[ stockid ] [ 0 ] [ E_SQL_ID ] ) );
// store temporary stock info
new temp_stock_price_data[ MAX_STOCKS ] [ STOCK_REPORTING_PERIODS ] [ E_STOCK_MARKET_PRICE_DATA ];
temp_stock_price_data = g_stockMarketReportData;
// shift all report data by one
for ( new r = 0; r < sizeof( g_stockMarketReportData[ ] ) - 2; r ++ ) {
g_stockMarketReportData[ stockid ] [ r + 1 ] [ E_SQL_ID ] = temp_stock_price_data[ stockid ] [ r ] [ E_SQL_ID ];
g_stockMarketReportData[ stockid ] [ r + 1 ] [ E_POOL ] = temp_stock_price_data[ stockid ] [ r ] [ E_POOL ];
2018-11-07 19:24:00 +11:00
g_stockMarketReportData[ stockid ] [ r + 1 ] [ E_DONATIONS ] = temp_stock_price_data[ stockid ] [ r ] [ E_DONATIONS ];
2018-11-02 09:57:08 +11:00
g_stockMarketReportData[ stockid ] [ r + 1 ] [ E_PRICE ] = temp_stock_price_data[ stockid ] [ r ] [ E_PRICE ];
}
// reset earnings
2018-10-31 09:42:10 +11:00
g_stockMarketReportData[ stockid ] [ 0 ] [ E_SQL_ID ] = cache_insert_id( );
2018-11-02 09:57:08 +11:00
g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ] = default_start_pool;
2018-11-07 19:24:00 +11:00
g_stockMarketReportData[ stockid ] [ 0 ] [ E_DONATIONS ] = default_donation_pool;
2018-11-02 09:57:08 +11:00
g_stockMarketReportData[ stockid ] [ 0 ] [ E_PRICE ] = default_start_price;
2018-10-26 04:58:12 +11:00
return 1;
}
2019-02-24 19:55:14 +11:00
thread StockMarket_PanicSell( stockid )
{
new
rows = cache_get_row_count( );
if ( rows )
{
new
Float: global_shares_forfeited = 0.0;
// remove the percentage of stock from the user
for ( new row = 0; row < rows; row ++ )
{
new user_id = cache_get_field_content_int( row, "USER_ID" );
2019-02-24 19:58:26 +11:00
if ( user_id == STOCK_MM_USER_ID )
continue; // ignore market maker account
2019-02-24 19:55:14 +11:00
new Float: held_shares = cache_get_field_content_float( row, "HELD_SHARES" );
new Float: owned_shares = cache_get_field_content_float( row, "OWNED_SHARES" );
new Float: player_shares_forfeited = floatround( ( held_shares + owned_shares ) * STOCK_BANKRUPTCY_LOSS_PERCENT / 100.0 );
// the amount forfeitted succeeds the amount in sale orders ... remove them
if ( ( held_shares - player_shares_forfeited ) <= 0.0 ) {
printf( "Strip sell orders if there is any for user %d, stock %d", user_id, stockid );
mysql_single_query( sprintf( "DELETE FROM `STOCK_SELL_ORDERS` WHERE `STOCK_ID`=%d AND `USER_ID`=%d", stockid, user_id ) );
player_shares_forfeited -= held_shares;
global_shares_forfeited += held_shares;
} else {
printf( "Deduct sell orders if there is any for user %d, stock %d", user_id, stockid );
mysql_single_query( sprintf( "UPDATE `STOCK_SELL_ORDERS` SET `SHARES`=%f WHERE `STOCK_ID`=%d AND `USER_ID`=%d", held_shares - player_shares_forfeited, stockid, user_id ) );
global_shares_forfeited += player_shares_forfeited;
player_shares_forfeited = 0.0;
}
// the amount forfeitted succeeds the amount in holdings ... remove them too
if ( ( owned_shares - player_shares_forfeited ) <= 0.0 ) {
printf( "Strip owners if there is any for user %d, stock %d", user_id, stockid );
mysql_single_query( sprintf( "DELETE FROM `STOCK_OWNERS` WHERE `STOCK_ID`=%d AND `USER_ID`=%d", stockid, user_id ) );
global_shares_forfeited += owned_shares;
} else {
printf( "Deduct owners if there is any for user %d, stock %d", user_id, stockid );
mysql_single_query( sprintf( "UPDATE `STOCK_OWNERS` SET `SHARES`=%f WHERE `STOCK_ID`=%d AND `USER_ID`=%d", owned_shares - player_shares_forfeited, stockid, user_id ) );
global_shares_forfeited += player_shares_forfeited;
}
}
// allow market maker to sell the shares again at IPO price
StockMarket_UpdateSellOrder( stockid, STOCK_MM_USER_ID, floatround( global_shares_forfeited ) );
// inform everyone of the sale
SendClientMessageToAllFormatted( -1, ""COL_GREY"[STOCKS]"COL_WHITE" %s has penalized its shareholders and has %d shares available for sale.", g_stockMarketData[ stockid ] [ E_NAME ], floatround( global_shares_forfeited ) );
printf( "[STOCK MAJOR BANKRUPTCY] %s(%d) has went partly bankrupt.", g_stockMarketData[ stockid ] [ E_NAME ], stockid );
}
return 1;
}
2018-10-26 11:44:45 +11:00
thread StockMarket_OnPurchaseOrder( playerid, stockid, Float: shares )
{
new
rows = cache_get_row_count( );
if ( ! rows ) {
return SendError( playerid, "This stock has no available shares for sale." );
}
2018-10-31 18:19:18 +11:00
// check if quantity is valid
new
Float: available_quantity = 0.0;
for ( new r = 0; r < rows; r ++ ) {
available_quantity += cache_get_field_content_float( r, "SHARES" );
}
if ( shares > available_quantity ) {
return SendError( playerid, "There are not that many shares available for sale." ), StockMarket_ShowBuySlip( playerid, stockid ), 1;
}
2018-10-26 11:44:45 +11:00
// check if the player has the money for the purchase
2018-10-31 09:42:10 +11:00
new Float: ask_price = g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ];
2018-10-26 11:44:45 +11:00
new purchase_cost = floatround( ask_price * shares );
2018-10-31 19:01:35 +11:00
new Float: purchase_fee = ask_price * shares * STOCK_MARKET_TRADING_FEE;
UpdateServerVariableFloat( "stock_trading_fees", GetServerVariableFloat( "stock_trading_fees" ) + ( purchase_fee / 1000.0 ) );
new purchase_cost_plus_fee = purchase_cost + floatround( purchase_fee );
if ( GetPlayerCash( playerid ) < purchase_cost_plus_fee ) {
return SendError( playerid, "You need at least %s to purchase this many shares.", cash_format( purchase_cost_plus_fee ) ), StockMarket_ShowBuySlip( playerid, stockid ), 1;
2018-10-26 11:44:45 +11:00
}
2018-10-31 18:19:18 +11:00
new
Float: amount_remaining = shares;
2018-10-26 11:44:45 +11:00
2018-10-31 18:19:18 +11:00
for ( new row = 0; row < rows; row ++ )
{
new sell_order_user_id = cache_get_field_content_int( row, "USER_ID" );
new Float: sell_order_shares = cache_get_field_content_float( row, "SHARES" );
2018-10-26 11:44:45 +11:00
2018-10-31 18:19:18 +11:00
// check if seller is online
new
sellerid;
foreach ( sellerid : Player ) if ( GetPlayerAccountID( sellerid ) == sell_order_user_id ) {
break;
}
new Float: sold_shares = amount_remaining > sell_order_shares ? sell_order_shares : amount_remaining;
2018-10-31 19:01:35 +11:00
2018-10-31 20:05:47 +11:00
StockMarket_CreateTradeLog( stockid, GetPlayerAccountID( playerid ), sell_order_user_id, sold_shares, ask_price );
2018-10-31 19:01:35 +11:00
new Float: sold_amount_fee = sold_shares * ask_price * STOCK_MARKET_TRADING_FEE;
UpdateServerVariableFloat( "stock_trading_fees", GetServerVariableFloat( "stock_trading_fees" ) + ( sold_amount_fee / 1000.0 ) );
new sold_amount_minus_fee = floatround( sold_shares * ask_price - sold_amount_fee );
2018-10-31 18:19:18 +11:00
2018-11-05 10:33:37 +11:00
if ( 0 <= sellerid < MAX_PLAYERS && Iter_Contains( Player, sellerid ) && IsPlayerLoggedIn( sellerid ) ) {
2018-10-31 19:01:35 +11:00
GivePlayerBankMoney( sellerid, sold_amount_minus_fee ), Beep( sellerid );
2018-11-07 19:24:00 +11:00
SendServerMessage( sellerid, "You have sold %s %s shares to %s(%d) for "COL_GOLD"%s"COL_WHITE" (plus %0.1f%s fee)!", number_format( sold_shares, .decimals = 0 ), g_stockMarketData[ stockid ] [ E_NAME ], ReturnPlayerName( playerid ), playerid, cash_format( sold_amount_minus_fee ), STOCK_MARKET_TRADING_FEE * 100.0, "%%" );
2018-10-31 18:19:18 +11:00
} else {
2018-10-31 19:01:35 +11:00
mysql_single_query( sprintf( "UPDATE `USERS` SET `BANKMONEY` = `BANKMONEY` + %d WHERE `ID` = %d", sold_amount_minus_fee, sell_order_user_id ) );
2018-10-31 18:19:18 +11:00
}
// remove the sell order if there is little to no shares available
if ( sell_order_shares - amount_remaining < 1.0 )
{
// get rid of this sell order
mysql_single_query( sprintf( "DELETE FROM `STOCK_SELL_ORDERS` WHERE `USER_ID`=%d and `STOCK_ID`=%d", sell_order_user_id, stockid ) );
// deduct the sell order amount from amount remaining
amount_remaining -= sell_order_shares;
2018-11-07 12:07:12 +11:00
// remove number of available shares
g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] -= sell_order_shares;
2018-10-31 18:19:18 +11:00
}
else
{
// reduce sell order quantity
StockMarket_UpdateSellOrder( stockid, sell_order_user_id, -amount_remaining );
// the player's buy order was filled in the single sell order ... prevent updating
break;
}
}
2018-10-26 11:44:45 +11:00
// increment the players shares
StockMarket_GiveShares( stockid, GetPlayerAccountID( playerid ), shares );
// reduce player balance and alert
2018-10-31 19:01:35 +11:00
GivePlayerCash( playerid, -purchase_cost_plus_fee );
2018-11-07 19:24:00 +11:00
SendServerMessage( playerid, "You have purchased %s shares of %s (@ %s/ea) for %s. (inc. %0.1f%s fee)", number_format( shares, .decimals = 0 ), g_stockMarketData[ stockid ] [ E_NAME ], cash_format( ask_price, .decimals = 2 ), cash_format( purchase_cost_plus_fee ), STOCK_MARKET_TRADING_FEE * 100.0, "%%" );
2018-10-26 11:44:45 +11:00
return 1;
}
thread StockMarket_OnShowBuySlip( playerid, stockid )
{
new
rows = cache_get_row_count( );
if ( ! rows ) {
2018-10-31 18:19:18 +11:00
return SendError( playerid, "This stock does not currently have any shares available to buy." );
2018-10-26 11:44:45 +11:00
}
new
2018-10-31 18:19:18 +11:00
Float: available_quantity = cache_get_field_content_float( 0, "SALE_SHARES" );
2018-10-26 11:44:45 +11:00
format(
szBigString, sizeof ( szBigString ),
""COL_WHITE"You can buy shares of %s for "COL_GREEN"%s"COL_WHITE" each.\n\nThere are %s available shares to buy.",
g_stockMarketData[ stockid ] [ E_NAME ],
2018-10-31 09:42:10 +11:00
cash_format( g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ], .decimals = 2 ),
2018-11-07 19:24:00 +11:00
number_format( available_quantity, .decimals = 0 )
2018-10-26 11:44:45 +11:00
);
2018-11-07 12:07:12 +11:00
ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET_BUY, DIALOG_STYLE_INPUT, ""COL_WHITE"Stock Market", szBigString, "Buy", "Back" );
2018-10-26 11:44:45 +11:00
return 1;
}
thread StockMarket_OnShowShares( playerid )
{
new
rows = cache_get_row_count( );
if ( ! rows ) {
2018-10-31 18:19:18 +11:00
return SendError( playerid, "You are not holding any shares of any company." );
2018-10-26 11:44:45 +11:00
}
2018-11-07 12:07:12 +11:00
szLargeString = ""COL_WHITE"Stock\t"COL_WHITE"Active Shares\t"COL_WHITE"Shares Being Sold\t"COL_GREEN"Value ($)\n";
2018-10-26 11:44:45 +11:00
for ( new row = 0; row < rows; row ++ )
{
new
2018-11-02 09:57:08 +11:00
stockid = cache_get_field_content_int( row, "STOCK_ID" );
2018-10-26 11:44:45 +11:00
if ( Iter_Contains( stockmarkets, stockid ) )
{
2018-11-07 12:07:12 +11:00
new Float: shares = cache_get_field_content_float( row, "OWNED_SHARES" );
new Float: held_shares = cache_get_field_content_float( row, "HELD_SHARES" );
2018-10-26 11:44:45 +11:00
format(
szLargeString, sizeof( szLargeString ),
2018-11-07 12:07:12 +11:00
"%s%s (%s)\t%s (%0.2f%%)\t%s (%0.2f%%)\t"COL_GREEN"%s\n",
2018-10-26 11:44:45 +11:00
szLargeString,
g_stockMarketData[ stockid ] [ E_NAME ],
g_stockMarketData[ stockid ] [ E_SYMBOL ],
2018-11-07 12:07:12 +11:00
number_format( shares, .decimals = 0 ),
2018-11-02 09:57:08 +11:00
( shares / g_stockMarketData[ stockid ] [ E_MAX_SHARES ] ) * 100.0,
2018-11-07 12:07:12 +11:00
number_format( held_shares, .decimals = 0 ),
( held_shares / g_stockMarketData[ stockid ] [ E_MAX_SHARES ] ) * 100.0,
cash_format( floatround( ( shares + held_shares ) * g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ] ) )
2018-10-26 11:44:45 +11:00
);
2018-10-31 18:19:18 +11:00
// store player stocks in a variable for reference
p_PlayerShares[ playerid ] [ stockid ] = shares;
2018-11-07 12:07:12 +11:00
p_PlayerHasShare[ playerid ] { stockid } = true;
2018-10-26 11:44:45 +11:00
}
2018-11-09 18:52:48 +11:00
else
{
p_PlayerHasShare[ playerid ] { stockid } = false;
}
2018-10-26 11:44:45 +11:00
}
2018-11-07 12:07:12 +11:00
return ShowPlayerDialog( playerid, DIALOG_PLAYER_STOCKS, DIALOG_STYLE_TABLIST_HEADERS, ""COL_WHITE"Stock Market", szLargeString, "Select", "Close" ), 1;
2018-10-26 11:44:45 +11:00
}
2018-11-07 19:51:46 +11:00
thread Stock_OnDividendPayout( stockid, bool: is_ipo )
2018-10-31 09:42:10 +11:00
{
new
rows = cache_get_row_count( );
// pay out existing shareholders
if ( rows )
{
new
Float: total_shares = g_stockMarketData[ stockid ] [ E_MAX_SHARES ];
for ( new row = 0; row < rows; row ++ )
{
new account_id = cache_get_field_content_int( row, "USER_ID" );
new Float: shares_owned = cache_get_field_content_float( row, "SHARES" );
new Float: dividend_rate = shares_owned / total_shares;
new dividend_payout = floatround( g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ] * dividend_rate );
new
shareholder;
foreach ( shareholder : Player ) if ( GetPlayerAccountID( shareholder ) == account_id ) {
break;
}
if ( 0 <= shareholder < MAX_PLAYERS && Iter_Contains( Player, shareholder ) ) {
GivePlayerBankMoney( shareholder, dividend_payout ), Beep( shareholder );
2018-11-01 17:48:25 +11:00
SendServerMessage( shareholder, "You have been paid a "COL_GOLD"%s"COL_WHITE" dividend (%0.2f%s) for owning %s!", cash_format( dividend_payout ), dividend_rate * 100.0, "%%", g_stockMarketData[ stockid ] [ E_NAME ] );
2018-10-31 09:42:10 +11:00
} else {
mysql_single_query( sprintf( "UPDATE `USERS` SET `BANKMONEY` = `BANKMONEY` + %d WHERE `ID` = %d", dividend_payout, account_id ) );
}
}
}
2018-11-02 09:57:08 +11:00
// insert to database a new report
2018-11-07 19:24:00 +11:00
mysql_format( dbHandle, szBigString, sizeof ( szBigString ), "INSERT INTO `STOCK_REPORTS` (`STOCK_ID`, `POOL`, `DONATIONS`, `PRICE`) VALUES (%d, %f, %f, %f)", stockid, STOCK_DEFAULT_START_POOL, STOCK_DEFAULT_START_DONATIONS, STOCK_DEFAULT_START_PRICE );
2018-11-07 19:51:46 +11:00
mysql_tquery( dbHandle, szBigString, "StockMarket_InsertReport", "dfffd", stockid, STOCK_DEFAULT_START_POOL, STOCK_DEFAULT_START_DONATIONS, STOCK_DEFAULT_START_PRICE, bool: is_ipo );
2018-10-31 09:42:10 +11:00
return 1;
}
thread Stock_UpdateMaximumShares( stockid )
{
new
rows = cache_get_row_count( );
2018-10-31 18:34:50 +11:00
if ( rows )
{
2018-11-07 12:07:12 +11:00
g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] = cache_get_field_content_float( 0, "SHARES_HELD" );
g_stockMarketData[ stockid ] [ E_MAX_SHARES ] = g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] + cache_get_field_content_float( 0, "SHARES_OWNED" );
2018-10-31 18:34:50 +11:00
// rows shown but still showing as 0 maximum shares? set it to the ipo issued amount
if ( ! g_stockMarketData[ stockid ] [ E_MAX_SHARES ] )
{
2018-11-07 12:07:12 +11:00
g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] = g_stockMarketData[ stockid ] [ E_IPO_SHARES ];
2018-10-31 18:34:50 +11:00
g_stockMarketData[ stockid ] [ E_MAX_SHARES ] = g_stockMarketData[ stockid ] [ E_IPO_SHARES ];
}
}
else
{
2018-11-07 12:07:12 +11:00
g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] = g_stockMarketData[ stockid ] [ E_IPO_SHARES ];
2018-10-31 09:42:10 +11:00
g_stockMarketData[ stockid ] [ E_MAX_SHARES ] = g_stockMarketData[ stockid ] [ E_IPO_SHARES ];
}
return 1;
}
2018-10-31 18:19:18 +11:00
thread StockMarket_OnCancelOrder( playerid )
{
new
rows = cache_get_row_count( );
if ( rows )
{
new
player_account = GetPlayerAccountID( playerid );
for ( new row = 0; row < rows; row ++ )
{
new stockid = cache_get_field_content_int( row, "STOCK_ID" );
new Float: shares = cache_get_field_content_float( row, "SHARES" );
2018-11-07 12:07:12 +11:00
g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] -= shares;
2018-10-31 18:19:18 +11:00
mysql_single_query( sprintf( "DELETE FROM `STOCK_SELL_ORDERS` WHERE `STOCK_ID`=%d AND `USER_ID`=%d", stockid, player_account ) );
StockMarket_GiveShares( stockid, player_account, shares );
2018-10-31 18:34:50 +11:00
2018-11-07 19:24:00 +11:00
SendServerMessage( playerid, "You have cancelled your order of to sell %s shares of %s.", number_format( shares, .decimals = 0 ), g_stockMarketData[ stockid ] [ E_NAME ] );
2018-10-31 18:19:18 +11:00
}
return 1;
}
else
{
2018-11-07 12:07:12 +11:00
return ShowPlayerShares( playerid ), SendError( playerid, "You don't have any sell orders for this stock to cancel." ), 1;
2018-10-31 18:19:18 +11:00
}
}
2018-11-06 15:24:24 +11:00
thread StockMarket_ShowShareholders( playerid, stockid )
2018-10-25 08:32:18 +11:00
{
2018-11-06 15:24:24 +11:00
new rows = cache_get_row_count( );
2018-10-25 08:32:18 +11:00
2018-11-06 15:24:24 +11:00
// format dialog title
2018-11-06 15:40:20 +11:00
szHugeString = ""COL_WHITE"User\t"COL_WHITE"Shares\t"COL_WHITE"Percentage (%)\n";
2018-11-06 15:24:24 +11:00
// track the shares that are held by players
new Float: out_standing_shares = g_stockMarketData[ stockid ] [ E_MAX_SHARES ];
// show all the shareholders
if ( rows )
2018-10-25 08:32:18 +11:00
{
2018-11-06 15:24:24 +11:00
new
player_name[ 24 ];
2018-10-25 08:32:18 +11:00
2018-11-06 15:24:24 +11:00
for ( new row = 0; row < rows; row ++ )
{
cache_get_field_content( row, "NAME", player_name );
new is_online = cache_get_field_content_int( row, "ONLINE" );
new Float: shares = cache_get_field_content_float( row, "SHARES" );
out_standing_shares -= shares;
2018-11-06 15:40:20 +11:00
format( szHugeString, sizeof ( szHugeString ), "%s%s%s\t%s\t%s%%\n", szHugeString, is_online ? COL_GREEN : COL_WHITE, player_name, number_format( shares, .decimals = 0 ), number_format( shares / g_stockMarketData[ stockid ] [ E_MAX_SHARES ] * 100.0, .decimals = 3 ) );
2018-11-06 15:24:24 +11:00
}
2018-10-25 08:32:18 +11:00
}
2018-11-06 15:24:24 +11:00
// tell players the shares tied up in sell orders
if ( out_standing_shares > 0.0 ) {
2018-11-06 15:40:20 +11:00
format( szHugeString, sizeof ( szHugeString ), "%s{666666}Held In Sell Orders\t{666666}%s\t{666666}%s%%\n", szHugeString, number_format( out_standing_shares, .decimals = 0 ), number_format( out_standing_shares / g_stockMarketData[ stockid ] [ E_MAX_SHARES ] * 100.0, .decimals = 3 ) );
2018-11-06 15:24:24 +11:00
}
// format dialog
2018-11-06 15:40:20 +11:00
ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET_HOLDERS, DIALOG_STYLE_TABLIST_HEADERS, ""COL_WHITE"Stock Market - Shareholders", szHugeString, "Close", "Back" );
2018-11-06 15:24:24 +11:00
return 1;
}
2018-12-23 09:56:48 +11:00
thread StockMarket_ForceShareSale( )
{
new
rows = cache_get_row_count( );
if ( rows )
{
for ( new row = 0; row < rows; row ++ )
{
new accountid = cache_get_field_content_int( row, "USER_ID" );
new stockid = cache_get_field_content_int( row, "STOCK_ID" );
new Float: shares = cache_get_field_content_float( row, "SHARES" );
mysql_single_query( sprintf( "DELETE FROM `STOCK_OWNERS` WHERE `USER_ID`=%d AND `STOCK_ID`=%d", accountid, stockid ) );
StockMarket_UpdateSellOrder( stockid, accountid, shares );
printf("Inactive shares (user id: %d, stock id: %s, shares: %f)", accountid, g_stockMarketData[ stockid ] [ E_NAME ], shares);
}
}
return 1;
}
2018-11-06 15:24:24 +11:00
/* ** Command ** */
CMD:stocks( playerid, params[ ] ) return cmd_stockmarkets( playerid, params );
CMD:stockmarkets( playerid, params[ ] )
{
2018-10-31 09:42:10 +11:00
SendServerMessage( playerid, "The stock market will payout dividends in %s.", secondstotime( GetServerVariableInt( "stock_report_time" ) - GetServerTime( ) ) );
2018-11-06 15:24:24 +11:00
return ShowPlayerStockMarket( playerid );
2018-10-26 11:44:45 +11:00
}
CMD:shares( playerid, params[ ] )
{
2018-11-07 12:07:12 +11:00
return ShowPlayerShares( playerid );
}
2018-11-07 18:10:36 +11:00
CMD:astock( playerid, params[ ] )
{
if ( ! IsPlayerAdmin( playerid ) || ! IsPlayerLeadMaintainer( playerid ) ) {
return 0;
2018-10-31 18:19:18 +11:00
}
2018-11-07 18:10:36 +11:00
if ( strmatch( params, "update maxshares" ) ) {
foreach ( new s : stockmarkets ) {
UpdateStockMaxShares( s );
}
return SendServerMessage( playerid, "Max shares has been updated for all stocks." );
}
else if ( strmatch( params, "new report" ) ) {
foreach ( new s : stockmarkets ) {
StockMarket_ReleaseDividends( s );
}
UpdateServerVariableInt( "stock_report_time", GetServerTime( ) + STOCK_REPORTING_PERIOD );
return SendServerMessage( playerid, "All stocks have now had their dividends distributed." );
}
2018-12-23 09:56:48 +11:00
return SendUsage( playerid, "/astock [UPDATE MAXSHARES/NEW REPORT]" );
2018-10-25 08:32:18 +11:00
}
2018-10-25 01:19:48 +11:00
/* ** Functions ** */
2018-11-07 12:07:12 +11:00
static stock CreateStockMarket( stockid, const name[ 64 ], const symbol[ 4 ], Float: ipo_shares, Float: ipo_price, Float: max_price, Float: pool_factor, Float: price_factor, const description[ 72 ] )
2018-10-25 08:32:18 +11:00
{
if ( ! Iter_Contains( stockmarkets, stockid ) )
{
strcpy( g_stockMarketData[ stockid ] [ E_NAME ], name );
strcpy( g_stockMarketData[ stockid ] [ E_SYMBOL ], symbol );
2018-11-07 12:07:12 +11:00
strcpy( g_stockMarketData[ stockid ] [ E_DESCRIPTION ], description );
2018-10-25 08:32:18 +11:00
2018-10-26 11:44:45 +11:00
g_stockMarketData[ stockid ] [ E_IPO_SHARES ] = ipo_shares;
g_stockMarketData[ stockid ] [ E_IPO_PRICE ] = ipo_price;
2018-10-31 18:19:18 +11:00
g_stockMarketData[ stockid ] [ E_MAX_PRICE ] = max_price;
2018-11-06 13:52:23 +11:00
g_stockMarketData[ stockid ] [ E_POOL_FACTOR ] = pool_factor;
g_stockMarketData[ stockid ] [ E_PRICE_FACTOR ] = price_factor;
2018-10-25 08:32:18 +11:00
// reset stock price information
2018-10-31 09:42:10 +11:00
for ( new r = 0; r < sizeof( g_stockMarketReportData[ ] ); r ++ ) {
2018-10-31 18:19:18 +11:00
g_stockMarketReportData[ stockid ] [ r ] [ E_POOL ] = 0.0;
2018-11-07 19:24:00 +11:00
g_stockMarketReportData[ stockid ] [ r ] [ E_DONATIONS ] = 0.0;
2018-11-02 09:57:08 +11:00
g_stockMarketReportData[ stockid ] [ r ] [ E_PRICE ] = 0.0;
2018-10-25 08:32:18 +11:00
}
// load price information if there is
2018-11-02 09:57:08 +11:00
mysql_tquery( dbHandle, sprintf( "SELECT * FROM `STOCK_REPORTS` WHERE `STOCK_ID`=%d ORDER BY `REPORTING_TIME` DESC, `ID` DESC LIMIT %d", stockid, sizeof( g_stockMarketReportData[ ] ) ), "Stock_UpdateReportingPeriods", "d", stockid );
2018-10-31 09:42:10 +11:00
// load the maximum number of shares
2018-11-07 18:10:36 +11:00
UpdateStockMaxShares( stockid );
2018-10-25 08:32:18 +11:00
// add to iterator
Iter_Add( stockmarkets, stockid );
}
return stockid;
}
2018-11-07 19:51:46 +11:00
static stock StockMarket_ReleaseDividends( stockid, bool: is_ipo = false )
2018-10-25 08:32:18 +11:00
{
2018-10-31 18:19:18 +11:00
mysql_format( dbHandle, szBigString, sizeof ( szBigString ), "SELECT * FROM `STOCK_OWNERS` WHERE `STOCK_ID`=%d", stockid );
2018-11-07 19:51:46 +11:00
mysql_tquery( dbHandle, szBigString, "Stock_OnDividendPayout", "dd", stockid, is_ipo );
2018-10-26 04:58:12 +11:00
return 1;
}
2018-10-25 08:32:18 +11:00
2018-11-07 19:24:00 +11:00
stock StockMarket_UpdateEarnings( stockid, amount, Float: factor = 1.0, Float: donation_amount = 0.0 )
2018-10-26 04:58:12 +11:00
{
if ( ! Iter_Contains( stockmarkets, stockid ) )
return 0;
2018-10-25 08:32:18 +11:00
2018-11-01 21:55:21 +11:00
// ensure that pool remains always above 0 dollars
if ( ( g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ] += float( amount ) * factor ) < 0.0 ) {
g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ] = 0.0;
}
2018-11-07 19:24:00 +11:00
// update donation amount
g_stockMarketReportData[ stockid ] [ 0 ] [ E_DONATIONS ] += donation_amount;
2018-11-01 21:55:21 +11:00
// save to database
2018-11-07 19:24:00 +11:00
mysql_single_query( sprintf( "UPDATE `STOCK_REPORTS` SET `POOL`=%f, `DONATIONS`=%f WHERE `ID` = %d", g_stockMarketReportData[ stockid ] [ 0 ] [ E_POOL ], g_stockMarketReportData[ stockid ] [ 0 ] [ E_DONATIONS ], g_stockMarketReportData[ stockid ] [ 0 ] [ E_SQL_ID ] ) );
2018-10-26 04:58:12 +11:00
return 1;
}
2018-10-25 08:32:18 +11:00
2018-10-31 20:05:47 +11:00
static stock StockMarket_GiveShares( stockid, accountid, Float: shares )
2018-10-26 11:44:45 +11:00
{
mysql_format(
dbHandle, szBigString, sizeof ( szBigString ),
"INSERT INTO `STOCK_OWNERS` (`USER_ID`, `STOCK_ID`, `SHARES`) VALUES (%d, %d, %f) ON DUPLICATE KEY UPDATE `SHARES` = `SHARES` + %f",
accountid, stockid, shares, shares
);
mysql_single_query( szBigString );
2018-10-25 08:32:18 +11:00
}
2018-10-31 20:05:47 +11:00
static stock StockMarket_UpdateSellOrder( stockid, accountid, Float: shares )
2018-10-31 18:19:18 +11:00
{
mysql_format(
dbHandle, szBigString, sizeof ( szBigString ),
"INSERT INTO `STOCK_SELL_ORDERS` (`STOCK_ID`, `USER_ID`, `SHARES`) VALUES (%d, %d, %f) ON DUPLICATE KEY UPDATE `SHARES` = `SHARES` + %f",
stockid, accountid, shares, shares
);
mysql_single_query( szBigString );
2018-11-07 12:07:12 +11:00
// we are just using this variable to loosely track available shares
g_stockMarketData[ stockid ] [ E_AVAILABLE_SHARES ] += shares;
2018-10-31 18:19:18 +11:00
}
2018-10-31 20:05:47 +11:00
static stock StockMarket_CreateTradeLog( stockid, buyer_acc, seller_acc, Float: shares, Float: price )
{
mysql_format(
dbHandle, szBigString, sizeof ( szBigString ),
"INSERT INTO `STOCK_TRADE_LOG` (`STOCK_ID`, `BUYER_ID`, `SELLER_ID`, `SHARES`, `PRICE`) VALUES (%d, %d, %d, %f, %f)",
stockid, buyer_acc, seller_acc, shares, price
);
mysql_single_query( szBigString );
}
2018-10-26 11:44:45 +11:00
static stock StockMarket_ShowBuySlip( playerid, stockid )
2018-10-26 04:58:12 +11:00
{
2018-10-31 18:19:18 +11:00
mysql_tquery( dbHandle, sprintf( "SELECT SUM(`SHARES`) AS `SALE_SHARES` FROM `STOCK_SELL_ORDERS` WHERE `STOCK_ID`=%d", stockid ), "StockMarket_OnShowBuySlip", "dd", playerid, stockid );
return 1;
}
static stock StockMarket_ShowSellSlip( playerid, stockid )
{
format(
szLargeString, sizeof ( szLargeString ),
""COL_WHITE"You can sell shares of %s for "COL_GREEN"%s"COL_WHITE" each.\n\nThough, you will have to wait until a person buys them.\n\nYou have %s available shares to sell.",
g_stockMarketData[ stockid ] [ E_NAME ],
cash_format( g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ], .decimals = 2 ),
2018-11-07 19:24:00 +11:00
number_format( p_PlayerShares[ playerid ] [ stockid ], .decimals = 0 )
2018-10-31 18:19:18 +11:00
);
2018-11-07 12:07:12 +11:00
ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET_SELL, DIALOG_STYLE_INPUT, ""COL_WHITE"Stock Market", szLargeString, "Sell", "Back" );
2018-10-26 04:58:12 +11:00
return 1;
2018-10-25 08:32:18 +11:00
}
2018-11-07 19:24:00 +11:00
static stock StockMarket_ShowDonationSlip( playerid, stockid )
{
format(
szLargeString, sizeof ( szLargeString ),
2018-12-04 00:17:48 +11:00
""COL_WHITE"Donations can be used to prop up %s's stock price.\n\n50%% of the money donated is distributed as dividends.\n\n"COL_ORANGE"You do not get any shares for donating to a company!",
2018-11-07 19:24:00 +11:00
g_stockMarketData[ stockid ] [ E_NAME ]
);
ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET_DONATE, DIALOG_STYLE_INPUT, ""COL_WHITE"Stock Market", szLargeString, "Donate", "Back" );
return 1;
}
2018-11-06 15:24:24 +11:00
static stock ShowPlayerStockMarket( playerid )
{
2018-11-07 12:07:12 +11:00
szLargeString = ""COL_WHITE"Stock\t"COL_WHITE"Available Shares\t"COL_WHITE"Dividend Per Share ($)\t"COL_WHITE"Price ($)\n";
2018-11-06 15:24:24 +11:00
foreach ( new s : stockmarkets )
{
new Float: price_change = ( ( g_stockMarketReportData[ s ] [ 1 ] [ E_PRICE ] / g_stockMarketReportData[ s ] [ 2 ] [ E_PRICE ] ) - 1.0 ) * 100.0;
new Float: payout = g_stockMarketReportData[ s ] [ 0 ] [ E_POOL ] / g_stockMarketData[ s ] [ E_MAX_SHARES ];
format(
szLargeString, sizeof( szLargeString ),
"%s%s (%s)\t%s\t"COL_GREEN"%s\t%s%s (%s%%)\n",
szLargeString,
g_stockMarketData[ s ] [ E_NAME ],
g_stockMarketData[ s ] [ E_SYMBOL ],
2018-11-07 12:07:12 +11:00
number_format( g_stockMarketData[ s ] [ E_AVAILABLE_SHARES ], .decimals = 0 ),
2018-11-06 15:24:24 +11:00
cash_format( payout, .decimals = 2 ),
price_change >= 0.0 ? COL_GREEN : COL_RED,
cash_format( g_stockMarketReportData[ s ] [ 1 ] [ E_PRICE ], .decimals = 2 ),
number_format( price_change, .decimals = 2, .prefix = ( price_change >= 0.0 ? '+' : '\0' ) )
);
}
2018-11-07 12:07:12 +11:00
return ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET, DIALOG_STYLE_TABLIST_HEADERS, ""COL_WHITE"Stock Market", szLargeString, "Select", "Close" );
2018-11-06 15:24:24 +11:00
}
static stock ShowPlayerStockMarketOptions( playerid, stockid )
{
2018-11-07 19:24:00 +11:00
format( szBigString, sizeof( szBigString ), "Buy shares\t"COL_GREEN"%s\nDonate to company\t>>>\nView shareholders\t"COL_GREY">>>\nView stock information\t"COL_GREY">>>", cash_format( g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ], .decimals = 2 ) );
2018-11-06 15:24:24 +11:00
ShowPlayerDialog( playerid, DIALOG_STOCK_MARKET_OPTIONS, DIALOG_STYLE_TABLIST, sprintf( ""COL_WHITE"Stock Market - %s", g_stockMarketData[ stockid ] [ E_NAME ] ), szBigString, "Select", "Back" );
return 1;
}
2018-11-07 12:07:12 +11:00
static stock ShowPlayerShareOptions( playerid, stockid )
{
format( szBigString, sizeof( szBigString ), "Sell shares\t"COL_GREEN"%s\n"COL_RED"Cancel Sell Orders\t"COL_RED">>>", cash_format( g_stockMarketReportData[ stockid ] [ 1 ] [ E_PRICE ], .decimals = 2 ) );
ShowPlayerDialog( playerid, DIALOG_STOCK_POPTIONS, DIALOG_STYLE_TABLIST, sprintf( ""COL_WHITE"Stock Market - %s", g_stockMarketData[ stockid ] [ E_NAME ] ), szBigString, "Select", "Back" );
return 1;
}
static stock ShowPlayerShares( playerid )
{
new
accountid = GetPlayerAccountID( playerid );
mysql_format(
dbHandle, szLargeString, 512,
"SELECT STOCK_ID, SUM(HELD_SHARES) AS HELD_SHARES, SUM(OWNED_SHARES) AS OWNED_SHARES " \
"FROM ( " \
" (SELECT STOCK_ID, USER_ID, SHARES AS HELD_SHARES, 0 AS OWNED_SHARES FROM STOCK_SELL_ORDERS t1 WHERE USER_ID=%d) " \
" UNION ALL " \
" (SELECT STOCK_ID, USER_ID, 0 AS HELD_SHARES, SHARES AS OWNED_SHARES FROM STOCK_OWNERS t2 WHERE USER_ID=%d) " \
") t_union " \
"GROUP BY STOCK_ID",
accountid, accountid
);
mysql_tquery( dbHandle, szLargeString, "StockMarket_OnShowShares", "d", playerid );
return 1;
}
2018-11-07 18:10:36 +11:00
static stock UpdateStockMaxShares( stockid ) {
mysql_tquery( dbHandle, sprintf( "SELECT (SELECT SUM(`SHARES`) FROM `STOCK_OWNERS` WHERE `STOCK_ID`=%d) AS `SHARES_OWNED`, (SELECT SUM(`SHARES`) FROM `STOCK_SELL_ORDERS` WHERE `STOCK_ID`=%d) AS `SHARES_HELD`", stockid, stockid ), "Stock_UpdateMaximumShares", "d", stockid );
}
2018-10-25 08:32:18 +11:00
/*
2018-10-31 18:19:18 +11:00
DROP TABLE `STOCK_REPORTS`;
2018-10-25 08:32:18 +11:00
CREATE TABLE IF NOT EXISTS `STOCK_REPORTS` (
2018-10-26 04:58:12 +11:00
`ID` int(11) primary key auto_increment,
2018-10-25 08:32:18 +11:00
`STOCK_ID` int(11),
2018-10-31 09:42:10 +11:00
`POOL` float,
2018-10-31 18:19:18 +11:00
`PRICE` float,
2018-10-25 08:32:18 +11:00
`REPORTING_TIME` TIMESTAMP default CURRENT_TIMESTAMP
);
2018-11-07 19:24:00 +11:00
ALTER TABLE STOCK_REPORTS ADD DONATIONS float DEFAULT 0.0 AFTER POOL;
2018-10-25 08:32:18 +11:00
2018-10-31 18:19:18 +11:00
DROP TABLE `STOCK_OWNERS`;
2018-10-26 04:58:12 +11:00
CREATE TABLE IF NOT EXISTS `STOCK_OWNERS` (
`USER_ID` int(11),
`STOCK_ID` int(11),
`SHARES` float,
2018-10-26 11:44:45 +11:00
PRIMARY KEY (USER_ID, STOCK_ID)
2018-10-26 04:58:12 +11:00
);
2018-10-31 18:19:18 +11:00
DROP TABLE `STOCK_SELL_ORDERS`;
CREATE TABLE IF NOT EXISTS `STOCK_SELL_ORDERS` (
2018-10-26 04:58:12 +11:00
`STOCK_ID` int(11),
2018-10-31 18:19:18 +11:00
`USER_ID` int(11),
2018-10-26 04:58:12 +11:00
`SHARES` float,
2018-10-31 18:19:18 +11:00
`LIST_DATE` TIMESTAMP default CURRENT_TIMESTAMP,
PRIMARY KEY (STOCK_ID, USER_ID)
);
2018-10-31 20:05:47 +11:00
DROP TABLE `STOCK_TRADE_LOG`;
CREATE TABLE IF NOT EXISTS `STOCK_TRADE_LOG` (
`ID` int(11) primary key auto_increment,
`STOCK_ID` int(11),
`BUYER_ID` int(11),
`SELLER_ID` int(11),
`SHARES` float,
`PRICE` float,
2018-11-01 17:48:25 +11:00
`DATE` TIMESTAMP default CURRENT_TIMESTAMP
2018-10-31 20:05:47 +11:00
);
2018-10-25 08:32:18 +11:00
*/