Description
Hello friends, today I want to share a simple cBot that should be an example to show that simplicity is always the best solution. Developers can use the Template as a starting point, in fact only the logic of the triggers needs to be changed, the rest is handled. The trigger or market entry is done by just 2 simple EMAs, the real difference is the management of this trigger which can be customised as you see fit.
DESCRIPTION
Identity
- EMA Power 1.0.7 : The name and version of the cBot, this parameter contains the link to our website, you can change this in case you want to save a new preset.
- Label ( Magic Name ) : Similar to number magic for MetaTrader, it is used to identify a strategy.
- Preset information : Some references to the preset in use, useful if you want to specify a specific back test period.
Strategy
- Open Trade Type : The type of trade you want to open, e.g. Buy positions only.
- Stop Loss : The number of pips for the stop loss, e.g. 10 equals 10 pips.
- Take Profit R:R 1:? : The risk: reward ratio, e.g. 2 means that the take profit will be 2 times the stop loss, so if the stop loss is 10 the take profit will be 20.
- Close On Trigger? : With this option it is possible to close trades with the trigger, only those against the trigger, e.g. if we have sell and buy positions in the market when there is a new buy trigger it will only close sell positions.
- Use BreakEven? : Enables the breakeven protection check of positions, taking into consideration the parameters entered in the BreakEven section.
- Use Trailing? : Enables the trailing protection check of positions, taking into consideration the parameters entered in the Trailing section.
- Use Deviation Martingala? (bypass all) : Enables the deviation martingala, taking into consideration the parameters entered in the Deviation Martingala section, enabling this feature excludes controls for recovery positions but not triggers.
Pause
- From (18.0 = 18:00) : This parameter identifies the start time of the break, e.g. 18.0 the cBot will no longer open positions from 18:00.
- To (8.20 = 08:20) : Marks the end of the pause, e.g. 8.20 the cBot will start opening new trades from 08:20.
Filters
- Max Spread allowed : Only triggers within a certain limit are considered, e.g. 1.5 will only consider spreads equal to or less than 1.5 pips.
- Max GAP Allowed (pips) : Protects against high volatility, e.g. if the value is 1 then all trades that have a GAP greater than 1 will be ignored, GAP is the distance between the close of the previous candle and the open of the current one which must not exceed 1 pip.
- Max Number of Trades : The maximum number of trades generated by the triggers, e.g. if the value is 1 after the first trigger no more trades will be opened if there is already 1 trade in the market, no martingale deviation trades are calculated.
Money Target
- Percentage (zero = disabled) : If the value of this parameter is equal to zero then no Money Target is considered, but if for example it is equal to 1 then it means that it will close all positions at the moment when the net profit reaches the value of 1% of the current balance.
- Minimum Trades to Activation : The minimum number of trades in the market before the control starts, e.g. if the value was 2 then the money target would only be controlled if there were 2 trades or more.
Money Management
- Fixed Lots (bypass all Capital) : If the value of this parameter is equal to zero then it will not enter fixed lots, otherwise it will exclude any calculation, e.g. if the value was 0.02 then it will always open with 0.02 lots.
- Capital : The capital with which to do the risk calculation if Fixed Lots is zero, you can choose between Balance or Current Equity.
- % Risk : The percentage of risk to be calculated in proportion to the selected Capital.
- Pips To Calculate ( empty = stoploss ) : To know how many lots to calculate we need to have the distance in pips, this value serves this purpose, if a value of zero is entered then the cBot will take into consideration the value of the stoploss, if this is also set to zero then a fictitious value of 100 pips will be considered.
Drawdown
- % Max (zero = disabled) : The percentage in proportion to the Maximum Acceptable Loss Balance, if for example the value was 30 out of €1000 Balance then you would close all positions if losses exceed -€300, this is not taken into consideration if in a martingale.
EMA
- Fast : The number of periods to be considered of Exponential Moving Average fast.
- Slow : The number of periods to be considered of Exponential Moving Average slow.
Break Even
- Activation (zero = disabled) : If the value of this parameter is equal to zero then it is not taken into consideration even if it is enabled, it is the number of pips beyond which the control is activated, for example if it were 12 then when 12 pips are reached it moves the stoploss to entry price + the value entered in the next Distance parameter.
- Distance : The number of pips to be added to the trade's entry price.
Trailing
- Activation (zero = disabled) : If the value of this parameter is equal to zero then it is not taken into consideration even if it is enabled, it is the number of pips beyond which the control is activated, for example if it were 12 then when 12 pips are reached it moves the stoploss to high price (if buy) - the value entered in the next Distance parameter and this follow the price.
- Distance : The number of pips to be added to the trade's entry price.
Deviation Martingala
- Multiplier (zero = disabled) : If the value of this parameter is zero then the control will not be considered even if enabled, it is the number of times the volume of the previous trade is multiplied to create exponentiality as positions increase.
- Max Consecutive Loss (zero = infinite) : If the value of this parameter is equal to zero then it will open an unlimited number of consecutive losing trades, otherwise if it is greater than zero e.g. 3 it would mean that it will get a maximum of 3 consecutive losses.
I suggest you optimize only the parameters you need, for example it may be important to optimize the number of EMA periods, or scenarios where there are recoveries with Martingale but also without, this task is up to you. As mentioned at the beginning, use this template as an example of development by changing only the triggers, do not forget to share interesting presets and follow cTrader guru on all social networks.
- Homepage : https://ctrader.guru/
- Telegram : https://t.me/ctraderguru
- Twitter : https://twitter.com/cTraderGURU/
- Facebook : https://www.facebook.com/ctrader.guru/
- YouTube : https://www.youtube.com/cTraderGuru
- GitHub : https://github.com/ctrader-guru
/* --> cTrader Guru | Template 'Extend cBot' 1.0.8
Homepage : https://ctrader.guru/
Telegram : https://t.me/ctraderguru
Twitter : https://twitter.com/cTraderGURU/
Facebook : https://www.facebook.com/ctrader.guru/
YouTube : https://www.youtube.com/cTraderGuru
GitHub : https://github.com/ctrader-guru
Edit only :
. NAME and VERSION of this cbot according to cBot Name "public class ExtendcBot : Strategy{...}"
. All methods of "public class Strategy : Robot{...}"
. Then intialize "public void StrategyInitialize(){...}";
. If necessary, add the configuration parameters of any indicators
*/
using System;
using System.Linq;
using cAlgo.API;
using cAlgo.API.Indicators;
using cAlgo.API.Internals;
using cAlgo.Indicators;
using System.Globalization;
namespace cAlgo
{
/// <summary>
/// <para><b>cTrader Guru | Extensios</b></para>
/// <para>A group of generic extensions that make the developer's life easier</para>
/// </summary>
public static class Extensions
{
#region Enum
/// <summary>
/// List of available colours to be displayed in the options
/// </summary>
public enum ColorNameEnum
{
AliceBlue,
AntiqueWhite,
Aqua,
Aquamarine,
Azure,
Beige,
Bisque,
Black,
BlanchedAlmond,
Blue,
BlueViolet,
Brown,
BurlyWood,
CadetBlue,
Chartreuse,
Chocolate,
Coral,
CornflowerBlue,
Cornsilk,
Crimson,
Cyan,
DarkBlue,
DarkCyan,
DarkGoldenrod,
DarkGray,
DarkGreen,
DarkKhaki,
DarkMagenta,
DarkOliveGreen,
DarkOrange,
DarkOrchid,
DarkRed,
DarkSalmon,
DarkSeaGreen,
DarkSlateBlue,
DarkSlateGray,
DarkTurquoise,
DarkViolet,
DeepPink,
DeepSkyBlue,
DimGray,
DodgerBlue,
Firebrick,
FloralWhite,
ForestGreen,
Fuchsia,
Gainsboro,
GhostWhite,
Gold,
Goldenrod,
Gray,
Green,
GreenYellow,
Honeydew,
HotPink,
IndianRed,
Indigo,
Ivory,
Khaki,
Lavender,
LavenderBlush,
LawnGreen,
LemonChiffon,
LightBlue,
LightCoral,
LightCyan,
LightGoldenrodYellow,
LightGray,
LightGreen,
LightPink,
LightSalmon,
LightSeaGreen,
LightSkyBlue,
LightSlateGray,
LightSteelBlue,
LightYellow,
Lime,
LimeGreen,
Linen,
Magenta,
Maroon,
MediumAquamarine,
MediumBlue,
MediumOrchid,
MediumPurple,
MediumSeaGreen,
MediumSlateBlue,
MediumSpringGreen,
MediumTurquoise,
MediumVioletRed,
MidnightBlue,
MintCream,
MistyRose,
Moccasin,
NavajoWhite,
Navy,
OldLace,
Olive,
OliveDrab,
Orange,
OrangeRed,
Orchid,
PaleGoldenrod,
PaleGreen,
PaleTurquoise,
PaleVioletRed,
PapayaWhip,
PeachPuff,
Peru,
Pink,
Plum,
PowderBlue,
Purple,
Red,
RosyBrown,
RoyalBlue,
SaddleBrown,
Salmon,
SandyBrown,
SeaGreen,
SeaShell,
Sienna,
Silver,
SkyBlue,
SlateBlue,
SlateGray,
Snow,
SpringGreen,
SteelBlue,
Tan,
Teal,
Thistle,
Tomato,
Transparent,
Turquoise,
Violet,
Wheat,
White,
WhiteSmoke,
Yellow,
YellowGreen
}
/// <summary>
/// List of capital for Money Management
/// </summary>
public enum CapitalTo
{
Balance,
Equity
}
public enum OpenTradeType
{
All,
Buy,
Sell
}
#endregion
#region Helper
/// <param name="colorName">The name of the colour to be converted, example : ColorNameEnum.Black</param>
/// <returns>Color : The colour converted to the correct format</returns>
public static Color ColorFromEnum(ColorNameEnum colorName)
{
return Color.FromName(colorName.ToString("G"));
}
#endregion
#region Bars
/// <summary>
/// It searches within the bars loaded in memory for the precise opening.
/// </summary>
/// <param name="thisTime">The precise date and time of the candle to be selected</param>
/// <returns>int : The index of the corresponding candle or if not found -1</returns>
public static int GetIndexByDate(this Bars thisBars, DateTime thisTime)
{
for (int i = thisBars.OpenTimes.Count - 1; i >= 0; i--)
{
if (thisTime == thisBars.OpenTimes[i])
return i;
}
return -1;
}
/// <param name="thisDigits">Symbol Digits</param>
/// <returns>double : GAP from last candle and current, example : 0,00350</returns>
public static double LastGAP(this Bars thisBars, int thisDigits = 5)
{
return Math.Round(Math.Abs(thisBars.ClosePrices.Last(1) - thisBars.OpenPrices.Last(0)), thisDigits);
}
#endregion
#region Bar
/// <param name="thisDigits">Symbol Digits</param>
/// <returns>double : The body of selected candle, example : 0,00350</returns>
public static double Body(this Bar thisBar, int thisDigits = 5)
{
return Math.Round(Math.Abs(thisBar.Close - thisBar.Open), thisDigits);
}
/// <returns>bool : True if the candle is bullish</returns>
public static bool IsBullish(this Bar thisBar)
{
return thisBar.Close > thisBar.Open;
}
/// <returns>bool : True if the candle is bearish</returns>
public static bool IsBearish(this Bar thisBar)
{
return thisBar.Close < thisBar.Open;
}
/// <returns>bool : True if the candle is a doji</returns>
public static bool IsDoji(this Bar thisBar)
{
return thisBar.Close == thisBar.Open;
}
#endregion
#region Symbol
/// <param name="Pips">The number of pips in the format Digits</param>
/// <returns>double : The representation of Digits in pips, example 0,00035 = 3,5 pips</returns>
public static double DigitsToPips(this Symbol thisSymbol, double Pips)
{
return Math.Round(Pips / thisSymbol.PipSize, 2);
}
/// <param name="Pips">The number of pips in Double (2) format</param>
/// <returns>double : The representation of Pips in Digits, example 3,5 = 0,00035 pips</returns>
public static double PipsToDigits(this Symbol thisSymbol, double Pips)
{
return Math.Round(Pips * thisSymbol.PipSize, thisSymbol.Digits);
}
/// <returns>double : A correct rappresantation of Spread</returns>
public static double RealSpread(this Symbol thisSymbol)
{
return Math.Round(thisSymbol.Spread / thisSymbol.PipSize, 2);
}
#endregion
#region TimeFrame
/// <returns>int : Returns the current timeframe in minutes</returns>
public static int ToMinutes(this TimeFrame thisTimeFrame)
{
if (thisTimeFrame == TimeFrame.Daily)
return 60 * 24;
if (thisTimeFrame == TimeFrame.Day2)
return 60 * 24 * 2;
if (thisTimeFrame == TimeFrame.Day3)
return 60 * 24 * 3;
if (thisTimeFrame == TimeFrame.Hour)
return 60;
if (thisTimeFrame == TimeFrame.Hour12)
return 60 * 12;
if (thisTimeFrame == TimeFrame.Hour2)
return 60 * 2;
if (thisTimeFrame == TimeFrame.Hour3)
return 60 * 3;
if (thisTimeFrame == TimeFrame.Hour4)
return 60 * 4;
if (thisTimeFrame == TimeFrame.Hour6)
return 60 * 6;
if (thisTimeFrame == TimeFrame.Hour8)
return 60 * 8;
if (thisTimeFrame == TimeFrame.Minute)
return 1;
if (thisTimeFrame == TimeFrame.Minute10)
return 10;
if (thisTimeFrame == TimeFrame.Minute15)
return 15;
if (thisTimeFrame == TimeFrame.Minute2)
return 2;
if (thisTimeFrame == TimeFrame.Minute20)
return 20;
if (thisTimeFrame == TimeFrame.Minute3)
return 3;
if (thisTimeFrame == TimeFrame.Minute30)
return 30;
if (thisTimeFrame == TimeFrame.Minute4)
return 4;
if (thisTimeFrame == TimeFrame.Minute45)
return 45;
if (thisTimeFrame == TimeFrame.Minute5)
return 5;
if (thisTimeFrame == TimeFrame.Minute6)
return 6;
if (thisTimeFrame == TimeFrame.Minute7)
return 7;
if (thisTimeFrame == TimeFrame.Minute8)
return 8;
if (thisTimeFrame == TimeFrame.Minute9)
return 9;
if (thisTimeFrame == TimeFrame.Monthly)
return 60 * 24 * 30;
if (thisTimeFrame == TimeFrame.Weekly)
return 60 * 24 * 7;
return 0;
}
#endregion
#region DateTime
/// <param name="Culture">Localization of double value</param>
/// <returns>double : Time representation in double format (example : 10:34:07 = 10,34)</returns>
public static double ToDouble(this DateTime thisDateTime, string Culture = "en-EN")
{
string nowHour = (thisDateTime.Hour < 10) ? string.Format("0{0}", thisDateTime.Hour) : string.Format("{0}", thisDateTime.Hour);
string nowMinute = (thisDateTime.Minute < 10) ? string.Format("0{0}", thisDateTime.Minute) : string.Format("{0}", thisDateTime.Minute);
return string.Format("{0}.{1}", nowHour, nowMinute).ToDouble(Culture);
}
#endregion
#region Position
/// <param name="Symbol">The symbol for which the changes are being made</param>
/// <param name="Activation">The number of pips to activate the control</param>
/// <param name="Distance">The number of pips for the distance of the stoploss displacement</param>
/// <returns>TradeResult : if no changes are made returns <b>null</b></returns>
public static TradeResult BreakEven(this Position thisPosition, Symbol Symbol, double Activation, double Distance)
{
if (Activation == 0)
return null;
double breakeven;
double distance = Symbol.PipsToDigits(Distance);
double activation = Symbol.PipsToDigits(Activation);
TradeResult result = null;
switch (thisPosition.TradeType)
{
case TradeType.Buy:
breakeven = Math.Round(thisPosition.EntryPrice + distance, Symbol.Digits);
if (thisPosition.StopLoss == breakeven || thisPosition.TakeProfit == breakeven)
break;
if (Symbol.Bid > breakeven && Symbol.Bid >= thisPosition.EntryPrice + activation && (thisPosition.StopLoss == null || thisPosition.StopLoss < breakeven))
{
result = thisPosition.ModifyStopLossPrice(breakeven);
}
break;
case TradeType.Sell:
breakeven = Math.Round(thisPosition.EntryPrice - distance, Symbol.Digits);
if (thisPosition.StopLoss == breakeven || thisPosition.TakeProfit == breakeven)
break;
if (Symbol.Bid < breakeven && Symbol.Ask <= thisPosition.EntryPrice - activation && (thisPosition.StopLoss == null || thisPosition.StopLoss > breakeven))
{
result = thisPosition.ModifyStopLossPrice(breakeven);
}
break;
}
return result;
}
/// <param name="Symbol">The symbol for which the changes are being made</param>
/// <param name="Activation">The number of pips to activate the control</param>
/// <param name="Distance">The number of pips for the distance of the stoploss displacement</param>
/// <returns>TradeResult : if no changes are made returns <b>null</b></returns>
public static TradeResult TrailingStop(this Position thisPosition, Symbol Symbol, double Activation, double Distance)
{
if (Activation == 0)
return null;
double trailing;
double distance = Symbol.PipsToDigits(Distance);
double activation = Symbol.PipsToDigits(Activation);
TradeResult result = null;
switch (thisPosition.TradeType)
{
case TradeType.Buy:
trailing = Math.Round(Symbol.Bid - distance, Symbol.Digits);
if (thisPosition.StopLoss == trailing || thisPosition.TakeProfit == trailing)
break;
if ((Symbol.Bid >= (thisPosition.EntryPrice + activation)) && (thisPosition.StopLoss == null || thisPosition.StopLoss < trailing))
{
result = thisPosition.ModifyStopLossPrice(trailing);
}
break;
case TradeType.Sell:
trailing = Math.Round(Symbol.Ask + distance, Symbol.Digits);
if (thisPosition.StopLoss == trailing || thisPosition.TakeProfit == trailing)
break;
if ((Symbol.Ask <= (thisPosition.EntryPrice - activation)) && (thisPosition.StopLoss == null || thisPosition.StopLoss > trailing))
{
result = thisPosition.ModifyStopLossPrice(trailing);
}
break;
}
return result;
}
#endregion
#region String
/// <param name="Culture">Localization of double value</param>
/// <returns>double : Time representation in double format (example : "10:34:07" = 10,34)</returns>
public static double ToDouble(this string thisString, string Culture = "en-EN")
{
var culture = CultureInfo.GetCultureInfo(Culture);
return double.Parse(thisString.Replace(',', '.').ToString(CultureInfo.InvariantCulture), culture);
}
#endregion
#region Class
public class MonenyManagement
{
private readonly double _minSize = 0.01;
private double _percentage = 0;
private double _fixedSize = 0;
private double _pipToCalc = 30;
private IAccount _account = null;
public readonly Symbol Symbol;
public CapitalTo CapitalType = CapitalTo.Balance;
public double Percentage
{
get { return _percentage; }
set { _percentage = (value > 0 && value <= 100) ? value : 0; }
}
public double FixedSize
{
get { return _fixedSize; }
set { _fixedSize = (value >= _minSize) ? value : 0; }
}
public double PipToCalc
{
get { return _pipToCalc; }
set { _pipToCalc = (value > 0) ? value : 100; }
}
public double Capital
{
get
{
switch (CapitalType)
{
case CapitalTo.Equity:
return _account.Equity;
default:
return _account.Balance;
}
}
}
public MonenyManagement(IAccount NewAccount, CapitalTo NewCapitalTo, double NewPercentage, double NewFixedSize, double NewPipToCalc, Symbol NewSymbol)
{
_account = NewAccount;
Symbol = NewSymbol;
CapitalType = NewCapitalTo;
Percentage = NewPercentage;
FixedSize = NewFixedSize;
PipToCalc = NewPipToCalc;
}
public double GetLotSize()
{
if (FixedSize > 0)
return FixedSize;
double moneyrisk = Capital / 100 * Percentage;
double sl_double = PipToCalc * Symbol.PipSize;
// --> 0.01 = microlotto double lots = Math.Round(Symbol.VolumeInUnitsToQuantity(moneyrisk / ((sl_double * Symbol.TickValue) / Symbol.TickSize)), 2);
// --> volume 1K = 1000 Math.Round((moneyrisk / ((sl_double * Symbol.TickValue) / Symbol.TickSize)), 2);
double lots = Math.Round(Symbol.VolumeInUnitsToQuantity(moneyrisk / ((sl_double * Symbol.TickValue) / Symbol.TickSize)), 2);
if (lots < _minSize)
return _minSize;
return lots;
}
}
#endregion
}
}
namespace cAlgo.Robots
{
public class Strategy : Robot
{
public ExponentialMovingAverage FastEMA;
public ExponentialMovingAverage SlowEMA;
/* Override Example
public virtual double EMAFilter { get; set; } // <-- If you want to use parameters, you have to overwrite them.
*/
public bool TriggerBuy
{
get { return FastEMA.Result.Last(2) < SlowEMA.Result.Last(2) && FastEMA.Result.Last(1) > SlowEMA.Result.Last(1); }
}
public bool TriggerSell
{
get { return FastEMA.Result.Last(2) > SlowEMA.Result.Last(2) && FastEMA.Result.Last(1) < SlowEMA.Result.Last(1); }
}
public bool FilterBuy
{
get { return true; }
}
public bool FilterSell
{
get { return true; }
}
public bool Buy
{
get { return FilterBuy && TriggerBuy; }
}
public bool Sell
{
get { return FilterSell && TriggerSell; }
}
}
[Robot(TimeZone = TimeZones.UTC, AccessRights = AccessRights.None)]
public class EMAPower : Strategy
{
#region Identity
public const string NAME = "EMA Power";
public const string VERSION = "1.0.2";
#endregion
#region Params
#region Identity
[Parameter(NAME + " " + VERSION, Group = "Identity", DefaultValue = "https://ctrader.guru/")]
public string ProductInfo { get; set; }
[Parameter("Label ( Magic Name )", Group = "Identity", DefaultValue = NAME)]
public string MyLabel { get; set; }
[Parameter("Preset information", Group = "Identity", DefaultValue = "USDJPY 5m | 29.04.2021 to 29.04.2022 | €1000")]
public string PresetInfo { get; set; }
#endregion
#region Strategy
[Parameter("Open Trade Type", Group = "Strategy", DefaultValue = Extensions.OpenTradeType.Buy)]
public Extensions.OpenTradeType MyOpenTradeType { get; set; }
[Parameter("Stop Loss", Group = "Strategy", DefaultValue = 30, MinValue = 0, Step = 0.1)]
public double StopLoss { get; set; }
[Parameter("Take Profit R:R 1:?", Group = "Strategy", DefaultValue = 5, MinValue = 0, Step = 0.1)]
public double TakeProfitRR { get; set; }
public double TakeProfit
{
get { return Math.Round(StopLoss * TakeProfitRR, 1); }
}
[Parameter("Close On Trigger?", Group = "Strategy", DefaultValue = false)]
public bool CloseOnTrigger { get; set; }
[Parameter("Use BreakEven?", Group = "Strategy", DefaultValue = false)]
public bool UseBreakEven { get; set; }
[Parameter("Use Trailing?", Group = "Strategy", DefaultValue = false)]
public bool UseTrailing { get; set; }
[Parameter("Use Deviation Martingala? (bypass all)", Group = "Strategy", DefaultValue = true)]
public bool UseDM { get; set; }
#endregion
#region Pausa
[Parameter("From (18.0 = 18:00)", Group = "Pause", DefaultValue = 0, MinValue = 0, MaxValue = 23.59, Step = 0.01)]
public double PauseFrom { get; set; }
[Parameter("To (8.20 = 08:20)", Group = "Pause", DefaultValue = 0, MinValue = 0, MaxValue = 23.59, Step = 0.01)]
public double PauseTo { get; set; }
public bool IAmInPause
{
get
{
if (PauseFrom == 0 && PauseTo == 0)
return false;
double now = Server.Time.ToDouble();
bool intraday = (PauseFrom < PauseTo && now >= PauseFrom && now <= PauseTo);
bool overnight = (PauseFrom > PauseTo && ((now >= PauseFrom && now <= 23.59) || now <= PauseTo));
return intraday || overnight;
}
}
#endregion
#region Filters
[Parameter("Max Spread allowed", Group = "Filters", DefaultValue = 1.5, MinValue = 0.1, Step = 0.1)]
public double SpreadToTrigger { get; set; }
[Parameter("Max GAP Allowed (pips)", Group = "Filters", DefaultValue = 2, MinValue = 0, Step = 0.01)]
public double GAP { get; set; }
[Parameter("Max Number of Trades", Group = "Filters", DefaultValue = 1, MinValue = 1, Step = 1)]
public int MaxTrades { get; set; }
#endregion
#region Money Target
[Parameter("Percentage (zero = disabled)", Group = "Money Target", DefaultValue = 0, MinValue = 0, Step = 0.1)]
public double MoneyTargetPercentage { get; set; }
public double MoneyTarget
{
get { return Math.Round((Account.Balance / 100) * MoneyTargetPercentage, 2); }
}
[Parameter("Minimum Trades to Activation", Group = "Money Target", DefaultValue = 1, MinValue = 1, Step = 1)]
public int MoneyTargetTrades { get; set; }
#endregion
#region Money Management
[Parameter("Fixed Lots (bypass all Capital)", Group = "Money Management", DefaultValue = 0, MinValue = 0, Step = 0.01)]
public double FixedLots { get; set; }
[Parameter("Capital", Group = "Money Management", DefaultValue = Extensions.CapitalTo.Balance)]
public Extensions.CapitalTo MyCapital { get; set; }
[Parameter("% Risk", Group = "Money Management", DefaultValue = 1, MinValue = 0.1, Step = 0.1)]
public double MyRisk { get; set; }
[Parameter("Pips To Calculate ( empty = stoploss )", Group = "Money Management", DefaultValue = 30, MinValue = 0, Step = 0.1)]
public double FakeSL { get; set; }
[Parameter("% Max (zero = disabled)", Group = "Drawdown", DefaultValue = 0, MinValue = 0, MaxValue = 100, Step = 0.1)]
public double DDPercentage { get; set; }
#endregion
#region Indicators Setup
[Parameter("Fast", Group = "EMA", DefaultValue = 30, MinValue = 1)]
public int PeriodFastEMA { get; set; }
[Parameter("Slow", Group = "EMA", DefaultValue = 50, MinValue = 2)]
public int PeriodSlowEMA { get; set; }
/* Override Example
[Parameter("Filter", Group = "EMA", DefaultValue = 20, MinValue = 2)]
public override double EMAFilter { get; set; }
*/
#endregion
#region BreakEven
[Parameter("Activation (zero = disabled)", Group = "Break Even", DefaultValue = 10, MinValue = 0, Step = 0.1)]
public double BreakEvenActivation { get; set; }
[Parameter("Distance", Group = "Break Even", DefaultValue = 1.1, MinValue = 0, Step = 0.1)]
public double BreakEvenDistance { get; set; }
#endregion
#region Trailing
[Parameter("Activation (zero = disabled)", Group = "Trailing", DefaultValue = 15, MinValue = 0, Step = 0.1)]
public double TrailingActivation { get; set; }
[Parameter("Distance", Group = "Trailing", DefaultValue = 10, MinValue = 1, Step = 0.1)]
public double TrailingDistance { get; set; }
#endregion
#region Deviation Martingala
[Parameter("Multiplier (zero = disabled)", Group = "Deviation Martingala", DefaultValue = 1.5, MinValue = 0, Step = 0.1)]
public double DMMultiplier { get; set; }
[Parameter("Max Consecutive Loss (zero = infinite)", Group = "Deviation Martingala", DefaultValue = 6, MinValue = 0, Step = 1)]
public int DMMaxLoss { get; set; }
#endregion
#endregion
#region Property
public bool OpenedInThisBar = false;
public double StrategyNetProfit = 0;
public Position[] StrategyPositions =
{
};
Extensions.MonenyManagement MonenyManagement1;
public int ConsecutiveLoss = 0;
public DateTime PreventGlitch;
#endregion
#region cBot Events
public void StrategyInitialize()
{
FastEMA = Indicators.ExponentialMovingAverage(Bars.ClosePrices, PeriodFastEMA);
SlowEMA = Indicators.ExponentialMovingAverage(Bars.ClosePrices, PeriodSlowEMA);
}
public void StrategyRun()
{
bool UsingRecovery = UseDM && DMMultiplier > 0 && ConsecutiveLoss > 0;
bool SharedConditions = !UsingRecovery && !OpenedInThisBar && StrategyPositions.Length < MaxTrades && Bars.LastGAP(Symbol.Digits) <= Symbol.PipsToDigits(GAP) && Symbol.RealSpread() <= SpreadToTrigger;
if (Buy && Sell)
{
Print("Trigger Buy and Sell, strategy error.");
return;
}
MonenyManagement1 = new Extensions.MonenyManagement(Account, MyCapital, MyRisk, FixedLots, FakeSL > 0 ? FakeSL : StopLoss, Symbol);
double lotSize = MonenyManagement1.GetLotSize();
double volumeInUnits = Symbol.QuantityToVolumeInUnits(lotSize);
if (Buy)
{
if (SharedConditions && MyOpenTradeType != Extensions.OpenTradeType.Sell)
{
ExecuteMarketRangeOrder(TradeType.Buy, SymbolName, volumeInUnits, 2, Ask, MyLabel, StopLoss, TakeProfit);
Print("Open on trigger, consecutive loss {0}", ConsecutiveLoss);
}
}
else if (Sell)
{
if (SharedConditions && MyOpenTradeType != Extensions.OpenTradeType.Buy)
{
ExecuteMarketRangeOrder(TradeType.Sell, SymbolName, volumeInUnits, 2, Bid, MyLabel, StopLoss, TakeProfit);
Print("Open on trigger, consecutive loss {0}", ConsecutiveLoss);
}
}
}
protected override void OnStart()
{
Print(NAME, " ", VERSION);
Positions.Opened += _onOpenPositions;
Positions.Closed += _onClosePositions;
StrategyInitialize();
}
protected override void OnTick()
{
bool OnMoneyTargetClose = MoneyTargetPercentage > 0 && StrategyPositions.Length >= MoneyTargetTrades && StrategyNetProfit >= MoneyTarget;
double DDControl = Math.Round((Account.Balance / 100) * DDPercentage, 2) * -1;
bool OnDrawDownClose = DDControl < 0 && StrategyNetProfit <= DDControl;
bool UsingRecovery = UseDM && DMMultiplier > 0 && ConsecutiveLoss > 0;
StrategyNetProfit = 0;
StrategyPositions = Positions.FindAll(MyLabel, SymbolName);
foreach (Position position in StrategyPositions)
{
if (!UsingRecovery)
{
bool OnTriggerClose = CloseOnTrigger && ((Buy && position.TradeType == TradeType.Sell) || (Sell && position.TradeType == TradeType.Buy));
if (OnTriggerClose || OnMoneyTargetClose || OnDrawDownClose)
{
position.Close();
continue;
}
TradeResult result = null;
if (UseBreakEven)
{
result = position.BreakEven(Symbol, BreakEvenActivation, BreakEvenDistance);
if (result != null)
{
if (result.IsSuccessful)
{
// --> Break Even successfully modified!!!
}
else
{
// --> Print("Error: {0}", result.Error);
}
}
}
if (UseTrailing)
{
result = position.TrailingStop(Symbol, TrailingActivation, TrailingDistance);
if (result != null)
{
if (result.IsSuccessful)
{
// --> TrailingStop successfully modified!!!
}
else
{
// --> Print("Error: {0}", result.Error);
}
}
}
}
StrategyNetProfit += position.NetProfit;
}
StrategyRun();
}
protected override void OnBar()
{
OpenedInThisBar = false;
}
protected override void OnStop()
{
Positions.Opened -= _onOpenPositions;
Positions.Closed -= _onClosePositions;
}
#endregion
#region Methods
private void _onOpenPositions(PositionOpenedEventArgs eventArgs)
{
Position position = eventArgs.Position;
if (position.SymbolName != SymbolName || position.Label != MyLabel)
return;
OpenedInThisBar = true;
}
private void _onClosePositions(PositionClosedEventArgs eventArgs)
{
Position position = eventArgs.Position;
if (position.SymbolName != SymbolName || position.Label != MyLabel)
return;
if (PreventGlitch == Server.Time)
{
Print("Glitch when close position, it is probably a bug of the cTrader and not of this cBot because 2 positions (Martingala) were opened simultaneously.");
return;
}
PreventGlitch = Server.Time;
if (position.NetProfit < 0)
{
ConsecutiveLoss++;
bool UseRecovery = UseDM && DMMultiplier > 0 && (DMMaxLoss == 0 || ConsecutiveLoss < DMMaxLoss);
if (UseRecovery)
{
TradeType reversed = (position.TradeType == TradeType.Sell) ? TradeType.Buy : TradeType.Sell;
double tmpSL = position.StopLoss == null ? 0 : Math.Abs(Math.Round(position.EntryPrice - (double)position.StopLoss, Symbol.Digits));
ExecuteMarketOrder(reversed, SymbolName, Symbol.QuantityToVolumeInUnits(Math.Round(position.Quantity * DMMultiplier, 2)), MyLabel, Symbol.DigitsToPips(tmpSL), Symbol.DigitsToPips(tmpSL));
Print("Open Martingala Deviation, consecutive loss {0}", ConsecutiveLoss);
}
else
{
ConsecutiveLoss = 0;
}
}
else
{
ConsecutiveLoss = 0;
}
}
#endregion
}
}
ctrader.guru
Joined on 19.07.2018
- Distribution: Free
- Language: C#
- Trading platform: cTrader Automate
- File name: EMA Power.algo
- Rating: 5
- Installs: 3116
- Modified: 30/04/2022 15:21
Comments
this cbot uses a template that includes methods for educational purposes only, I like to make my students understand how things work, I deliberately left some old methods, which are now natively included, to teach how to extend objects.
If you are good enough, you will know how to modify them.
Update 1.0.2 Increased precision
Update 1.0.1 fixed minor bug
I have the R:R set to 2. But when the Martingale is applied the R:R becomes 1. Why?