Открыть на GitHub

Стратегия ColorMETRO Stochastic

C# порт эксперта MQL5 exp_colormetro_stochastic.mq5. В оригинале использовался собственный индикатор ColorMETRO Stochastic, здесь его заменяет стандартный StochasticOscillator StockSharp.

Логика

  • По умолчанию подписывается на 8‑часовые свечи (можно изменить).
  • Расчитывает стохастик с параметрами:
    • период %K (KPeriod)
    • период %D (DPeriod)
    • дополнительное сглаживание (Slowing)
  • Сохраняет предыдущие значения %K и %D для поиска пересечений.
  • Покупка при пересечении %K выше %D.
  • Продажа при пересечении %K ниже %D.
  • Через StartProtection используется стоп‑лосс и тейк‑профит по 2%.

Параметры

Имя Описание
KPeriod Длина расчёта линии %K (по умолчанию 5).
DPeriod Сглаживание линии %D (по умолчанию 3).
Slowing Дополнительное сглаживание (по умолчанию 3).
CandleType Таймфрейм свечей, по умолчанию 8 часов.

Примечания

Оригинальная версия MQL использовала шаговые линии ColorMETRO. В данном порте их сигналы аппроксимированы стандартным стохастиком.

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>
/// ColorMETRO Stochastic crossover strategy based on the Stochastic Oscillator.
/// Generates market entries when %K crosses %D.
/// </summary>
public class ColorMetroStochasticStrategy : Strategy
{
	private readonly StrategyParam<int> _kPeriod;
	private readonly StrategyParam<int> _dPeriod;
	private readonly StrategyParam<int> _slowing;
	private readonly StrategyParam<DataType> _candleType;

	private decimal? _prevK;
	private decimal? _prevD;

	/// <summary>
	/// %K calculation period.
	/// </summary>
	public int KPeriod
	{
		get => _kPeriod.Value;
		set => _kPeriod.Value = value;
	}

	/// <summary>
	/// %D smoothing period.
	/// </summary>
	public int DPeriod
	{
		get => _dPeriod.Value;
		set => _dPeriod.Value = value;
	}

	/// <summary>
	/// Additional smoothing value.
	/// </summary>
	public int Slowing
	{
		get => _slowing.Value;
		set => _slowing.Value = value;
	}

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

	/// <summary>
	/// Initializes a new instance of the strategy.
	/// </summary>
	public ColorMetroStochasticStrategy()
	{
		_kPeriod = Param(nameof(KPeriod), 5)
			.SetGreaterThanZero()
			.SetDisplay("K Period", "K calculation period", "Indicator")
			.SetOptimize(3, 15, 1);

		_dPeriod = Param(nameof(DPeriod), 3)
			.SetGreaterThanZero()
			.SetDisplay("D Period", "D smoothing period", "Indicator")
			.SetOptimize(2, 10, 1);

		_slowing = Param(nameof(Slowing), 3)
			.SetGreaterThanZero()
			.SetDisplay("Slowing", "Additional smoothing", "Indicator")
			.SetOptimize(1, 5, 1);

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

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

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_prevK = null;
		_prevD = null;
	}

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

		_prevK = null;
		_prevD = null;

		var stoch = new StochasticOscillator();
		stoch.K.Length = KPeriod;
		stoch.D.Length = DPeriod;

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

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

		StartProtection(
			takeProfit: new Unit(2, UnitTypes.Percent),
			stopLoss: new Unit(2, UnitTypes.Percent));
	}

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

		if (!IsFormedAndOnlineAndAllowTrading())
			return;

		var stoch = (IStochasticOscillatorValue)stochValue;

		if (stoch.K is not decimal k || stoch.D is not decimal d)
			return;

		if (_prevK is decimal prevK && _prevD is decimal prevD)
		{
			// Bullish crossover: %K rises above %D
			if (prevK <= prevD && k > d && Position <= 0)
				BuyMarket();

			// Bearish crossover: %K falls below %D
			else if (prevK >= prevD && k < d && Position >= 0)
				SellMarket();
		}

		_prevK = k;
		_prevD = d;
	}
}