Открыть на GitHub

Стратегия Long/Short Expert MACD

Обзор

Long/Short Expert MACD — это конвертация экспертной системы MetaTrader «LongShortExpertMACD» на StockSharp. Логика базируется на пересечениях MACD и его сигнальной линии, дополненных фиксированными стоп-лоссами и тейк-профитами. Стратегия может работать только в лонг, только в шорт или в обе стороны и автоматически рассчитывает защитные уровни в ценовых пунктах.

Реализация использует высокоуровневый API StockSharp: подписку на свечи и привязку индикаторов. Заявки регистрируются как рыночные, поэтому стратегию можно одинаково применять как на реальном потоке, так и в историческом тесте.

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

  • Свечи — один таймфрейм, задаётся параметром CandleType (по умолчанию минутные свечи). Подписка выполняется через SubscribeCandles.
  • MovingAverageConvergenceDivergenceSignal — встроенный MACD с настраиваемыми периодами быстрого, медленного и сигнального EMA. Гистограмма implicitly получается как разница между линией MACD и сигналом.

Торговая логика

  1. Подготовка сигнала

    • На каждой завершённой свече стратегия получает значения MACD и сигнальной линии из привязанного индикатора.
    • Переменная _prevIsMacdAboveSignal запоминает положение MACD относительно сигнальной линии на предыдущей свече.
  2. Условия входа

    • Бычье пересечение: MACD переходит выше сигнальной линии — открываем лонг, если выбранный режим допускает длинные позиции.
      • При активном шорте и разрешённых реверсах (AllowedPosition = Both) объём рыночной заявки включает величину текущего шорта, что обеспечивает закрытие и переворот в одном ордере.
      • В режиме «только лонг» активный шорт закрывается, но новый вход произойдёт только после следующего согласованного сигнала.
    • Медвежье пересечение: зеркальные правила для входа в короткую позицию.
  3. Условия выхода

    • Управление рисками: стоп-лосс и тейк-профит пересчитываются от средней цены входа при каждой фиксации позиции. Расстояния выражаются в ценовых пунктах (PriceStep * параметр), поэтому логика корректно переносится на разные инструменты.
      • Лонги закрываются, если минимум свечи достигает стоп-лосса или максимум — тейк-профита.
      • Шорты закрываются, если максимум свечи пробивает стоп-лосс или минимум — тейк-профит.
    • Обратное пересечение: при разрешённой торговле в противоположную сторону позиция закрывается (и при необходимости разворачивается) сразу после смены соотношения линий MACD/сигнала.
  4. Дополнительные ограничения

    • Логика выполняется только в состоянии IsFormedAndOnlineAndAllowTrading.
    • Защитные уровни обнуляются при отсутствии позиции, чтобы исключить использование устаревших значений.

Параметры

Имя Значение по умолчанию Описание
AllowedPosition Both Разрешённые направления торговли: только лонг, только шорт или оба варианта.
FastLength 12 Период быстрого EMA внутри MACD.
SlowLength 24 Период медленного EMA внутри MACD.
SignalLength 9 Период сигнального EMA, определяющего пересечения.
TakeProfitPoints 50 Дистанция до тейк-профита в ценовых пунктах (PriceStep * значение). 0 отключает уровень.
StopLossPoints 20 Дистанция до стоп-лосса в ценовых пунктах. 0 отключает уровень.
CandleType TimeFrame(1 minute) Тип свечей, используемых для расчётов.
Volume 1 Объём каждой рыночной заявки.

Все числовые параметры имеют заранее заданные диапазоны оптимизации, что упрощает перебор в Designer или Runner.

Управление позицией

  • Реверс позиций: при AllowedPosition = Both стратегия отправляет комбинированный объём, закрывая текущую позицию и открывая противоположную одним ордером — аналогично оригинальному эксперту MetaTrader.
  • Режимы long-only / short-only: позиции, не соответствующие выбранному направлению, закрываются сразу, но новое открытие будет только при сигнале в допустимую сторону.
  • Актуализация стопов: уровни стоп-лосса и тейк-профита пересчитываются каждый раз на основе PositionAvgPrice, что корректно отражает среднюю цену после частичных закрытий или доливок.

Рекомендации по применению

  • Убедитесь, что инструмент имеет корректный PriceStep. При его отсутствии стратегия использует значение 1.0, что подходит для акций, но может потребовать корректировок на валютном рынке.
  • Стратегия работает только по завершённым свечам. Для минимизации задержек выберите подходящий таймфрейм.
  • Рыночные заявки не учитывают проскальзывание, поэтому на малоликвидных инструментах целесообразно закладывать дополнительный запас риска.
  • При наличии графиков стратегия автоматически выводит свечи, индикатор MACD и собственные сделки.

Замечания по конвертации

  • В StockSharp сохранены все настраиваемые параметры MACD, стоп-лосс/тейк-профит и переключатель разрешённых позиций из MQL5.
  • Модули трейлинг-стопа и управления капиталом исключены, так как в исходном эксперте использовались заглушки "None".
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>
/// Long/short MACD expert strategy converted from the MetaTrader example.
/// The strategy opens positions on MACD crossovers and applies fixed stop-loss and take-profit distances.
/// Allowed trade direction can be restricted to long only, short only, or both sides.
/// </summary>
public class LongShortExpertMacdStrategy : Strategy
{
	/// <summary>
	/// Trade directions supported by the strategy.
	/// </summary>
	public enum AllowedPositionTypes
	{
		/// <summary>
		/// Long trades only.
		/// </summary>
		Long,

		/// <summary>
		/// Short trades only.
		/// </summary>
		Short,

		/// <summary>
		/// Long and short trades are allowed.
		/// </summary>
		Both
	}

	private readonly StrategyParam<AllowedPositionTypes> _allowedPosition;
	private readonly StrategyParam<int> _fastLength;
	private readonly StrategyParam<int> _slowLength;
	private readonly StrategyParam<int> _signalLength;
	private readonly StrategyParam<int> _takeProfitPoints;
	private readonly StrategyParam<int> _stopLossPoints;
	private readonly StrategyParam<DataType> _candleType;

	private MovingAverageConvergenceDivergenceSignal _macd = null!;

	private bool? _prevIsMacdAboveSignal;
	private decimal _longStopPrice;
	private decimal _longTakePrice;
	private decimal _shortStopPrice;
	private decimal _shortTakePrice;
	private decimal? _entryPrice;

	/// <summary>
	/// Initializes a new instance of <see cref="LongShortExpertMacdStrategy"/>.
	/// </summary>
	public LongShortExpertMacdStrategy()
	{
		_allowedPosition = Param(nameof(AllowedPosition), AllowedPositionTypes.Both)
			.SetDisplay("Allowed Positions", "Permitted trade direction", "General");

		_fastLength = Param(nameof(FastLength), 12)
			.SetGreaterThanZero()
			.SetDisplay("Fast EMA", "Fast MACD EMA length", "MACD")

			.SetOptimize(8, 16, 2);

		_slowLength = Param(nameof(SlowLength), 24)
			.SetGreaterThanZero()
			.SetDisplay("Slow EMA", "Slow MACD EMA length", "MACD")

			.SetOptimize(20, 40, 2);

		_signalLength = Param(nameof(SignalLength), 9)
			.SetGreaterThanZero()
			.SetDisplay("Signal EMA", "MACD signal EMA length", "MACD")

			.SetOptimize(5, 15, 1);

		_takeProfitPoints = Param(nameof(TakeProfitPoints), 50)
			.SetNotNegative()
			.SetDisplay("Take Profit", "Take profit distance in price points", "Risk")

			.SetOptimize(0, 150, 10);

		_stopLossPoints = Param(nameof(StopLossPoints), 20)
			.SetNotNegative()
			.SetDisplay("Stop Loss", "Stop loss distance in price points", "Risk")

			.SetOptimize(0, 100, 10);

		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
			.SetDisplay("Candle Type", "Type of candles to process", "General");

		Volume = 1;
	}

	/// <summary>
	/// Allowed trade direction.
	/// </summary>
	public AllowedPositionTypes AllowedPosition
	{
		get => _allowedPosition.Value;
		set => _allowedPosition.Value = value;
	}

	/// <summary>
	/// Fast EMA length used by MACD.
	/// </summary>
	public int FastLength
	{
		get => _fastLength.Value;
		set => _fastLength.Value = value;
	}

	/// <summary>
	/// Slow EMA length used by MACD.
	/// </summary>
	public int SlowLength
	{
		get => _slowLength.Value;
		set => _slowLength.Value = value;
	}

	/// <summary>
	/// Signal EMA length used by MACD.
	/// </summary>
	public int SignalLength
	{
		get => _signalLength.Value;
		set => _signalLength.Value = value;
	}

	/// <summary>
	/// Take-profit distance expressed in price points.
	/// </summary>
	public int TakeProfitPoints
	{
		get => _takeProfitPoints.Value;
		set => _takeProfitPoints.Value = value;
	}

	/// <summary>
	/// Stop-loss distance expressed in price points.
	/// </summary>
	public int StopLossPoints
	{
		get => _stopLossPoints.Value;
		set => _stopLossPoints.Value = value;
	}

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

	private bool CanEnterLong => AllowedPosition != AllowedPositionTypes.Short;
	private bool CanEnterShort => AllowedPosition != AllowedPositionTypes.Long;
	private bool AllowReverse => AllowedPosition == AllowedPositionTypes.Both;

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

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

		_prevIsMacdAboveSignal = null;
		_entryPrice = null;
		ResetProtection();
	}

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

		_macd = new MovingAverageConvergenceDivergenceSignal
		{
			Macd =
			{
				ShortMa = { Length = FastLength },
				LongMa = { Length = SlowLength },
			},
			SignalMa = { Length = SignalLength }
		};

		var subscription = SubscribeCandles(CandleType);
		subscription
			.BindEx(_macd, ProcessCandle)
			.Start();

		var area = CreateChartArea();
		if (area != null)
		{
			DrawCandles(area, subscription);
			DrawIndicator(area, _macd);
			DrawOwnTrades(area);
		}
	}

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

		var macdTyped = (MovingAverageConvergenceDivergenceSignalValue)macdValue;

		if (macdTyped.Macd is not decimal macd || macdTyped.Signal is not decimal signal)
			return;

		UpdateProtectionLevels();

		var isMacdAboveSignal = macd > signal;

		if (!_macd.IsFormed)
		{
			_prevIsMacdAboveSignal = isMacdAboveSignal;
			return;
		}

		if (TryExitWithProtection(candle))
		{
			_prevIsMacdAboveSignal = isMacdAboveSignal;
			return;
		}

		if (_prevIsMacdAboveSignal is null)
		{
			_prevIsMacdAboveSignal = isMacdAboveSignal;
			return;
		}

		var crossUp = isMacdAboveSignal && _prevIsMacdAboveSignal == false;
		var crossDown = !isMacdAboveSignal && _prevIsMacdAboveSignal == true;

		if (crossUp)
		{
			if (CanEnterLong)
			{
				if (Position < 0)
				{
					if (AllowReverse)
					{
						var volume = Volume + Math.Abs(Position);

						if (volume > 0)
						{
							ResetProtection();
							BuyMarket();
							_entryPrice = candle.ClosePrice;
						}
					}
					else
					{
						var volume = Math.Abs(Position);
						if (volume > 0)
						{
							BuyMarket();
							ResetProtection();
							_entryPrice = null;
						}
					}
				}
				else if (Position == 0)
				{
					if (Volume > 0)
					{
						ResetProtection();
						BuyMarket();
						_entryPrice = candle.ClosePrice;
					}
				}
			}
			else if (Position < 0)
			{
				var volume = Math.Abs(Position);
				if (volume > 0)
				{
					BuyMarket();
					ResetProtection();
					_entryPrice = null;
				}
			}
		}
		else if (crossDown)
		{
			if (CanEnterShort)
			{
				if (Position > 0)
				{
					if (AllowReverse)
					{
						var volume = Volume + Math.Abs(Position);

						if (volume > 0)
						{
							ResetProtection();
							SellMarket();
							_entryPrice = candle.ClosePrice;
						}
					}
					else
					{
						var volume = Math.Abs(Position);
						if (volume > 0)
						{
							SellMarket();
							ResetProtection();
							_entryPrice = null;
						}
					}
				}
				else if (Position == 0)
				{
					if (Volume > 0)
					{
						ResetProtection();
						SellMarket();
						_entryPrice = candle.ClosePrice;
					}
				}
			}
			else if (Position > 0)
			{
				var volume = Math.Abs(Position);
				if (volume > 0)
				{
					SellMarket();
					ResetProtection();
					_entryPrice = null;
				}
			}
		}

		_prevIsMacdAboveSignal = isMacdAboveSignal;
	}

	private void UpdateProtectionLevels()
	{
		if (_entryPrice is not decimal entry)
		{
			ResetProtection();
			return;
		}

		if (Position > 0)
		{
			var step = GetPriceStep();
			_longStopPrice = StopLossPoints > 0 ? entry - StopLossPoints * step : 0m;
			_longTakePrice = TakeProfitPoints > 0 ? entry + TakeProfitPoints * step : 0m;
			_shortStopPrice = 0m;
			_shortTakePrice = 0m;
		}
		else if (Position < 0)
		{
			var step = GetPriceStep();
			_shortStopPrice = StopLossPoints > 0 ? entry + StopLossPoints * step : 0m;
			_shortTakePrice = TakeProfitPoints > 0 ? entry - TakeProfitPoints * step : 0m;
			_longStopPrice = 0m;
			_longTakePrice = 0m;
		}
		else
		{
			ResetProtection();
		}
	}

	private bool TryExitWithProtection(ICandleMessage candle)
	{
		if (Position > 0)
		{
			var volume = Math.Abs(Position);

			if (volume > 0)
			{
				if (StopLossPoints > 0 && _longStopPrice > 0m && candle.LowPrice <= _longStopPrice)
				{
					SellMarket();
					ResetProtection();
					_entryPrice = null;
					return true;
				}

				if (TakeProfitPoints > 0 && _longTakePrice > 0m && candle.HighPrice >= _longTakePrice)
				{
					SellMarket();
					ResetProtection();
					_entryPrice = null;
					return true;
				}
			}
		}
		else if (Position < 0)
		{
			var volume = Math.Abs(Position);

			if (volume > 0)
			{
				if (StopLossPoints > 0 && _shortStopPrice > 0m && candle.HighPrice >= _shortStopPrice)
				{
					BuyMarket();
					ResetProtection();
					_entryPrice = null;
					return true;
				}

				if (TakeProfitPoints > 0 && _shortTakePrice > 0m && candle.LowPrice <= _shortTakePrice)
				{
					BuyMarket();
					ResetProtection();
					_entryPrice = null;
					return true;
				}
			}
		}
		return false;
	}

	private void ResetProtection()
	{
		_longStopPrice = 0m;
		_longTakePrice = 0m;
		_shortStopPrice = 0m;
		_shortTakePrice = 0m;
	}

	private decimal GetPriceStep()
	{
		var step = Security?.PriceStep ?? 0m;
		return step > 0m ? step : 1m;
	}
}