Открыть на GitHub

Стратегия пробоя предыдущей свечи

Стратегия воспроизводит советник BreakOut автора Soubra2003 для MetaTrader. Алгоритм отслеживает максимум и минимум последней завершённой свечи и при закрытии новой свечи выше/ниже этих уровней разворачивает позицию. Поддерживаются симметричные сделки в обе стороны, а также опциональные стоп-лосс и тейк-профит, задаваемые в абсолютных ценовых шагах.

Общая логика

  • Подписка на один поток свечей (по умолчанию — часовой таймфрейм).
  • После каждой завершённой свечи сохраняются её High и Low, которые становятся уровнями пробоя для следующей свечи.
  • Сделки открываются только по цене закрытия свечи, что повторяет тиковую логику оригинала без обращения к данным внутри свечи.

Правила торговли

  1. Вход и разворот по пробою
    • Закрытие текущей свечи выше максимума предыдущей:
      • Любой открытый шорт закрывается рыночной покупкой.
      • Затем сразу открывается новая длинная позиция (разворот происходит в рамках обработки одной свечи).
    • Закрытие текущей свечи ниже минимума предыдущей:
      • Любой открытый лонг закрывается рыночной продажей.
      • Затем открывается новая короткая позиция.
  2. Защитные выходы (опционально)
    • При ненулевом стоп-лоссе позиция закрывается, если цена закрытия отклонилась от цены входа больше чем на указанное значение (ниже для лонга, выше для шорта).
    • При ненулевом тейк-профите позиция закрывается, если цена закрытия ушла в прибыль на указанное количество ценовых пунктов.
  3. Обновление уровней
    • После обработки свечи её максимум/минимум становятся эталоном для следующего шага.

Параметры

  • Candle Type – тип данных свечи (по умолчанию часовая). Установите тот таймфрейм, который использовался в MetaTrader для исходного эксперта.
  • Stop Loss – расстояние в абсолютных ценовых единицах от цены входа до защитного стопа. Значение 0 отключает стоп-лосс.
  • Take Profit – расстояние в абсолютных ценовых единицах до цели по прибыли. Значение 0 отключает тейк-профит.

Примечания

  • В оригинале SL/TP выставлялись брокеру. В StockSharp закрытие реализовано через рыночные ордера после фиксации условия на цене закрытия.
  • Настройки стопа и цели задавайте с учётом минимального шага цены инструмента (например, при шаге 0.01 и стопе 20 пунктов укажите значение 0.20).
  • Стратегия особенно хорошо подходит для волатильных инструментов, где пробои максимумов/минимумов предыдущей свечи дают направленное движение.

Источник

using System;
using System.Linq;
using System.Collections.Generic;

using Ecng.Common;
using Ecng.Collections;
using Ecng.Serialization;

using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;

namespace StockSharp.Samples.Strategies;

/// <summary>
/// Breakout strategy that trades when the close crosses the previous candle's high or low.
/// Ported from the BreakOut.mq4 expert by Soubra2003.
/// </summary>
public class PreviousCandleBreakoutStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<decimal> _stopLossOffset;
	private readonly StrategyParam<decimal> _takeProfitOffset;

	private decimal? _previousHigh;
	private decimal? _previousLow;
	private decimal _entryPrice;

	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
	public decimal StopLossOffset { get => _stopLossOffset.Value; set => _stopLossOffset.Value = value; }
	public decimal TakeProfitOffset { get => _takeProfitOffset.Value; set => _takeProfitOffset.Value = value; }

	public PreviousCandleBreakoutStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromDays(1).TimeFrame())
			.SetDisplay("Candle Type", "Timeframe for candle subscription", "General");

		_stopLossOffset = Param(nameof(StopLossOffset), 1000m)
			.SetDisplay("Stop Loss", "Price distance for the stop-loss. Set 0 to disable.", "Risk")
			;

		_takeProfitOffset = Param(nameof(TakeProfitOffset), 1500m)
			.SetDisplay("Take Profit", "Price distance for the take-profit. Set 0 to disable.", "Risk")
			;
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_previousHigh = null;
		_previousLow = null;
		_entryPrice = 0m;
	}

	public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
	{
		yield return (Security, CandleType);
	}

	/// <inheritdoc />
	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);

		var subscription = SubscribeCandles(CandleType);
		subscription.Bind(ProcessCandle).Start();
	}

	private void ProcessCandle(ICandleMessage candle)
	{
		if (candle.State != CandleStates.Finished)
			return;

		if (_previousHigh is null || _previousLow is null)
		{
			// Store the first finished candle to obtain reference high/low levels.
			_previousHigh = candle.HighPrice;
			_previousLow = candle.LowPrice;
			return;
		}

		var previousHigh = _previousHigh.Value;
		var previousLow = _previousLow.Value;
		var close = candle.ClosePrice;

		var breakoutAbove = close > previousHigh;
		var breakoutBelow = close < previousLow;

		// Manage protective exits while a position is open.
		if (Position > 0)
		{
			if (StopLossOffset > 0m && close <= _entryPrice - StopLossOffset)
			{
				if (Position > 0) SellMarket(); else if (Position < 0) BuyMarket();
				_entryPrice = 0m;
			}
			else if (TakeProfitOffset > 0m && close >= _entryPrice + TakeProfitOffset)
			{
				if (Position > 0) SellMarket(); else if (Position < 0) BuyMarket();
				_entryPrice = 0m;
			}
		}
		else if (Position < 0)
		{
			if (StopLossOffset > 0m && close >= _entryPrice + StopLossOffset)
			{
				if (Position > 0) SellMarket(); else if (Position < 0) BuyMarket();
				_entryPrice = 0m;
			}
			else if (TakeProfitOffset > 0m && close <= _entryPrice - TakeProfitOffset)
			{
				if (Position > 0) SellMarket(); else if (Position < 0) BuyMarket();
				_entryPrice = 0m;
			}
		}

		// Breakout above the previous high opens or reverses into a long position.
		if (breakoutAbove)
		{
			if (Position < 0)
			{
				if (Position > 0) SellMarket(); else if (Position < 0) BuyMarket();
				_entryPrice = 0m;
			}

			if (Position <= 0)
			{
				BuyMarket();
				_entryPrice = close;
			}
		}
		else if (breakoutBelow)
		{
			// Breakout below the previous low opens or reverses into a short position.
			if (Position > 0)
			{
				if (Position > 0) SellMarket(); else if (Position < 0) BuyMarket();
				_entryPrice = 0m;
			}

			if (Position >= 0)
			{
				SellMarket();
				_entryPrice = close;
			}
		}

		_previousHigh = candle.HighPrice;
		_previousLow = candle.LowPrice;
	}
}