Открыть на GitHub

Стратегия Elite eFibo Trader

Описание

Elite eFibo Trader Strategy — конвертация советника MQL5 «Elite eFibo Trader». Стратегия строит сетку усредняющих заявок по числам Фибоначчи: открывает стартовую рыночную позицию и размещает цепочку стоп-заявок через фиксированные интервалы. Работа ведётся на тиковых данных, а уровни стоп-лосса автоматически подтягиваются по мере расширения сетки.

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

  1. При отсутствии открытых позиций и активных заявок, когда торговля разрешена, запускается новый цикл в выбранном направлении (покупка или продажа).
  2. Первая сделка исполняется по рынку объёмом LotsLevel1. Ещё тринадцать стоп-заявок выставляются на расстояниях, кратных LevelDistance, их объёмы задаются параметрами LotsLevel2LotsLevel14 (последовательность Фибоначчи).
  3. Для каждой сработавшей заявки фиксируется индивидуальный стоп-уровень на StopLossPoints шагов цены от входа. Максимальный (для лонга) или минимальный (для шорта) из этих уровней становится общим трейлинг-стопом для всей позиции.
  4. При достижении цены трейлинг-уровня позиция закрывается полностью, оставшиеся стоп-заявки снимаются.
  5. Плавающая прибыль рассчитывается в валюте счёта. Когда она достигает MoneyTakeProfit, сетка закрывается. В зависимости от TradeAgainAfterProfit цикл либо запускается повторно, либо стратегия ожидает ручного включения.

Параметры

  • OpenBuy — включить сетку только на покупку.
  • OpenSell — включить сетку только на продажу.
  • TradeAgainAfterProfit — автоматически перезапускать сетку после фиксации прибыли.
  • LevelDistance — расстояние между уровнями сетки в шагах цены инструмента.
  • StopLossPoints — расстояние стоп-лосса от цены входа в шагах цены.
  • MoneyTakeProfit — целевая плавающая прибыль в валюте счёта.
  • LotsLevel1LotsLevel14 — объёмы для каждого уровня сетки (по умолчанию последовательность Фибоначчи 1, 1, 2, 3, 5, …, 377).

Особенности реализации

  • Смещения по цене вычисляются через PriceStep инструмента. Если шаг цены не задан (0), стратегия не отправляет заявки.
  • Одновременно активен только один торговый цикл; все стоп-заявки создаются в момент запуска и остаются до исполнения или отмены.
  • При каждом новом исполнении пересчитывается общий трейлинг-стоп, чтобы все позиции имели лучший доступный уровень защиты.
  • Контроль прибыли основан на текущей позиции (Position, PositionPrice) и параметрах шага/стоимости шага (PriceStep, StepPrice).
  • Если TradeAgainAfterProfit = false, после достижения цели стратегия блокирует новые циклы до ручного включения.

Рекомендации по использованию

  • Перед запуском включите только одно направление (покупка или продажа). При одновременном включении обоих сетка не стартует.
  • Корректируйте расстояния и объёмы в соответствии с волатильностью инструмента и размером контракта. Большие объёмы по Фибоначчи ведут к агрессивному наращиванию позиции и требуют тщательного тестирования.
  • Убедитесь, что брокер поддерживает выставление стоп-заявок по рассчитанным ценам, иначе возможно отклонение заявок.
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>
/// Elite eFibo Trader grid strategy converted from MQL5.
/// Builds a Fibonacci-based sequence of market entries at price levels.
/// Buys or sells at progressively worse prices with increasing volume (Fibonacci sequence).
/// Exits when total PnL target is reached or stop loss is hit.
/// </summary>
public class EliteEFiboTraderStrategy : Strategy
{
	private readonly StrategyParam<int> _levelsCount;
	private readonly StrategyParam<bool> _openBuy;
	private readonly StrategyParam<bool> _openSell;
	private readonly StrategyParam<decimal> _levelDistance;
	private readonly StrategyParam<decimal> _stopLossPoints;
	private readonly StrategyParam<decimal> _takeProfitPoints;
	private readonly StrategyParam<DataType> _candleType;

	private decimal _entryPrice;
	private int _currentLevel;
	private int _activeDirection;
	private bool _cycleActive;

	// Fibonacci volumes for grid levels
	private static readonly decimal[] FibVolumes = { 1, 1, 2, 3, 5, 8, 13, 21, 34, 55 };

	/// <summary>
	/// Enable buy-only mode.
	/// </summary>
	public bool OpenBuy
	{
		get => _openBuy.Value;
		set => _openBuy.Value = value;
	}

	/// <summary>
	/// Enable sell-only mode.
	/// </summary>
	public bool OpenSell
	{
		get => _openSell.Value;
		set => _openSell.Value = value;
	}

	/// <summary>
	/// Number of Fibonacci grid levels.
	/// </summary>
	public int LevelsCount
	{
		get => _levelsCount.Value;
		set => _levelsCount.Value = value;
	}

	/// <summary>
	/// Distance between successive pending levels in price steps.
	/// </summary>
	public decimal LevelDistance
	{
		get => _levelDistance.Value;
		set => _levelDistance.Value = value;
	}

	/// <summary>
	/// Stop-loss size in price steps.
	/// </summary>
	public decimal StopLossPoints
	{
		get => _stopLossPoints.Value;
		set => _stopLossPoints.Value = value;
	}

	/// <summary>
	/// Take profit size in price steps.
	/// </summary>
	public decimal TakeProfitPoints
	{
		get => _takeProfitPoints.Value;
		set => _takeProfitPoints.Value = value;
	}

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

	/// <summary>
	/// Initializes strategy parameters.
	/// </summary>
	public EliteEFiboTraderStrategy()
	{
		_levelsCount = Param(nameof(LevelsCount), 6)
			.SetGreaterThanZero()
			.SetDisplay("Levels Count", "Number of Fibonacci levels", "Grid");

		_openBuy = Param(nameof(OpenBuy), true)
			.SetDisplay("Open Buy", "Enable buying", "General");

		_openSell = Param(nameof(OpenSell), true)
			.SetDisplay("Open Sell", "Enable selling", "General");

		_levelDistance = Param(nameof(LevelDistance), 50m)
			.SetGreaterThanZero()
			.SetDisplay("Level Distance", "Distance between orders in price steps", "Grid");

		_stopLossPoints = Param(nameof(StopLossPoints), 200m)
			.SetGreaterThanZero()
			.SetDisplay("Stop Loss", "Stop-loss size in price steps", "Risk");

		_takeProfitPoints = Param(nameof(TakeProfitPoints), 100m)
			.SetGreaterThanZero()
			.SetDisplay("Take Profit", "Take profit size in price steps", "Risk");

		_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(5).TimeFrame())
			.SetDisplay("Candle Type", "Type of candles for calculations", "General");
	}

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

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

		_cycleActive = false;
		_currentLevel = 0;
		_activeDirection = 0;
		_entryPrice = 0;

		var sma = new SMA { Length = 20 };

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

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

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

		if (!IsFormedAndOnlineAndAllowTrading())
			return;

		var close = candle.ClosePrice;
		var step = Security.PriceStep ?? 1m;

		if (!_cycleActive && Position == 0)
		{
			// Start a new cycle based on trend direction
			if (OpenBuy && close > smaValue)
			{
				_activeDirection = 1;
				_entryPrice = close;
				_currentLevel = 0;
				_cycleActive = true;
				BuyMarket();
			}
			else if (OpenSell && close < smaValue)
			{
				_activeDirection = -1;
				_entryPrice = close;
				_currentLevel = 0;
				_cycleActive = true;
				SellMarket();
			}
		}
		else if (_cycleActive)
		{
			var stopDistance = StopLossPoints * step;
			var tpDistance = TakeProfitPoints * step;
			var levelDist = LevelDistance * step;

			// Check stop loss
			if (_activeDirection == 1 && close <= _entryPrice - stopDistance)
			{
				SellMarket(Math.Abs(Position));
				ResetCycle();
				return;
			}
			else if (_activeDirection == -1 && close >= _entryPrice + stopDistance)
			{
				BuyMarket(Math.Abs(Position));
				ResetCycle();
				return;
			}

			// Check take profit
			if (_activeDirection == 1 && close >= _entryPrice + tpDistance)
			{
				SellMarket(Math.Abs(Position));
				ResetCycle();
				return;
			}
			else if (_activeDirection == -1 && close <= _entryPrice - tpDistance)
			{
				BuyMarket(Math.Abs(Position));
				ResetCycle();
				return;
			}

			// Check for grid level additions (averaging into losing positions)
			var nextLevel = _currentLevel + 1;
			if (nextLevel < LevelsCount && nextLevel < FibVolumes.Length)
			{
				if (_activeDirection == 1 && close <= _entryPrice - levelDist * nextLevel)
				{
					BuyMarket(FibVolumes[nextLevel]);
					_currentLevel = nextLevel;
				}
				else if (_activeDirection == -1 && close >= _entryPrice + levelDist * nextLevel)
				{
					SellMarket(FibVolumes[nextLevel]);
					_currentLevel = nextLevel;
				}
			}
		}
	}

	private void ResetCycle()
	{
		_cycleActive = false;
		_currentLevel = 0;
		_activeDirection = 0;
		_entryPrice = 0;
	}
}