Библиотеки: Report - страница 5

 
fxsaber:

Это хорошо, что не загружается, т.к. в архиве лежат файлы из MT5, что требуются для запуска на MT4.

И как это может навредить, даже если развернуть архив в MQL5? Пусть была бы кроссплатформенная библиотека (какой она и есть).

 
Andrey Khatimlianskii:

И как это может навредить, даже если развернуть архив в MQL5? Пусть была бы кроссплатформенная библиотека (какой она и есть).

Автоматическая КБ-проверка не пустит. Там же MT4Orders используется для пятерки.

Сама библиотека в одном только файле - Report.mqh.

Не совсем понимаю, для чего может быть нужно именно штатное расположение скрипта CustomReport в КБ. Если есть какой-то интересный сценарий использования, дайте знать.

 
fxsaber:

Автоматическая КБ-проверка не пустит. Там же MT4Orders используется для пятерки.

Сама библиотека в одном только файле - Report.mqh.

Не совсем понимаю, для чего может быть нужно именно штатное расположение скрипта CustomReport в КБ. Если есть какой-то интересный сценарий использования, дайте знать.

Понял. Давно не публиковал коды, не сталкивался с проверкой.

Сценарий простой — установить библиотеку в терминал одним движением.

 
Andrey Khatimlianskii:

Понял. Давно не публиковал коды, не сталкивался с проверкой.

Сценарий простой — установить библиотеку в терминал одним движением.

Только CustomReport - не библиотека, а скрипт на ее основе. Думаю, если он кому-то и интересен, то только на MT4 и на определенных брокерах. А MT5-КБ не поставится в MT4. Поэтому zip. Его и передать легко.

 

Стал некоторой неожиданностью один сценарий применения библиотеки. Т.к. он совсем не предусматривался и родился по ходу пьесы.


Когда ставишь на чарт советник, сразу открывается отчет бэктеста до текущих свежих данных. Это позволяет сразу увидеть (взгляд на графики многое определяет за секунду), что и как советник с заданными входными параметрами "наторговал". И в случае какой-то случайной ошибки во входных параметрах, быстро определить это.


У меня это самое востребованное применение Report.mqh. Как оказалось, получать свежий бэктест при запуске советника - это очень удобно!

Саму библиотеку не рекомендую, но данный сценарий запуска советника - да. Это совершенно другое, чем просто запустить советник на чарте.
 
Кроссплатформенный альтернативный график баланса

Forum on trading, automated trading systems and testing trading strategies

Scripts: Balance Graph HTML

fxsaber, 2019.04.08 09:16

For MT4/5.
// MQL4&5-code

#property strict

// Copy "Graph.txt" to "Files\Graph.txt".
// https://www.mql5.com/en/code/25199
// #resource "\\Scripts\\Graph_HTML\\Resource\\Graph.txt" as string htm_file

#import "shell32.dll"
  int ShellExecuteW( int hwnd, string lpOperation, string lpFile, string lpParameters, string lpDirectory, int nShowCmd );
#import

#ifdef __MQL5__
  #include <MT4Orders.mqh> // https://www.mql5.com/en/code/16006

  #define BASEPATH (TerminalInfoString(TERMINAL_PATH) + "\\MQL5\\Files\\")
#else // __MQL5__
  #define BASEPATH (TerminalInfoString(TERMINAL_PATH) + "\\MQL4\\Files\\")
#endif // __MQL5__

bool CreateBalanceData( const string FileName = "exdat.txt" )
{
  const int handle = FileOpen(FileName, FILE_WRITE | FILE_TXT | FILE_ANSI);
  const bool Res = (handle != INVALID_HANDLE);

  if (Res)
  {
  // https://www.mql5.com/ru/forum/170953/page14#comment_10550116
  #define CLOSETIME_INDEX 0
  #define TICKET_INDEX 1
    long Tickets[][2];
    int Total = OrdersHistoryTotal();    

  #ifdef __MQL5__
    static const bool Sort = false;
  #else // __MQL5__
    #ifdef __VIRTUAL__
      static const bool IsTester = ::MQLInfoInteger(MQL_TESTER);

      const bool Sort = !IsTester && !VIRTUAL::GetHandle();
    #else // __VIRTUAL__
      static const bool Sort = !::MQLInfoInteger(MQL_TESTER);
    #endif // __VIRTUAL__

    if (Sort)
    {
      int Amount = 0;

      for (int i = (::ArrayResize(Tickets, Total) >> 1) - 1; i >= 0; i--) // https://www.mql5.com/ru/forum/170953/page16#comment_10604064
        if (OrderSelect(i, SELECT_BY_POS, MODE_HISTORY))
        {
          Tickets[Amount][CLOSETIME_INDEX] = OrderCloseTime();
          Tickets[Amount++][TICKET_INDEX] = OrderTicket();
        }

      if ((bool)(Total = ::ArrayResize(Tickets, Amount) >> 1)) // https://www.mql5.com/ru/forum/170953/page16#comment_10604064
        ::ArraySort(Tickets);
    }
  #endif // __MQL5__
    
    double Balance = 0;

    FileWriteString(handle, "var dat1=[\n");

    for (int i = 0; i < Total; i++)
      if ((Sort ? OrderSelect((int)Tickets[i][TICKET_INDEX], SELECT_BY_TICKET) : (OrderSelect(i, SELECT_BY_POS, MODE_HISTORY))) && // int - (Sort == true) only for MQL4.
          ((OrderType() <= OP_SELL) || (OrderType() > OP_SELLSTOP)))
    #undef TICKET_INDEX
    #undef CLOSETIME_INDEX
      {
        Balance += OrderProfit() + OrderCommission() + OrderSwap();

        FileWriteString(handle, "[" + (string)((long)OrderCloseTime() * 1000) + "," + DoubleToString(Balance, 2) + "],\n");
      }

    FileWriteString(handle, "];\n");
    FileWriteString(handle, "var T1=dat1[0][0];\n");
    FileWriteString(handle, "var T2=dat1[dat1.length-1][0];\n");
    FileWriteString(handle, "var nTrades=dat1.length;\n");
    FileWriteString(handle, "var Balance=" + DoubleToString(Balance, 2) + ";\n");
    FileWriteString(handle, "var Currency=\"" + AccountInfoString(ACCOUNT_CURRENCY) + "\";\n");

    FileClose(handle);
  }

  return(Res);
}

string GetHTM( const string FileName = "Graph.txt" )
{
  uchar Data[];

  const int handle = FileOpen(FileName, FILE_READ | FILE_BIN);
  const bool Res = (handle != INVALID_HANDLE);

  if (Res)
  {
    FileReadArray(handle, Data);

    FileClose(handle);
  }

  return(Res ? CharArrayToString(Data) : NULL);
}

//+------------------------------------------------------------------+
//| Creates 'HighCharts' balance report on trades in .html format    |
//+------------------------------------------------------------------+
void CreateHighChartsBalanceReport( const string FileName = "Graph.htm" )
{
  const int handle = FileOpen(FileName, FILE_WRITE | FILE_TXT | FILE_ANSI);

  if (handle != INVALID_HANDLE)
  {
//    FileWriteString(handle, htm_file);
    FileWriteString(handle, GetHTM());
    FileClose(handle);

    if (CreateBalanceData())
      ShellExecuteW(0, "Open", BASEPATH + FileName, NULL, NULL, 3);
  }
}

void OnStart()
{
  CreateHighChartsBalanceReport();
}



Отличие от Report - интерактивный график с различными наворотами.

Balance Graph HTML
Balance Graph HTML
  • www.mql5.com
Pan PrizMA Sin leverage 72 Builds a sliding line with a polynomial of 4 degrees. Extrapolates as a section of a given function with a sinusoid and its axial one. One value is taken from the constructed lines on each bar and a line of extrapolated values is plotted...
 
Добавлена новая возможность. Пример применения к стандартному советнику.
#define REPORT_TESTER             // В тестере будут автоматически записываться отчеты
#define REPORT_TESTER_INPUTS      // В отчете одиночного прохода будут видны входные параметры советника - требует разрешения DLL.
#define REPORT_INTERACTIVE_CHARTS // Добавляет в отчет интерактивные графики.
#define REPORT_BROWSER            // Создание отчета с запуском браузера - требует разрешения DLL.
#include <Report.mqh> // https://www.mql5.com/ru/code/18801

#include <..\Experts\Examples\Moving Average\Moving Average.mq5>


В прицепе результат запуска одиночного прохода. Теперь видны входные параметры советника и интерактивный график баланса.


ЗЫ Обновлен скрипт CustomReport. Попробуйте не статичный торговый отчет на своем счете. Пример работы.

Файлы:
Reports.zip  138 kb
 

Наконец-то одиночные проходы стали полноценными - появились все графики в отчете.

Так же в MT4 для всех брокеров в отчет добавлены данные о проскальзываниях при исполнении TP/SL. Для этого у себя нужно обновить библиотеку.

 

В отчеты одиночного прохода добавлены данные о настройках Тестера.