Создание простого привода на S#

Создание простого привода на S#
Atom
5/25/2012
M.Kovaleva


Бесплатная библиотека StockSharp предоставляет широкие возможности при разработке торговых роботов различной сложности. Но самое трудное для многих пользователей - сделать первые шаги при разработке своего робота. Поэтому представляем вашему вниманию статью-инструкцию по созданию простого привода для терминала QUIK с использованием S#.API.

Для создания простого привода нам понадобится:
1) Quik;
2) Библиотека S#;
3) Visual Studio Express;
4) Немного навыков программирования.

Подготовка рабочего места



После установки Quik, его нужно настроить для работы со S#. Для этого нужно проделать 3 базовых шага:

1) Включить обработку внешних транзакций. "Торговля" → "Внешние Транзакции" → "Начать обработку" и поставить галочку "Запускать процесс обработки внешних транзацкий автоматически"

2) Далее нужно открыть соответствующий Wnd-файл с настройками: "Настройки" → "Загрузить настройки из файла" → выбрать wnd-файл по пути "Разахивированная папка"\Samples\Quik\info.wnd

3) Далее настраиваем счета : "Торговля" → "Настройка счетов" → добавить все счета в выбранные счета депо.

4) В итоге можно настроить окошки - подтянуть их, уменьшить, но названия столбцов менять нельзя, так же как и удалять их. У разных брокеров (квиков) есть особенности , стоят определённые фильтры на таблицах "Мои сделки", "Мои заявки". Нажав правой кнопкой мышки на таблицу "Мои сделки", выделить все. Аналогично и в таблице “Мои заявки”.

Добавьте торгующийся на данный момент инструмент в таблицу "Инструменты"; по нему в дальнейшем мы будем отправлять заявки.

Создание торгового привода

Открываем Visual Studio.
Создаём новый проект → "Приложение Wpf".
Настраиваем наш проект на работу S#:
  • Правой кнопкой мыши на "Проект" → "Свойства" и меняем "Требуемая версия NetFrameWork" (Net FrameWork 4 Client Profile ) на просто Net FrameWork 4(после этого меняется целевая среда разарботки).
  • Добавляем нужные библиотеки в наш проект "Ссылки" → "Добавить ссылку" → "Обзор".
  • Выбираем нужные нам библиотеки, которые лежат в "Разахрихированая папка"\References:
    1)StockSharp.Quik
    2)StockSharp.BusinessEntites
    3)StockSharp.Algo
    4)Ecng.Common
    5)Ecng.ComponentModel
    6)Ecng.Xaml
  • Прописываем соответсвующие using на наши библиотеки в классе MainWindow.xaml.cs

Code
using System.Windows;
using StockSharp.Quik;
using Ecng.Xaml;
using StockSharp.BusinessEntities;

Заходим в MainWindow.Xaml, создаём соответсвующие кнопки и другие элементы на форме нашего приложения, перетягивая их из панели элементов. Получаем следующую форму:

Даем каждому элементу на нашей форме название, чтобы потом из кода обращаться к нему.
Нажимаем два раза на кнопку, чтобы создать метод, который будет вызываться по нажатию на эту кнопку (метод обработчик).

Прописываем код под кнопку подключения.
Это простой код демонстрирующий подключение и выгрузку данных из Quik в асинхронном режиме. Теперь осталось дополнить код выводом информации на главную форму, то есть сделать вывод в наши выпадающие списки "Портфели" и "Инструменты".
Code
//обьявляем переменную
        private QuikTrader _trader;
        /// <summary>
        /// Подключение к квику
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="e"></param>
        private void Connect_Click(object sender, RoutedEventArgs e)
        {
            //создаём квик трейдера, передавая ему сразу место расположения нашего терминала
            _trader= new QuikTrader(QuikTerminal.GetDefaultPath());
            //подписываемся на событие подключения, как только подлючимся, сразу запустим Экспорт
            //Connect- просто подключение к потоку
            //StartExport- получение он-лайн данных из квика Инструменты,Заявки , Портфели и так далее
            _trader.Connected += () => _trader.StartExport();
            // подключаем квик 
            _trader.Connect();

        }

Дописываем графический вывод на нашу форму в выпадающие списки:
Code
private void Connect_Click(object sender, RoutedEventArgs e)
        {
            //создаём квик трейдера, передавая ему сразу место расположения нашего терминала
            _trader= new QuikTrader(QuikTerminal.GetDefaultPath());
            //подписываемся на событие появление новых инструментов
            //оно сработает когда включится экспорт 
            _trader.NewSecurities += securities =>this.GuiAsync(()=>//GuiASync используется чтобы выводить графику из другого потока
                                                                    {
                                                                        //заполняем коллекцию у нашего выпадающего списка (ComboBox)
                                                                        Securitites.ItemsSource = _trader.Securities;
                                                                    });
            //подписываемся на событие появления новых портфелей
            //сработает после запуска экспорта
            _trader.NewPortfolios += portfolios =>this.GuiAsync(()=>
                                                                    {
                                                                        //заполняем коллекцию у нашего выпадающего списка (ComboBox)
                                                                        Portfolios.ItemsSource = _trader.Portfolios;
                                                                    });
            
            //подписываемся на событие подключения, как только подлючимся, сразу запустим Экспорт
            //Connect - просто подключение к потоку
            //StartExport- получение онлайн данных из квика Инструменты, Заявки, Портфели и так далее
            _trader.Connected += () => _trader.StartExport();
            // подключаем квик 
            _trader.Connect();

        }

Запускаем проект (F5), нажимаем на кнопку "Подключиться" (в это время у нас уже работает Quik, который подключен к котировкам). После этого наш робот включает эскпорт DDE в Quik и через некоторое время мы получаем заполненные выпадающие списки в нашем проекте.

Остается дописать функционал под две оставшиеся кнопки "Купить", "Снять все заявки":
  • "Купить" - отправляем заявку (портфель и инструмент берется из выпадающих списков) по той цене, которая будет указана в нашем текстовом окне;
  • "Снять все заявки" - снимаем все выставленные нами заявки.

Код для отправки заявки:
Code
private void Buy_Click(object sender, RoutedEventArgs e)
        {
            //создаём ордер
            //заполняем его нужными свойствами
            // портфель и инструмент вынимаем из выпадающих списков
            //цену для отправки заявки вынимаем из тесктового окна
            var order = new Order
                            {
                                Trader = _trader,
                                Portfolio = (Portfolio)Portfolios.SelectedItem,
                                Security = (Security) Securitites.SelectedItem,
                                Volume = 1,
                                Price = decimal.Parse(PRICE.Text),
                                Direction = OrderDirections.Buy
                            };
            //регистрируем ордер
            _trader.RegisterOrder(order);
        }

Код для снятия всех заявок:
Code
private void CancelOrders_Click(object sender, RoutedEventArgs e)
        {
            //отменить все заявки
            _trader.CancelOrders();
        }

Пример работы простого привода:

Скачать исходники также можно здесь.

Видео-урок по созданию простого торгового робота с использованием библиотеки StockSharp:



Автор статьи: Самунджян Артём


<< < 2 3 4 5 6  >
Smelov

Avatar
Date: 6/12/2012
Reply


Кот Матроскин

Это используется внутри стратегии, и данные о портфеле, инструменте и шлюзе берутся из нее же.
Если класс стратегий не используете, то попробуйте следующее (сам ни разу не использовал, но может сработать):

var _marketQuotingStrategy = new MarketQuotingStrategy(_order, new Unit(), new Unit());
_marketQuotingStrategy.Start();

где в _order указываете портфель, инструмент и шлюз. Цену не указываете

А как указывать шлюз? В коде ниже как раз ругается на отсутствие шлюза.
Code

            if (_trader != null)
            {
                var order = new Order
                {
                    Trader = _trader,
                    Portfolio = (Portfolio)Portfolios.SelectedItem,
                    Security = (Security)Securities.SelectedItem,
                    Volume = decimal.Parse(Amount.Text),
                    //Price = ((Security)Securities.SelectedItem).GetCurrentPrice(OrderDirections.Buy).Value,
                    Direction = OrderDirections.Buy
                };
                var _marketQuotingStrategy = new MarketQuotingStrategy(order, new Unit(), new Unit());
                _marketQuotingStrategy.Start();
                //регистрируем ордера
                //_trader.RegisterOrder(order);
            }
Thanks:

esper

Avatar
Date: 6/12/2012
Reply


Smelov
Создал кнопку для выставления заявки по рыночной цене
Code

            if (_trader != null)
            {
                var order = new Order
                {
                    Trader = _trader,
                    Portfolio = (Portfolio)Portfolios.SelectedItem,
                    Security = (Security)Securities.SelectedItem,
                    Volume = decimal.Parse(Amount.Text),
                    Price = ((Security)Securities.SelectedItem).GetMarketPrice(OrderDirections.Buy),
                    Direction = OrderDirections.Buy
                };
                //регистрируем орде
                _trader.RegisterOrder(order);
            }

GetMarketPrice всегда возвращает 0 и система не может регистрировать заявки


Если торгуете акциями, то 0 и есть рыночная цена. Просто надо выставить тип заявки в Market, а не Limit.
Thanks: Smelov

Кот Матроскин

Avatar
Date: 6/12/2012
Reply


Smelov
У коде ниже как раз ругается на отсутствие шлюза.

Да вроде правильно указали шлюз...
Thanks:

esper

Avatar
Date: 6/12/2012
Reply


Кот Матроскин
Smelov
У коде ниже как раз ругается на отсутствие шлюза.

Да вроде правильно указали шлюз...

Надо не
Code
 _marketQuotingStrategy.Start();
а
Code
ChildStrategies.Add(marketQuotingStrategy);


Хотя да, Кот Матроскин правильно сказал, первый вариант в случае отсутствия стратегии. Но тогда, для marketQuotingStrategy тоже надо шлюз устанавливать.
Thanks: Smelov

Smelov

Avatar
Date: 6/12/2012
Reply


esper

Если торгуете акциями, то 0 и есть рыночная цена. Просто надо выставить тип заявки в Market, а не Limit.


Все, наконец-то получилось. Всем спасибо! покопался в справке и просто добавил Type = OrderTypes.Market в order. Скажите пожалуйста, а где можно почитать про то, как создавать заявки других типов (например conditional)?
Thanks:

Кот Матроскин

Avatar
Date: 6/12/2012
Reply


Smelov
esper

Если торгуете акциями, то 0 и есть рыночная цена. Просто надо выставить тип заявки в Market, а не Limit.


Все, наконец-то получилось. Всем спасибо! покопался в справке и просто добавил Type = OrderTypes.Market в order. Скажите пожалуйста, а где можно почитать про то, как создавать заявки других типов (например conditional)?

http://stocksharp.com/do...79-8f48-dd8f806c39e7.htm
Thanks: Smelov

Smelov

Avatar
Date: 6/12/2012
Reply


Возвращаясь к вопросу о windows forms.

Создал WinFroms проект, прицепил библиотеки: все нормально коннектится/дисконнектится.
Но вот с получением портфелей и списка активов возникли проблемы:
Code

            _trader.NewSecurities += securities => this.GuiAsync(() =>//GuiASync используется чтобы выводить графику из другого потока
            {
                //заполняем коллекцию у нашего выпадающего списка (ComboBox)
                Securities.ItemsSource = _trader.Securities;
            });
            //подписываемся на событие появления новых портфелей
            //сработает после запуска экспорта
            _trader.NewPortfolios += portfolios => this.GuiAsync(() =>
            {
                //заполняем коллекцию у нашего выпадающего списка (ComboBox)
                Portfolios.ItemsSource = _trader.Portfolios;
            });

ругается на GuiAsync. Пробовал написать "в лоб":
Code

            //создаём квик трейдера , передавая ему сразу место расположения нашего терминала
            _trader = new QuikTrader(QuikTerminal.GetDefaultPath());
            //StartExport- получение он-лайн данных из квика Инструменты,Заявки , Портфели и так далее
            _trader.Connected += () => _trader.StartExport();
            // подключаем квик 
            _trader.Connect();

            foreach (var portfolio in _trader.Portfolios)
                Portfolios.Items.Add(portfolio.Name.ToString());

получаю пустой список. Можно ли вообще в WinForms проектах реализовать подобное?
Thanks:

esper

Avatar
Date: 6/12/2012
Reply


Например тут описано решение. Пользуйтесь поиском по форуму, многие вопросы сразу отпадут.
Thanks: Smelov

Кот Матроскин

Avatar
Date: 6/12/2012
Reply


Совет про изучения Форума остается в силе)))
http://stocksharp.com/posts/m/15374/
Thanks: Smelov

Smelov

Avatar
Date: 6/12/2012
Reply


Еще один вопрос. Вряд ли на него есть ответ на форуме:) Я создаю datagridview, в котором создаю столбец с comboox'ами. Пытаюсь передать в combobox данные об активах (чтобы можно было выбрать из выпадающего списка)

Code

                DataGridViewComboBoxColumn dgvColumn = new DataGridViewComboBoxColumn();
                dgvColumn.Name = "A1";
                dgvColumn.HeaderText = "A5";
                dgvColumn.DefaultCellStyle = dataGridView1.DefaultCellStyle;
                dgvColumn.DataSource = _trader.Securities;
                dataGridView1.Columns.Add(dgvColumn);
                dataGridView1.Rows.Add();


Приложение запускается и даже в Combobox вижу список всех активов. Но когда пытаюсь что-то выбрать из этого списка, то datagridview выдает ошибку: недопустимое значение ячейки. В списке находятся значения типа SBER@QJSIM
Thanks:
<< < 2 3 4 5 6  >

Attach files by dragging & dropping, , or pasting from the clipboard.

loading
clippy