Стратегия XD Range Switch переносит в StockSharp советник MetaTrader 5 Exp_XD-RangeSwitch и использует высокоуровневый API платформы. В основе лежит пользовательский индикатор XD-RangeSwitch, который строит попеременно верхнюю и нижнюю границы канала и рисует стрелки в момент смены доминирующей стороны. В зависимости от параметра TradeDirection стратегия может торговать против появившейся стрелки (контртрендовый подход) или в сторону пробоя канала. Размер сделки определяется через базовое свойство Strategy.Volume, а исходные алгоритмы мани-менеджмента заменены штатными средствами управления позицией в StockSharp.
Восстановление индикатора XD-RangeSwitch
Индикатор отслеживает последние Peaks завершённых свечей и вычисляет экстремумы (максимумы и минимумы) в этом окне.
Нижняя граница канала появляется, если цена закрытия текущей свечи выше наибольшего максимума предыдущих Peaks баров. Значение границы соответствует минимальному минимуму в указанном окне вместе с текущей свечой.
Верхняя граница канала появляется, если цена закрытия ниже наименьшего минимума предыдущих Peaks баров. Значение границы равно максимальному максимуму в том же окне.
Если пробоя не произошло, значения канала просто переносятся вперёд.
Как только граница снова становится ненулевой после паузы, стратегия фиксирует стрелку на уровне этой границы — аналог MT5-буферов 2 и 3 в оригинальном советнике.
Обрабатываются только полностью сформированные свечи, что обеспечивает одинаковые значения в реальном времени и при тестах.
Логика торговли
Стратегия подписывается на свечи с таймфрейма CandleType и формирует буферы индикатора в точности как в MT5.
Для каждого нового бара анализируется значение индикатора, которое находится на SignalBar свечей позади (совпадает с аргументом CopyBuffer в исходном коде).
Интерпретация сигналов зависит от параметра TradeDirection:
AgainstSignal воспроизводит поведение по умолчанию — восходящая стрелка открывает лонг и требует закрыть шорт, нисходящая стрелка открывает шорт и закрывает лонг.
WithSignal меняет трактовку: восходящая стрелка воспринимается как выход из лонга и вход в шорт, то есть торговля ведётся в сторону пробоя канала.
Даже без стрелок границы канала используются как сигналы к закрытию позиции, полностью повторяя флаги SELL_Close и BUY_Close.
Закрытия позиций выполняются до открытия новых сделок, чтобы при смене направления сначала обнулить противоположную позицию.
Сделки исполняются рыночными заявками (BuyMarket/SellMarket). При наличии противоположной позиции объём автоматически расширяется, чтобы сначала закрыть текущую экспозицию и только затем открыть позицию в новом направлении.
Управление рисками
Встроено опциональное сопровождение стоп-лосса и тейк-профита через пары параметров UseStopLoss/StopLossPoints и UseTakeProfit/TakeProfitPoints.
Расстояния задаются в абсолютных ценовых единицах, что соответствует "пунктам" из MT5.
Проверка уровней выполняется на каждой закрытой свече по значениям High/Low, чтобы имитировать внутридневное срабатывание.
Если одновременно активны и стоп, и тейк, приоритет отдаётся стоп-лоссу: позиция закрывается при достижении любого из уровней.
Параметры
Параметр
Значение по умолчанию
Описание
CandleType
Свечи H4
Таймфрейм, на котором вычисляется индикатор XD-RangeSwitch.
Peaks
4
Количество экстремумов в окне (длина анализа) индикатора.
SignalBar
1
Сдвиг по завершённым свечам при чтении буферов индикатора.
TradeDirection
AgainstSignal
Выбор между контртрендовой и трендовой интерпретацией стрелок.
AllowBuyEntry / AllowSellEntry
true
Разрешение открывать новые позиции соответствующего типа.
AllowBuyExit / AllowSellExit
true
Разрешение закрывать текущие позиции по сигналу индикатора.
UseStopLoss / StopLossPoints
true / 1000
Включение стоп-лосса и расстояние до него в ценовых пунктах.
UseTakeProfit / TakeProfitPoints
true / 2000
Включение тейк-профита и расстояние до него в ценовых пунктах.
Примечания
Внутренние буферы по максимумам и минимумам ведутся прямо в стратегии, без дополнительных коллекций, что сохраняет логику MT5 и соответствует требованиям по конверсии.
Сигналы учитывают только завершённые свечи. При SignalBar > 0 сделка совершается на следующем баре после того, который сформировал стрелку — так же, как в исходном советнике.
Объём хранимой истории ограничен максимумом из Peaks и SignalBar, плюс небольшой запас, поэтому потребление памяти остаётся стабильным на длинных прогонах.
Стандартные настройки совпадают с MT5: H4, Peaks = 4, SignalBar = 1, контртрендовый режим и коридор риска 1000/2000 пунктов.
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;
/// <summary>
/// XD-RangeSwitch strategy using Highest/Lowest channel breakouts.
/// Buys on breakout above channel high, sells on breakout below channel low.
/// Uses stop-loss and take-profit for risk management.
/// </summary>
public class XdRangeSwitchStrategy : Strategy
{
private readonly StrategyParam<int> _channelPeriod;
private readonly StrategyParam<int> _stopLossPoints;
private readonly StrategyParam<int> _takeProfitPoints;
private Highest _highest;
private Lowest _lowest;
private decimal _prevHigh;
private decimal _prevLow;
private decimal _entryPrice;
private int _cooldown;
/// <summary>
/// Channel lookback period.
/// </summary>
public int ChannelPeriod
{
get => _channelPeriod.Value;
set => _channelPeriod.Value = value;
}
/// <summary>
/// Stop-loss distance in price steps.
/// </summary>
public int StopLossPoints
{
get => _stopLossPoints.Value;
set => _stopLossPoints.Value = value;
}
/// <summary>
/// Take-profit distance in price steps.
/// </summary>
public int TakeProfitPoints
{
get => _takeProfitPoints.Value;
set => _takeProfitPoints.Value = value;
}
/// <summary>
/// Initializes strategy parameters.
/// </summary>
public XdRangeSwitchStrategy()
{
_channelPeriod = Param(nameof(ChannelPeriod), 200)
.SetGreaterThanZero()
.SetDisplay("Channel Period", "Lookback for highest/lowest channel", "Indicator");
_stopLossPoints = Param(nameof(StopLossPoints), 200)
.SetNotNegative()
.SetDisplay("Stop Loss", "Stop-loss distance in price steps", "Risk");
_takeProfitPoints = Param(nameof(TakeProfitPoints), 400)
.SetNotNegative()
.SetDisplay("Take Profit", "Take-profit distance in price steps", "Risk");
}
/// <inheritdoc />
public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
{
yield return (Security, TimeSpan.FromMinutes(5).TimeFrame());
}
/// <inheritdoc />
protected override void OnReseted()
{
base.OnReseted();
_highest = null;
_lowest = null;
_prevHigh = 0;
_prevLow = 0;
_entryPrice = 0;
_cooldown = 0;
}
/// <inheritdoc />
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_highest = new Highest { Length = ChannelPeriod };
_lowest = new Lowest { Length = ChannelPeriod };
var subscription = SubscribeCandles(TimeSpan.FromMinutes(5).TimeFrame());
subscription.Bind(_highest, _lowest, ProcessCandle);
subscription.Start();
}
private void ProcessCandle(ICandleMessage candle, decimal highValue, decimal lowValue)
{
if (candle.State != CandleStates.Finished)
return;
if (!_highest.IsFormed || !_lowest.IsFormed)
{
_prevHigh = highValue;
_prevLow = lowValue;
return;
}
if (_cooldown > 0)
{
_cooldown--;
_prevHigh = highValue;
_prevLow = lowValue;
return;
}
var close = candle.ClosePrice;
var step = Security?.PriceStep ?? 1m;
// Check SL/TP
if (Position > 0 && _entryPrice > 0)
{
if (StopLossPoints > 0 && close <= _entryPrice - StopLossPoints * step)
{
SellMarket();
_entryPrice = 0;
_cooldown = 30;
_prevHigh = highValue;
_prevLow = lowValue;
return;
}
if (TakeProfitPoints > 0 && close >= _entryPrice + TakeProfitPoints * step)
{
SellMarket();
_entryPrice = 0;
_cooldown = 30;
_prevHigh = highValue;
_prevLow = lowValue;
return;
}
}
else if (Position < 0 && _entryPrice > 0)
{
if (StopLossPoints > 0 && close >= _entryPrice + StopLossPoints * step)
{
BuyMarket();
_entryPrice = 0;
_cooldown = 30;
_prevHigh = highValue;
_prevLow = lowValue;
return;
}
if (TakeProfitPoints > 0 && close <= _entryPrice - TakeProfitPoints * step)
{
BuyMarket();
_entryPrice = 0;
_cooldown = 30;
_prevHigh = highValue;
_prevLow = lowValue;
return;
}
}
// Breakout above previous channel high
if (close > _prevHigh && _prevHigh > 0 && Position <= 0)
{
if (Position < 0)
BuyMarket();
BuyMarket();
_entryPrice = close;
_cooldown = 30;
}
// Breakout below previous channel low
else if (close < _prevLow && _prevLow > 0 && Position >= 0)
{
if (Position > 0)
SellMarket();
SellMarket();
_entryPrice = close;
_cooldown = 30;
}
_prevHigh = highValue;
_prevLow = lowValue;
}
}
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 Highest, Lowest
from StockSharp.Algo.Strategies import Strategy
class xd_range_switch_strategy(Strategy):
def __init__(self):
super(xd_range_switch_strategy, self).__init__()
self._channel_period = self.Param("ChannelPeriod", 200) \
.SetDisplay("Channel Period", "Lookback for highest/lowest channel", "Indicator")
self._stop_loss_points = self.Param("StopLossPoints", 200) \
.SetDisplay("Stop Loss", "Stop-loss distance in price steps", "Risk")
self._take_profit_points = self.Param("TakeProfitPoints", 400) \
.SetDisplay("Take Profit", "Take-profit distance in price steps", "Risk")
self._highest = None
self._lowest = None
self._prev_high = 0.0
self._prev_low = 0.0
self._entry_price = 0.0
self._cooldown = 0
@property
def channel_period(self):
return self._channel_period.Value
@property
def stop_loss_points(self):
return self._stop_loss_points.Value
@property
def take_profit_points(self):
return self._take_profit_points.Value
def OnReseted(self):
super(xd_range_switch_strategy, self).OnReseted()
self._highest = None
self._lowest = None
self._prev_high = 0.0
self._prev_low = 0.0
self._entry_price = 0.0
self._cooldown = 0
def OnStarted2(self, time):
super(xd_range_switch_strategy, self).OnStarted2(time)
self._highest = Highest()
self._highest.Length = self.channel_period
self._lowest = Lowest()
self._lowest.Length = self.channel_period
subscription = self.SubscribeCandles(DataType.TimeFrame(TimeSpan.FromMinutes(5)))
subscription.Bind(self._highest, self._lowest, self._process_candle)
subscription.Start()
def _process_candle(self, candle, high_value, low_value):
if candle.State != CandleStates.Finished:
return
h_val = float(high_value)
l_val = float(low_value)
if not self._highest.IsFormed or not self._lowest.IsFormed:
self._prev_high = h_val
self._prev_low = l_val
return
if self._cooldown > 0:
self._cooldown -= 1
self._prev_high = h_val
self._prev_low = l_val
return
close = float(candle.ClosePrice)
step = float(self.Security.PriceStep) if self.Security is not None and self.Security.PriceStep is not None else 1.0
# Check SL/TP
if self.Position > 0 and self._entry_price > 0:
if self.stop_loss_points > 0 and close <= self._entry_price - self.stop_loss_points * step:
self.SellMarket()
self._entry_price = 0.0
self._cooldown = 30
self._prev_high = h_val
self._prev_low = l_val
return
if self.take_profit_points > 0 and close >= self._entry_price + self.take_profit_points * step:
self.SellMarket()
self._entry_price = 0.0
self._cooldown = 30
self._prev_high = h_val
self._prev_low = l_val
return
elif self.Position < 0 and self._entry_price > 0:
if self.stop_loss_points > 0 and close >= self._entry_price + self.stop_loss_points * step:
self.BuyMarket()
self._entry_price = 0.0
self._cooldown = 30
self._prev_high = h_val
self._prev_low = l_val
return
if self.take_profit_points > 0 and close <= self._entry_price - self.take_profit_points * step:
self.BuyMarket()
self._entry_price = 0.0
self._cooldown = 30
self._prev_high = h_val
self._prev_low = l_val
return
# Breakout above previous channel high
if close > self._prev_high and self._prev_high > 0 and self.Position <= 0:
if self.Position < 0:
self.BuyMarket()
self.BuyMarket()
self._entry_price = close
self._cooldown = 30
# Breakout below previous channel low
elif close < self._prev_low and self._prev_low > 0 and self.Position >= 0:
if self.Position > 0:
self.SellMarket()
self.SellMarket()
self._entry_price = close
self._cooldown = 30
self._prev_high = h_val
self._prev_low = l_val
def CreateClone(self):
return xd_range_switch_strategy()