File Name: BtDonchian_PositionManagement.efs
Description:
This formula is a back testing example that illustrates how to code a strategy with complete position management with profit targets, stops and closing out positions with more than one closing trade.
Formula Parameters:
NA
Notes:
This formula is a back testing formula to be used in conjunction with the Strategy Analyzer. It is not intended for real time analysis. This formula requires version 7.9.1 or later.
This strategy enters a position when a 10-period Donchian makes a new high on the upper channel or new low on the lower channel going long and short, respectively. The exit strategy for this sample uses the Average True Range study for the basis of the trailing stop logic. The initial stop is set to the Middle Donchian channel. The trailing stop will increment in the direction of the position by 25% of the previous bar's 10 period ATR. To complete the position management for this strategy, the positions will also be partially (50%) closed at a fixed profit target. This target will initially be set to 1.5X the difference between the entry price and the initial stop price, which creates a minimum 1.5 reward risk ratio for the first half of the position. The second half of the position will be closed by the trailing stop.
Download File:
BtDonchian_PositionManagement.efs
EFS Code:
Description:
This formula is a back testing example that illustrates how to code a strategy with complete position management with profit targets, stops and closing out positions with more than one closing trade.
Formula Parameters:
NA
Notes:
This formula is a back testing formula to be used in conjunction with the Strategy Analyzer. It is not intended for real time analysis. This formula requires version 7.9.1 or later.
This strategy enters a position when a 10-period Donchian makes a new high on the upper channel or new low on the lower channel going long and short, respectively. The exit strategy for this sample uses the Average True Range study for the basis of the trailing stop logic. The initial stop is set to the Middle Donchian channel. The trailing stop will increment in the direction of the position by 25% of the previous bar's 10 period ATR. To complete the position management for this strategy, the positions will also be partially (50%) closed at a fixed profit target. This target will initially be set to 1.5X the difference between the entry price and the initial stop price, which creates a minimum 1.5 reward risk ratio for the first half of the position. The second half of the position will be closed by the trailing stop.
Download File:
BtDonchian_PositionManagement.efs
EFS Code:
PHP Code:
/*********************************
Copyright © eSignal, a division of Interactive Data Corporation. 2006. All rights reserved.
This sample eSignal Formula Script (EFS) may be modified and saved under a new
filename; however, eSignal is no longer responsible for the functionality once modified.
eSignal reserves the right to modify and overwrite this EFS file with each new release.
This sample is for educational purposes only. It is not intended for trading.
Strategy Logic:
This Strategy example is a basic strategy that goes long when a new
10 period Donchian high occurs and goes short when a new 10 period Donchian
low occurs. The exit strategy for this sample uses the Average True Range study
for the basis of the trailing stop logic. The initial stop is set to the Middle Donchian
channel. The trailing stop will increment in the direction of the position by 25% of the
previous bar's 10 period ATR. To complete the position management for this strategy, the
positions will also be partially (50%) closed at a fixed profit target. This target will
initially be set to 1.5X the difference between the entry price and the initial stop price,
which creates a minimum 1.5 reward risk ratio for the first half of the position. The
second half of the position will be closed by the trailing stop.
*********************************/
function preMain() {
setPriceStudy(true);
setStudyTitle("Back Test: Donchian Position Management Example");
setCursorLabelName("Upper Donchian", 0);
setCursorLabelName("Lower Donchian", 1);
setCursorLabelName("Stop", 2);
setCursorLabelName("Target", 3);
setDefaultBarFgColor(Color.blue, 0);
setDefaultBarFgColor(Color.blue, 1);
setDefaultBarFgColor(Color.red, 2); // Stop color
setDefaultBarFgColor(Color.khaki, 3); // Target color
setDefaultBarThickness(2, 2); // Stop Thickness
setDefaultBarThickness(2, 3); // Target Thickness
setPlotType(PLOTTYPE_FLATLINES, 2); // Stop plot type
setPlotType(PLOTTYPE_FLATLINES, 3); // Target plot type
}
// Global Variables
var xUpper = null;
var xMiddle = null;
var xLower = null;
var xATR = null;
var nStop = null;
var nTarget = null;
var bTargetBreach = false;
var bInit = false; // initialization flag
function main() {
// Back Testing formulas are not for real time analysis.
// Therefore, prevent processing and exit at bar 0.
if (getCurrentBarIndex() == 0) return;
if(bInit == false) {
// This code block executes only once at the beginning of initialization.
// Initialize Donchian Series Objects.
xUpper = upperDonchian(10);
xMiddle = middleDonchian(10);
xLower = lowerDonchian(10);
xATR = atr(10);
bInit = true; // Prevents this code block from executing again.
}
// Retrieve previous bar's values of Donchian Series for back testing conditions.
var nU = xUpper.getValue(-1);
var nM = xMiddle.getValue(-1);
var nL = xLower.getValue(-1);
// Retrieve current bar's value of sar for the trailing stop condition.
var nATR = xATR.getValue(0);
var nInc = 0.25 * xATR.getValue(-1); // Stop increment.
// Validate the study variables to ensure they contain valid data.
if(nU == null || nM == null || nL == null || nATR == null || nInc == null) {
return; // Exit the formula if variables contain invalid data.
}
/*****
Reset nStop and nTarget to null so that they do not plot if no longer in a position.
Also reset the flag for the target breach so that this will be enabled for the next
position taken.
*****/
if (Strategy.isInTrade() == false) {
nStop = null;
nTarget = null;
bTargetBreach = false;
} else {
/*****
If the strategy is in a position, increment the trailing stop value.
*****/
if (Strategy.isLong() == true) {
nStop += nInc; // Moves up by nInc value
} else if (Strategy.isShort() == true) {
nStop -= nInc; // Moves down by nInc value
}
/*****
If we are still in a position, check to see if the profit target was
executed. The global bTargetBreach Boolean variable is used to track
this occurance. If it is set to true, reset the nTarget to null so
that it will no longer plot on the chart.
*****/
if (bTargetBreach == true) {
nTarget = null;
}
}
// Trailing Stop Exit Strategy
if (Strategy.isInTrade() == true) {
/*****
First check for a stop before looking for entry signals. If a stop
is found on the current bar, prevent a new entry from occuring on the
same bar. This is accomplished by placing the entry logic inside an else
statement that follows this trailing stop exit condition.
******/
if (Strategy.isLong() == true) { // Long Trailing Stop and Profit Target Exit
/*****
First check to see if the bar opened below the stop price. If so, exit
at the open of the bar.
*****/
if (open(0) <= nStop) {
Strategy.doSell("Long Stop", Strategy.MARKET, Strategy.THISBAR);
} else if (low(0) <= nStop) {
/*****
Next, check to see if the low of the bar breached the stop price.
If so, exit with a stop order at the stop price.
*****/
Strategy.doSell("Long Stop", Strategy.STOP, Strategy.THISBAR, Strategy.ALL, nStop);
} else if (bTargetBreach == false) { // Profit Target
/*****
If we get to this else statement, it means we did not get stopped out by
the preceeding conditions and that we have not yet hit the profit target.
Now check for a breach of the profit target.
*****/
var nLot = Math.round(Strategy.getPositionSize()/2);
if (open(0) >= nTarget) {
/*****
Bar opened above the profit target level, close half of the position
with a market order at the open. To get the number of shares or contracts
to sell, get the current position size, divide by 2 and round the result to
ensure that we pass a whole number to the .doSell() call. Also set the
Boolean bTargetBreach variable to true to prevent subsequent bars from
executing another profit target trade. This is only allowed to occur once
for each position taken.
*****/
bTargetBreach = true;
Strategy.doSell("Profit Target", Strategy.MARKET, Strategy.THISBAR, nLot);
} else if (high(0) >= nTarget) {
/*****
If the bar did not open above the target level, check to see if the high of
the bar breached the target level and exit half of the position with a limit
order at the target price. Again, set bTargetBreach to true.
*****/
bTargetBreach = true;
Strategy.doSell("Profit Target", Strategy.LIMIT, Strategy.THISBAR, nLot, nTarget);
}
}
}
if (Strategy.isShort() == true) { // Short Trailing Stop Exit
/*****
First check to see if the bar opened above the stop price. If so, exit
at the open of the bar.
*****/
if (open(0) >= nStop) {
Strategy.doCover("Short Stop", Strategy.MARKET, Strategy.THISBAR);
} else if (high(0) >= nStop) {
/*****
Next, check to see if the high of the bar breached the stop price.
If so, exit with a stop order at the stop price.
*****/
Strategy.doCover("Short Stop", Strategy.STOP, Strategy.THISBAR, Strategy.ALL, nStop);
} else if (bTargetBreach == false) { // Profit Target
/*****
If we get to this else statement, it means we did not get stopped out by
the preceeding conditions and that we have not yet hit the profit target.
Now check for a breach of the profit target.
*****/
var nLot = Math.abs(Math.round(Strategy.getPositionSize()/2));
if (open(0) <= nTarget) {
/*****
Bar opened below the profit target level, close half of the position
with a market order at the open. To get the number of shares or contracts
to sell, get the current position size, divide by 2, round the result to
ensure that we pass a whole number to the .doCover() call. When the position
is short, .getPositionSize() returns a negative number so the result above also
needs to be converted to a positive number by taking the absolute value of the
result (Math.abs() ). Also set the Boolean bTargetBreach variable to true to
prevent subsequent bars from executing another profit target trade. This is
only allowed to occur once for each position taken.
*****/
bTargetBreach = true;
Strategy.doCover("Profit Target", Strategy.MARKET, Strategy.THISBAR, nLot);
} else if (low(0) <= nTarget) {
/*****
If the bar did not open below the target level, check to see if the low of
the bar breached the target level and exit half of the position with a limit
order at the target price. Again, set bTargetBreach to true.
*****/
bTargetBreach = true;
Strategy.doCover("Profit Target", Strategy.LIMIT, Strategy.THISBAR, nLot, nTarget);
}
}
}
} else { // Entry Strategy
/*****
Look for new entry only when the Strategy is not holding a position. By including
this entry logic in this else block, we ensure that we are not in a position when
these entry conditions are evaluated.
*****/
if (high(0) >= nU) { // Long Trade Signal
/*****
If the current bar's high is equal to or greater than the previous bar's Upper
Donchian channel, go long with a limit order at the value of the previous bar's
Upper Donchian value or the open of the bar, which ever is greater.
*****/
var nEntry = Math.max(nU, open(0));
Strategy.doLong("Long Signal", Strategy.LIMIT, Strategy.THISBAR, Strategy.DEFAULT, nEntry );
// Set the initial stop value at the previous bar's value of the
// Middle Donchian channel.
nStop = nM;
// Set the profit target value at 1.5X the difference of the entry price
// and the initial stop price.
nTarget = nEntry + (1.5 * (nEntry - nStop));
} else if (low(0) <= nL) { // Short Trade Signal
/*****
If the current bar's low is equal to or lower than the previous bar's Lower
Donchian channel, go short with a limit order at the value of the previous bar's
Lower Donchian value or the open of the bar, which ever is smaller.
*****/
var nEntry = Math.min(nL, open(0));
Strategy.doShort("Short Signal", Strategy.LIMIT, Strategy.THISBAR, Strategy.DEFAULT, nEntry );
// Set the initial stop value at the previous bar's value of the
// Middle Donchian channel.
nStop = nM;
// Set the profit target value at 1.5X the difference of the entry price
// and the initial stop price.
nTarget = nEntry - (1.5 * (nStop - nEntry));
}
}
if(Strategy.isLong()) {
setBarBgColor(Color.darkgreen);
} else if(Strategy.isShort()) {
setBarBgColor(Color.maroon);
}
// Plot the current bar's Donchian Channel values and the trailing stop.
return new Array(xUpper.getValue(0), xLower.getValue(0), nStop, nTarget);
}