Открыть на GitHub

Стратегия Psi Proc EMA MACD

Стратегия реализует систему T4 из оригинального эксперта MQL e-PSI@PROC.mq4. Торговля ведётся по согласованности нескольких экспоненциальных средних и фильтру MACD.

Логика стратегии

  1. На каждой свече рассчитываются EMA(200), EMA(50) и EMA(10).
  2. Рассчитывается MACD с параметрами 12, 26, 9.
  3. Покупка, если:
    • EMA200 растёт и EMA50 > EMA200;
    • EMA50 растёт и EMA10 > EMA50;
    • MACD растёт и превышает LimitMACD.
  4. Продажа, если:
    • EMA200 падает и EMA50 < EMA200;
    • EMA50 падает и EMA10 < EMA50;
    • MACD падает и ниже -LimitMACD.
  5. Закрытие длинной позиции при закрытии цены ниже EMA50.
  6. Закрытие короткой позиции при закрытии цены выше EMA50.

Поддерживаются опциональные тейк-профит и трейлинг-стоп.

Параметры

Имя Описание
LimitMACD Минимальный модуль MACD для входа.
TakeProfitPoints Уровень тейк-профита в пунктах цены.
TrailStopPoints Уровень трейлинг-стопа в пунктах цены.
CandleType Таймфрейм свечей.

Примечания

  • Открытие позиций выполняется рыночными ордерами.
  • Обрабатываются только завершённые свечи.
  • Стратегия работает с одним инструментом.
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>
/// Strategy based on multiple EMA alignment with MACD confirmation.
/// Buys when EMA10 > EMA50 > EMA200 and MACD positive.
/// Sells on opposite alignment.
/// </summary>
public class PsiProcEmaMacdStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;

	private decimal _prevEma200;
	private decimal _prevEma50;
	private decimal _prevEma10;
	private bool _initialized;

	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }

	public PsiProcEmaMacdStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
			.SetDisplay("Candle Type", "Timeframe", "General");
	}

	public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
		=> [(Security, CandleType)];

	protected override void OnReseted()
	{
		base.OnReseted();
		_prevEma200 = 0;
		_prevEma50 = 0;
		_prevEma10 = 0;
		_initialized = false;
	}

	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);

		var ema200 = new ExponentialMovingAverage { Length = 50 };
		var ema50 = new ExponentialMovingAverage { Length = 20 };
		var ema10 = new ExponentialMovingAverage { Length = 10 };
		var macd = new MovingAverageConvergenceDivergence();

		var subscription = SubscribeCandles(CandleType);
		subscription
			.Bind(ema200, ema50, ema10, macd, ProcessCandle)
			.Start();
	}

	private void ProcessCandle(ICandleMessage candle, decimal ema200, decimal ema50, decimal ema10, decimal macdVal)
	{
		if (candle.State != CandleStates.Finished)
			return;

		if (!_initialized)
		{
			_prevEma200 = ema200;
			_prevEma50 = ema50;
			_prevEma10 = ema10;
			_initialized = true;
			return;
		}

		// Entry/reversal conditions - EMA alignment
		if (ema10 > ema50 && Position <= 0)
			BuyMarket();
		else if (ema10 < ema50 && Position >= 0)
			SellMarket();

		_prevEma200 = ema200;
		_prevEma50 = ema50;
		_prevEma10 = ema10;
	}
}