Table of Contents

Candle Chart

Chart is a graphical component that allows building stock charts: candles, indicators, and displaying order and trade markers on charts.

Below is an example of building a chart using the Chart component. The example is based on Samples/Common/SampleConnection, with some modifications.

Gui ChartSample

Example of building a chart using Chart

  1. In XAML, we create a window and add the Chart graphical component to it. We assign the name Chart to the component. Note that when creating the window, you need to add the namespace http://schemas.stocksharp.com/xaml.

    <Window x:Class="SampleCandles.ChartWindow"
            xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
            xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
            xmlns:charting="http://schemas.stocksharp.com/xaml"
            Title="ChartWindow" Height="300" Width="300">
       <charting:Chart x:Name="Chart" x:FieldModifier="public" />
    </Window>
    
  2. In the main window code, we declare variables for chart areas, chart elements, indicators, and subscriptions.

    private readonly Dictionary<Subscription, ChartWindow> _chartWindows = new Dictionary<Subscription, ChartWindow>();
    private readonly Connector _connector = new Connector();
    private readonly LogManager _logManager;
    private ChartArea _candlesArea;
    private ChartArea _indicatorsArea;
    private ChartIndicatorElement _smaChartElement;
    private ChartIndicatorElement _macdChartElement;
    private ChartCandleElement _candlesElem;
    private SimpleMovingAverage _sma;
    private MovingAverageConvergenceDivergence _macd;
    
  3. In the Connect button's Click event handler, along with subscribing to connector events and calling the IConnector.Connect method, we subscribe to the Connector.CandleReceived event. In this event handler, the chart will be drawn when a new candle is received.

    private void ConnectClick(object sender, RoutedEventArgs e)
    {
        _connector.CandleReceived += OnCandleReceived;
    
        // Subscribe to other necessary events
        _connector.Connected += () => this.GuiAsync(() => { /* Handle connection */ });
        _connector.Disconnected += () => this.GuiAsync(() => { /* Handle disconnection */ });
    
        // Connect to the trading system
        _connector.Connect();
    }
    
  4. In the ShowChart button handler, we create indicator objects, areas, and chart elements. We add elements to areas, and areas to the chart. We open the chart window and start a subscription to candles.

    private void ShowChartClick(object sender, RoutedEventArgs e)
    {
        var security = SelectedSecurity;
    
        // Create a subscription to candles
        var subscription = new Subscription(
            DataType.TimeFrame(TimeSpan.FromMinutes(5)),
            security)
        {
            MarketData = 
            {
                // Request historical data for 30 days
                From = DateTime.Today.Subtract(TimeSpan.FromDays(30)),
                To = DateTime.Now,
                // Get only finished candles
                IsFinishedOnly = true
            }
        };
    
        // Create a chart window
        _chartWindows.SafeAdd(subscription, key =>
        {
            var wnd = new ChartWindow
            {
                Title = $"{security.Code} {TimeSpan.FromMinutes(5)}"
            };
            wnd.MakeHideable();
    
            // Initialize indicators
            _sma = new SimpleMovingAverage() { Length = 11 };
            _macd = new MovingAverageConvergenceDivergence();
    
            // Initialize chart elements
            _smaChartElement = new ChartIndicatorElement();
            _macdChartElement = new ChartIndicatorElement();
            _candlesElem = new ChartCandleElement();
    
            // Set MACD display style as histogram
            _macdChartElement.DrawStyle = DrawStyles.Histogram;
    
            // Initialize chart areas
            _candlesArea = new ChartArea();
            _indicatorsArea = new ChartArea();
    
            // Add areas to the chart
            wnd.Chart.Areas.Add(_candlesArea);
            wnd.Chart.Areas.Add(_indicatorsArea);
    
            // Add elements to areas
            _candlesArea.Elements.Add(_candlesElem);
            _candlesArea.Elements.Add(_smaChartElement);
            _indicatorsArea.Elements.Add(_macdChartElement);
    
            // Bind chart elements to subscription for automatic drawing
            wnd.Chart.AddElement(_candlesArea, _candlesElem, subscription);
            wnd.Chart.AddElement(_candlesArea, _smaChartElement, subscription);
            wnd.Chart.AddElement(_indicatorsArea, _macdChartElement, subscription);
    
            return wnd;
        }).Show();
    
        // Start subscription to candles
        _connector.Subscribe(subscription);
    }
    
  5. In the Connector.CandleReceived event handler, we draw the candle and indicator values for each finished candle.

    private void OnCandleReceived(Subscription subscription, ICandleMessage candle)
    {
        var wnd = _chartWindows.TryGetValue(subscription);
        if (wnd == null)
            return;
    
        // Process only finished candles
        if (candle.State != CandleStates.Finished)
            return;
    
        // Calculate indicator values
        var smaValue = _sma.Process(candle);
        var macdValue = _macd.Process(candle);
    
        // Create data for drawing
        var data = new ChartDrawData();
        data
            .Group(candle.OpenTime)
                .Add(_candlesElem, candle)
                .Add(_smaChartElement, smaValue)
                .Add(_macdChartElement, macdValue);
    
        // Draw data on the chart in the user interface thread
        this.GuiAsync(() => wnd.Chart.Draw(data));
    }
    

Example with automatic chart drawing

Starting with the latest versions of StockSharp, it is possible to set up automatic chart drawing without the need to explicitly call the Draw method. For this, when configuring Chart, the AddElement method is used, which links the chart element with a subscription:

private void SetupAutoDrawingChart()
{
    var security = SelectedSecurity;
    
    // Create chart elements
    var candleElement = new ChartCandleElement();
    var smaElement = new ChartIndicatorElement { Title = "SMA" };
    
    // Create chart areas
    var area = new ChartArea();
    
    // Add area to the chart
    Chart.Areas.Add(area);
    
    // Create a subscription to candles
    var subscription = new Subscription(
        DataType.TimeFrame(TimeSpan.FromMinutes(5)),
        security)
    {
        MarketData = 
        {
            From = DateTime.Today.Subtract(TimeSpan.FromDays(30)),
            To = DateTime.Now
        }
    };
    
    // Bind elements to the chart area and subscription
    Chart.AddElement(area, candleElement, subscription);
    Chart.AddElement(area, smaElement, subscription);
    
    // Create an indicator
    var sma = new SimpleMovingAverage { Length = 14 };
    
    // Subscribe to the candle receiving event for indicator processing
    _connector.CandleReceived += (sub, candle) => 
    {
        if (sub == subscription && candle.State == CandleStates.Finished)
        {
            // Process the candle with the indicator and get the value
            var smaValue = sma.Process(candle);
            
            // Draw the indicator value
            var data = new ChartDrawData();
            data
                .Group(candle.OpenTime)
                    .Add(smaElement, smaValue);
            
            this.GuiAsync(() => Chart.Draw(data));
        }
    };
    
    // Start subscription
    _connector.Subscribe(subscription);
}

Displaying orders and trades on the chart

You can display order and trade markers directly on the chart:

// Create elements for displaying orders and trades
var orderElement = new ChartOrderElement();
var tradeElement = new ChartTradeElement();

// Add elements to the chart area
_candlesArea.Elements.Add(orderElement);
_candlesArea.Elements.Add(tradeElement);

// Subscribe to order and trade receiving events
_connector.OrderReceived += (sub, order) => 
{
    if (order.Security != _security)
        return;
    
    // Draw the order on the chart
    var data = new ChartDrawData();
    data.Group(order.Time).Add(orderElement, order);
    
    this.GuiAsync(() => Chart.Draw(data));
};

_connector.OwnTradeReceived += (sub, trade) => 
{
    if (trade.Order.Security != _security)
        return;
    
    // Draw the trade on the chart
    var data = new ChartDrawData();
    data.Group(trade.Time).Add(tradeElement, trade);
    
    this.GuiAsync(() => Chart.Draw(data));
};

Clearing the chart

To clear the chart, you can use the Reset method:

// Clear the entire chart
Chart.Reset();

// Clear a specific area
_candlesArea.Reset();

// Clear a specific element
_candlesElem.Reset();