Открыть на GitHub

Стратегия Ichimoku Cloud Retrace

Стратегия представляет собой перенос эксперта MetaTrader «ichimok2005» на платформу StockSharp. Логика основана на откатах внутрь облака Ichimoku с входом в сторону текущего наклона кума. Сигналы рассчитываются только по завершённым свечам.

Общая информация

  • Подходит для любого инструмента и таймфрейма, в котором доступны свечные данные.
  • По умолчанию использует стандартные периоды Ichimoku (9/26/52), но все параметры настраиваются.
  • Работает в обе стороны (лонг и шорт). Размер позиции задаётся свойством Volume стратегии.
  • Стоп-лосс и тейк-профит задаются в абсолютных ценовых единицах и могут быть отключены установкой значения 0.

Индикаторы и параметры

  • Ichimoku: периоды линий Tenkan, Kijun и Senkou Span B вынесены в параметры.
  • Тип свечей: можно выбирать любой агрегирующий тип свечей, поддерживаемый подключением (по умолчанию часовые свечи).
  • Stop Loss Offset: расстояние от цены входа для защитного стопа. 0 — отключено.
  • Take Profit Offset: расстояние от цены входа для фиксации прибыли. 0 — отключено.

Правила входа

Открытие длинной позиции

  1. Senkou Span A располагается выше Senkou Span B, что указывает на бычье облако.
  2. Текущая завершённая свеча бычья (Close > Open).
  3. Закрытие находится внутри облака (Close между линиями Senkou Span A и Senkou Span B).
  4. При выполнении условий и отсутствии длинной позиции отправляется рыночная заявка на покупку. Объём заявки включает закрытие возможного короткого остатка и набор новой длинной позиции.

Открытие короткой позиции

  1. Senkou Span B располагается выше Senkou Span A, что указывает на медвежье облако.
  2. Текущая завершённая свеча медвежья (Open > Close).
  3. Закрытие находится внутри облака (Close между двумя линиями облака).
  4. При выполнении условий и отсутствии короткой позиции отправляется рыночная заявка на продажу, совмещающая закрытие лонга и открытие шорта.

Правила выхода

  • Обратный сигнал разворачивает позицию: стратегия рассчитывает объём так, чтобы одновременно закрыть старую позицию и открыть новую.
  • При включённом Stop Loss Offset выход из лонга выполняется на уровне EntryPrice - Offset, из шорта — на EntryPrice + Offset (проверка по цене закрытия свечи).
  • При включённом Take Profit Offset выход из лонга выполняется на EntryPrice + Offset, из шорта — на EntryPrice - Offset.
  • Принудительное закрытие (остановка стратегии) сбрасывает запомненную цену входа.

Управление рисками

  • Смещения указываются в абсолютной цене. Перед настройкой переведите значение стопа из пунктов или тиков в цену инструмента.
  • Так как контроль стопов и тейков происходит по цене закрытия свечи, для мелких таймфреймов имеет смысл задавать меньшие значения смещений.
  • Частичных выходов и трейлинг-стопа нет: позиция закрывается полностью.

Дополнительные детали реализации

  • Стратегия подписывается на свечи через высокоуровневый API и связывает индикатор Ichimoku с помощью BindEx.
  • Обрабатываются только завершённые свечи; промежуточные обновления игнорируются.
  • При наличии графика автоматически рисуются цены, облако Ichimoku и сделки стратегии.
  • Метод ManageRisk вызывается до проверки сигналов, чтобы защитные выходы выполнялись приоритетно.
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>
/// Ichimoku cloud retrace strategy.
/// Takes trades when price pulls back inside the cloud in the direction of the kumo slope.
/// Uses optional fixed offsets for stop-loss and take-profit management.
/// </summary>
public class IchimokuCloudRetraceStrategy : Strategy
{
	private readonly StrategyParam<int> _tenkanPeriod;
	private readonly StrategyParam<int> _kijunPeriod;
	private readonly StrategyParam<int> _senkouSpanBPeriod;
	private readonly StrategyParam<decimal> _stopLossOffset;
	private readonly StrategyParam<decimal> _takeProfitOffset;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _entryPrice;

	/// <summary>
	/// Tenkan-sen period.
	/// </summary>
	public int TenkanPeriod
	{
		get => _tenkanPeriod.Value;
		set => _tenkanPeriod.Value = value;
	}

	/// <summary>
	/// Kijun-sen period.
	/// </summary>
	public int KijunPeriod
	{
		get => _kijunPeriod.Value;
		set => _kijunPeriod.Value = value;
	}

	/// <summary>
	/// Senkou Span B period.
	/// </summary>
	public int SenkouSpanBPeriod
	{
		get => _senkouSpanBPeriod.Value;
		set => _senkouSpanBPeriod.Value = value;
	}

	/// <summary>
	/// Stop-loss offset in price units. Set to zero to disable.
	/// </summary>
	public decimal StopLossOffset
	{
		get => _stopLossOffset.Value;
		set => _stopLossOffset.Value = value;
	}

	/// <summary>
	/// Take-profit offset in price units. Set to zero to disable.
	/// </summary>
	public decimal TakeProfitOffset
	{
		get => _takeProfitOffset.Value;
		set => _takeProfitOffset.Value = value;
	}

	/// <summary>
	/// Candle type used for signal calculations.
	/// </summary>
	public DataType CandleType
	{
		get => _candleType.Value;
		set => _candleType.Value = value;
	}

	/// <summary>
	/// Initializes strategy parameters.
	/// </summary>
	public IchimokuCloudRetraceStrategy()
	{
		_tenkanPeriod = Param(nameof(TenkanPeriod), 9)
			.SetGreaterThanZero()
			.SetDisplay("Tenkan Period", "Tenkan-sen length", "Ichimoku Settings")
			
			.SetOptimize(5, 15, 1);

		_kijunPeriod = Param(nameof(KijunPeriod), 26)
			.SetGreaterThanZero()
			.SetDisplay("Kijun Period", "Kijun-sen length", "Ichimoku Settings")
			
			.SetOptimize(20, 40, 2);

		_senkouSpanBPeriod = Param(nameof(SenkouSpanBPeriod), 52)
			.SetGreaterThanZero()
			.SetDisplay("Senkou Span B Period", "Senkou Span B length", "Ichimoku Settings")
			
			.SetOptimize(40, 70, 5);

		_stopLossOffset = Param(nameof(StopLossOffset), 0m)
			.SetDisplay("Stop Loss Offset", "Distance from entry for stop-loss (price units)", "Risk Management");

		_takeProfitOffset = Param(nameof(TakeProfitOffset), 0m)
			.SetDisplay("Take Profit Offset", "Distance from entry for take-profit (price units)", "Risk Management");

		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(1).TimeFrame())
			.SetDisplay("Candle Type", "Type of candles for analysis", "General");
	}

	/// <inheritdoc />
	public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
	{
		return [(Security, CandleType)];
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();

		// Reset internal state values.
		_entryPrice = 0m;
	}

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

		// Prepare Ichimoku indicator with user-defined lengths.
		var ichimoku = new Ichimoku
		{
			Tenkan = { Length = TenkanPeriod },
			Kijun = { Length = KijunPeriod },
			SenkouB = { Length = SenkouSpanBPeriod }
		};

		// Subscribe to candle data and bind the indicator for processing.
		var subscription = SubscribeCandles(CandleType);

		subscription
			.BindEx(ichimoku, ProcessCandle)
			.Start();

		// Draw helper visuals if a chart is available.
		var area = CreateChartArea();
		if (area != null)
		{
			DrawCandles(area, subscription);
			DrawIndicator(area, ichimoku);
			DrawOwnTrades(area);
		}
	}

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

		if (!ichimokuValue.IsFinal)
			return;

		// Manage open positions using the latest close before looking for new entries.
		ManageRisk(candle);

		if (Position == 0)
			_entryPrice = 0m;

		var ichimoku = (IchimokuValue)ichimokuValue;

		if (ichimoku.SenkouA is not decimal senkouA ||
			ichimoku.SenkouB is not decimal senkouB)
			return;

		var open = candle.OpenPrice;
		var close = candle.ClosePrice;

		var lowerSpan = Math.Min(senkouA, senkouB);
		var upperSpan = Math.Max(senkouA, senkouB);

		var priceInsideCloud = close > lowerSpan && close < upperSpan;

		var bullishCloud = senkouA > senkouB;
		var bearishCloud = senkouB > senkouA;

		var shouldBuy = bullishCloud && close > open && priceInsideCloud;
		var shouldSell = bearishCloud && open > close && priceInsideCloud;

		if (shouldBuy && Position <= 0)
		{
			// Combine reversal and new entry volume in a single market order.
			var volume = Volume + (Position < 0 ? Math.Abs(Position) : 0m);

			if (volume > 0)
			{
				_entryPrice = close;
				BuyMarket(volume);
			}
		}
		else if (shouldSell && Position >= 0)
		{
			// Combine reversal and new entry volume in a single market order.
			var volume = Volume + (Position > 0 ? Math.Abs(Position) : 0m);

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

	private void ManageRisk(ICandleMessage candle)
	{
		if (_entryPrice == 0m)
			return;

		var close = candle.ClosePrice;

		if (Position > 0)
		{
			if (StopLossOffset > 0m && close <= _entryPrice - StopLossOffset)
			{
				var volumeToClose = Math.Abs(Position);

				if (volumeToClose > 0m)
				{
					SellMarket(volumeToClose);
					_entryPrice = 0m;
					return;
				}
			}

			if (TakeProfitOffset > 0m && close >= _entryPrice + TakeProfitOffset)
			{
				var volumeToClose = Math.Abs(Position);

				if (volumeToClose > 0m)
				{
					SellMarket(volumeToClose);
					_entryPrice = 0m;
				}
			}
		}
		else if (Position < 0)
		{
			if (StopLossOffset > 0m && close >= _entryPrice + StopLossOffset)
			{
				var volumeToClose = Math.Abs(Position);

				if (volumeToClose > 0m)
				{
					BuyMarket(volumeToClose);
					_entryPrice = 0m;
					return;
				}
			}

			if (TakeProfitOffset > 0m && close <= _entryPrice - TakeProfitOffset)
			{
				var volumeToClose = Math.Abs(Position);

				if (volumeToClose > 0m)
				{
					BuyMarket(volumeToClose);
					_entryPrice = 0m;
				}
			}
		}
	}
}