Открыть на GitHub

Простая стратегия разворота по пивоту

Обзор

Стратегия представляет собой высокоуровневый порт советника MetaTrader 4 из файла MQL/7610/Simplepivot_www_forex-instruments_info.mq4. В оригинале на каждом новом баре берётся его цена открытия и сравнивается с диапазоном предыдущей свечи, после чего советник открывает или переворачивает позицию. В версии для StockSharp та же логика реализована с помощью высокоуровневых вызовов SubscribeCandles, Bind, BuyMarket, SellMarket и ClosePosition.

Пошаговая схема работы:

  1. Ожидание закрытия свечи, чтобы получить её open/high/low.
  2. Вычисление пивота как середины диапазона предыдущего бара.
  3. Если новый бар открылся в нижней половине диапазона либо сделал разрыв вверх, открывается длинная позиция.
  4. Если открытие находится в верхней половине предыдущего диапазона, открывается короткая позиция.
  5. Перед сменой направления текущая позиция закрывается — тем самым сохраняется логика одного тикета, как и в MQL.

В оригинальном советнике нет стоп-лоссов и тейк-профитов, поэтому позиция разворачивается только на следующем сигнале.

Параметры

Имя Значение по умолчанию Описание
OrderVolume 1 Объём рыночного ордера при входе.
CandleType Таймфрейм 1 минута Тип свечей, которые запрашиваются у источника данных.

Детали торговой логики

  1. Первая полностью сформированная свеча сохраняется и используется как эталон. Пока нет предыдущей свечи, стратегия не торгует.
  2. Для каждой следующей завершённой свечи:
    • Рассчитывается pivot = (previousHigh + previousLow) / 2.
    • Если Open < previousHigh и одновременно Open > pivot, выбирается короткое направление.
    • Во всех остальных случаях выбирается длинное направление (нижняя половина диапазона, равенство пивоту, гэпы вверх/вниз).
  3. Если уже открыта позиция в нужную сторону, сигнал пропускается, чтобы не платить спрэд повторно. Это соответствует проверке prevTradeOp в MQL.
  4. При смене направления вызывается ClosePosition(), после чего отправляется новый рыночный ордер объёмом OrderVolume.
  5. Диапазон предыдущей свечи обновляется значениями текущего бара и используется в следующей итерации.

Управление риском

Алгоритм не содержит защитных стопов и целей по прибыли. Управление риском сводится к настройке объёма OrderVolume или внешним мерам контроля со стороны портфеля/счёта.

Визуализация

При наличии области графика стратегия отображает свечи и собственные сделки, что облегчает проверку корректности разворотов по пивоту в тестах.

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;

using StockSharp.Algo;

namespace StockSharp.Samples.Strategies;

/// <summary>
/// Simple pivot-based strategy converted from the MetaTrader expert advisor in MQL/7610.
/// The strategy compares the current candle open with the previous candle range to decide
/// whether the next trade should be long or short.
/// </summary>
public class SimplePivotFlipStrategy : Strategy
{
	private readonly StrategyParam<decimal> _orderVolume;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _previousHigh;
	private decimal _previousLow;
	private bool _hasPreviousCandle;

	/// <summary>
	/// Order volume used for market entries.
	/// </summary>
	public decimal OrderVolume
	{
		get => _orderVolume.Value;
		set => _orderVolume.Value = value;
	}

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

	/// <summary>
	/// Initializes a new instance of the <see cref="SimplePivotFlipStrategy"/> class.
	/// </summary>
	public SimplePivotFlipStrategy()
	{
		_orderVolume = Param(nameof(OrderVolume), 1m)
			.SetGreaterThanZero()
			.SetDisplay("Order Volume", "Market order volume used for entries.", "General");

		_candleType = Param(nameof(CandleType), TimeSpan.FromDays(1).TimeFrame())
			.SetDisplay("Candle Type", "Type of candles used for pivot calculation.", "Data");
	}

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

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

		_previousHigh = 0m;
		_previousLow = 0m;
		_hasPreviousCandle = false;
	}

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

		var subscription = SubscribeCandles(CandleType);
		subscription
			.Bind(ProcessCandle)
			.Start();

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

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

		

		if (!_hasPreviousCandle)
		{
			// Store the first completed candle to build the reference range.
			_previousHigh = candle.HighPrice;
			_previousLow = candle.LowPrice;
			_hasPreviousCandle = true;
			return;
		}

		// Calculate the pivot as the midpoint of the previous candle range.
		var pivot = (_previousHigh + _previousLow) / 2m;
		var desiredSide = Sides.Buy;

		// If the new candle opens inside the upper half of the previous range we go short.
		if (candle.OpenPrice < _previousHigh && candle.OpenPrice > pivot)
			desiredSide = Sides.Sell;

		// Skip re-entry if we already hold a position in the desired direction.
		if ((desiredSide == Sides.Buy && Position > 0) || (desiredSide == Sides.Sell && Position < 0))
		{
			_previousHigh = candle.HighPrice;
			_previousLow = candle.LowPrice;
			return;
		}

		if (Position > 0)
			SellMarket();
		else if (Position < 0)
			BuyMarket();

		if (desiredSide == Sides.Buy)
		{
			BuyMarket();
		}
		else
		{
			SellMarket();
		}

		// Update reference range for the next candle.
		_previousHigh = candle.HighPrice;
		_previousLow = candle.LowPrice;
	}
}