SR Breakout отслеживает уровни поддержки и сопротивления, рассчитанные индикатором Donchian Channels на двух таймфреймах (H1 и H4). Когда завершившаяся свеча закрывается выше сопротивления или ниже поддержки, стратегия записывает информационное сообщение в лог. Реализация повторяет логику уведомлений исходного эксперта MQL4 и не совершает сделок.
Как это работает
Создаются две подписки на свечи: для таймфрейма 1 час и для таймфрейма 4 часа.
Каждая подписка связывается со своим индикатором DonchianChannels с настраиваемым периодом (по умолчанию 26).
После формирования индикатора стратегия запоминает цену закрытия предыдущей свечи на каждом таймфрейме.
При закрытии очередной свечи её цена сравнивается с верхней и нижней границей Donchian:
Если цена переходит из области ниже верхней границы в область выше неё, в лог пишется сообщение «пересечение сопротивления».
Если цена переходит из области выше нижней границы в область ниже неё, в лог пишется сообщение «пересечение поддержки».
Для оповещений используются вызовы LogInfo, что соответствует поведению исходного советника.
Параметры
Имя
Описание
Значение по умолчанию
LookbackLength
Количество свечей для расчёта уровней поддержки/сопротивления Donchian.
26
Hour1CandleType
Тип свечей для часового таймфрейма.
TimeFrame(1h)
Hour4CandleType
Тип свечей для четырёхчасового таймфрейма.
TimeFrame(4h)
Сигналы
Пробой H1 – лог при пересечении ценой закрытия часовой свечи сопротивления или поддержки.
Пробой H4 – лог при пересечении ценой закрытия четырёхчасовой свечи сопротивления или поддержки.
Примечания
Стратегия предназначена для оповещений и не отправляет торговых заявок.
Для корректной работы индикатора Donchian необходимы данные High и Low в обеих подписках.
При необходимости измените период расчёта или типы свечей под конкретный инструмент.
using System;
using System.Collections.Generic;
using Ecng.Common;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
namespace StockSharp.Samples.Strategies;
/// <summary>
/// Support and resistance breakout strategy using Donchian channels.
/// Buys when price breaks above resistance (upper band), sells when breaks below support (lower band).
/// </summary>
public class SrBreakoutStrategy : Strategy
{
private readonly StrategyParam<int> _lookbackLength;
private readonly StrategyParam<DataType> _candleType;
private readonly Queue<decimal> _highHistory = new();
private readonly Queue<decimal> _lowHistory = new();
public int LookbackLength
{
get => _lookbackLength.Value;
set => _lookbackLength.Value = value;
}
public DataType CandleType
{
get => _candleType.Value;
set => _candleType.Value = value;
}
public SrBreakoutStrategy()
{
_lookbackLength = Param(nameof(LookbackLength), 20)
.SetGreaterThanZero()
.SetDisplay("Lookback", "Number of candles for Donchian channel", "Indicators");
_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(60).TimeFrame())
.SetDisplay("Candle Type", "Timeframe for analysis", "General");
}
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_highHistory.Clear();
_lowHistory.Clear();
var subscription = SubscribeCandles(CandleType);
subscription
.Bind(ProcessCandle)
.Start();
var area = CreateChartArea();
if (area != null)
{
DrawCandles(area, subscription);
DrawOwnTrades(area);
}
}
private void ProcessCandle(ICandleMessage candle)
{
if (candle.State != CandleStates.Finished)
return;
if (_highHistory.Count < LookbackLength)
{
EnqueueCandle(candle);
return;
}
var highs = _highHistory.ToArray();
var lows = _lowHistory.ToArray();
var upper = GetMax(highs);
var lower = GetMin(lows);
var close = candle.ClosePrice;
var range = upper - lower;
var volume = Volume;
if (volume <= 0)
volume = 1;
var breakoutPadding = range * 0.05m;
// Break above resistance
if (close > upper + breakoutPadding)
{
if (Position <= 0)
BuyMarket(Position < 0 ? Math.Abs(Position) + volume : volume);
}
// Break below support
else if (close < lower - breakoutPadding)
{
if (Position >= 0)
SellMarket(Position > 0 ? Math.Abs(Position) + volume : volume);
}
EnqueueCandle(candle);
}
private void EnqueueCandle(ICandleMessage candle)
{
_highHistory.Enqueue(candle.HighPrice);
_lowHistory.Enqueue(candle.LowPrice);
if (_highHistory.Count > LookbackLength)
{
_highHistory.Dequeue();
_lowHistory.Dequeue();
}
}
private static decimal GetMax(IEnumerable<decimal> values)
{
var max = decimal.MinValue;
foreach (var value in values)
{
if (value > max)
max = value;
}
return max;
}
private static decimal GetMin(IEnumerable<decimal> values)
{
var min = decimal.MaxValue;
foreach (var value in values)
{
if (value < min)
min = value;
}
return min;
}
/// <inheritdoc />
protected override void OnReseted()
{
_highHistory.Clear();
_lowHistory.Clear();
base.OnReseted();
}
}