Открыть на GitHub

Стратегия «Weekly Rebound Corridor»

Описание

Стратегия «Weekly Rebound Corridor» является конверсией эксперта MetaTrader 4 2_Otkat_Sys_v1_1. Алгоритм ищет ценовой разрыв между закрытием предыдущей свечи и ценой открытия, наблюдавшейся 24 свечи назад. Если разрыв превышает заданный порог и наступил выбранный день недели, стратегия открывает позицию в первые минуты нового торгового дня, устанавливает защитные стоп-приказы и принудительно закрывает позицию перед завершением сессии.

Логика работы

  1. Подготовка данных
    • По умолчанию используются минутные свечи, однако тип свечей можно изменить параметром.
    • Хранится значение закрытия предыдущей свечи и поддерживается циклический буфер, возвращающий цену открытия 24 свечи назад.
  2. Поиск сигнала
    • В выбранный день недели (нумерация MetaTrader: 0 = воскресенье, 6 = суббота) анализируются свечи, открывшиеся между 00:00 и 00:03 по локальному времени площадки.
    • Если историческая цена открытия (24 свечи назад) превышает закрытие предыдущей свечи больше, чем на установленный порог, открывается длинная позиция. При обратной ситуации открывается короткая позиция.
    • На один торговый день допускается только одна сделка.
  3. Управление позицией
    • Стоп-лосс и тейк-профит задаются в пунктах и переводятся в цену на основе минимального шага цены инструмента.
    • Для длинных позиций тейк-профит увеличивается на дополнительные 3 пункта (как в исходном советнике).
    • Каждая закрытая свеча проверяется на достижение стоп-уровней; при срабатывании позиция закрывается рыночным ордером.
    • Если позиция открыта после 22:45 локального времени, она закрывается принудительно.

Параметры

Параметр Описание Значение по умолчанию
TakeProfitPoints Размер тейк-профита в пунктах. Для длинных сделок добавляются 3 дополнительных пункта. 5
StopLossPoints Размер стоп-лосса в пунктах. 49
TradeVolume Торговый объём рыночных заявок; корректируется по шагу объёма инструмента. 1
CorridorPoints Минимальный разрыв между историческим открытием и последним закрытием. 10
TradeDayOfWeek День недели в формате MetaTrader (0 = воскресенье6 = суббота). 5 (пятница)
CandleType Тип свечей, используемый в расчётах. 1 минута

Дополнительные замечания

  • Стратегия работает только с завершёнными свечами, что соответствует требованиям проекта.
  • Перед началом торговли необходимо обеспечить наличие достаточной истории для заполнения буфера из 24 свечей.
  • Параметры стратегии следует адаптировать под спецификацию инструмента (шаг цены, шаг объёма, расписание торгов).
using System;

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

namespace StockSharp.Samples.Strategies;

public class WeeklyReboundCorridorStrategy : Strategy
{
	private readonly StrategyParam<int> _rsiPeriod;
	private readonly StrategyParam<int> _emaPeriod;
	private readonly StrategyParam<decimal> _oversold;
	private readonly StrategyParam<decimal> _overbought;
	private readonly StrategyParam<int> _cooldownCandles;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _prevRsi;
	private bool _hasPrev;
	private int _cooldownRemaining;

	public int RsiPeriod { get => _rsiPeriod.Value; set => _rsiPeriod.Value = value; }
	public int EmaPeriod { get => _emaPeriod.Value; set => _emaPeriod.Value = value; }
	public decimal Oversold { get => _oversold.Value; set => _oversold.Value = value; }
	public decimal Overbought { get => _overbought.Value; set => _overbought.Value = value; }
	public int CooldownCandles { get => _cooldownCandles.Value; set => _cooldownCandles.Value = value; }
	public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }

	public WeeklyReboundCorridorStrategy()
	{
		_rsiPeriod = Param(nameof(RsiPeriod), 14).SetDisplay("RSI Period", "RSI lookback", "Indicators");
		_emaPeriod = Param(nameof(EmaPeriod), 50).SetDisplay("EMA Period", "EMA filter", "Indicators");
		_oversold = Param(nameof(Oversold), 25m).SetDisplay("Oversold", "RSI oversold level", "Levels");
		_overbought = Param(nameof(Overbought), 75m).SetDisplay("Overbought", "RSI overbought level", "Levels");
		_cooldownCandles = Param(nameof(CooldownCandles), 50).SetDisplay("Cooldown", "Candles between signals", "General");
		_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(5).TimeFrame()).SetDisplay("Candle Type", "Candle timeframe", "General");
	}

	/// <inheritdoc />
	protected override void OnReseted()
	{
		base.OnReseted();
		_prevRsi = default;
		_hasPrev = default;
		_cooldownRemaining = default;
	}

	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);
		_prevRsi = 0;
		_hasPrev = false;
		_cooldownRemaining = 0;

		var rsi = new RelativeStrengthIndex { Length = RsiPeriod };
		var ema = new ExponentialMovingAverage { Length = EmaPeriod };
		var subscription = SubscribeCandles(CandleType);
		subscription.Bind(rsi, ema, ProcessCandle).Start();
	}

	private void ProcessCandle(ICandleMessage candle, decimal rsi, decimal ema)
	{
		if (candle.State != CandleStates.Finished) return;
		var close = candle.ClosePrice;
		if (!_hasPrev) { _prevRsi = rsi; _hasPrev = true; return; }

		if (_cooldownRemaining > 0)
		{
			_cooldownRemaining--;
			_prevRsi = rsi;
			return;
		}

		var oversold = Oversold;
		var overbought = Overbought;

		if (_prevRsi >= oversold && rsi < oversold && Position <= 0)
		{
			if (Position < 0) BuyMarket();
			BuyMarket();
			_cooldownRemaining = CooldownCandles;
		}
		else if (_prevRsi <= overbought && rsi > overbought && Position >= 0)
		{
			if (Position > 0) SellMarket();
			SellMarket();
			_cooldownRemaining = CooldownCandles;
		}
		_prevRsi = rsi;
	}
}