Volume Surge
Volume Surge identifies unusually high volume relative to the moving average. When the ratio exceeds the defined multiplier, it signals strong interest and potential continuation in the direction of price relative to its moving average.
Testing indicates an average annual return of about 52%. It performs best in the crypto market.
Trades are initiated only on a surge and closed once volume falls back below average or when the stop-loss is reached.
This simple approach captures momentum sparked by sudden participation.
Details
- Entry Criteria: Volume ratio above
VolumeSurgeMultiplier. - Long/Short: Both directions.
- Exit Criteria: Volume drops below average or stop.
- Stops: Yes.
- Default Values:
MAPeriod= 20VolumeAvgPeriod= 20VolumeSurgeMultiplier= 2.0mCandleType= TimeSpan.FromMinutes(5)
- Filters:
- Category: Breakout
- Direction: Both
- Indicators: Volume
- Stops: Yes
- Complexity: Basic
- Timeframe: Intraday
- Seasonality: No
- Neural Networks: No
- Divergence: No
- Risk Level: Medium
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>
/// Volume Surge strategy.
/// Long: Price above MA with volume confirmation.
/// Short: Price below MA with volume confirmation.
/// Exit: Price crosses MA.
/// </summary>
public class VolumeSurgeStrategy : Strategy
{
private readonly StrategyParam<int> _maPeriod;
private readonly StrategyParam<DataType> _candleType;
private readonly StrategyParam<int> _cooldownBars;
private decimal _prevClose;
private decimal _prevMa;
private decimal _prevVolume;
private int _cooldown;
/// <summary>
/// MA Period.
/// </summary>
public int MAPeriod
{
get => _maPeriod.Value;
set => _maPeriod.Value = value;
}
/// <summary>
/// Candle type for strategy calculation.
/// </summary>
public DataType CandleType
{
get => _candleType.Value;
set => _candleType.Value = value;
}
/// <summary>
/// Cooldown bars between trades.
/// </summary>
public int CooldownBars
{
get => _cooldownBars.Value;
set => _cooldownBars.Value = value;
}
/// <summary>
/// Initialize <see cref="VolumeSurgeStrategy"/>.
/// </summary>
public VolumeSurgeStrategy()
{
_maPeriod = Param(nameof(MAPeriod), 20)
.SetGreaterThanZero()
.SetDisplay("MA Period", "Period for price MA", "Indicators")
.SetOptimize(10, 50, 10);
_candleType = Param(nameof(CandleType), TimeSpan.FromMinutes(1).TimeFrame())
.SetDisplay("Candle Type", "Type of candles to use", "General");
_cooldownBars = Param(nameof(CooldownBars), 500)
.SetRange(1, 1000)
.SetDisplay("Cooldown Bars", "Bars to wait between trades", "General");
}
/// <inheritdoc />
public override IEnumerable<(Security sec, DataType dt)> GetWorkingSecurities()
{
return [(Security, CandleType)];
}
/// <inheritdoc />
protected override void OnReseted()
{
base.OnReseted();
_prevClose = default;
_prevMa = default;
_prevVolume = default;
_cooldown = default;
}
/// <inheritdoc />
protected override void OnStarted2(DateTime time)
{
base.OnStarted2(time);
_prevClose = 0;
_prevMa = 0;
_prevVolume = 0;
_cooldown = 0;
var ma = new SimpleMovingAverage { Length = MAPeriod };
var subscription = SubscribeCandles(CandleType);
subscription
.Bind(ma, ProcessCandle)
.Start();
var area = CreateChartArea();
if (area != null)
{
DrawCandles(area, subscription);
DrawIndicator(area, ma);
DrawOwnTrades(area);
}
}
private void ProcessCandle(ICandleMessage candle, decimal maValue)
{
if (candle.State != CandleStates.Finished)
return;
if (!IsFormedAndOnlineAndAllowTrading())
return;
if (_prevClose == 0)
{
_prevClose = candle.ClosePrice;
_prevMa = maValue;
_prevVolume = candle.TotalVolume;
return;
}
if (_cooldown > 0)
{
_cooldown--;
_prevClose = candle.ClosePrice;
_prevMa = maValue;
_prevVolume = candle.TotalVolume;
return;
}
var crossUp = _prevClose <= _prevMa && candle.ClosePrice > maValue;
var crossDown = _prevClose >= _prevMa && candle.ClosePrice < maValue;
var volumeRising = candle.TotalVolume > _prevVolume;
if (Position == 0 && crossUp && volumeRising)
{
BuyMarket();
_cooldown = CooldownBars;
}
else if (Position == 0 && crossDown && volumeRising)
{
SellMarket();
_cooldown = CooldownBars;
}
else if (Position > 0 && crossDown)
{
SellMarket();
_cooldown = CooldownBars;
}
else if (Position < 0 && crossUp)
{
BuyMarket();
_cooldown = CooldownBars;
}
_prevClose = candle.ClosePrice;
_prevMa = maValue;
_prevVolume = candle.TotalVolume;
}
}
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
from StockSharp.Algo.Strategies import Strategy
class volume_surge_strategy(Strategy):
"""
Volume Surge strategy.
Long: Price crosses above MA with volume confirmation.
Short: Price crosses below MA with volume confirmation.
Exit: Price crosses MA.
"""
def __init__(self):
super(volume_surge_strategy, self).__init__()
self._ma_period = self.Param("MAPeriod", 20).SetDisplay("MA Period", "Period for price MA", "Indicators")
self._candle_type = self.Param("CandleType", DataType.TimeFrame(TimeSpan.FromMinutes(1))).SetDisplay("Candle Type", "Type of candles to use", "General")
self._cooldown_bars = self.Param("CooldownBars", 500).SetDisplay("Cooldown Bars", "Bars to wait between trades", "General")
self._prev_close = 0.0
self._prev_ma = 0.0
self._prev_volume = 0.0
self._cooldown = 0
@property
def candle_type(self):
return self._candle_type.Value
def OnReseted(self):
super(volume_surge_strategy, self).OnReseted()
self._prev_close = 0.0
self._prev_ma = 0.0
self._prev_volume = 0.0
self._cooldown = 0
def OnStarted2(self, time):
super(volume_surge_strategy, self).OnStarted2(time)
self._prev_close = 0.0
self._prev_ma = 0.0
self._prev_volume = 0.0
self._cooldown = 0
ma = SimpleMovingAverage()
ma.Length = self._ma_period.Value
subscription = self.SubscribeCandles(self.candle_type)
subscription.Bind(ma, self._process_candle).Start()
area = self.CreateChartArea()
if area is not None:
self.DrawCandles(area, subscription)
self.DrawIndicator(area, ma)
self.DrawOwnTrades(area)
def _process_candle(self, candle, ma_val):
if candle.State != CandleStates.Finished:
return
close = float(candle.ClosePrice)
mv = float(ma_val)
vol = float(candle.TotalVolume)
if self._prev_close == 0:
self._prev_close = close
self._prev_ma = mv
self._prev_volume = vol
return
if self._cooldown > 0:
self._cooldown -= 1
self._prev_close = close
self._prev_ma = mv
self._prev_volume = vol
return
cd = self._cooldown_bars.Value
cross_up = self._prev_close <= self._prev_ma and close > mv
cross_down = self._prev_close >= self._prev_ma and close < mv
volume_rising = vol > self._prev_volume
if self.Position == 0 and cross_up and volume_rising:
self.BuyMarket()
self._cooldown = cd
elif self.Position == 0 and cross_down and volume_rising:
self.SellMarket()
self._cooldown = cd
elif self.Position > 0 and cross_down:
self.SellMarket()
self._cooldown = cd
elif self.Position < 0 and cross_up:
self.BuyMarket()
self._cooldown = cd
self._prev_close = close
self._prev_ma = mv
self._prev_volume = vol
def CreateClone(self):
return volume_surge_strategy()