// This source code is subject to the terms of the Mozilla Public License 2.
0 at
https://mozilla.org/MPL/2.0/
// © ZenAndTheArtOfTrading
// Version 2.1 (Last Updated 14th November, 2024)
// Huge thanks to Marc (https://www.tradingview.com/u/Mrcrbw/) for providing
// major improvements and fixes to the code that made this script much better!
// @version=6
//////////////////////////////////////////////////////////////////////////
// EXAMPLE STRATEGY (REPLACE THIS CODE WITH YOUR OWN) //
// Replace lines 15 -> 69 with your own strategy script code! //
// This example system is my Simple Pullback Strategy //
// Check my TradingView profile for more info about this script //
//////////////////////////////////////////////////////////////////////////
strategy("Monthly Performance Table",
overlay=true,
initial_capital=100000,
default_qty_type=strategy.percent_of_equity,
default_qty_value=100,
commission_type=strategy.commission.cash_per_contract,
commission_value=0.005)
// Get user input
i_ma1 = input.int(title="MA 1 Length", defval=200, step=10,
group="Strategy Parameters", tooltip="Long-term MA", display=display.none)
i_ma2 = input.int(title="MA 2 Length", defval=10, step=10,
group="Strategy Parameters", tooltip="Short-term MA", display=display.none)
i_stopPercent = input.float(title="Stop Loss Percent", defval=0.10, step=0.1,
group="Strategy Parameters", tooltip="Failsafe Stop Loss Percent Decline",
display=display.none)
i_lowerClose = input.bool(title="Exit On Lower Close", defval=false,
group="Strategy Parameters", tooltip="Wait for a lower-close before exiting above
MA2", display=display.none)
i_startTime = input.time(title="Start Filter", defval=timestamp("01 Jan 1995
13:30 +0000"), group="Time Filter", tooltip="Start date & time to begin searching
for setups", display=display.none)
i_endTime = input.time(title="End Filter", defval=timestamp("1 Jan 2099 19:30
+0000"), group="Time Filter", tooltip="End date & time to stop searching for
setups", display=display.none)
// Get indicator values
ma1 = ta.sma(close, i_ma1)
ma2 = ta.sma(close, i_ma2)
// Check filter(s)
f_dateFilter = time >= i_startTime and time <= i_endTime
// Check buy/sell conditions
var float buyPrice = 0
buyCondition = close > ma1 and close < ma2 and strategy.position_size == 0 and
f_dateFilter
sellCondition = close > ma2 and strategy.position_size > 0 and (not i_lowerClose
or close < low[1])
stopDistance = strategy.position_size > 0 ? ((buyPrice - close) / close) : na
stopPrice = strategy.position_size > 0 ? buyPrice - (buyPrice *
i_stopPercent) : na
stopCondition = strategy.position_size > 0 and stopDistance > i_stopPercent
// Enter positions
if buyCondition
strategy.entry(id="Long", direction=strategy.long, comment="Equity = $" +
str.tostring(strategy.equity, "#.##"))
if buyCondition[1]
buyPrice := strategy.position_avg_price
// Exit positions
if sellCondition or stopCondition
strategy.close(id="Long", comment="Exit" + (stopCondition ? "SL=true" : ""))
buyPrice := na
// Draw pretty colors
plot(buyPrice, color=color.lime, style=plot.style_linebr)
plot(stopPrice, color=color.red, style=plot.style_linebr, offset=-1)
plot(ma1, color=color.blue)
plot(ma2, color=color.orange)
////////////////////////////////////////////////////////////
// END EXAMPLE STRATEGY (REPLACE THIS CODE WITH YOUR OWN) //
////////////////////////////////////////////////////////////
//-------------------------------------------------------------------------//
///////////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////
// MONTHLY TABLE based on the script written by QuantNomad - Copy & Paste code from
here down into your strategy script /{
// Original Script: https://tradingview.com/script/kzp8e4X3-Monthly-Returns-in-
PineScript-Strategies/ //
// ZenAndTheArtOfTrading Script: https://www.tradingview.com/script/aHwncfep-
Monthly-Strategy-Performance-Table/ //
///////////////////////////////////////////////////////////////////////////////////
///////////////////////////////////////
var string GROUP_PERFORMANCE_TABLE = "Monthly Performance Table"
bool mptable_on = input.bool(title="Turn On |", defval=false,
display=display.none, group=GROUP_PERFORMANCE_TABLE, inline="MPT_Toggles")
bool mptable_debug = input.bool(title="Debug Mode |", defval=false,
display=display.none, group=GROUP_PERFORMANCE_TABLE, inline="MPT_Toggles")
int mptable_precision = input.int(title="Decimal Precision", defval=2,
minval=1, display=display.none, group=GROUP_PERFORMANCE_TABLE,
inline="MPT_Toggles", tooltip="Decimal precision of each cell")
color mptable_titleColor = input.color(title="Title Cell Color",
defval=#cccccc, display=display.none, group=GROUP_PERFORMANCE_TABLE,
inline="MPT_Colors")
color mptable_titleTextColor = input.color(title="Title Text Color",
defval=#363a45, display=display.none, group=GROUP_PERFORMANCE_TABLE,
inline="MPT_Colors")
color rmptable_textColor = input.color(title="Cell Text Color",
defval=color.white, display=display.none, group=GROUP_PERFORMANCE_TABLE,
inline="MPT_Colors")
color mptable_ProfitColor = input.color(title="Year Profit Color",
defval=color.new(color.green, 50), display=display.none,
group=GROUP_PERFORMANCE_TABLE, inline="MPT_Colors")
color mptable_LossColor = input.color(title="Year Loss Color",
defval=color.new(color.red, 50), display=display.none,
group=GROUP_PERFORMANCE_TABLE, inline="MPT_Colors")
color mptable_BreakEvenColor = input.color(title="Year B/E Color",
defval=color.new(color.white, 50), display=display.none,
group=GROUP_PERFORMANCE_TABLE, inline="MPT_Colors")
int mptable_pageNumber = input.int(title="Page Number", defval=1,
minval=1, step=1, maxval=10, display=display.none, group=GROUP_PERFORMANCE_TABLE,
tooltip="Which page of results to display") - 1 // -1 is for corection for arrays.
Aaray index start with 0
int mptable_pageSize = input.int(title="Page Size (Rows)", defval=20,
minval=1, display=display.none, group=GROUP_PERFORMANCE_TABLE, tooltip="How many
rows to display") - 1 // 9 is used to show 10 rows of data due to array start from
0
string mptable_tableTextSize = input.string(title="Text Size", defval="Small",
options=["Auto", "Huge", "Large", "Normal", "Small", "Tiny"],
display=display.none, group=GROUP_PERFORMANCE_TABLE)
// Custom function for getting table text sized based on user input
table_text_size(_size) =>
switch _size
"Auto" => size.auto
"Huge" => size.huge
"Large" => size.large
"Normal" => size.normal
"Small" => size.small
=> size.tiny
tableTextSize = table_text_size(mptable_tableTextSize)
// Custom function for getting decimal precision based on given number
// (eg. if number is > 0 but < 0.05 or < 0 and > -0.05, set precision to 3 to avoid
rounding to 0 which is misleading)
getRoundingPrecision(float num) =>
if (num > 0 and num < 0.05) or (num < 0 and num > -0.05)
3
else
mptable_precision
// Define a monthly/yearly return type
type StrategyReturn
float profit
float drawdown
float peak
int timestamp
// Store accumulated P&L values
var float accumulatedMonthlyPL = 0
var float accumulatedYearlyPL = 0
var float bestAccumulatedMonthlyPL = 0
var float bestAccumulatedYearlyPL = 0
// Store drawdown values
var float equityPeak = strategy.initial_capital
var float yearlyEquityHigh = 0
var float currentYearlyDrawdown = 0
var float yearlyMaxDrawdown = 0
var float worstDrawdown = 0
var float monthlyEquityHigh = 0
var float currentMonthlyDrawdown = 0
var float monthlyMaxDrawdown = 0
var int currentDrawdownBars = 0
var int maxDrawdownBars = 0
// Store stat arrays
var array<int> totalDrawdownBars = array.new<int>(0)
var array<float> totalDrawdowns = array.new<float>(0)
// Store long & short trade count
var int totalBreakEvenTrades = 0
var int totalLongTrades = 0
var int totalLongTradeWins = 0
var int totalShortTrades = 0
var int totalShortTradeWins = 0
var float lastProfitLoss = na
var float lastCumulativeProfitPercent = na
var float lastTradeCost = na
// Detect a closed trade
// TV Documentation: Trade List's Cumulative Profit % Formula = TradeProfit /
(InitialCapital + Cumulative Profit of the previous trades) * 100%
if strategy.closedtrades != strategy.closedtrades[1]
// Retrieve trade cost for the closed trade
lastTradeCost := strategy.closedtrades.commission(strategy.closedtrades - 1)
// Calculate equity before trade closed (strategy.equity will not do, because
it changes bar-by-bar based on open P&L not realized P&L)
float preEquity = strategy.initial_capital + strategy.netprofit[1]
// Calculate P&L + cost of this trade
lastProfitLoss := strategy.closedtrades.profit(strategy.closedtrades - 1)
//(strategy.grossloss - (strategy.grossloss[1] - tradeCost)) * -1
// Check if this was a long or short trade and if it won or lost
if strategy.position_size[1] > 0
totalLongTrades := totalLongTrades + 1
if lastProfitLoss > 0
totalLongTradeWins := totalLongTradeWins + 1
else if strategy.position_size[1] < 0
totalShortTrades := totalShortTrades + 1
if lastProfitLoss > 0
totalShortTradeWins := totalShortTradeWins + 1
// Check if the trade broke even
if lastProfitLoss == 0
totalBreakEvenTrades := totalBreakEvenTrades + 1
// Calculate cumulative profit % for this trade
lastCumulativeProfitPercent := (lastProfitLoss / preEquity) * 100
// Store highest peak value of equity (we can now use strategy.equity since
equity has updated to realized P&L on this bar)
if strategy.equity > equityPeak
equityPeak := strategy.equity
// Calculate total system drawdown %
float equityDD = ((strategy.equity - equityPeak) / equityPeak) * 100
if equityDD < worstDrawdown
worstDrawdown := equityDD
// Store accumulated monthly + yearly P&L
accumulatedMonthlyPL := lastCumulativeProfitPercent + accumulatedMonthlyPL[1]
accumulatedYearlyPL := accumulatedYearlyPL + lastCumulativeProfitPercent
// Save max favourable excursion for this month (ie. peak return as %)
if accumulatedMonthlyPL > bestAccumulatedMonthlyPL
bestAccumulatedMonthlyPL := accumulatedMonthlyPL
// Save max favourable excursion for this year (ie. peak return as %)
if accumulatedYearlyPL > bestAccumulatedYearlyPL
bestAccumulatedYearlyPL := accumulatedYearlyPL
// Track max equity high over current year for max yearly drawdown calculation
if accumulatedYearlyPL > yearlyEquityHigh
yearlyEquityHigh := accumulatedYearlyPL
// Check if our yearly realized equity high minus current realized equity
exceeds our stored max drawdown for the year, update if necessary, and save worst
drawdown
if accumulatedYearlyPL - yearlyEquityHigh < 0
currentYearlyDrawdown := accumulatedYearlyPL - yearlyEquityHigh
if currentYearlyDrawdown < yearlyMaxDrawdown
yearlyMaxDrawdown := currentYearlyDrawdown
// Track max equity high over current month for max monthly drawdown
calculation
if accumulatedMonthlyPL > monthlyEquityHigh
monthlyEquityHigh := accumulatedMonthlyPL
// Check if our monthly realized equity high minus current realized equity
exceeds our stored max drawdown for the month, update if necessary, and save worst
drawdown
if accumulatedMonthlyPL - monthlyEquityHigh < 0
currentMonthlyDrawdown := accumulatedMonthlyPL - monthlyEquityHigh
if currentMonthlyDrawdown < monthlyMaxDrawdown
monthlyMaxDrawdown := currentMonthlyDrawdown
// Debug label
if mptable_debug
string debugTip = "Equity = $" + str.tostring(strategy.equity, "#.##") +
"\nP&L=" + str.tostring(lastCumulativeProfitPercent) + "%" +
"\nAccumMonthlyP&L=" + str.tostring(math.round(accumulatedMonthlyPL,
getRoundingPrecision(accumulatedMonthlyPL))) + "%" +
"\nAccumYearlyP&L=" + str.tostring(math.round(accumulatedYearlyPL,
getRoundingPrecision(accumulatedYearlyPL))) + "%" +
"\nMonthlyMaxDD=" + str.tostring(math.round(monthlyMaxDrawdown,
getRoundingPrecision(monthlyMaxDrawdown))) + "%" +
"\nYearlyMaxDD=" + str.tostring(math.round(yearlyMaxDrawdown,
getRoundingPrecision(yearlyMaxDrawdown))) + "%" +
"\nTotalMaxDD=" + str.tostring(math.round(worstDrawdown,
getRoundingPrecision(worstDrawdown))) + "%" +
"\nCurrentDDBars=" + str.tostring(currentDrawdownBars) +
"\nMaxDDBars=" + str.tostring(maxDrawdownBars) +
"\nTotalBreakEven=" + str.tostring(totalBreakEvenTrades) +
"\nTotalLongs=" + str.tostring(totalLongTrades) +
"\nTotalLongWins=" + str.tostring(totalLongTradeWins) +
"\nTotalShorts=" + str.tostring(totalShortTrades) +
"\nTotalShortWins=" + str.tostring(totalShortTradeWins)
label.new(bar_index, high + (high * 0.01), "P&L " +
str.tostring(math.round(lastCumulativeProfitPercent,
getRoundingPrecision(lastCumulativeProfitPercent))) + "%", tooltip=debugTip,
textcolor=color.white)
// Calculate drawdown since last equity high (NOT max drawdown, just the current
max DD since we were out of DD)
float t_equityDD = ((strategy.equity - equityPeak) / equityPeak) * 100
var float currentMaxDrawdownSinceLast = 0
// Update Max Drawdown bar count and current DD if equity is under water, check
isconfirmed to prevent double-counting bars with recalc_on_order_fills on
if strategy.equity < equityPeak and barstate.isconfirmed
currentDrawdownBars := currentDrawdownBars + 1
if currentDrawdownBars > maxDrawdownBars
maxDrawdownBars := currentDrawdownBars
if t_equityDD < currentMaxDrawdownSinceLast
currentMaxDrawdownSinceLast := t_equityDD
else
if currentDrawdownBars > 0
totalDrawdownBars.push(currentDrawdownBars)
totalDrawdowns.push(currentMaxDrawdownSinceLast)
currentDrawdownBars := 0
currentMaxDrawdownSinceLast := 0
// Prepare arrays to store Yearly and Monthly P&Ls
var array<StrategyReturn> monthlyReturns = array.new<StrategyReturn>(0)
var array<StrategyReturn> yearlyReturns = array.new<StrategyReturn>(0)
var bool firstEntryTime = false
// Retrieve entry time of initial entry in open position
if not firstEntryTime and not na(strategy.opentrades.entry_time(0))
firstEntryTime := true
// Detect new month and year
bool new_month = month(time) != month(time[1])
bool new_year = year(time) != year(time[1])
// Detect a new month and store its return profile
if not barstate.isfirst and new_month and firstEntryTime or
barstate.islastconfirmedhistory
StrategyReturn mr = StrategyReturn.new(accumulatedMonthlyPL,
monthlyMaxDrawdown, bestAccumulatedMonthlyPL, time[1])
monthlyReturns.push(mr)
accumulatedMonthlyPL := 0
monthlyMaxDrawdown := 0
monthlyEquityHigh := 0
currentMonthlyDrawdown := 0
bestAccumulatedMonthlyPL := 0
// Detect a new year and reset tracking variables
if not barstate.isfirst and new_year and firstEntryTime or
barstate.islastconfirmedhistory
StrategyReturn yr = StrategyReturn.new(accumulatedYearlyPL, yearlyMaxDrawdown,
bestAccumulatedYearlyPL, time[1])
yearlyReturns.push(yr)
accumulatedYearlyPL := 0
yearlyMaxDrawdown := 0
yearlyEquityHigh := 0
currentYearlyDrawdown := 0
bestAccumulatedYearlyPL := 0
// DEBUG code
bgcolor(mptable_debug and new_month ? color.lime : na, title="New Month",
display=display.none)
bgcolor(mptable_debug and new_year ? color.red : na, title="New Year",
display=display.none)
// END DEBUG CODE
// Define Monthly P&L Table
var table performance_table = table(na)
//Adjust mptable_pageSize if the years are less than the mptable_pageSize
if yearlyReturns.size() < mptable_pageSize
mptable_pageSize := yearlyReturns.size()
// Caluclate the start and end of page to display
int startIndex = math.max(math.min(yearlyReturns.size() - 1, yearlyReturns.size() -
1 - (mptable_pageSize + 1) * mptable_pageNumber), mptable_pageSize - 1)
int endIndex = math.max(startIndex - mptable_pageSize, 0)
mptable_pageSize := endIndex <= mptable_pageSize ? endIndex : mptable_pageSize
// If this is the last bar on our chart, display the performance table
var int EXTRA_STAT_ROWS = 5 // This ensures table includes enough rows for CAGR etc
if mptable_on and monthlyReturns.size() > 0 and barstate.islastconfirmedhistory
// Create table (100 rows = 100 years of data, should be plenty for all
markets!)
performance_table := table.new(position.bottom_right, columns=17,
rows=yearlyReturns.size() + EXTRA_STAT_ROWS, border_width=1)
// Set column headers
performance_table.cell(0, 0, "Year", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(1, 0, "Jan", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(2, 0, "Feb", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(3, 0, "Mar", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(4, 0, "Apr", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(5, 0, "May", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(6, 0, "Jun", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(7, 0, "Jul", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(8, 0, "Aug", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(9, 0, "Sep", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(10, 0, "Oct", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(11, 0, "Nov", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(12, 0, "Dec", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(13, 0, "TOTAL", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
performance_table.cell(14, 0, "MaxDD", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
// Set yearly values
for year_index = startIndex to yearlyReturns.size() == 0 ? na : endIndex
// Get yearly return for this loop, set year number in first column,
determine color of cell
StrategyReturn yearlyReturn = yearlyReturns.get(year_index)
// Set year title and determine color
performance_table.cell(0, year_index + 1,
str.tostring(year(yearlyReturn.timestamp)), bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_size=tableTextSize)
color y_color = yearlyReturn.profit > 0 ? mptable_ProfitColor :
yearlyReturn.profit == 0 ? mptable_BreakEvenColor : mptable_LossColor
// Populate yearly cell values
string yearlyTip = "MaxDD: " +
str.tostring(math.round(yearlyReturn.drawdown,
getRoundingPrecision(yearlyReturn.drawdown))) + "%" +
"\nMFE: " + str.tostring(math.round(yearlyReturn.peak,
getRoundingPrecision(yearlyReturn.peak))) + "%"
performance_table.cell(13, year_index + 1, (yearlyReturn.profit > 0 ? "+" :
"") + str.tostring(math.round(yearlyReturn.profit,
getRoundingPrecision(yearlyReturn.profit))) + "%", bgcolor=y_color,
text_color=color.white, text_size=tableTextSize, tooltip=yearlyTip)
performance_table.cell(14, year_index + 1,
str.tostring(math.round(yearlyReturn.drawdown,
getRoundingPrecision(yearlyReturn.drawdown))) + "%",
bgcolor=mptable_BreakEvenColor, text_color=color.white, text_size=tableTextSize)
// Set monthly values
for month_index = 0 to monthlyReturns.size() - 1
// Get monthly return for this loop, get current year for this loop,
then calculate the corresponding table column and row
StrategyReturn monthlyReturn = monthlyReturns.get(month_index)
int yearOfMonth = year(monthlyReturn.timestamp)
int monthCol = month(monthlyReturn.timestamp)
// populate monthly profit only if the years of the yearly return match
with the monthly return.
if yearOfMonth == year(yearlyReturn.timestamp)
// Determine color for monthly P&L
color m_color = monthlyReturn.profit > 0 ?
color.new(mptable_ProfitColor, color.t(mptable_ProfitColor) + 20) :
monthlyReturn.profit == 0 ? color.new(mptable_BreakEvenColor,
color.t(mptable_BreakEvenColor) + 20) : color.new(mptable_LossColor,
color.t(mptable_LossColor) + 20)
// Set monthly P&L cell
string monthlyTip = "MaxDD: " +
str.tostring(math.round(monthlyReturn.drawdown,
getRoundingPrecision(monthlyReturn.drawdown))) + "%" +
"\nMFE: " + str.tostring(math.round(monthlyReturn.peak,
getRoundingPrecision(monthlyReturn.peak))) + "%"
performance_table.cell(monthCol, year_index + 1,
str.tostring(math.round(monthlyReturn.profit,
getRoundingPrecision(monthlyReturn.profit))), bgcolor=m_color,
text_color=color.white, text_size=tableTextSize, tooltip=monthlyTip)
// Calculate the time difference in milliseconds
float start_time = strategy.closedtrades.entry_time(0)
float end_time = strategy.closedtrades.exit_time(strategy.closedtrades - 1)
float time_diff = end_time - start_time
// Convert time difference from milliseconds to years
float years_traded = time_diff / 31536000000.0
// Calculate CAGR
float final_value = strategy.netprofit + strategy.initial_capital
float cagr = (math.pow(final_value / strategy.initial_capital, 1 /
years_traded) - 1) * 100
float percentReturn = (strategy.netprofit / strategy.initial_capital) * 100
float mar = cagr / math.abs(worstDrawdown)
lastMonthRowIndex = startIndex < 5 ? 5 : startIndex
// Populate table data
float totalWinRate = (strategy.wintrades / strategy.closedtrades) * 100
float longWinRate = nz((totalLongTradeWins / totalLongTrades) * 100)
float shortWinRate = nz((totalShortTradeWins / totalShortTrades) * 100)
int avgTradesYr = math.ceil(strategy.closedtrades / years_traded)
float totalCommissionCost = 0
for i = 0 to strategy.closedtrades
totalCommissionCost := totalCommissionCost +
strategy.closedtrades.commission(i)
float commissionDrag = ((totalCommissionCost / strategy.initial_capital) * 100)
/ years_traded
string returnTip = "Based on a total of " + str.tostring(strategy.closedtrades)
+ " trades" +
"\nWin Rate = " + str.tostring(math.round(totalWinRate,
getRoundingPrecision(totalWinRate))) + "%" +
"\nLong Trades = " + str.tostring(totalLongTrades) + " (Win " +
str.tostring(math.round(longWinRate, getRoundingPrecision(longWinRate))) + "%)" +
"\nShort Trades = " + str.tostring(totalShortTrades) + " (Win " +
str.tostring(math.round(shortWinRate, getRoundingPrecision(shortWinRate))) + "%)" +
"\nAvg Trades Per Year = " + str.tostring(avgTradesYr) + " (" +
str.tostring(years_traded, "#.#") + " years)" +
"\nCommission Drag = " + str.tostring(commissionDrag, "#.##") + "% per
year"
performance_table.cell(15, lastMonthRowIndex, "Return: " + (percentReturn > 0 ?
"+" : "") + str.tostring(math.round(percentReturn,
getRoundingPrecision(percentReturn))) + "%", bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_font_family=font.family_monospace,
text_size=tableTextSize, tooltip=returnTip)
performance_table.cell(15, lastMonthRowIndex - 1, "MAR: " + str.tostring(mar,
"#.##"), bgcolor=mptable_titleColor, text_color=mptable_titleTextColor,
text_font_family=font.family_monospace, text_size=tableTextSize, tooltip="Measure
of return adjusted for risk: CAGR divided by Max Drawdown. Indicates how
comfortable the system might be to trade. Higher than 0.5 is ideal, 1.0 and above
is very good, and anything 3.0 or above should be considered suspicious.")
performance_table.cell(15, lastMonthRowIndex - 2, "DD Bars: " +
str.tostring(maxDrawdownBars), bgcolor=mptable_titleColor,
text_color=mptable_titleTextColor, text_font_family=font.family_monospace,
text_size=tableTextSize, tooltip="Average Drawdown Bars: " +
str.tostring(totalDrawdownBars.avg(), "#.#") + "\n\nThis is how many bars it took
to recover the longest drawdown (note: this is different to the MAX drawdown, and
represents time drawdown)")
performance_table.cell(15, lastMonthRowIndex - 3, "MaxDD: " +
str.tostring(math.round(worstDrawdown, getRoundingPrecision(worstDrawdown))) + "%",
bgcolor=mptable_titleColor, text_color=mptable_titleTextColor,
text_font_family=font.family_monospace, text_size=tableTextSize, tooltip="Average
Drawdown: " + str.tostring(totalDrawdowns.avg(), "#.##") + "%\n\nThis number is
different to the Strategy Tester because this number is based on closed trade
equity while the Tester's MaxDD is based on open equity.")
performance_table.cell(15, lastMonthRowIndex - 4, "CAGR: " + (cagr > 0 ? "+" :
"") + str.tostring(math.round(cagr, getRoundingPrecision(cagr))) + "%",
bgcolor=mptable_titleColor, text_color=mptable_titleTextColor,
text_font_family=font.family_monospace, text_size=tableTextSize,
tooltip="Compounded annual growth rate")
performance_table.cell(15, lastMonthRowIndex - 5, "REALIZED P&L",
bgcolor=mptable_titleColor, text_color=mptable_titleTextColor,
text_font_family=font.family_monospace, text_size=tableTextSize, tooltip="These
numbers are based on Realized equity (closed trades)")
// } END MONTHLY TABLE
// Debug plots
plot(na, title="##### PERFORMANCE DEBUG #####", display=display.data_window)
plot(lastTradeCost, title="Last Trade Cost $", display=display.data_window)
plot(lastProfitLoss, title="Last Trade P&L $", display=display.data_window)
plot(lastCumulativeProfitPercent, title="Last Trade P&L %",
display=display.data_window)
plot(accumulatedMonthlyPL, title="Accum. Monthly P&L", display=display.data_window)
plot(accumulatedYearlyPL, title="Accum. Yearly P&L", display=display.data_window)
/////////// END OF PERFORMANCE TABLE CODE. ///////////