Стратегия Manadi Buy Sell EMA MACD RSI
Стратегия пересечения EMA с подтверждением MACD и фильтром RSI. Входы по рынку и выход по фиксированным процентным стоп-лоссу и тейк-профиту.
Подробности
- Критерии входа: Пересечение EMA с подтверждением MACD и RSI в пределах.
- Длинные/Короткие: Оба направления.
- Критерии выхода: Процентный стоп-лосс или тейк-профит.
- Стопы: Процентные.
- Значения по умолчанию:
FastEmaLength= 9SlowEmaLength= 21RsiLength= 14RsiUpperLong= 70RsiLowerLong= 40RsiUpperShort= 60RsiLowerShort= 30MacdFast= 12MacdSlow= 26MacdSignal= 9TakeProfitPercent= 0.03StopLossPercent= 0.015CandleType= TimeSpan.FromMinutes(1)
- Фильтры:
- Категория: Trend Following
- Направление: Оба
- Индикаторы: EMA, MACD, RSI
- Стопы: Да
- Сложность: Низкая
- Таймфрейм: Внутридневной (1м)
- Сезонность: Нет
- Нейросети: Нет
- Дивергенция: Нет
- Уровень риска: Средний
using System;
using System.Collections.Generic;
using Ecng.Common;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
namespace StockSharp.Samples.Strategies;
public class ManadiBuySellStrategy : Strategy
{
private readonly StrategyParam<int> _fastEmaLength;
private readonly StrategyParam<int> _slowEmaLength;
private readonly StrategyParam<int> _rsiLength;
private readonly StrategyParam<decimal> _takeProfitPercent;
private readonly StrategyParam<decimal> _stopLossPercent;
private readonly StrategyParam<int> _cooldownBars;
private readonly StrategyParam<DataType> _candleType;
private ExponentialMovingAverage _emaFast;
private ExponentialMovingAverage _emaSlow;
private RelativeStrengthIndex _rsi;
private decimal _prevEmaFast;
private decimal _prevEmaSlow;
private decimal _stopPrice;
private decimal _takeProfitPrice;
private int _barsFromTrade;
public int FastEmaLength { get => _fastEmaLength.Value; set => _fastEmaLength.Value = value; }
public int SlowEmaLength { get => _slowEmaLength.Value; set => _slowEmaLength.Value = value; }
public int RsiLength { get => _rsiLength.Value; set => _rsiLength.Value = value; }
public decimal TakeProfitPercent { get => _takeProfitPercent.Value; set => _takeProfitPercent.Value = value; }
public decimal StopLossPercent { get => _stopLossPercent.Value; set => _stopLossPercent.Value = value; }
public int CooldownBars { get => _cooldownBars.Value; set => _cooldownBars.Value = value; }
public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
public ManadiBuySellStrategy()
{
_fastEmaLength = Param(nameof(FastEmaLength), 9);
_slowEmaLength = Param(nameof(SlowEmaLength), 21);
_rsiLength = Param(nameof(RsiLength), 14);
_takeProfitPercent = Param(nameof(TakeProfitPercent), 0.15m);
_stopLossPercent = Param(nameof(StopLossPercent), 0.06m);
_cooldownBars = Param(nameof(CooldownBars), 100);
_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(30).TimeFrame());
}
/// <inheritdoc />
public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
{
return [(Security, CandleType)];
}
/// <inheritdoc />
protected override void OnReseted()
{
base.OnReseted();
_emaFast = null;
_emaSlow = null;
_rsi = null;
_prevEmaFast = 0m;
_prevEmaSlow = 0m;
_stopPrice = 0m;
_takeProfitPrice = 0m;
_barsFromTrade = CooldownBars;
}
/// <inheritdoc />
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_emaFast = new ExponentialMovingAverage { Length = FastEmaLength };
_emaSlow = new ExponentialMovingAverage { Length = SlowEmaLength };
_rsi = new RelativeStrengthIndex { Length = RsiLength };
_prevEmaFast = 0m;
_prevEmaSlow = 0m;
_stopPrice = 0m;
_takeProfitPrice = 0m;
_barsFromTrade = CooldownBars;
var subscription = SubscribeCandles(CandleType);
subscription
.Bind(_emaFast, _emaSlow, _rsi, ProcessCandle)
.Start();
}
private void ProcessCandle(ICandleMessage candle, decimal fastEma, decimal slowEma, decimal rsi)
{
if (candle.State != CandleStates.Finished)
return;
if (!_emaFast.IsFormed || !_emaSlow.IsFormed || !_rsi.IsFormed)
{
_prevEmaFast = fastEma;
_prevEmaSlow = slowEma;
return;
}
if (_prevEmaFast == 0 || _prevEmaSlow == 0)
{
_prevEmaFast = fastEma;
_prevEmaSlow = slowEma;
return;
}
var bullCross = _prevEmaFast <= _prevEmaSlow && fastEma > slowEma;
var bearCross = _prevEmaFast >= _prevEmaSlow && fastEma < slowEma;
var longCondition = bullCross && rsi < 70 && rsi > 40;
var shortCondition = bearCross && rsi > 30 && rsi < 60;
_barsFromTrade++;
var canEnter = _barsFromTrade >= CooldownBars;
if (canEnter && longCondition && Position <= 0)
{
BuyMarket();
var close = candle.ClosePrice;
_stopPrice = close * (1m - StopLossPercent);
_takeProfitPrice = close * (1m + TakeProfitPercent);
_barsFromTrade = 0;
}
else if (canEnter && shortCondition && Position >= 0)
{
SellMarket();
var close = candle.ClosePrice;
_stopPrice = close * (1m + StopLossPercent);
_takeProfitPrice = close * (1m - TakeProfitPercent);
_barsFromTrade = 0;
}
else if (Position > 0)
{
if (candle.LowPrice <= _stopPrice || candle.HighPrice >= _takeProfitPrice)
{
SellMarket();
_barsFromTrade = 0;
}
}
else if (Position < 0)
{
if (candle.HighPrice >= _stopPrice || candle.LowPrice <= _takeProfitPrice)
{
BuyMarket();
_barsFromTrade = 0;
}
}
_prevEmaFast = fastEma;
_prevEmaSlow = slowEma;
}
}
import clr
clr.AddReference("StockSharp.Messages")
clr.AddReference("StockSharp.Algo")
clr.AddReference("StockSharp.Algo.Indicators")
clr.AddReference("StockSharp.Algo.Strategies")
from System import TimeSpan
from StockSharp.Messages import DataType, CandleStates
from StockSharp.Algo.Indicators import ExponentialMovingAverage, RelativeStrengthIndex
from StockSharp.Algo.Strategies import Strategy
class manadi_buy_sell_strategy(Strategy):
"""
Manadi Buy Sell: EMA crossover with RSI filter and percent-based SL/TP.
"""
def __init__(self):
super(manadi_buy_sell_strategy, self).__init__()
self._fast_ema_length = self.Param("FastEmaLength", 9).SetDisplay("Fast EMA", "Fast EMA", "Indicators")
self._slow_ema_length = self.Param("SlowEmaLength", 21).SetDisplay("Slow EMA", "Slow EMA", "Indicators")
self._rsi_length = self.Param("RsiLength", 14).SetDisplay("RSI", "RSI period", "Indicators")
self._take_profit_pct = self.Param("TakeProfitPercent", 0.15).SetDisplay("TP %", "Take profit percent", "Risk")
self._stop_loss_pct = self.Param("StopLossPercent", 0.06).SetDisplay("SL %", "Stop loss percent", "Risk")
self._cooldown_bars = self.Param("CooldownBars", 100).SetDisplay("Cooldown", "Bars between trades", "Risk")
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromMinutes(30))).SetDisplay("Candle Type", "Candles", "General")
self._prev_fast = 0.0
self._prev_slow = 0.0
self._stop_price = 0.0
self._take_profit_price = 0.0
self._bars_from_trade = 100
@property
def candle_type(self):
return self._candle_type.Value
def OnReseted(self):
super(manadi_buy_sell_strategy, self).OnReseted()
self._prev_fast = 0.0
self._prev_slow = 0.0
self._stop_price = 0.0
self._take_profit_price = 0.0
self._bars_from_trade = self._cooldown_bars.Value
def OnStarted2(self, time):
super(manadi_buy_sell_strategy, self).OnStarted2(time)
self._prev_fast = 0.0
self._prev_slow = 0.0
self._stop_price = 0.0
self._take_profit_price = 0.0
self._bars_from_trade = self._cooldown_bars.Value
self._ema_fast = ExponentialMovingAverage()
self._ema_fast.Length = self._fast_ema_length.Value
self._ema_slow = ExponentialMovingAverage()
self._ema_slow.Length = self._slow_ema_length.Value
self._rsi = RelativeStrengthIndex()
self._rsi.Length = self._rsi_length.Value
subscription = self.SubscribeCandles(self.candle_type)
subscription.Bind(self._ema_fast, self._ema_slow, self._rsi, self._process_candle).Start()
area = self.CreateChartArea()
if area is not None:
self.DrawCandles(area, subscription)
self.DrawIndicator(area, self._ema_fast)
self.DrawIndicator(area, self._ema_slow)
self.DrawOwnTrades(area)
def _process_candle(self, candle, fast_val, slow_val, rsi_val):
if candle.State != CandleStates.Finished:
return
fast = float(fast_val)
slow = float(slow_val)
rsi = float(rsi_val)
if not self._ema_fast.IsFormed or not self._ema_slow.IsFormed or not self._rsi.IsFormed:
self._prev_fast = fast
self._prev_slow = slow
return
if self._prev_fast == 0.0 or self._prev_slow == 0.0:
self._prev_fast = fast
self._prev_slow = slow
return
close = float(candle.ClosePrice)
bull_cross = self._prev_fast <= self._prev_slow and fast > slow
bear_cross = self._prev_fast >= self._prev_slow and fast < slow
long_cond = bull_cross and rsi < 70 and rsi > 40
short_cond = bear_cross and rsi > 30 and rsi < 60
self._bars_from_trade += 1
can_enter = self._bars_from_trade >= self._cooldown_bars.Value
sl_pct = float(self._stop_loss_pct.Value)
tp_pct = float(self._take_profit_pct.Value)
if can_enter and long_cond and self.Position <= 0:
self.BuyMarket()
self._stop_price = close * (1.0 - sl_pct)
self._take_profit_price = close * (1.0 + tp_pct)
self._bars_from_trade = 0
elif can_enter and short_cond and self.Position >= 0:
self.SellMarket()
self._stop_price = close * (1.0 + sl_pct)
self._take_profit_price = close * (1.0 - tp_pct)
self._bars_from_trade = 0
elif self.Position > 0:
if float(candle.LowPrice) <= self._stop_price or float(candle.HighPrice) >= self._take_profit_price:
self.SellMarket()
self._bars_from_trade = 0
elif self.Position < 0:
if float(candle.HighPrice) >= self._stop_price or float(candle.LowPrice) <= self._take_profit_price:
self.BuyMarket()
self._bars_from_trade = 0
self._prev_fast = fast
self._prev_slow = slow
def CreateClone(self):
return manadi_buy_sell_strategy()