Стратегия Forex Fraus M1 переносит советника MetaTrader 5 «Forex Fraus M1» в инфраструктуру StockSharp. Это контртрендовая система, которая отслеживает осциллятор Williams %R с большим периодом (360) на минутных свечах. Когда индикатор достигает экстремальных значений, алгоритм пытается развернуть движение, рассчитывая на быстрый возврат к среднему уровню диапазона. В реализации сохранены все элементы управления рисками из оригинального советника: ограничение по времени торговли, фиксированные стоп-лосс и тейк-профит в пунктах, а также трейлинг-стоп.
Торговая логика
Индикатор: Williams %R с периодом 360.
Покупка: при значении Williams %R ниже -99.9 рынок считается сильно перепроданным. Стратегия отправляет рыночную заявку на покупку, если длинной позиции нет. При активном флаге CloseOppositePositions короткий объём закрывается в той же заявке.
Продажа: при значении Williams %R выше -0.1 рынок перекуплен. Отправляется рыночная заявка на продажу; при необходимости длинная позиция закрывается сразу.
Фильтр по времени: при включённом UseTimeControl сигналы обрабатываются только между StartHour (включительно) и EndHour (не включая). Если начало сессии позже её окончания (StartHour > EndHour), торговля разрешена с StartHour до 23 часов и с 0 до EndHour - 1.
Управление рисками
Стоп-лосс: уровень рассчитывается как StopLossPips * PipSize ниже (для покупок) или выше (для продаж) цены входа. Если минимум свечи достигает уровня стопа, позиция закрывается по рынку.
Тейк-профит: уровень определяется как TakeProfitPips * PipSize выше (для покупок) или ниже (для продаж) цены входа. При достижении максимума/минимума свечи фиксируется прибыль.
Трейлинг-стоп: при положительных TrailingStopPips и TrailingStepPips стоп подтягивается каждый раз, когда цена проходит не менее TrailingStopPips + TrailingStepPips пунктов в прибыльную сторону. Для длинных позиций стоп следует за закрытием свечи на расстоянии TrailingStopPips; для коротких — над закрытием.
Размер пункта: параметр PipSize задаёт стоимость одного пункта. Для пятизначных валютных пар используйте 0.0001, для трёхзначных пар с JPY — 0.01 и т. д.
Проверка уровней стоп-лосса и тейк-профита выполняется по максимумам и минимумам завершённых свечей. Если в пределах одной свечи задеты оба уровня, приоритет отдаётся стоп-лоссу — это повторяет консервативный подход исходного советника.
Параметры
Параметр
Значение по умолчанию
Описание
OrderVolume
0.1
Объём сделки для открытия новых позиций.
StopLossPips
50
Расстояние стоп-лосса в пунктах от цены входа. Ноль отключает стоп.
TakeProfitPips
150
Расстояние тейк-профита в пунктах. Ноль отключает тейк.
TrailingStopPips
1
Базовое расстояние трейлинг-стопа в пунктах. Ноль отключает сопровождение.
TrailingStepPips
1
Минимальное дополнительное движение цены, после которого стоп подтягивается.
UseTimeControl
true
Включает фильтр торговых часов.
StartHour
7
Час начала торговой сессии (0-23).
EndHour
17
Час окончания сессии (1-24, не включая).
CloseOppositePositions
true
При включении стратегия полностью разворачивает позицию одной заявкой.
WilliamsPeriod
360
Период расчёта Williams %R.
CandleType
1 minute
Тип свечей, используемых в вычислениях.
PipSize
0.0001
Стоимость одного пункта в ценовых единицах.
Дополнительные сведения
Используется высокоуровневый API StockSharp: подписка на свечи и привязка индикатора позволяют избежать ручной работы с буферами.
Все проверки стопов и трейлинга выполняются только на завершённых свечах, чтобы не реагировать на незакрытые данные.
Метод StartProtection() вызывается один раз при запуске стратегии согласно требованиям проекта; основной контроль рисков реализован внутри стратегии.
Обязательно адаптируйте PipSize под торгуемый инструмент, чтобы пункты корректно пересчитывались в цену.
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>
/// Forex Fraus M1 strategy using fast EMA crossover on short timeframes.
/// </summary>
public class ForexFrausM1Strategy : Strategy
{
private readonly StrategyParam<DataType> _candleType;
private readonly StrategyParam<int> _fastPeriod;
private readonly StrategyParam<int> _slowPeriod;
private decimal? _prevFast;
private decimal? _prevSlow;
public DataType CandleType
{
get => _candleType.Value;
set => _candleType.Value = value;
}
public int FastPeriod
{
get => _fastPeriod.Value;
set => _fastPeriod.Value = value;
}
public int SlowPeriod
{
get => _slowPeriod.Value;
set => _slowPeriod.Value = value;
}
public ForexFrausM1Strategy()
{
_candleType = Param(nameof(CandleType), TimeSpan.FromHours(1).TimeFrame())
.SetDisplay("Candle Type", "Timeframe", "General");
_fastPeriod = Param(nameof(FastPeriod), 5)
.SetGreaterThanZero()
.SetDisplay("Fast Period", "Fast EMA", "Indicators");
_slowPeriod = Param(nameof(SlowPeriod), 15)
.SetGreaterThanZero()
.SetDisplay("Slow Period", "Slow EMA", "Indicators");
}
public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
{
return [(Security, CandleType)];
}
/// <inheritdoc />
protected override void OnReseted()
{
base.OnReseted();
_prevFast = null;
_prevSlow = null;
}
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_prevFast = null;
_prevSlow = null;
var fast = new ExponentialMovingAverage { Length = FastPeriod };
var slow = new ExponentialMovingAverage { Length = SlowPeriod };
var subscription = SubscribeCandles(CandleType);
subscription
.Bind(fast, slow, ProcessCandle)
.Start();
var area = CreateChartArea();
if (area != null)
{
DrawCandles(area, subscription);
DrawIndicator(area, fast);
DrawIndicator(area, slow);
DrawOwnTrades(area);
}
}
private void ProcessCandle(ICandleMessage candle, decimal fastVal, decimal slowVal)
{
if (candle.State != CandleStates.Finished)
return;
if (!IsFormedAndOnlineAndAllowTrading())
{
_prevFast = fastVal;
_prevSlow = slowVal;
return;
}
if (_prevFast == null || _prevSlow == null)
{
_prevFast = fastVal;
_prevSlow = slowVal;
return;
}
var prevAbove = _prevFast.Value > _prevSlow.Value;
var currAbove = fastVal > slowVal;
_prevFast = fastVal;
_prevSlow = slowVal;
if (!prevAbove && currAbove)
{
if (Position < 0)
BuyMarket();
if (Position <= 0)
BuyMarket();
}
else if (prevAbove && !currAbove)
{
if (Position > 0)
SellMarket();
if (Position >= 0)
SellMarket();
}
}
}
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 ExponentialMovingAverage
from StockSharp.Algo.Strategies import Strategy
class forex_fraus_m1_strategy(Strategy):
def __init__(self):
super(forex_fraus_m1_strategy, self).__init__()
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromHours(1))) \
.SetDisplay("Candle Type", "Timeframe", "General")
self._fast_period = self.Param("FastPeriod", 5) \
.SetDisplay("Fast Period", "Fast EMA", "Indicators")
self._slow_period = self.Param("SlowPeriod", 15) \
.SetDisplay("Slow Period", "Slow EMA", "Indicators")
self._prev_fast = None
self._prev_slow = None
@property
def CandleType(self):
return self._candle_type.Value
@property
def FastPeriod(self):
return self._fast_period.Value
@property
def SlowPeriod(self):
return self._slow_period.Value
def OnReseted(self):
super(forex_fraus_m1_strategy, self).OnReseted()
self._prev_fast = None
self._prev_slow = None
def OnStarted2(self, time):
super(forex_fraus_m1_strategy, self).OnStarted2(time)
self._prev_fast = None
self._prev_slow = None
fast = ExponentialMovingAverage()
fast.Length = self.FastPeriod
slow = ExponentialMovingAverage()
slow.Length = self.SlowPeriod
subscription = self.SubscribeCandles(self.CandleType)
subscription.Bind(fast, slow, self._on_process).Start()
area = self.CreateChartArea()
if area is not None:
self.DrawCandles(area, subscription)
self.DrawIndicator(area, fast)
self.DrawIndicator(area, slow)
self.DrawOwnTrades(area)
def _on_process(self, candle, fast_value, slow_value):
if candle.State != CandleStates.Finished:
return
fv = float(fast_value)
sv = float(slow_value)
if self._prev_fast is None or self._prev_slow is None:
self._prev_fast = fv
self._prev_slow = sv
return
prev_above = self._prev_fast > self._prev_slow
curr_above = fv > sv
self._prev_fast = fv
self._prev_slow = sv
if not prev_above and curr_above:
if self.Position < 0:
self.BuyMarket()
if self.Position <= 0:
self.BuyMarket()
elif prev_above and not curr_above:
if self.Position > 0:
self.SellMarket()
if self.Position >= 0:
self.SellMarket()
def CreateClone(self):
return forex_fraus_m1_strategy()