//------------------------------------------------------------------------------
//
// Индикатор Histogram. Copyright (c) 2023 Tiger Trade Capital AG. All rights reserved.
//
//------------------------------------------------------------------------------
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Runtime.Serialization;
using System.Windows;
using System.Windows.Media;
using TigerTrade.Chart.Base.Enums;
using TigerTrade.Chart.Data;
using TigerTrade.Chart.Indicators.Common;
using TigerTrade.Chart.Indicators.Enums;
using TigerTrade.Core.UI.Converters;
using TigerTrade.Core.Utils.Time;
using TigerTrade.Dx;
namespace TigerTrade.Chart.Indicators.Custom
{
[DataContract(Name = "HistogramPeriodType", Namespace = "http://schemas.datacontract.org/2004/07/TigerTrade.Chart.Indicators.Custom")]
[TypeConverter(typeof(EnumDescriptionTypeConverter))]
public enum HistogramPeriodType
{
[EnumMember(Value = "Minute"), Description("Минута")]
Minute,
[EnumMember(Value = "Hour"), Description("Час")]
Hour,
[EnumMember(Value = "Day"), Description("День")]
Day,
[EnumMember(Value = "Week"), Description("Неделя")]
Week,
[EnumMember(Value = "Month"), Description("Месяц")]
Month,
[EnumMember(Value = "AllBars"), Description("Все бары")]
AllBars,
[EnumMember(Value = "CustomDate"), Description("Свой интервал")]
CustomDate
}
[DataContract(Name = "HistogramViewType", Namespace = "http://schemas.datacontract.org/2004/07/TigerTrade.Chart.Indicators.Custom")]
[TypeConverter(typeof(EnumDescriptionTypeConverter))]
public enum HistogramViewType
{
[EnumMember(Value = "Volume"), Description("Volume")]
Volume,
[EnumMember(Value = "Trades"), Description("Trades")]
Trades,
[EnumMember(Value = "Delta"), Description("Delta")]
Delta,
[EnumMember(Value = "BidAsk"), Description("Bid x Ask")]
BidAsk,
}
[DataContract(Name = "HistogramCellViewType", Namespace = "http://schemas.datacontract.org/2004/07/TigerTrade.Chart.Indicators.Custom")]
[TypeConverter(typeof(EnumDescriptionTypeConverter))]
public enum HistogramCellViewType
{
[EnumMember(Value = "Solid"), Description("Без отступов")]
Solid,
[EnumMember(Value = "Bars"), Description("С отступами")]
Bars,
[EnumMember(Value = "BorderedBars"), Description("С границей")]
BorderedBars
}
[DataContract(Name = "HistogramIndicator", Namespace = "http://schemas.datacontract.org/2004/07/TigerTrade.Chart.Indicators.Custom")]
[Indicator("Z_Histogram", "*Histogram", true, Type = typeof(HistogramIndicator))]
internal sealed class HistogramIndicator : IndicatorBase
{
private HistogramPeriodType _periodType;
[DataMember(Name = "PeriodType")]
[Category("Период"), DisplayName("Интервал")]
public HistogramPeriodType PeriodType
{
get => _periodType;
set
{
if (value == _periodType)
{
return;
}
_periodType = value;
_periodValue = _periodType == HistogramPeriodType.Minute ? 15 : 1;
Clear();
OnPropertyChanged();
OnPropertyChanged(nameof(PeriodValue));
OnPropertyChanged(nameof(PeriodRevers));
OnPropertyChanged(nameof(StartDate));
OnPropertyChanged(nameof(EndDate));
}
}
private int _periodValue;
[DataMember(Name = "PeriodValue")]
[Category("Период"), DisplayName("Значение")]
public int PeriodValue
{
get => _periodValue;
set
{
value = Math.Max(1, value);
if (value == _periodValue)
{
return;
}
_periodValue = value;
Clear();
OnPropertyChanged();
}
}
private bool _periodRevers;
[DataMember(Name = "PeriodRevers")]
[Category("Период"), DisplayName("От последнего бара")]
public bool PeriodRevers
{
get => _periodRevers;
set
{
if (value == _periodRevers)
{
return;
}
_periodRevers = value;
Clear();
OnPropertyChanged();
}
}
private DateTime? _startDate;
[DataMember(Name = "StartDate")]
[Category("Период"), DisplayName("Начальная дата")]
public DateTime? StartDate
{
get => _startDate;
set
{
if (value.Equals(_startDate))
{
return;
}
_startDate = value;
Clear();
OnPropertyChanged();
}
}
private DateTime? _endDate;
[DataMember(Name = "EndDate")]
[Category("Период"), DisplayName("Конечная дата")]
public DateTime? EndDate
{
get => _endDate;
set
{
if (value.Equals(_endDate))
{
return;
}
_endDate = value;
Clear();
OnPropertyChanged();
}
}
private HistogramViewType _histogramViewType;
[DataMember(Name = "HistogramViewType"), DefaultValue(HistogramViewType.Volume)]
[Category("Параметры"), DisplayName("Вид гистограммы")]
public HistogramViewType HistogramViewType
{
get => _histogramViewType;
set
{
if (value == _histogramViewType)
{
return;
}
_histogramViewType = value;
OnPropertyChanged();
}
}
private HistogramCellViewType _histogramCellViewType;
[DataMember(Name = "HistogramCellViewType"), DefaultValue(HistogramCellViewType.Solid)]
[Category("Параметры"), DisplayName("Вид ячейки")]
public HistogramCellViewType HistogramCellViewType
{
get => _histogramCellViewType;
set
{
if (value == _histogramCellViewType)
{
return;
}
_histogramCellViewType = value;
OnPropertyChanged();
}
}
private bool _histogramGradient;
[DataMember(Name = "HistogramGradient"), DefaultValue(true)]
[Category("Параметры"), DisplayName("Градиент")]
public bool HistogramGradient
{
get => _histogramGradient;
set
{
if (value == _histogramGradient)
{
return;
}
_histogramGradient = value;
OnPropertyChanged();
}
}
private bool _histogramShowValues;
[DataMember(Name = "HistogramShowValues"), DefaultValue(true)]
[Category("Параметры"), DisplayName("Отображать значения")]
public bool HistogramShowValues
{
get => _histogramShowValues;
set
{
if (value == _histogramShowValues)
{
return;
}
_histogramShowValues = value;
OnPropertyChanged();
}
}
private bool _histogramMinimizeValues;
[DataMember(Name = "HistogramMinimizeValues")]
[Category("Параметры"), DisplayName("Минимизировать значения")]
public bool HistogramMinimizeValues
{
get => _histogramMinimizeValues;
set
{
if (value == _histogramMinimizeValues)
{
return;
}
_histogramMinimizeValues = value;
OnPropertyChanged();
}
}
private IndicatorIntParam _histogramRoundValueParam;
[DataMember(Name = "HistogramRoundValuesParam")]
public IndicatorIntParam HistogramRoundValuesParam
{
get => _histogramRoundValueParam ?? (_histogramRoundValueParam = new IndicatorIntParam(0));
set => _histogramRoundValueParam = value;
}
[DefaultValue(0)]
[Category("Параметры"), DisplayName("Округлять значения")]
public int HistogramRoundValues
{
get => HistogramRoundValuesParam.Get(SettingsLongKey);
set
{
if (!HistogramRoundValuesParam.Set(SettingsLongKey, value, -4, 4))
{
return;
}
OnPropertyChanged();
}
}
private bool _histogramShowValueArea;
[DataMember(Name = "HistogramShowValueArea"), DefaultValue(true)]
[Category("Параметры"), DisplayName("Отображать Value Area")]
public bool HistogramShowValueArea
{
get => _histogramShowValueArea;
set
{
if (value == _histogramShowValueArea)
{
return;
}
_histogramShowValueArea = value;
OnPropertyChanged();
}
}
private bool _histogramExtendValueArea;
[DataMember(Name = "HistogramExtendValueArea"), DefaultValue(false)]
[Category("Параметры"), DisplayName("Продлить Value Area")]
public bool HistogramExtendValueArea
{
get => _histogramExtendValueArea;
set
{
if (value == _histogramExtendValueArea)
{
return;
}
_histogramExtendValueArea = value;
OnPropertyChanged();
}
}
private int _histogramValueAreaPercent;
[DataMember(Name = "HistogramValueAreaPercent")]
[Category("Параметры"), DisplayName("ValueArea %")]
public int HistogramValueAreaPercent
{
get => _histogramValueAreaPercent;
set
{
value = Math.Max(0, Math.Min(100, value));
if (value == 0)
{
value = 70;
}
if (value == _histogramValueAreaPercent)
{
return;
}
_histogramValueAreaPercent = value;
Clear();
OnPropertyChanged();
}
}
private bool _histogramShowPoc;
[DataMember(Name = "HistogramShowPoc"), DefaultValue(true)]
[Category("Параметры"), DisplayName("Отображать POC")]
public bool HistogramShowPoc
{
get => _histogramShowPoc;
set
{
if (value == _histogramShowPoc)
{
return;
}
_histogramShowPoc = value;
OnPropertyChanged();
}
}
private bool _histogramExtendPoc;
[DataMember(Name = "HistogramExtendPoc"), DefaultValue(false)]
[Category("Параметры"), DisplayName("Продлить POC")]
public bool HistogramExtendPoc
{
get => _histogramExtendPoc;
set
{
if (value == _histogramExtendPoc)
{
return;
}
_histogramExtendPoc = value;
OnPropertyChanged();
}
}
private XBrush _volumeBrush;
private XColor _volumeColor;
[DataMember(Name = "VolumeColor")]
[Category("Стиль"), DisplayName("Volume")]
public XColor VolumeColor
{
get => _volumeColor;
set
{
if (value == _volumeColor)
{
return;
}
_volumeColor = value;
_volumeBrush = new XBrush(_volumeColor);
OnPropertyChanged();
}
}
private XBrush _tradesBrush;
private XColor _tradesColor;
[DataMember(Name = "TradesColor")]
[Category("Стиль"), DisplayName("Trades")]
public XColor TradesColor
{
get => _tradesColor;
set
{
if (value == _tradesColor)
{
return;
}
_tradesColor = value;
_tradesBrush = new XBrush(_tradesColor);
OnPropertyChanged();
}
}
private XBrush _deltaPlusBrush;
private XColor _deltaPlusColor;
[DataMember(Name = "DeltaPlusColor")]
[Category("Стиль"), DisplayName("Delta+")]
public XColor DeltaPlusColor
{
get => _deltaPlusColor;
set
{
if (value == _deltaPlusColor)
{
return;
}
_deltaPlusColor = value;
_deltaPlusBrush = new XBrush(_deltaPlusColor);
OnPropertyChanged();
}
}
private XBrush _deltaMinusBrush;
private XColor _deltaMinusColor;
[DataMember(Name = "DeltaMinusColor")]
[Category("Стиль"), DisplayName("Delta-")]
public XColor DeltaMinusColor
{
get => _deltaMinusColor;
set
{
if (value == _deltaMinusColor)
{
return;
}
_deltaMinusColor = value;
_deltaMinusBrush = new XBrush(_deltaMinusColor);
OnPropertyChanged();
}
}
private XBrush _bidBrush;
private XColor _bidColor;
[DataMember(Name = "BidColor")]
[Category("Стиль"), DisplayName("Bid")]
public XColor BidColor
{
get => _bidColor;
set
{
if (value == _bidColor)
{
return;
}
_bidColor = value;
_bidBrush = new XBrush(_bidColor);
OnPropertyChanged();
}
}
private XBrush _askBrush;
private XColor _askColor;
[DataMember(Name = "AskColor")]
[Category("Стиль"), DisplayName("Ask")]
public XColor AskColor
{
get => _askColor;
set
{
if (value == _askColor)
{
return;
}
_askColor = value;
_askBrush = new XBrush(_askColor);
OnPropertyChanged();
}
}
private XBrush _valueAreaBrush;
private XColor _valueAreaColor;
[DataMember(Name = "ValueAreaColor")]
[Category("Стиль"), DisplayName("Value Area")]
public XColor ValueAreaColor
{
get => _valueAreaColor;
set
{
if (value == _valueAreaColor)
{
return;
}
_valueAreaColor = value;
_valueAreaBrush = new XBrush(_valueAreaColor);
OnPropertyChanged();
}
}
private XBrush _maximumBrush;
private XColor _maximumColor;
[DataMember(Name = "MaximumColor")]
[Category("Стиль"), DisplayName("POC")]
public XColor MaximumColor
{
get => _maximumColor;
set
{
if (value == _maximumColor)
{
return;
}
_maximumColor = value;
_maximumBrush = new XBrush(_maximumColor);
OnPropertyChanged();
}
}
private XBrush _cellBorderBrush;
private XPen _cellBorderPen;
private XColor _cellBorderColor;
[DataMember(Name = "CellBorderColor")]
[Category("Стиль"), DisplayName("Граница ячейки")]
public XColor CellBorderColor
{
get => _cellBorderColor;
set
{
if (value == _cellBorderColor)
{
return;
}
_cellBorderColor = value;
_cellBorderBrush = new XBrush(_cellBorderColor);
_cellBorderPen = new XPen(_cellBorderBrush, 1);
OnPropertyChanged();
}
}
private XBrush _textBrush;
private XColor _textColor;
[DataMember(Name = "TextColor")]
[Category("Стиль"), DisplayName("Текст")]
public XColor TextColor
{
get => _textColor;
set
{
if (value == _textColor)
{
return;
}
_textColor = value;
_textBrush = new XBrush(_textColor);
OnPropertyChanged();
}
}
[Browsable(false)]
public override bool ShowIndicatorValues => false;
[Browsable(false)]
public override IndicatorCalculation Calculation => IndicatorCalculation.OnEachTick;
private List<IndicatorLabelInfo> _labels;
private long _lastSequense;
private RawCluster _histogram;
public HistogramIndicator()
{
ShowIndicatorTitle = false;
PeriodType = HistogramPeriodType.Day;
PeriodValue = 1;
HistogramViewType = HistogramViewType.Volume;
HistogramCellViewType = HistogramCellViewType.Solid;
HistogramGradient = true;
HistogramShowValues = true;
HistogramMinimizeValues = false;
HistogramShowValueArea = true;
HistogramValueAreaPercent = 70;
HistogramExtendValueArea = false;
HistogramShowPoc = true;
HistogramExtendPoc = false;
VolumeColor = Color.FromArgb(255, 70, 130, 180);
TradesColor = Color.FromArgb(255, 70, 130, 180);
DeltaPlusColor = Color.FromArgb(255, 46, 139, 87);
DeltaMinusColor = Color.FromArgb(255, 178, 34, 34);
BidColor = Color.FromArgb(255, 178, 34, 34);
AskColor = Color.FromArgb(255, 70, 130, 180);
ValueAreaColor = Color.FromArgb(255, 128, 128, 128);
MaximumColor = Color.FromArgb(255, 178, 34, 34);
CellBorderColor = Color.FromArgb(255, 127, 127, 127);
TextColor = Color.FromArgb(255, 255, 255, 255);
}
protected override void Execute()
{
if (_labels == null)
{
_labels = new List<IndicatorLabelInfo>();
}
if (_histogram == null)
{
_histogram = new RawCluster(DateTime.MinValue);
}
if (ClearData)
{
Clear();
}
if (PeriodType == HistogramPeriodType.CustomDate)
{
if (_lastSequense == -1)
{
_histogram = new RawCluster(DateTime.MinValue);
for (var i = 0; i < DataProvider.Count; i++)
{
var cluster = DataProvider.GetRawCluster(i);
if (cluster.Time < StartDate || cluster.Time > EndDate)
{
continue;
}
_histogram.AddCluster(cluster);
}
_lastSequense = 1;
}
else
{
var ticks = DataProvider.GetRawTicks();
foreach (var tick in ticks)
{
_histogram.AddTick(tick, DataProvider.Scale);
}
}
_histogram.UpdateData();
return;
}
if (PeriodRevers && PeriodType != HistogramPeriodType.AllBars)
{
if (_lastSequense != DataProvider.Count)
{
var interval = GetTimeSeconds();
var lastBar = DataProvider.GetRawCluster(DataProvider.Count - 1);
var startTime = lastBar?.OpenTime ?? TimeHelper.GetCurrTime(DataProvider.Symbol.Exchange);
startTime = startTime.AddSeconds(-interval);
_histogram = new RawCluster(DateTime.MinValue);
for (var i = 0; i < DataProvider.Count; i++)
{
var cluster = DataProvider.GetRawCluster(i);
if (cluster.Time < startTime)
{
continue;
}
_histogram.AddCluster(cluster);
}
_lastSequense = DataProvider.Count;
}
else
{
var ticks = DataProvider.GetRawTicks();
foreach (var tick in ticks)
{
_histogram.AddTick(tick, DataProvider.Scale);
}
}
_histogram.UpdateData();
return;
}
var timeOffset = TimeHelper.GetSessionOffset(DataProvider.Symbol.Exchange);
var lastCluster = DataProvider.GetRawCluster(DataProvider.Count - 1);
if (lastCluster == null)
{
return;
}
var currSequence = GetSequence(lastCluster.Time, timeOffset);
if (_lastSequense != currSequence)
{
_histogram = new RawCluster(DateTime.MinValue);
_lastSequense = currSequence;
for (var i = DataProvider.Count - 1; i >= 0; i--)
{
var cluster = DataProvider.GetRawCluster(i);
var newSequence = GetSequence(cluster.Time, timeOffset);
if (_lastSequense != newSequence)
{
break;
}
_histogram.AddCluster(cluster);
}
}
else
{
var ticks = DataProvider.GetRawTicks();
foreach (var tick in ticks)
{
_histogram.AddTick(tick, DataProvider.Scale);
}
}
_histogram.UpdateData();
}
private int GetSequence(DateTime date, double offset)
{
var cycleBase = ChartPeriodType.Hour;
switch (PeriodType)
{
case HistogramPeriodType.Minute:
cycleBase = ChartPeriodType.Minute;
break;
case HistogramPeriodType.Hour:
cycleBase = ChartPeriodType.Hour;
break;
case HistogramPeriodType.Day:
cycleBase = ChartPeriodType.Day;
break;
case HistogramPeriodType.Week:
cycleBase = ChartPeriodType.Week;
break;
case HistogramPeriodType.Month:
cycleBase = ChartPeriodType.Month;
break;
case HistogramPeriodType.AllBars:
return 0;
}
return DataProvider.Period.GetSequence(cycleBase, PeriodValue, date, offset);
}
public int GetTimeSeconds()
{
switch (PeriodType)
{
case HistogramPeriodType.Minute:
return PeriodValue * 60;
case HistogramPeriodType.Hour:
return PeriodValue * 60 * 60;
case HistogramPeriodType.Day:
return PeriodValue * 60 * 60 * 24;
case HistogramPeriodType.Week:
return PeriodValue * 60 * 60 * 24 * 7;
case HistogramPeriodType.Month:
return PeriodValue * 60 * 60 * 24 * 30;
default:
return 0;
}
}
private void Clear()
{
_lastSequense = -1;
}
public override void Render(DxVisualQueue visual)
{
_labels.Clear();
if (_histogram == null)
{
return;
}
var dp = DataProvider;
var step = dp.Step;
var symbol = dp.Symbol;
var rect = Canvas.Rect;
var width = rect.Width * 0.2;
var heightCorrect = HistogramCellViewType == HistogramCellViewType.Bars ||
HistogramCellViewType == HistogramCellViewType.BorderedBars
? -1
: 0;
var height = Math.Max(GetY(0.0) - GetY(dp.Step), 1);
var fontSize = Math.Min(height + heightCorrect - 2, 18) * 96.0 / 72.0;
fontSize = Math.Min(fontSize, Canvas.ChartFont.Size);
var textFont = new XFont(Canvas.ChartFont.Name, fontSize);
var minPrice = (long)(Canvas.MinY / step) - 1;
var maxPrice = (long)(Canvas.MaxY / step) + 1;
if (maxPrice - minPrice > 100000)
{
return;
}
var valueArea = HistogramShowValueArea ? _histogram.GetValueArea(HistogramValueAreaPercent) : null;
maxPrice = Math.Min(_histogram.High, maxPrice + 1);
minPrice = Math.Max(_histogram.Low, minPrice - 1);
var prevY = (int)GetY(maxPrice * step + step / 2.0);
var roundValues = HistogramRoundValues;
var maxValues = _histogram.MaxValues;
for (var price = maxPrice; price >= minPrice; price--)
{
var currY = (int)GetY(price * step - step / 2.0);
var currHeight = Math.Max((currY - prevY) + heightCorrect, 1);
var y = prevY;
prevY = currY;
var item = _histogram.GetItem(price);
if (item == null)
{
continue;
}
var barWidth = 0.0;
var valueText = "";
switch (HistogramViewType)
{
case HistogramViewType.Volume:
{
barWidth = width / maxValues.MaxVolume * item.Volume;
var fillBrush = HistogramGradient
? new XBrush(VolumeColor.ChangeOpacity(item.Volume, maxValues.MaxVolume,
Canvas.Theme.ChartBackColor))
: _volumeBrush;
var fillRect = new Rect(rect.Left, y, barWidth, currHeight);
visual.FillRectangle(fillBrush, fillRect);
valueText = symbol.FormatRawSize(item.Volume, roundValues, HistogramMinimizeValues);
break;
}
case HistogramViewType.Trades:
{
barWidth = width / maxValues.MaxTrades * item.Trades;
var fillBrush = HistogramGradient
? new XBrush(TradesColor.ChangeOpacity(item.Trades, maxValues.MaxTrades,
Canvas.Theme.ChartBackColor))
: _tradesBrush;
var fillRect = new Rect(rect.Left, y, barWidth, currHeight);
visual.FillRectangle(fillBrush, fillRect);
valueText = symbol.FormatTrades(item.Trades, roundValues, HistogramMinimizeValues);
break;
}
case HistogramViewType.Delta:
{
var maxDelta = Math.Max(maxValues.MaxDelta, -maxValues.MinDelta);
barWidth = width / maxDelta * Math.Abs(item.Delta);
var fillBrush = item.Delta > 0
? (HistogramGradient
? new XBrush(DeltaPlusColor.ChangeOpacity(item.Delta, maxDelta,
Canvas.Theme.ChartBackColor))
: _deltaPlusBrush)
: (HistogramGradient
? new XBrush(DeltaMinusColor.ChangeOpacity(-item.Delta, maxDelta,
Canvas.Theme.ChartBackColor))
: _deltaMinusBrush);
var fillRect = new Rect(rect.Left, y, barWidth, currHeight);
visual.FillRectangle(fillBrush, fillRect);
valueText = symbol.FormatRawSize(item.Delta, roundValues, HistogramMinimizeValues);
break;
}
case HistogramViewType.BidAsk:
{
barWidth = width / maxValues.MaxVolume * item.Volume;
var redWidth = barWidth / item.Volume * item.Bid;
var bidBrush = HistogramGradient
? new XBrush(BidColor.ChangeOpacity(item.Bid,
Math.Max(maxValues.MaxBid, maxValues.MaxAsk), Canvas.Theme.ChartBackColor))
: _bidBrush;
var bidRect = new Rect(rect.Left, y, redWidth, currHeight);
var askBrush = HistogramGradient
? new XBrush(AskColor.ChangeOpacity(item.Ask,
Math.Max(maxValues.MaxBid, maxValues.MaxAsk), Canvas.Theme.ChartBackColor))
: _askBrush;
var askRect = new Rect(rect.Left + redWidth, y, barWidth - redWidth, currHeight);
visual.FillRectangle(bidBrush, bidRect);
visual.FillRectangle(askBrush, askRect);
valueText = symbol.FormatRawSize(item.Volume, roundValues, HistogramMinimizeValues);
break;
}
}
if (HistogramCellViewType == HistogramCellViewType.BorderedBars &&
height + heightCorrect > 3)
{
visual.DrawRectangle(_cellBorderPen,
new Rect(rect.Left - 1, y, barWidth, currHeight - 1));
}
if (price == maxValues.Poc && HistogramShowPoc)
{
var pocRect = new Rect(rect.Left, y, width, currHeight);
if (HistogramExtendPoc)
{
visual.DrawLine(new XPen(_maximumBrush, 3),
new Point(rect.Left, y + currHeight / 2),
new Point(rect.Right, y + currHeight / 2));