Открыть на GitHub

Стратегия Pipso

Описание

Pipso — это ночная пробойная система, перенесённая из советника MetaTrader Pipso.mq4. Стратегия отслеживает максимум и минимум предыдущих завершённых свечей и при выходе цены за эти границы немедленно переворачивает позицию: при обновлении максимума закрываются лонги и открывается шорт, при пробое минимума закрываются шорты и открывается лонг. Дистанция защитного стопа вычисляется исходя из ширины диапазона, благодаря чему стоп автоматически подстраивается под текущую волатильность.

Алгоритм работы

  1. Подписаться на выбранный таймфрейм (по умолчанию 15 минут) и дождаться формирования истории для индикаторов.
  2. Для каждой новой завершённой свечи рассчитать максимальное и минимальное значение по предыдущим BreakoutPeriod свечам. Текущая свеча не попадает в расчёт, что повторяет вызов iHighest(..., shift = 1) в оригинальном коде.
  3. Пересчитать расстояние до стоп-лосса как (high - low) * StopLossMultiplier, применяя ограничение MinStopDistance на минимально допустимую дистанцию.
  4. Поддерживать торговое окно, заданное параметрами SessionStartHour и SessionLengthHours. Если окно переходит через ночь с пятницы на субботу, оно продлевается на 48 часов, чтобы сделки переживали выходные так же, как в MetaTrader.
  5. При пробое верхней границы диапазона:
    • Закрыть все активные длинные позиции и, если торговля разрешена, открыть короткую позицию объёмом OrderVolume.
    • Установить защитный стоп выше цены входа на рассчитанное расстояние.
  6. При пробое нижней границы диапазона:
    • Закрыть все активные короткие позиции и, если торговля разрешена, открыть длинную позицию объёмом OrderVolume.
    • Установить защитный стоп ниже цены входа на рассчитанное расстояние.
  7. На каждой завершённой свече проверять защитные стопы. Если минимум свечи опускается ниже стопа лонга или максимум поднимается выше стопа шорта, позиция немедленно закрывается.

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

  • SessionStartHour задаёт час открытия окна в часах торговой площадки, продолжительность определяется SessionLengthHours.
  • Если сумма SessionStartHour + SessionLengthHours превышает 24 часа и текущий день — пятница, конец окна переносится вперёд на двое суток, чтобы новые сделки стали доступны только в понедельник, как и в MQL4.
  • Вне торгового окна стратегия только фиксирует существующие позиции, но не открывает новые до повторного входа в окно.

Параметры

Имя Описание Значение по умолчанию
CandleType Тип свечей, используемых для расчёта сигналов. Таймфрейм 15 минут
OrderVolume Фиксированный объём рыночных заявок. 1
SessionStartHour Час начала торгового окна. 21
SessionLengthHours Продолжительность торгового окна в часах. 9
BreakoutPeriod Количество завершённых свечей, формирующих пробойный диапазон. 36
StopLossMultiplier Множитель диапазона для расчёта стоп-лосса (значение 3 соответствует исходному SLpp = 300). 3
MinStopDistance Минимальная дистанция от цены входа до стопа, имитирующая брокерский StopLevel. 0

Примечания

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

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

namespace StockSharp.Samples.Strategies;

/// <summary>
/// Range breakout strategy that uses Highest/Lowest channel.
/// Enters on breakouts above/below the channel and exits on reversion.
/// </summary>
public class PipsoNightBreakoutStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<int> _breakoutPeriod;

	private decimal _entryPrice;
	private decimal _prevHighest;
	private decimal _prevLowest;
	private bool _hasPrev;

	public PipsoNightBreakoutStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(5).TimeFrame())
			.SetDisplay("Candle Type", "Timeframe for analysis.", "General");

		_breakoutPeriod = Param(nameof(BreakoutPeriod), 36)
			.SetDisplay("Breakout Period", "Period for Highest/Lowest channel.", "Indicators");
	}

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

	public int BreakoutPeriod
	{
		get => _breakoutPeriod.Value;
		set => _breakoutPeriod.Value = value;
	}

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

		_entryPrice = 0;
		_prevHighest = 0;
		_prevLowest = 0;
		_hasPrev = false;
	}

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

		_entryPrice = 0;
		_prevHighest = 0;
		_prevLowest = 0;
		_hasPrev = false;

		var highest = new Highest { Length = BreakoutPeriod };
		var lowest = new Lowest { Length = BreakoutPeriod };
		var ema = new ExponentialMovingAverage { Length = BreakoutPeriod };

		var subscription = SubscribeCandles(CandleType);
		subscription
			.Bind(highest, lowest, ema, ProcessCandle)
			.Start();

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

	private void ProcessCandle(ICandleMessage candle, decimal highestValue, decimal lowestValue, decimal emaValue)
	{
		if (candle.State != CandleStates.Finished)
			return;

		var close = candle.ClosePrice;
		var mid = (highestValue + lowestValue) / 2m;

		// Exit conditions
		if (Position > 0)
		{
			// Exit when price reverts to middle or stop-loss
			if (close < mid || (_entryPrice > 0 && close < _entryPrice * 0.98m))
			{
				SellMarket();
			}
		}
		else if (Position < 0)
		{
			if (close > mid || (_entryPrice > 0 && close > _entryPrice * 1.02m))
			{
				BuyMarket();
			}
		}

		// Entry conditions: breakout above previous highest or below previous lowest
		if (Position == 0 && _hasPrev)
		{
			if (close > _prevHighest && close > emaValue)
			{
				_entryPrice = close;
				BuyMarket();
			}
			else if (close < _prevLowest && close < emaValue)
			{
				_entryPrice = close;
				SellMarket();
			}
		}

		_prevHighest = highestValue;
		_prevLowest = lowestValue;
		_hasPrev = true;
	}
}