VLT Trader — это порт стратегии MetaTrader 4 "VLT_TRADER" на платформу StockSharp. Алгоритм ищет участки с минимальной волатильностью и выставляет отложенные стоп-заявки вокруг последней свечи. Когда диапазон завершившейся свечи меньше минимального диапазона за предыдущие LookbackCandles баров, стратегия готовит симметричный «страддл» на пробой.
Логика торговли
Подписка на выбранный таймфрейм свечей и расчёт диапазона (High − Low) для каждой свечи.
Отслеживание минимального диапазона среди предыдущих LookbackCandles свечей с помощью индикатора Lowest.
Если последняя завершённая свеча уже исторического минимума, стратегия подготавливает новую пару отложенных ордеров.
Выставление Buy Stop выше максимума предыдущей свечи на EntryOffsetPoints пунктов и Sell Stop ниже минимума на такое же расстояние.
Каждому ордеру сразу задаются уровни Stop Loss и Take Profit согласно параметрам StopLossPoints и TakeProfitPoints.
Оба ордера остаются активными: когда один из них исполняется и открывает позицию, противоположный ордер продолжает ожидать возможного разворота.
После исполнения или отмены ордеров внутренние ссылки очищаются, чтобы стратегия смогла вновь построить страддл, когда не останется позиций и активных заявок.
Управление рисками
Объём позиций задаётся параметром OrderVolume и автоматически приводится к биржевым ограничениям по шагу и минимуму лота.
Расстояния стопов и тейков задаются в пунктах и переводятся в цену через PriceStep инструмента.
Параметры
Параметр
Описание
OrderVolume
Объём ордеров, выставляемых стратегией.
EntryOffsetPoints
Дополнительное расстояние от максимума/минимума предыдущей свечи при размещении стоп-заявок.
TakeProfitPoints
Дистанция до уровня Take Profit для каждой сделки.
StopLossPoints
Дистанция до уровня Stop Loss для каждой сделки.
LookbackCandles
Количество свечей для поиска минимального диапазона.
CandleType
Таймфрейм свечей, используемых в расчётах.
Примечания
Для работы стратегии у инструмента должен быть задан корректный PriceStep; без него заявки не выставляются.
Передача стопов и тейков вместе с отложенными ордерами может приводить к небольшим расхождениям с результатами MetaTrader из-за особенностей исполнения у брокера.
Реализация использует только высокоуровневый API (SubscribeCandles + Bind) и стандартный индикатор Lowest, полностью повторяя проверку волатильности оригинального советника.
using System;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
namespace StockSharp.Samples.Strategies;
public class VltTraderStraddleStrategy : Strategy
{
private readonly StrategyParam<int> _channelPeriod;
private readonly StrategyParam<int> _atrPeriod;
private readonly StrategyParam<int> _cooldownCandles;
private readonly StrategyParam<DataType> _candleType;
private decimal _prevClose;
private decimal _prevMid;
private bool _hasPrev;
private int _cooldownRemaining;
public int ChannelPeriod { get => _channelPeriod.Value; set => _channelPeriod.Value = value; }
public int AtrPeriod { get => _atrPeriod.Value; set => _atrPeriod.Value = value; }
public int CooldownCandles { get => _cooldownCandles.Value; set => _cooldownCandles.Value = value; }
public DataType CandleType { get => _candleType.Value; set => _candleType.Value = value; }
public VltTraderStraddleStrategy()
{
_channelPeriod = Param(nameof(ChannelPeriod), 48).SetDisplay("Channel Period", "Breakout lookback", "Indicators");
_atrPeriod = Param(nameof(AtrPeriod), 14).SetDisplay("ATR Period", "ATR lookback", "Indicators");
_cooldownCandles = Param(nameof(CooldownCandles), 150).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();
_prevClose = default;
_prevMid = default;
_hasPrev = default;
_cooldownRemaining = default;
}
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_prevClose = 0;
_prevMid = 0;
_hasPrev = false;
_cooldownRemaining = 0;
var highest = new Highest { Length = ChannelPeriod };
var lowest = new Lowest { Length = ChannelPeriod };
var atr = new AverageTrueRange { Length = AtrPeriod };
var subscription = SubscribeCandles(CandleType);
subscription.Bind(highest, lowest, atr, ProcessCandle).Start();
}
private void ProcessCandle(ICandleMessage candle, decimal highest, decimal lowest, decimal atr)
{
if (candle.State != CandleStates.Finished) return;
var close = candle.ClosePrice;
var mid = (highest + lowest) / 2;
if (!_hasPrev) { _prevClose = close; _prevMid = mid; _hasPrev = true; return; }
if (_cooldownRemaining > 0)
{
_cooldownRemaining--;
_prevClose = close;
_prevMid = mid;
return;
}
if (_prevClose <= _prevMid && close > mid && Position <= 0)
{
if (Position < 0) BuyMarket();
BuyMarket();
_cooldownRemaining = CooldownCandles;
}
else if (_prevClose >= _prevMid && close < mid && Position >= 0)
{
if (Position > 0) SellMarket();
SellMarket();
_cooldownRemaining = CooldownCandles;
}
_prevClose = close;
_prevMid = mid;
}
}