Russian20 Momentum MA — это прямой порт оригинального советника MetaTrader 5 Russian20-hp1.mq5. Скрипт компании Gordago Software Corp. использует двухчасовые свечи, простое скользящее среднее с периодом 20 и индикатор Momentum с периодом 5 для отслеживания продолжения краткосрочного тренда. Версия для StockSharp сохраняет исходную логику анализа и адаптирует управление позициями под высокоуровневый API стратегий.
Торговая логика
Таймфрейм: по умолчанию 2-часовые свечи (аналог MQL5 PERIOD_H2). Обработка выполняется только после закрытия свечи.
Индикаторы:
Простое скользящее среднее (SMA) с настраиваемым периодом, по умолчанию 20.
Индикатор Momentum с настраиваемым периодом, по умолчанию 5. Уровень 100 используется как нейтральное значение, как и в MQL5.
Условия входа в длинную позицию (все пункты должны выполниться):
Цена закрытия выше SMA.
Значение Momentum превышает 100 и подтверждает положительное ускорение.
Цена закрытия выше закрытия предыдущей свечи, что фиксирует восходящий импульс.
Условия входа в короткую позицию (все пункты должны выполниться):
Цена закрытия ниже SMA.
Значение Momentum ниже 100 и показывает отрицательное ускорение.
Цена закрытия ниже закрытия предыдущей свечи.
Выход из длинной позиции: при падении Momentum ниже 100 либо при достижении рассчитанного стоп-лосса или тейк-профита.
Выход из короткой позиции: при росте Momentum выше 100 либо при достижении соответствующих уровней стоп-лосса и тейк-профита.
Управление рисками
В MQL5 версия использует фиксированные стоп-лосс и тейк-профит в пунктах с поправкой на 4- и 5-значные котировки. Реализация на C# повторяет этот подход:
Расчитывает размер пункта на основе PriceStep. Для инструментов с тремя или пятью десятичными знаками пункт равен PriceStep * 10, в остальных случаях — PriceStep.
Переводит заданные пользователем значения стоп-лосса и тейк-профита в абсолютные ценовые расстояния.
На закрытии каждой свечи контролирует пересечение цены с вычисленными уровнями и закрывает позицию при необходимости.
Параметры
Параметр
Значение по умолчанию
Описание
CandleType
2-часовые свечи
Тип данных, используемый для расчётов.
MovingAverageLength
20
Период SMA-фильтра.
MomentumPeriod
5
Период индикатора Momentum.
StopLossBuyPips
50
Размер стоп-лосса для покупок в пунктах (0 — отключить).
TakeProfitBuyPips
50
Размер тейк-профита для покупок в пунктах (0 — отключить).
StopLossSellPips
50
Размер стоп-лосса для продаж в пунктах (0 — отключить).
TakeProfitSellPips
50
Размер тейк-профита для продаж в пунктах (0 — отключить).
Все числовые параметры реализованы через StrategyParam<T> и при необходимости помечены как оптимизируемые, что позволяет легко запускать тесты и подбор параметров в StockSharp.
Особенности реализации
Используется высокоуровневый вызов SubscribeCandles().Bind(...), который одновременно предоставляет свечи и значения SMA/Momentum без ручного копирования буферов.
Пороговые значения Momentum полностью совпадают с оригинальной версией; при выходе цены за преобразованные уровни стоп-лосса или тейк-профита позиция закрывается рыночным ордером.
Для оценки импульса хранится только предыдущее значение закрытия, что избавляет от дорогостоящих выборок по историям и соответствует правилам производительности проекта.
При наличии графического окружения доступны функции DrawCandles, DrawIndicator, DrawOwnTrades для визуального контроля сигналов и сделок.
Рекомендации по применению
Стандартные настройки соответствуют исходной стратегии. При работе с другими инструментами скорректируйте CandleType и периоды индикаторов.
Для инструментов с нестандартным шагом цены убедитесь, что пересчитанный размер пункта соответствует ожиданиям перед установкой стоп-уровней.
Стратегия рассчитана на единственную открытую позицию. Ручные сделки или параллельные алгоритмы по тому же инструменту могут нарушить логику выхода.
using System;
using System.Collections.Generic;
using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;
namespace StockSharp.Samples.Strategies;
/// <summary>
/// Russian20 Momentum MA strategy. Combines SMA filter with momentum confirmation.
/// </summary>
public class Russian20MomentumMaStrategy : Strategy
{
private readonly StrategyParam<DataType> _candleType;
private readonly StrategyParam<int> _maPeriod;
private readonly StrategyParam<int> _momPeriod;
private decimal? _prevMom;
public DataType CandleType
{
get => _candleType.Value;
set => _candleType.Value = value;
}
public int MaPeriod
{
get => _maPeriod.Value;
set => _maPeriod.Value = value;
}
public int MomPeriod
{
get => _momPeriod.Value;
set => _momPeriod.Value = value;
}
public Russian20MomentumMaStrategy()
{
_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
.SetDisplay("Candle Type", "Timeframe", "General");
_maPeriod = Param(nameof(MaPeriod), 20)
.SetGreaterThanZero()
.SetDisplay("MA Period", "SMA period for trend filter", "Indicators");
_momPeriod = Param(nameof(MomPeriod), 10)
.SetGreaterThanZero()
.SetDisplay("Momentum Period", "Momentum lookback", "Indicators");
}
public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
{
return [(Security, CandleType)];
}
/// <inheritdoc />
protected override void OnReseted()
{
base.OnReseted();
_prevMom = null;
}
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_prevMom = null;
var sma = new SimpleMovingAverage { Length = MaPeriod };
var mom = new Momentum { Length = MomPeriod };
var subscription = SubscribeCandles(CandleType);
subscription
.Bind(sma, mom, ProcessCandle)
.Start();
var area = CreateChartArea();
if (area != null)
{
DrawCandles(area, subscription);
DrawIndicator(area, sma);
DrawOwnTrades(area);
}
}
private void ProcessCandle(ICandleMessage candle, decimal maVal, decimal momVal)
{
if (candle.State != CandleStates.Finished)
return;
if (!IsFormedAndOnlineAndAllowTrading())
{
_prevMom = momVal;
return;
}
if (_prevMom == null)
{
_prevMom = momVal;
return;
}
var close = candle.ClosePrice;
// Price above MA + momentum crosses above zero → buy
if (close > maVal && _prevMom.Value <= 100m && momVal > 100m && Position <= 0)
{
if (Position < 0)
BuyMarket();
BuyMarket();
}
// Price below MA + momentum crosses below zero → sell
else if (close < maVal && _prevMom.Value >= 100m && momVal < 100m && Position >= 0)
{
if (Position > 0)
SellMarket();
SellMarket();
}
_prevMom = momVal;
}
}
import clr
clr.AddReference("StockSharp.Messages")
clr.AddReference("StockSharp.Algo")
clr.AddReference("StockSharp.Algo.Indicators")
clr.AddReference("StockSharp.Algo.Strategies")
from System import TimeSpan
from StockSharp.Messages import DataType, CandleStates
from StockSharp.Algo.Indicators import SimpleMovingAverage, Momentum
from StockSharp.Algo.Strategies import Strategy
class russian20_momentum_ma_strategy(Strategy):
def __init__(self):
super(russian20_momentum_ma_strategy, self).__init__()
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromHours(4))) \
.SetDisplay("Candle Type", "Timeframe", "General")
self._ma_period = self.Param("MaPeriod", 20) \
.SetDisplay("MA Period", "SMA period for trend filter", "Indicators")
self._mom_period = self.Param("MomPeriod", 10) \
.SetDisplay("Momentum Period", "Momentum lookback", "Indicators")
self._prev_mom = None
@property
def CandleType(self):
return self._candle_type.Value
@property
def MaPeriod(self):
return self._ma_period.Value
@property
def MomPeriod(self):
return self._mom_period.Value
def OnReseted(self):
super(russian20_momentum_ma_strategy, self).OnReseted()
self._prev_mom = None
def OnStarted2(self, time):
super(russian20_momentum_ma_strategy, self).OnStarted2(time)
self._prev_mom = None
sma = SimpleMovingAverage()
sma.Length = self.MaPeriod
mom = Momentum()
mom.Length = self.MomPeriod
subscription = self.SubscribeCandles(self.CandleType)
subscription.Bind(sma, mom, self._on_process).Start()
area = self.CreateChartArea()
if area is not None:
self.DrawCandles(area, subscription)
self.DrawIndicator(area, sma)
self.DrawOwnTrades(area)
def _on_process(self, candle, ma_value, mom_value):
if candle.State != CandleStates.Finished:
return
mv = float(ma_value)
momv = float(mom_value)
if not self.IsFormedAndOnlineAndAllowTrading():
self._prev_mom = momv
return
if self._prev_mom is None:
self._prev_mom = momv
return
close = float(candle.ClosePrice)
# Price above MA + momentum crosses above 100
if close > mv and self._prev_mom <= 100.0 and momv > 100.0 and self.Position <= 0:
if self.Position < 0:
self.BuyMarket()
self.BuyMarket()
# Price below MA + momentum crosses below 100
elif close < mv and self._prev_mom >= 100.0 and momv < 100.0 and self.Position >= 0:
if self.Position > 0:
self.SellMarket()
self.SellMarket()
self._prev_mom = momv
def CreateClone(self):
return russian20_momentum_ma_strategy()