Открыть на GitHub

Стратегия SilverTrend CrazyChart

Стратегия является портом советника MetaTrader «Exp_SilverTrend_CrazyChart» на платформу StockSharp. Логика использует два буфер a индикатора SilverTrend CrazyChart: текущую линию и смещённую линию. Когда смещённая линия пересекает текущую, стратегия открыв ает позицию в сторону доминирующего канала и закрывает противоположную.

Детали

  • Условия входа:
    • Покупка: на предыдущем завершённом сигналом баре текущая линия располагалась выше смещённой, а на анализируемом баре теку щая линия опустилась ниже или коснулась смещённой. Входы можно отключить параметром AllowBuyEntry.
    • Продажа: на предыдущем сигналом баре текущая линия была ниже смещённой, а на текущем баре поднялась выше или коснулась см ещённой. Входы можно отключить параметром AllowSellEntry.
  • Направление торговли: длинные и короткие позиции.
  • Условия выхода:
    • Длинные позиции закрываются при пересечении линий в противоположном направлении (AllowBuyExit) либо по стоп-лоссу/тейк-про iту.
    • Короткие позиции закрываются, когда текущая линия обгоняет смещённую (AllowSellExit), а также при срабатывании стоп-лосса и i тейк-профита.
  • Стопы: используются абсолютные расстояния от цены входа (StopLossPoints и TakeProfitPoints). Значение 0 отключает соотв eтствующий предел.
  • Фильтры:
    • SignalBar задаёт, сколько полностью сформированных свечей назад оценивается пересечение.
    • CandleType определяет таймфрейм расчёта индикатора.

Параметры

  • CandleType – тип свечей, подаваемых на индикатор (по умолчанию часовые).
  • Length – период качания (SSP) для индикатора SilverTrend CrazyChart.
  • KMin – коэффициент смещения для расчёта смещённой линии.
  • KMax – коэффициент смещения для расчёта текущей линии.
  • SignalBar – количество завершённых свечей назад для оценки сигнала.
  • AllowBuyEntry / AllowSellEntry – разрешение на открытие длинных/коротких позиций.
  • AllowBuyExit / AllowSellExit – разрешение на закрытие длинных/коротких позиций.
  • StopLossPoints – абсолютная дистанция для стоп-лосса длинных и тейк-профита коротких позиций.
  • TakeProfitPoints – абсолютная дистанция для тейк-профита длинных и стоп-лосса коротких позиций.
  • Volume – базовый размер заявки стратегии.

Логика индикатора

Встроенный SilverTrendCrazyChartIndicator повторяет буферы оригинала:

  • По параметрам Length, KMin и KMax формируется канал на основе максимума и минимума за период.
  • «Текущая» линия соответствует нулевому буферу MetaTrader и реагирует на последнюю свечу без задержки.
  • «Смещённая» линия соответствует первому буферу и отстаёт на Length + 1 бар для совпадения с рисунком исходного индикатора.

Покупка возникает, когда смещённая линия пересекает текущую снизу вверх, а продажа – при обратном пересечении. Параметр SignalB a гарантирует использование только закрытых свечей, как и в оригинальном советнике.

using System;
using System.Collections.Generic;

using Ecng.Common;

using StockSharp.Algo.Indicators;
using StockSharp.Algo.Strategies;
using StockSharp.BusinessEntities;
using StockSharp.Messages;

namespace StockSharp.Samples.Strategies;

/// <summary>
/// SilverTrend CrazyChart strategy (simplified).
/// Uses Highest/Lowest channel inversions for entries.
/// </summary>
public class SilverTrendCrazyChartStrategy : Strategy
{
	private readonly StrategyParam<DataType> _candleType;
	private readonly StrategyParam<int> _length;

	public DataType CandleType
	{
		get => _candleType.Value;
		set => _candleType.Value = value;
	}

	public int Length
	{
		get => _length.Value;
		set => _length.Value = value;
	}

	public SilverTrendCrazyChartStrategy()
	{
		_candleType = Param(nameof(CandleType), TimeSpan.FromHours(4).TimeFrame())
			.SetDisplay("Candle Type", "Candles", "General");

		_length = Param(nameof(Length), 14)
			.SetGreaterThanZero()
			.SetDisplay("Length", "Channel length", "Indicators");
	}

	protected override void OnStarted2(DateTime time)
	{
		base.OnStarted2(time);

		var highest = new Highest { Length = Length };
		var lowest = new Lowest { Length = Length };

		decimal prevHigh = 0, prevLow = 0;
		bool hasPrev = false;

		var subscription = SubscribeCandles(CandleType);
		subscription
			.Bind(highest, lowest, (ICandleMessage candle, decimal highValue, decimal lowValue) =>
			{
				if (candle.State != CandleStates.Finished)
					return;

				if (!hasPrev)
				{
					prevHigh = highValue;
					prevLow = lowValue;
					hasPrev = true;
					return;
				}

				if (!IsFormedAndOnlineAndAllowTrading())
				{
					prevHigh = highValue;
					prevLow = lowValue;
					return;
				}

				var close = candle.ClosePrice;
				var mid = (highValue + lowValue) / 2m;
				var prevMid = (prevHigh + prevLow) / 2m;

				// Price crosses above channel midpoint
				if (close > mid && candle.OpenPrice <= prevMid && Position <= 0)
				{
					BuyMarket();
				}
				// Price crosses below channel midpoint
				else if (close < mid && candle.OpenPrice >= prevMid && Position >= 0)
				{
					SellMarket();
				}

				prevHigh = highValue;
				prevLow = lowValue;
			})
			.Start();

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