From 6859c3a321e191639388b20bcc8a75c6a06dc0a7 Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Mon, 6 May 2024 21:39:30 +0200 Subject: [PATCH 1/7] Fixed problem with casting string to ENUM_LOG_LEVEL via indicator params' Get() and Set() methods. Also fixed formatting of files. --- EA.mqh | 25 +++++++++++++------------ Exchange/tests/Exchange.test.mq5 | 4 ++-- Log.mqh | 9 +++++++-- Strategy.mqh | 25 ++++++++++++------------- Trade.mqh | 15 ++++++++------- 5 files changed, 42 insertions(+), 36 deletions(-) diff --git a/EA.mqh b/EA.mqh index 3d7bdb6d4..16cccca33 100644 --- a/EA.mqh +++ b/EA.mqh @@ -30,8 +30,8 @@ #define EA_MQH // Includes. -#include "Chart.mqh" #include "./Chart.struct.static.h" +#include "Chart.mqh" #include "Data.struct.h" #include "Dict.mqh" #include "DictObject.mqh" @@ -114,12 +114,12 @@ class EA : public Taskable { Init(); // Initialize a trade instance for the current chart and symbol. Ref _source = Platform::FetchDefaultCandleIndicator(_Symbol, PERIOD_CURRENT); - TradeParams _tparams(0, 1.0f, 0, eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); + TradeParams _tparams(0, 1.0f, 0, (ENUM_LOG_LEVEL)eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); Trade _trade(_tparams, _source.Ptr()); trade.Set(_Symbol, _trade); logger.Link(_trade.GetLogger()); - logger.SetLevel(eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); - //_trade.GetLogger().SetLevel(eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); + logger.SetLevel((ENUM_LOG_LEVEL)eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); + //_trade.GetLogger().SetLevel((ENUM_LOG_LEVEL)eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); } /** @@ -409,11 +409,11 @@ class EA : public Taskable { _strat.OnOrderOpen(_oparams); // Send the request. _result = _trade.RequestSend(_request, _oparams); - if (!_result) { // && _strade.IsTradeRecommended( - logger.Debug( - StringFormat("Error while sending a trade request! Entry: %s", - SerializerConverter::FromObject(MqlTradeRequestProxy(_request)).ToString()), - __FUNCTION_LINE__, StringFormat("Code: %d, Msg: %s", _LastError, Terminal::GetErrorText(_LastError))); + if (!_result) { // && _strade.IsTradeRecommended( + logger.Debug( + StringFormat("Error while sending a trade request! Entry: %s", + SerializerConverter::FromObject(MqlTradeRequestProxy(_request)).ToString()), + __FUNCTION_LINE__, StringFormat("Code: %d, Msg: %s", _LastError, Terminal::GetErrorText(_LastError))); if (_trade.IsTradeRecommended()) { logger.Debug( StringFormat("Error while sending a trade request! Entry: %s", @@ -421,7 +421,8 @@ class EA : public Taskable { __FUNCTION_LINE__, StringFormat("Code: %d, Msg: %s", _LastError, Terminal::GetErrorText(_LastError))); } #ifdef __debug_ea__ - Print(__FUNCTION_LINE__ + "(): " + SerializerConverter::FromObject(MqlTradeRequestProxy(_request)).ToString()); + Print(__FUNCTION_LINE__ + + "(): " + SerializerConverter::FromObject(MqlTradeRequestProxy(_request)).ToString()); #endif } return _result; @@ -766,8 +767,8 @@ class EA : public Taskable { Ref _strat = ((SClass *)NULL).Init(_tf, THIS_PTR); _strat.Ptr().Set(STRAT_PARAM_ID, _magic_no); _strat.Ptr().Set(TRADE_PARAM_MAGIC_NO, _magic_no); - _strat.Ptr().Set(STRAT_PARAM_LOG_LEVEL, - eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); + _strat.Ptr().Set(STRAT_PARAM_LOG_LEVEL, + (ENUM_LOG_LEVEL)eparams.Get(STRUCT_ENUM(EAParams, EA_PARAM_PROP_LOG_LEVEL))); _strat.Ptr().Set(STRAT_PARAM_TF, _tf); _strat.Ptr().Set(STRAT_PARAM_TYPE, _type); _strat.Ptr().OnInit(); diff --git a/Exchange/tests/Exchange.test.mq5 b/Exchange/tests/Exchange.test.mq5 index 498c0c791..797c44274 100644 --- a/Exchange/tests/Exchange.test.mq5 +++ b/Exchange/tests/Exchange.test.mq5 @@ -96,8 +96,8 @@ bool TestExchange01() { exchange REF_DEREF SymbolAdd(symbol02.Ptr(), "Symbol02"); // Attach instances of dummy trades. - Ref trade01 = new TradeDummy(Platform::FetchDefaultCandleIndicator(_Symbol, PERIOD_CURRENT)); - Ref trade02 = new TradeDummy(Platform::FetchDefaultCandleIndicator(_Symbol, PERIOD_CURRENT)); + Ref trade01 = new TradeDummy(Platform::FetchDefaultCandleIndicator(_Symbol, Period())); + Ref trade02 = new TradeDummy(Platform::FetchDefaultCandleIndicator(_Symbol, Period())); exchange REF_DEREF TradeAdd(trade01.Ptr(), "Trade01"); exchange REF_DEREF TradeAdd(trade02.Ptr(), "Trade02"); diff --git a/Log.mqh b/Log.mqh index c55c526d6..13bc13fd4 100644 --- a/Log.mqh +++ b/Log.mqh @@ -76,8 +76,7 @@ class Log : public Object { /** * Class copy constructor. */ - Log(const Log &_log) : filename(_log.filename), last_entry(_log.last_entry), log_level(_log.log_level) { - } + Log(const Log &_log) : filename(_log.filename), last_entry(_log.last_entry), log_level(_log.log_level) {} /** * Class deconstructor. @@ -345,4 +344,10 @@ bool Log::AddLastError(string prefix, long suffix) { return Add(V_ERROR, Terminal::GetLastErrorText(), prefix, StringFormat("%d", suffix)); } +// Specialization of StringToType() for enum. +void StringToType(string _value, ENUM_LOG_LEVEL &_out) { + // Maybe parse the string? + _out = V_NONE; +} + #endif diff --git a/Strategy.mqh b/Strategy.mqh index 115b7660a..79ab81eba 100644 --- a/Strategy.mqh +++ b/Strategy.mqh @@ -668,7 +668,7 @@ class Strategy : public Taskable { virtual void OnInit() { // Link log instances. logger.Link(trade.Ptr().GetLogger()); - trade.Ptr().GetLogger().SetLevel(sparams.Get(STRAT_PARAM_LOG_LEVEL)); + trade.Ptr().GetLogger().SetLevel((ENUM_LOG_LEVEL)sparams.Get(STRAT_PARAM_LOG_LEVEL)); // Sets strategy stops. SetStops(THIS_PTR, THIS_PTR); // trade.SetStrategy(&this); // @fixme @@ -926,12 +926,12 @@ class Strategy : public Taskable { bool _result = true; if (_method != 0) { int _shift = _method / 64; - if (METHOD(_method, 0)) _result &= !trade REF_DEREF HasBarOrder(_cmd, _shift); // 1 - if (METHOD(_method, 1)) _result &= IsTrend(_cmd); // 2 - if (METHOD(_method, 2)) _result &= trade REF_DEREF IsPivot(_cmd, _shift); // 4 - if (METHOD(_method, 3)) _result &= !trade REF_DEREF HasOrderOppositeType(_cmd); // 8 - if (METHOD(_method, 4)) _result &= trade REF_DEREF IsPeak(_cmd, _shift); // 16 - if (METHOD(_method, 5)) _result &= !trade REF_DEREF HasOrderBetter(_cmd); // 32 + if (METHOD(_method, 0)) _result &= !trade REF_DEREF HasBarOrder(_cmd, _shift); // 1 + if (METHOD(_method, 1)) _result &= IsTrend(_cmd); // 2 + if (METHOD(_method, 2)) _result &= trade REF_DEREF IsPivot(_cmd, _shift); // 4 + if (METHOD(_method, 3)) _result &= !trade REF_DEREF HasOrderOppositeType(_cmd); // 8 + if (METHOD(_method, 4)) _result &= trade REF_DEREF IsPeak(_cmd, _shift); // 16 + if (METHOD(_method, 5)) _result &= !trade REF_DEREF HasOrderBetter(_cmd); // 32 if (METHOD(_method, 6)) _result &= trade REF_DEREF CalcActiveProfitInValue() <= 0.0f; // 64 /* if (METHOD(_method, 6)) @@ -1054,15 +1054,14 @@ class Strategy : public Taskable { virtual bool SignalCloseFilter(ENUM_ORDER_TYPE _cmd, int _method = 0, int _shift = 0) { bool _result = _method == 0; if (_method != 0) { - if (METHOD(_method, 0)) _result |= _result || !trade REF_DEREF HasBarOrder(_cmd, _shift); // 1 - if (METHOD(_method, 1)) _result |= _result || !IsTrend(_cmd); // 2 + if (METHOD(_method, 1)) _result |= _result || !IsTrend(_cmd); // 2 if (METHOD(_method, 2)) _result |= _result || !trade REF_DEREF IsPivot(_cmd, _shift); // 4 if (METHOD(_method, 3)) - _result |= _result || Open[_shift] > High[_shift + 1] || Open[_shift] < Low[_shift + 1]; // 8 - if (METHOD(_method, 4)) _result |= _result || trade REF_DEREF IsPeak(_cmd, _shift); // 16 - if (METHOD(_method, 5)) _result |= _result || trade REF_DEREF HasOrderBetter(_cmd); // 32 - if (METHOD(_method, 6)) _result |= _result || trade REF_DEREF CalcActiveProfitInValue() > 0.0f; // 64 + _result |= _result || Open[_shift] > High[_shift + 1] || Open[_shift] < Low[_shift + 1]; // 8 + if (METHOD(_method, 4)) _result |= _result || trade REF_DEREF IsPeak(_cmd, _shift); // 16 + if (METHOD(_method, 5)) _result |= _result || trade REF_DEREF HasOrderBetter(_cmd); // 32 + if (METHOD(_method, 6)) _result |= _result || trade REF_DEREF CalcActiveProfitInValue() > 0.0f; // 64 /* if (METHOD(_method, 6)) _result |= diff --git a/Trade.mqh b/Trade.mqh index 4c4daf932..4ec0a0ec5 100644 --- a/Trade.mqh +++ b/Trade.mqh @@ -830,7 +830,7 @@ HistorySelect(0, TimeCurrent()); // Select history for access. // @todo: _order.IsPending()? _result &= orders_active.Set(_order PTR_DEREF Get(ORDER_PROP_TICKET), _order_ref); logger.Link(_order.GetLogger()); - _order PTR_DEREF GetLogger().SetLevel(tparams.Get(TRADE_PARAM_LOG_LEVEL)); + _order PTR_DEREF GetLogger().SetLevel((ENUM_LOG_LEVEL)tparams.Get(TRADE_PARAM_LOG_LEVEL)); } else { _result &= orders_history.Set(_order PTR_DEREF Get(ORDER_PROP_TICKET), _order_ref); } @@ -894,11 +894,11 @@ HistorySelect(0, TimeCurrent()); // Select history for access. OrderMoveToHistory(_order.Ptr()); order_last = _order; } else { - logger.Error( - StringFormat("Failed to close the order: %d! Error: %d (%s)", _order REF_DEREF Get(ORDER_PROP_TICKET), - _order REF_DEREF Get(ORDER_PROP_LAST_ERROR), - Terminal::GetErrorText(_order REF_DEREF Get(ORDER_PROP_LAST_ERROR))), - __FUNCTION_LINE__); + logger.Error(StringFormat("Failed to close the order: %d! Error: %d (%s)", + _order REF_DEREF Get(ORDER_PROP_TICKET), + _order REF_DEREF Get(ORDER_PROP_LAST_ERROR), + Terminal::GetErrorText(_order REF_DEREF Get(ORDER_PROP_LAST_ERROR))), + __FUNCTION_LINE__); continue; } } else { @@ -932,7 +932,8 @@ HistorySelect(0, TimeCurrent()); // Select history for access. order_last = _order; } else { logger.Error( - StringFormat("Failed to close the order: %d! Error: %d (%s)", _order REF_DEREF Get(ORDER_PROP_TICKET), + StringFormat("Failed to close the order: %d! Error: %d (%s)", + _order REF_DEREF Get(ORDER_PROP_TICKET), _order REF_DEREF Get(ORDER_PROP_LAST_ERROR), Terminal::GetErrorText(_order REF_DEREF Get(ORDER_PROP_LAST_ERROR))), __FUNCTION_LINE__); From d8eb065e8be78fa2da05757a688a82a68e3965ed Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Wed, 22 May 2024 19:39:38 +0200 Subject: [PATCH 2/7] WIP. Trying to make indicators to work with historic ticks. Also changed .clang-format in order to indent macros. Changed Indi_Bands's OnCalculate param names in order to prevent warnings. --- .clang-format | 1 + Indicator/Indicator.h | 11 +- Indicator/IndicatorTf.provider.h | 8 +- IndicatorLegacy.h | 413 +++++++++++++++++------------- Indicators/Indi_Bands.mqh | 34 +-- Indicators/Indi_MA.mqh | 13 +- Indicators/Tick/Indi_TickMt.mqh | 24 +- Std.h | 426 +++++++++++++++++-------------- 8 files changed, 522 insertions(+), 408 deletions(-) diff --git a/.clang-format b/.clang-format index 3442bc765..d64b4533c 100644 --- a/.clang-format +++ b/.clang-format @@ -2,3 +2,4 @@ Language: Cpp BasedOnStyle: Google ColumnLimit: 120 +IndentPPDirectives: BeforeHash diff --git a/Indicator/Indicator.h b/Indicator/Indicator.h index 53474ff19..ee68fe330 100644 --- a/Indicator/Indicator.h +++ b/Indicator/Indicator.h @@ -687,10 +687,13 @@ class Indicator : public IndicatorData { } if (_LastError != ERR_SUCCESS) { - datetime _bar_dt = (datetime)_bar_time; - Print("Error: Code ", _LastError, " while trying to retrieve entry at shift ", _rel_shift, " (absolute ", - ToAbsShift(_rel_shift), "), mode ", _mode, ", time ", _bar_dt); - DebugBreak(); + if (_LastError != 4806) { + // Error occured and it's not "4806 Requested data not found". + datetime _bar_dt = (datetime)_bar_time; + Print("Error: Code ", _LastError, " while trying to retrieve entry at shift ", _rel_shift, " (absolute ", + ToAbsShift(_rel_shift), "), mode ", _mode, ", time ", _bar_dt); + DebugBreak(); + } } } GetEntryAlter(_entry, _rel_shift); diff --git a/Indicator/IndicatorTf.provider.h b/Indicator/IndicatorTf.provider.h index 63669b218..3318b58da 100644 --- a/Indicator/IndicatorTf.provider.h +++ b/Indicator/IndicatorTf.provider.h @@ -25,8 +25,8 @@ #define INDICATOR_TF_PROVIDER_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif /** @@ -159,9 +159,11 @@ class ItemsHistoryTfCandleProvider : public ItemsHistoryCandleProvider { // Adding candle to the output array. ArrayPushObject(_out_arr, _candle); - --_num_items; } + // Even if we don't form an item (a candle), we assume we've done one item. + --_num_items; + if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { _from_time_ms += _candle_length_ms; } else { diff --git a/IndicatorLegacy.h b/IndicatorLegacy.h index 188d02fd7..31ee7226e 100644 --- a/IndicatorLegacy.h +++ b/IndicatorLegacy.h @@ -4,86 +4,138 @@ */ #ifndef __MQL__ -#pragma once + #pragma once #endif +// Includes. +#include "Platform.h" + #ifdef INDICATOR_LEGACY_VERSION_MT4 -#define INDICATOR_LEGACY_VERSION_DEFINED + #define INDICATOR_LEGACY_VERSION_DEFINED #endif #ifdef INDICATOR_LEGACY_VERSION_MT5 -#define INDICATOR_LEGACY_VERSION_DEFINED + #define INDICATOR_LEGACY_VERSION_DEFINED #endif -#ifndef INDICATOR_LEGACY_VERSION_DEFINED -#define INDICATOR_LEGACY_VERSION_MT5 -#define INDICATOR_LEGACY_VERSION_DEFINED +#ifdef INDICATOR_STANDALONE_VERSION_LONG + #define INDICATOR_STANDALONE_VERSION_DEFINED #endif -#ifdef __MQL4__ +#ifdef INDICATOR_STANDALONE_VERSION_SHORT + #define INDICATOR_STANDALONE_VERSION_DEFINED +#endif -#include -#include -#include -#include -#include +#ifdef INDICATOR_STANDALONE_VERSION_DEFINED + +/** + * Wrapper for future OnInit(). We may need to init indicator with candle and + * tick inidicator in the hierarchy. + */ +int OnInit() { + int _result = OnInitOriginal(); + Platform::AddWithDefaultBindings(INDICATOR_STANDALONE_INDI_PTR, Symbol(), (ENUM_TIMEFRAMES)Period()); + return _result; +} + + #define OnInit OnInitOriginal + + // In standalone mode without legacy mode we also need to wrap OnCalculate() as + // we need to call Platform::OnCalculate(). The only difference is that we + // don't need to change buffer's AsSeries flag in MT5. + #ifndef INDICATOR_LEGACY_VERSION_DEFINED + #ifndef INDICATOR_LEGACY_EMIT_ONCALCULATE_WRAPPER + #define INDICATOR_LEGACY_EMIT_ONCALCULATE_WRAPPER + #endif + + // No work required for buffers. + #define INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER + #define INDICATOR_LEGACY_VERSION_RELEASE_BUFFER + #ifdef INDICATOR_STANDALONE_VERSION_LONG + // We don't want to write two OnCalculate() wrappers. One is enough. + #define INDICATOR_LEGACY_VERSION_LONG + #endif // INDICATOR_STANDALONE_VERSION_LONG + #ifdef INDICATOR_STANDALONE_VERSION_SHORT + // We don't want to write two OnCalculate() wrappers. One is enough. + #define INDICATOR_LEGACY_VERSION_SHORT + #endif // INDICATOR_STANDALONE_VERSION_LONG + #endif // INDICATOR_LEGACY_VERSION_DEFINED + +#endif // INDICATOR_STANDALONE_VERSION_DEFINED #ifndef INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER -#define INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER + #define INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER #endif #ifndef INDICATOR_LEGACY_VERSION_RELEASE_BUFFER -#define INDICATOR_LEGACY_VERSION_RELEASE_BUFFER + #define INDICATOR_LEGACY_VERSION_RELEASE_BUFFER #endif -#ifdef INDICATOR_LEGACY_VERSION_MT5 +#ifdef INDICATOR_LEGACY_EMIT_ONCALCULATE_WRAPPER -#ifndef INDICATOR_LEGACY_VERSION_SHORT + #include + #include + #include + #include + #include + + #ifdef INDICATOR_LEGACY_VERSION_SHORT /** - * Replacement for future OHLC-based OnCalculate(). + * Wrapper for future price-based OnCalculate(). */ -int OnCalculate(const int rates_total, const int prev_calculated, const datetime& time[], const double& open[], - const double& high[], const double& low[], const double& close[], const long& tick_volume[], - const long& volume[], const int& spread[]) { +int OnCalculate(const int rates_total, const int prev_calculated, const int begin, const double& price[]) { // We need to call Platform::Tick() and maybe also IndicatorData::EmitHistory() before. Platform::OnCalculate(rates_total, prev_calculated); INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER; - int _num_calculated = - OnCalculateMT5(rates_total, prev_calculated, time, open, high, low, close, tick_volume, volume, spread); + // NOTE: If compiler sees an error here about 'time' parameter conversion + // then you probably must do: + // #define INDICATOR_LEGACY_VERSION_LONG + // before including IndicatorLegacy.h + int _num_calculated = OnCalculateMT5(rates_total, prev_calculated, begin, price); INDICATOR_LEGACY_VERSION_RELEASE_BUFFER; return _num_calculated; } -#else + #endif // INDICATOR_LEGACY_VERSION_SHORT + + #ifdef INDICATOR_LEGACY_VERSION_LONG /** - * Replacement for future price-based OnCalculate(). + * Wrapper for future OHLC-based OnCalculate(). */ -int OnCalculate(const int rates_total, const int prev_calculated, const int begin, const double& price[]) { +int OnCalculate(const int rates_total, const int prev_calculated, const datetime& time[], const double& open[], + const double& high[], const double& low[], const double& close[], const long& tick_volume[], + const long& volume[], const int& spread[]) { // We need to call Platform::Tick() and maybe also IndicatorData::EmitHistory() before. Platform::OnCalculate(rates_total, prev_calculated); - INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER; - - // NOTE: If compiler sees an error here about parameter conversion then you - // probably must do: + // NOTE: If compiler sees an error here about parameter conversion + // then you probably must do: // #define INDICATOR_LEGACY_VERSION_SHORT // before including IndicatorLegacy.h - int _num_calculated = OnCalculateMT5(rates_total, prev_calculated, begin, price); + INDICATOR_LEGACY_VERSION_ACQUIRE_BUFFER; + + int _num_calculated = + OnCalculateMT5(rates_total, prev_calculated, time, open, high, low, close, tick_volume, volume, spread); INDICATOR_LEGACY_VERSION_RELEASE_BUFFER; return _num_calculated; } -#endif + #endif // INDICATOR_LEGACY_VERSION_LONG -#define OnCalculate OnCalculateMT5 + #define OnCalculate OnCalculateMT5 + +#endif // INDICATOR_LEGACY_EMIT_ONCALCULATE_WRAPPER + +#ifdef __MQL4__ + #ifdef INDICATOR_LEGACY_VERSION_MT5 /** * Wrapper class to be used by MQL4 code to allow calling MQL5's indicator functions like iMA() in MQL4. @@ -188,10 +240,10 @@ int CopyBuffer(int _handle, int _mode, int _start, int _count, double& _buffer[] return _num_copied; } -/** - * Defines wrapper class and global iNAME() indicator function (e.g., iMA(), iATR()). - */ -// Print(#FN_NAME " key = ", _key); \ + /** + * Defines wrapper class and global iNAME() indicator function (e.g., iMA(), iATR()). + */ + // Print(#FN_NAME " key = ", _key); \ #define DEFINE_LEGACY_INDICATOR(FN_NAME, BUILTIN_NAME, TYPED_PARAMS_COMMA, TYPED_PARAMS_NO_UDL_SEMICOLON, UNTYPED_PARAMS_COMMA_KEY, UNTYPED_PARAMS_COMMA_VALUES, ASSIGNMENTS_COMMA, UNTYPED_PARAMS_NO_UDL_COMMA_VALUES) \ class BUILTIN_NAME##Legacy : public IndicatorLegacy { \ TYPED_PARAMS_NO_UDL_SEMICOLON; \ @@ -218,92 +270,95 @@ int FN_NAME(TYPED_PARAMS_COMMA) { \ return PTR_ATTRIB(_indi.Ptr(), GetHandle()); \ } -/** - * 1-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_1(FN_NAME, INDI_NAME, T1, N1) \ - DEFINE_LEGACY_INDICATOR(INDI_NAME, T1 _##N1, T1 N1, _##N1, _##N1, N1(_##N1), N1); - -/** - * 2-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_2(FN_NAME, INDI_NAME, T1, N1, T2, N2) \ - DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2, T1 N1 SEMICOLON T2 N2, _##N1 COMMA _##N2, \ - _##N1 COMMA _##N2, N1(_##N1) COMMA N2(_##N2), N1 COMMA N2); - -/** - * 3-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_3(FN_NAME, INDI_NAME, T1, N1, T2, N2, T3, N3) \ - DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3, \ - T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3, _##N1 COMMA _##N2 COMMA _##N3, \ - _##N1 COMMA _##N2 COMMA _##N3, N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3), \ - N1 COMMA N2 COMMA N3); - -/** - * 4-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_4(FN_NAME, INDI_NAME) \ - DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4, \ - T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4, _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4, \ - N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4), N1 COMMA N2 COMMA N3 COMMA N4); - -/** - * 5-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_5(FN_NAME, INDI_NAME) \ - DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5, \ - T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5, \ - N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5), \ - N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5); - -/** - * 6-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_6(FN_NAME, INDI_NAME) \ - DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, \ - T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5 COMMA T6 _##N6, \ - T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5 SEMICOLON T6 N6, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6, \ - N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5) COMMA N6(_##N6), \ - N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5 COMMA N6); - -/** - * 7-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_7(FN_NAME, INDI_NAME) \ - DEFINE_LEGACY_INDICATOR( \ - FN_NAME, INDI_NAME, \ - T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5 COMMA T6 _##N6 COMMA T7 _##N7, \ - T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5 SEMICOLON T6 N6 SEMICOLON T7 N7, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7, \ - N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5) COMMA N6(_##N6) COMMA N7(_##N7), \ - N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5 COMMA N6 COMMA N7); - -/** - * 8-parameter helper for DEFINE_LEGACY_INDICATOR. - */ -#define DEFINE_LEGACY_INDICATOR_8(FN_NAME, INDI_NAME) \ - DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, \ - T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5 COMMA T6 _##N6 COMMA T7 \ - _##N7 COMMA T8 _##N8, \ - T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5 SEMICOLON T6 N6 \ - SEMICOLON T7 N7 SEMICOLON T8 N8, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7 COMMA _##N8, \ - _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7 COMMA _##N8, \ - N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5) COMMA N6(_##N6) \ - COMMA N7(_##N7) COMMA N8(_##N8), \ - N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5 COMMA N6 COMMA N7 COMMA N8); - -/** - * Replacement for future StringConcatenate(). - */ -#define StringConcatenate StringConcatenateMT5 + /** + * 1-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_1(FN_NAME, INDI_NAME, T1, N1) \ + DEFINE_LEGACY_INDICATOR(INDI_NAME, T1 _##N1, T1 N1, _##N1, _##N1, N1(_##N1), N1); + + /** + * 2-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_2(FN_NAME, INDI_NAME, T1, N1, T2, N2) \ + DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2, T1 N1 SEMICOLON T2 N2, _##N1 COMMA _##N2, \ + _##N1 COMMA _##N2, N1(_##N1) COMMA N2(_##N2), N1 COMMA N2); + + /** + * 3-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_3(FN_NAME, INDI_NAME, T1, N1, T2, N2, T3, N3) \ + DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3, \ + T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3, _##N1 COMMA _##N2 COMMA _##N3, \ + _##N1 COMMA _##N2 COMMA _##N3, N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3), \ + N1 COMMA N2 COMMA N3); + + /** + * 4-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_4(FN_NAME, INDI_NAME) \ + DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4, \ + T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4, _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4, \ + N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4), \ + N1 COMMA N2 COMMA N3 COMMA N4); + + /** + * 5-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_5(FN_NAME, INDI_NAME) \ + DEFINE_LEGACY_INDICATOR(FN_NAME, INDI_NAME, \ + T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5, \ + T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5, \ + N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5), \ + N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5); + + /** + * 6-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_6(FN_NAME, INDI_NAME) \ + DEFINE_LEGACY_INDICATOR( \ + FN_NAME, INDI_NAME, T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5 COMMA T6 _##N6, \ + T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5 SEMICOLON T6 N6, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6, \ + N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5) COMMA N6(_##N6), \ + N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5 COMMA N6); + + /** + * 7-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_7(FN_NAME, INDI_NAME) \ + DEFINE_LEGACY_INDICATOR( \ + FN_NAME, INDI_NAME, \ + T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5 COMMA T6 _##N6 COMMA T7 _##N7, \ + T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5 SEMICOLON T6 N6 SEMICOLON T7 N7, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7, \ + N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5) COMMA N6(_##N6) COMMA N7(_##N7), \ + N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5 COMMA N6 COMMA N7); + + /** + * 8-parameter helper for DEFINE_LEGACY_INDICATOR. + */ + #define DEFINE_LEGACY_INDICATOR_8(FN_NAME, INDI_NAME) \ + DEFINE_LEGACY_INDICATOR( \ + FN_NAME, INDI_NAME, \ + T1 _##N1 COMMA T2 _##N2 COMMA T3 _##N3 COMMA T4 _##N4 COMMA T5 _##N5 COMMA T6 _##N6 COMMA T7 _##N7 COMMA T8 \ + _##N8, \ + T1 N1 SEMICOLON T2 N2 SEMICOLON T3 N3 SEMICOLON T4 N4 SEMICOLON T5 N5 SEMICOLON T6 N6 SEMICOLON T7 N7 \ + SEMICOLON T8 N8, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7 COMMA _##N8, \ + _##N1 COMMA _##N2 COMMA _##N3 COMMA _##N4 COMMA _##N5 COMMA _##N6 COMMA _##N7 COMMA _##N8, \ + N1(_##N1) COMMA N2(_##N2) COMMA N3(_##N3) COMMA N4(_##N4) COMMA N5(_##N5) COMMA N6(_##N6) COMMA N7(_##N7) \ + COMMA N8(_##N8), \ + N1 COMMA N2 COMMA N3 COMMA N4 COMMA N5 COMMA N6 COMMA N7 COMMA N8); + + /** + * Replacement for future StringConcatenate(). + */ + #define StringConcatenate StringConcatenateMT5 /** * MQL4 wrapper of MQL5's StringConcatenate(). @@ -395,66 +450,66 @@ DEFINE_LEGACY_INDICATOR_2(iAD, iAD, string, symbol, int, period); // int iATR(string symbol, ENUM_TIMEFRAMES period, int ma_period); DEFINE_LEGACY_INDICATOR_3(iATR, iATR, string, symbol, int, period, int, ma_period); -// int iRSI(string symbol, ENUM_TIMEFRAMES period, int ma_period, int applied_price); -#define T1 string -#define N1 symbol -#define T2 int -#define N2 period -#define T3 int -#define N3 ma_period -#define T4 int -#define N4 applied_price + // int iRSI(string symbol, ENUM_TIMEFRAMES period, int ma_period, int applied_price); + #define T1 string + #define N1 symbol + #define T2 int + #define N2 period + #define T3 int + #define N3 ma_period + #define T4 int + #define N4 applied_price DEFINE_LEGACY_INDICATOR_4(iRSI, iRSI) -#undef T1 -#undef N1 -#undef T2 -#undef N2 -#undef T3 -#undef N3 -#undef T4 -#undef N4 -#undef T5 -#undef N5 -#undef T6 -#undef N6 - -// int iMA(string symbol, ENUM_TIMEFRAMES period, int ma_period, int ma_shift, ENUM_MA_METHOD ma_method, -#define T1 string -#define N1 symbol -#define T2 int -#define N2 period -#define T3 int -#define N3 ma_period -#define T4 int -#define N4 ma_shift -#define T5 int -#define N5 ma_method -#define T6 int -#define N6 applied_price + #undef T1 + #undef N1 + #undef T2 + #undef N2 + #undef T3 + #undef N3 + #undef T4 + #undef N4 + #undef T5 + #undef N5 + #undef T6 + #undef N6 + + // int iMA(string symbol, ENUM_TIMEFRAMES period, int ma_period, int ma_shift, ENUM_MA_METHOD ma_method, + #define T1 string + #define N1 symbol + #define T2 int + #define N2 period + #define T3 int + #define N3 ma_period + #define T4 int + #define N4 ma_shift + #define T5 int + #define N5 ma_method + #define T6 int + #define N6 applied_price DEFINE_LEGACY_INDICATOR_6(iMA, iMA) -#undef T1 -#undef N1 -#undef T2 -#undef N2 -#undef T3 -#undef N3 -#undef T4 -#undef N4 -#undef T5 -#undef N5 -#undef T6 -#undef N6 - -#endif // INDICATOR_LEGACY_VERSION_MT5 -#endif // __MQL4__ + #undef T1 + #undef N1 + #undef T2 + #undef N2 + #undef T3 + #undef N3 + #undef T4 + #undef N4 + #undef T5 + #undef N5 + #undef T6 + #undef N6 + + #endif // INDICATOR_LEGACY_VERSION_MT5 +#endif // __MQL4__ #ifdef __MQL5__ -#ifdef INDICATOR_LEGACY_VERSION_MT4 + #ifdef INDICATOR_LEGACY_VERSION_MT4 -/** - * Replacement for future StringConcatenate(). - */ -#define StringConcatenate StringConcatenateMT4 + /** + * Replacement for future StringConcatenate(). + */ + #define StringConcatenate StringConcatenateMT4 /** * MQL5 wrapper of MQL4's StringConcatenate(). @@ -522,5 +577,5 @@ string StringConcatenateMT4(string& _result, A _a) { return (string)_a; } -#endif // INDICATOR_LEGACY_VERSION_MT4 -#endif // __MQL5__ + #endif // INDICATOR_LEGACY_VERSION_MT4 +#endif // __MQL5__ diff --git a/Indicators/Indi_Bands.mqh b/Indicators/Indi_Bands.mqh index 9300cb9c5..fcd390dbd 100644 --- a/Indicators/Indi_Bands.mqh +++ b/Indicators/Indi_Bands.mqh @@ -176,33 +176,33 @@ class Indi_Bands : public Indicator { /** * OnCalculate() method for Bands indicator. */ - static int Calculate(INDICATOR_CALCULATE_METHOD_PARAMS_SHORT, ValueStorage &ExtMLBuffer, - ValueStorage &ExtTLBuffer, ValueStorage &ExtBLBuffer, - ValueStorage &ExtStdDevBuffer, int InpBandsPeriod, int InpBandsShift, - double InpBandsDeviations) { + static int Calculate(INDICATOR_CALCULATE_METHOD_PARAMS_SHORT, ValueStorage &_ExtMLBuffer, + ValueStorage &_ExtTLBuffer, ValueStorage &_ExtBLBuffer, + ValueStorage &_ExtStdDevBuffer, int _InpBandsPeriod, int _InpBandsShift, + double _InpBandsDeviations) { int ExtBandsPeriod, ExtBandsShift; double ExtBandsDeviations; int ExtPlotBegin = 0; - if (InpBandsPeriod < 2) { + if (_InpBandsPeriod < 2) { ExtBandsPeriod = 20; PrintFormat("Incorrect value for input variable InpBandsPeriod=%d. Indicator will use value=%d for calculations.", - InpBandsPeriod, ExtBandsPeriod); + _InpBandsPeriod, ExtBandsPeriod); } else - ExtBandsPeriod = InpBandsPeriod; - if (InpBandsShift < 0) { + ExtBandsPeriod = _InpBandsPeriod; + if (_InpBandsShift < 0) { ExtBandsShift = 0; PrintFormat("Incorrect value for input variable InpBandsShift=%d. Indicator will use value=%d for calculations.", - InpBandsShift, ExtBandsShift); + _InpBandsShift, ExtBandsShift); } else - ExtBandsShift = InpBandsShift; - if (InpBandsDeviations == 0.0) { + ExtBandsShift = _InpBandsShift; + if (_InpBandsDeviations == 0.0) { ExtBandsDeviations = 2.0; PrintFormat( "Incorrect value for input variable InpBandsDeviations=%f. Indicator will use value=%f for calculations.", - InpBandsDeviations, ExtBandsDeviations); + _InpBandsDeviations, ExtBandsDeviations); } else - ExtBandsDeviations = InpBandsDeviations; + ExtBandsDeviations = _InpBandsDeviations; if (rates_total < ExtPlotBegin) return (0); //--- indexes draw begin settings, when we've recieved previous begin @@ -221,13 +221,13 @@ class Indi_Bands : public Indicator { //--- main cycle for (int i = pos; i < rates_total && !IsStopped(); i++) { //--- middle line - ExtMLBuffer[i] = Indi_MA::SimpleMA(i, ExtBandsPeriod, price); + _ExtMLBuffer[i] = Indi_MA::SimpleMA(i, ExtBandsPeriod, price); //--- calculate and write down StdDev - ExtStdDevBuffer[i] = StdDev_Func(i, price, ExtMLBuffer, ExtBandsPeriod); + _ExtStdDevBuffer[i] = StdDev_Func(i, price, _ExtMLBuffer, ExtBandsPeriod); //--- upper line - ExtTLBuffer[i] = ExtMLBuffer[i] + ExtBandsDeviations * ExtStdDevBuffer[i].Get(); + _ExtTLBuffer[i] = _ExtMLBuffer[i] + ExtBandsDeviations * _ExtStdDevBuffer[i].Get(); //--- lower line - ExtBLBuffer[i] = ExtMLBuffer[i] - ExtBandsDeviations * ExtStdDevBuffer[i].Get(); + _ExtBLBuffer[i] = _ExtMLBuffer[i] - ExtBandsDeviations * _ExtStdDevBuffer[i].Get(); } //--- OnCalculate done. Return new prev_calculated. return (rates_total); diff --git a/Indicators/Indi_MA.mqh b/Indicators/Indi_MA.mqh index 1af92cd2e..051e8cd29 100644 --- a/Indicators/Indi_MA.mqh +++ b/Indicators/Indi_MA.mqh @@ -25,6 +25,8 @@ #define INDI_MA_MQH // Includes. +#include + #include "../Dict.mqh" #include "../DictObject.mqh" #include "../Indicator/Indicator.h" @@ -33,7 +35,14 @@ #include "../Storage/ValueStorage.h" #include "../String.mqh" -#ifndef __MQL4__ +#ifdef __MQL4__ +// MQL4 version of the method doesn't have last parameter. +int LinearWeightedMAOnBuffer(const int rates_total, const int prev_calculated, const int begin, const int period, + const double &price[], double &buffer[]) { + int _weight_sum; + return LinearWeightedMAOnBuffer(rates_total, prev_calculated, begin, period, price, buffer, _weight_sum); +} +#else // !__MQL__4 // Defines global functions (for MQL4 backward compability). double iMA(string _symbol, int _tf, int _ma_period, int _ma_shift, int _ma_method, int _ap, int _shift) { ResetLastError(); @@ -45,7 +54,7 @@ double iMAOnArray(double &_arr[], int _total, int _period, int _ma_shift, int _m ResetLastError(); return Indi_MA::iMAOnArray(_arr, _total, _period, _ma_shift, _ma_method, _abs_shift, _cache); } -#endif +#endif // __MQL4__ // Structs. struct IndiMAParams : IndicatorParams { diff --git a/Indicators/Tick/Indi_TickMt.mqh b/Indicators/Tick/Indi_TickMt.mqh index 98e33a884..320c3f298 100644 --- a/Indicators/Tick/Indi_TickMt.mqh +++ b/Indicators/Tick/Indi_TickMt.mqh @@ -26,8 +26,8 @@ */ #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -196,10 +196,10 @@ class Indi_TickMt : public IndicatorTick _tick(_tmp_ticks[i]); -#ifdef __debug_verbose__ + #ifdef __debug_verbose__ Print("Fetched tick at ", TimeToString(_tmp_ticks[i].time, TIME_DATE | TIME_MINUTES | TIME_SECONDS), ": ", _tmp_ticks[i].ask, ", ", _tmp_ticks[i].bid); -#endif + #endif ArrayPushObject(_out_ticks, _tick); } @@ -232,6 +232,9 @@ class Indi_TickMt : public IndicatorTick -#include -#include -#include + #include + #include + #include + #include #endif #ifndef __MQL__ -#define __FUNCSIG__ __FUNCTION__ + #define __FUNCSIG__ __FUNCTION__ #endif #ifdef __MQL__ -#define ASSIGN_TO_THIS(TYPE, VALUE) ((TYPE)this) = ((TYPE)VALUE) + #define ASSIGN_TO_THIS(TYPE, VALUE) ((TYPE)this) = ((TYPE)VALUE) #else -#define ASSIGN_TO_THIS(TYPE, VALUE) ((TYPE&)*this) = ((TYPE&)VALUE) + #define ASSIGN_TO_THIS(TYPE, VALUE) ((TYPE&)*this) = ((TYPE&)VALUE) #endif // Pointers. #ifdef __MQL__ -#define GET_PTR(obj) GetPointer(obj) -#define THIS_ATTR -#define THIS_PTR (&this) -#define THIS_REF this -#define PTR_DEREF . -#define PTR_ATTRIB(O, A) O.A -#define PTR_ATTRIB2(O, A, B) O.A.B -#define PTR_TO_REF(PTR) PTR -#define MAKE_REF_FROM_PTR(TYPE, NAME, PTR) TYPE* NAME = PTR -#define REF_DEREF .Ptr(). -#define int64 long + #define GET_PTR(obj) GetPointer(obj) + #define THIS_ATTR + #define THIS_PTR (&this) + #define THIS_REF this + #define PTR_DEREF . + #define PTR_ATTRIB(O, A) O.A + #define PTR_ATTRIB2(O, A, B) O.A.B + #define PTR_TO_REF(PTR) PTR + #define MAKE_REF_FROM_PTR(TYPE, NAME, PTR) TYPE* NAME = PTR + #define REF_DEREF .Ptr(). + #define int64 long #else -#define GET_PTR(obj) (*obj) -#define THIS_ATTR this-> -#define THIS_PTR (this) -#define THIS_REF (*this) -#define PTR_DEREF -> -#define PTR_ATTRIB(O, A) O->A -#define PTR_ATTRIB2(O, A, B) O->A->B -#define PTR_TO_REF(PTR) (*PTR) -#define MAKE_REF_FROM_PTR(TYPE, NAME, PTR) TYPE& NAME = PTR -#define REF_DEREF .Ptr()-> -#define int64 long long + #define GET_PTR(obj) (*obj) + #define THIS_ATTR this-> + #define THIS_PTR (this) + #define THIS_REF (*this) + #define PTR_DEREF -> + #define PTR_ATTRIB(O, A) O->A + #define PTR_ATTRIB2(O, A, B) O->A->B + #define PTR_TO_REF(PTR) (*PTR) + #define MAKE_REF_FROM_PTR(TYPE, NAME, PTR) TYPE& NAME = PTR + #define REF_DEREF .Ptr()-> + #define int64 long long #endif // References. #ifdef __cplusplus -#define REF(X) (&X) + #define REF(X) (&X) #else -#define REF(X) X& + #define REF(X) X& #endif // Arrays and references to arrays. #define _COMMA , #ifdef __MQL__ -#define ARRAY_DECLARATION_BRACKETS [] + #define ARRAY_DECLARATION_BRACKETS [] #else -// C++'s _cpp_array is an object, so no brackets are needed. -#define ARRAY_DECLARATION_BRACKETS + // C++'s _cpp_array is an object, so no brackets are needed. + #define ARRAY_DECLARATION_BRACKETS #endif #ifdef __MQL__ -/** - * Reference to object. - */ -#define CONST_REF_TO(T) const T + /** + * Reference to object. + */ + #define CONST_REF_TO(T) const T -/** - * Reference to the array. - * - * @usage - * ARRAY_REF(, ) - */ -#define ARRAY_TYPE(T) T[] -#define ARRAY_REF(T, N) REF(T) N ARRAY_DECLARATION_BRACKETS -#define FIXED_ARRAY_REF(T, N, S) ARRAY_REF(T, N) + /** + * Reference to the array. + * + * @usage + * ARRAY_REF(, ) + */ + #define ARRAY_TYPE(T) T[] + #define ARRAY_REF(T, N) REF(T) N ARRAY_DECLARATION_BRACKETS + #define FIXED_ARRAY_REF(T, N, S) ARRAY_REF(T, N) -#define CONST_ARRAY_REF(T, N) const N ARRAY_DECLARATION_BRACKETS + #define CONST_ARRAY_REF(T, N) const N ARRAY_DECLARATION_BRACKETS -/** - * Array definition. - * - * @usage - * ARRAY(, ) - */ -#define ARRAY(T, N) T N[] + /** + * Array definition. + * + * @usage + * ARRAY(, ) + */ + #define ARRAY(T, N) T N[] #else -/** - * Reference to object. - */ -#define CONST_REF_TO(T) const T& + /** + * Reference to object. + */ + #define CONST_REF_TO(T) const T& -/** + /** - * Reference to the array. - * - * @usage - * ARRAY_REF(, ) - */ -#define ARRAY_TYPE(T) _cpp_array -#define ARRAY_REF(T, N) ARRAY_TYPE(T)& N -#define FIXED_ARRAY_REF(T, N, S) T(&N)[S] + * Reference to the array. + * + * @usage + * ARRAY_REF(, ) + */ + #define ARRAY_TYPE(T) _cpp_array + #define ARRAY_REF(T, N) ARRAY_TYPE(T) & N + #define FIXED_ARRAY_REF(T, N, S) T(&N)[S] -#define CONST_ARRAY_REF(T, N) const _cpp_array& N + #define CONST_ARRAY_REF(T, N) const _cpp_array& N -/** - * Array definition. - * - * @usage - * ARRAY(, ) - */ -#define ARRAY(T, N) ::_cpp_array N + /** + * Array definition. + * + * @usage + * ARRAY(, ) + */ + #define ARRAY(T, N) ::_cpp_array N #endif // typename(T) #ifndef __MQL__ -#define typename(T) typeid(T).name() + #define typename(T) typeid(T).name() #endif // C++ array class. @@ -235,19 +235,19 @@ class _cpp_array { void setIsSeries(bool _isSeries) { m_isSeries = _isSeries; } }; -#ifdef EMSCRIPTEN -#include + #ifdef EMSCRIPTEN + #include -#define REGISTER_ARRAY_OF(N, T, D) \ - EMSCRIPTEN_BINDINGS(N) { \ - emscripten::register_vector(D "CppVector"); \ - emscripten::class_<_cpp_array>(D) \ - .constructor() \ - .function("Push", &_cpp_array::push) \ - .function("Size", &_cpp_array::size); \ - } + #define REGISTER_ARRAY_OF(N, T, D) \ + EMSCRIPTEN_BINDINGS(N) { \ + emscripten::register_vector(D "CppVector"); \ + emscripten::class_<_cpp_array>(D) \ + .constructor() \ + .function("Push", &_cpp_array::push) \ + .function("Size", &_cpp_array::size); \ + } -#endif + #endif template class _cpp_array; @@ -270,14 +270,14 @@ class color { // MQL defines. #ifndef __MQL__ -#define WHOLE_ARRAY -1 // For processing the entire array. + #define WHOLE_ARRAY -1 // For processing the entire array. #endif // Converts string into C++-style string pointer. #ifdef __MQL__ -#define C_STR(S) S + #define C_STR(S) S #else -#define C_STR(S) cstring_from(S) + #define C_STR(S) cstring_from(S) inline const char* cstring_from(const std::string& _value) { return _value.c_str(); } #endif @@ -295,19 +295,19 @@ inline bool IsNull(const string& str) { return str == ""; } * STRUCT_ENUM(, ) */ #ifdef __MQL4__ -#define STRUCT_ENUM(S, E) E + #define STRUCT_ENUM(S, E) E #else -#define STRUCT_ENUM(S, E) S::E + #define STRUCT_ENUM(S, E) S::E #endif #ifndef __MQL__ -// Additional enum values for ENUM_SYMBOL_INFO_DOUBLE -#define SYMBOL_MARGIN_LIMIT ((ENUM_SYMBOL_INFO_DOUBLE)46) -#define SYMBOL_MARGIN_MAINTENANCE ((ENUM_SYMBOL_INFO_DOUBLE)43) -#define SYMBOL_MARGIN_LONG ((ENUM_SYMBOL_INFO_DOUBLE)44) -#define SYMBOL_MARGIN_SHORT ((ENUM_SYMBOL_INFO_DOUBLE)45) -#define SYMBOL_MARGIN_STOP ((ENUM_SYMBOL_INFO_DOUBLE)47) -#define SYMBOL_MARGIN_STOPLIMIT ((ENUM_SYMBOL_INFO_DOUBLE)48) + // Additional enum values for ENUM_SYMBOL_INFO_DOUBLE + #define SYMBOL_MARGIN_LIMIT ((ENUM_SYMBOL_INFO_DOUBLE)46) + #define SYMBOL_MARGIN_MAINTENANCE ((ENUM_SYMBOL_INFO_DOUBLE)43) + #define SYMBOL_MARGIN_LONG ((ENUM_SYMBOL_INFO_DOUBLE)44) + #define SYMBOL_MARGIN_SHORT ((ENUM_SYMBOL_INFO_DOUBLE)45) + #define SYMBOL_MARGIN_STOP ((ENUM_SYMBOL_INFO_DOUBLE)47) + #define SYMBOL_MARGIN_STOPLIMIT ((ENUM_SYMBOL_INFO_DOUBLE)48) #endif template @@ -360,14 +360,14 @@ template <> _NULL_VALUE::operator string() const { return _empty_string; } -#define NULL_STRING "" + #define NULL_STRING "" #else -#define NULL_VALUE NULL -#define NULL_STRING NULL + #define NULL_VALUE NULL + #define NULL_STRING NULL #endif #ifndef __MQL__ -#include "Chart.enum.h" + #include "Chart.enum.h" /** * Returns currently selected period for platform. */ @@ -389,9 +389,9 @@ extern ENUM_TIMEFRAMES Period(); #define SET_BUFFER_AS_SERIES_FOR_TARGET(A) ArraySetAsSeries(A, false); #ifdef __MQL4__ -#define SET_BUFFER_AS_SERIES_FOR_HOST(A) ArraySetAsSeries(A, true); + #define SET_BUFFER_AS_SERIES_FOR_HOST(A) ArraySetAsSeries(A, true); #else -#define SET_BUFFER_AS_SERIES_FOR_HOST(A) ArraySetAsSeries(A, false); + #define SET_BUFFER_AS_SERIES_FOR_HOST(A) ArraySetAsSeries(A, false); #endif // Ensures that we do RELEASE_BUFFERx after ACQUIRE_BUFFERx. @@ -427,109 +427,141 @@ struct AsSeriesReleaseEnsurer { #define SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(NUM_BUFFS) _as_series_release_ensurer.done(NUM_BUFFS); // Acquiring buffer is preparing it to be used as in MQL5. -#define ACQUIRE_BUFFER1(A) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ +#define ACQUIRE_BUFFER1_NO_ENSURE(A) SET_BUFFER_AS_SERIES_FOR_TARGET(A); +#define ACQUIRE_BUFFER2_NO_ENSURE(A, B) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); +#define ACQUIRE_BUFFER3_NO_ENSURE(A, B, C) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(C); +#define ACQUIRE_BUFFER4_NO_ENSURE(A, B, C, D) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(D); +#define ACQUIRE_BUFFER5_NO_ENSURE(A, B, C, D, E) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(E); +#define ACQUIRE_BUFFER6_NO_ENSURE(A, B, C, D, E, F) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(F); +#define ACQUIRE_BUFFER7_NO_ENSURE(A, B, C, D, E, F, G) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(F); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(G); +#define ACQUIRE_BUFFER8_NO_ENSURE(A, B, C, D, E, F, G, H) \ + SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(F); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(G); \ + SET_BUFFER_AS_SERIES_FOR_TARGET(H); + +#define ACQUIRE_BUFFER1(A) \ + ACQUIRE_BUFFER1_NO_ENSURE(A); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(1); -#define ACQUIRE_BUFFER2(A, B) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ +#define ACQUIRE_BUFFER2(A, B) \ + ACQUIRE_BUFFER2_NO_ENSURE(A, B); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(2); #define ACQUIRE_BUFFER3(A, B, C) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ + ACQUIRE_BUFFER3_NO_ENSURE(A, B, C); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(3); -#define ACQUIRE_BUFFER4(A, B, C, D) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ +#define ACQUIRE_BUFFER4(A, B, C, D) \ + ACQUIRE_BUFFER4_NO_ENSURE(A, B, C, D); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(4); -#define ACQUIRE_BUFFER5(A, B, C, D, E) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ +#define ACQUIRE_BUFFER5(A, B, C, D, E) \ + ACQUIRE_BUFFER5_NO_ENSURE(A, B, C, D, E); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(5); -#define ACQUIRE_BUFFER6(A, B, C, D, E, F) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(F); \ +#define ACQUIRE_BUFFER6(A, B, C, D, E, F) \ + ACQUIRE_BUFFER6_NO_ENSURE(A, B, C, D, E, F); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(6); -#define ACQUIRE_BUFFER7(A, B, C, D, E, F, G) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(F); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(G); \ +#define ACQUIRE_BUFFER7(A, B, C, D, E, F, G) \ + ACQUIRE_BUFFER7_NO_ENSURE(A, B, C, D, E, F, G); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(7); -#define ACQUIRE_BUFFER8(A, B, C, D, E, F, G, H) \ - SET_BUFFER_AS_SERIES_FOR_TARGET(A); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(B); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(C); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(D); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(E); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(F); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(G); \ - SET_BUFFER_AS_SERIES_FOR_TARGET(H); \ +#define ACQUIRE_BUFFER8(A, B, C, D, E, F, G, H) \ + ACQUIRE_BUFFER8_NO_ENSURE(A, B, C, D, E, F, G, H); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_BEGIN(8); // Releasing buffer is setting its AsSeries as the default in the host language. -#define RELEASE_BUFFER1(A) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ +#define RELEASE_BUFFER1_NO_ENSURE(A) SET_BUFFER_AS_SERIES_FOR_HOST(A) +#define RELEASE_BUFFER2_NO_ENSURE(A, B) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); +#define RELEASE_BUFFER3_NO_ENSURE(A, B, C) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); \ + SET_BUFFER_AS_SERIES_FOR_HOST(C); +#define RELEASE_BUFFER4_NO_ENSURE(A, B, C, D) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); \ + SET_BUFFER_AS_SERIES_FOR_HOST(C); \ + SET_BUFFER_AS_SERIES_FOR_HOST(D); +#define RELEASE_BUFFER5_NO_ENSURE(A, B, C, D, E) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); \ + SET_BUFFER_AS_SERIES_FOR_HOST(C); \ + SET_BUFFER_AS_SERIES_FOR_HOST(D); \ + SET_BUFFER_AS_SERIES_FOR_HOST(E); +#define RELEASE_BUFFER6_NO_ENSURE(A, B, C, D, E, F) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); \ + SET_BUFFER_AS_SERIES_FOR_HOST(C); \ + SET_BUFFER_AS_SERIES_FOR_HOST(D); \ + SET_BUFFER_AS_SERIES_FOR_HOST(E); \ + SET_BUFFER_AS_SERIES_FOR_HOST(F); +#define RELEASE_BUFFER7_NO_ENSURE(A, B, C, D, E, F, G) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); \ + SET_BUFFER_AS_SERIES_FOR_HOST(C); \ + SET_BUFFER_AS_SERIES_FOR_HOST(D); \ + SET_BUFFER_AS_SERIES_FOR_HOST(E); \ + SET_BUFFER_AS_SERIES_FOR_HOST(F); \ + SET_BUFFER_AS_SERIES_FOR_HOST(G); +#define RELEASE_BUFFER8_NO_ENSURE(A, B, C, D, E, F, G, H) \ + SET_BUFFER_AS_SERIES_FOR_HOST(A); \ + SET_BUFFER_AS_SERIES_FOR_HOST(B); \ + SET_BUFFER_AS_SERIES_FOR_HOST(C); \ + SET_BUFFER_AS_SERIES_FOR_HOST(D); \ + SET_BUFFER_AS_SERIES_FOR_HOST(E); \ + SET_BUFFER_AS_SERIES_FOR_HOST(F); \ + SET_BUFFER_AS_SERIES_FOR_HOST(G); \ + SET_BUFFER_AS_SERIES_FOR_HOST(H); + +#define RELEASE_BUFFER1(A) \ + RELEASE_BUFFER1_NO_ENSURE(A); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(1); -#define RELEASE_BUFFER2(A, B) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ +#define RELEASE_BUFFER2(A, B) \ + RELEASE_BUFFER2_NO_ENSURE(A, B); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(2); -#define RELEASE_BUFFER3(A, B, C) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ - SET_BUFFER_AS_SERIES_FOR_HOST(C); \ +#define RELEASE_BUFFER3(A, B, C) \ + RELEASE_BUFFER3_NO_ENSURE(A, B, C); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(3); -#define RELEASE_BUFFER4(A, B, C, D) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ - SET_BUFFER_AS_SERIES_FOR_HOST(C); \ - SET_BUFFER_AS_SERIES_FOR_HOST(D); \ +#define RELEASE_BUFFER4(A, B, C, D) \ + RELEASE_BUFFER4_NO_ENSURE(A, B, C, D); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(4); -#define RELEASE_BUFFER5(A, B, C, D, E) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ - SET_BUFFER_AS_SERIES_FOR_HOST(C); \ - SET_BUFFER_AS_SERIES_FOR_HOST(D); \ - SET_BUFFER_AS_SERIES_FOR_HOST(E); \ +#define RELEASE_BUFFER5(A, B, C, D, E) \ + RELEASE_BUFFER5_NO_ENSURE(A, B, C, D, E); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(5); -#define RELEASE_BUFFER6(A, B, C, D, E, F) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ - SET_BUFFER_AS_SERIES_FOR_HOST(C); \ - SET_BUFFER_AS_SERIES_FOR_HOST(D); \ - SET_BUFFER_AS_SERIES_FOR_HOST(E); \ - SET_BUFFER_AS_SERIES_FOR_HOST(F); \ +#define RELEASE_BUFFER6(A, B, C, D, E, F) \ + RELEASE_BUFFER6_NO_ENSURE(A, B, C, D, E, F); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(6); -#define RELEASE_BUFFER7(A, B, C, D, E, F, G) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ - SET_BUFFER_AS_SERIES_FOR_HOST(C); \ - SET_BUFFER_AS_SERIES_FOR_HOST(D); \ - SET_BUFFER_AS_SERIES_FOR_HOST(E); \ - SET_BUFFER_AS_SERIES_FOR_HOST(F); \ - SET_BUFFER_AS_SERIES_FOR_HOST(G); \ +#define RELEASE_BUFFER7(A, B, C, D, E, F, G) \ + RELEASE_BUFFER7_NO_ENSURE(A, B, C, D, E, F, G); \ SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(7); -#define RELEASE_BUFFER8(A, B, C, D, E, F, G, H) \ - SET_BUFFER_AS_SERIES_FOR_HOST(A); \ - SET_BUFFER_AS_SERIES_FOR_HOST(B); \ - SET_BUFFER_AS_SERIES_FOR_HOST(C); \ - SET_BUFFER_AS_SERIES_FOR_HOST(D); \ - SET_BUFFER_AS_SERIES_FOR_HOST(E); \ - SET_BUFFER_AS_SERIES_FOR_HOST(F); \ - SET_BUFFER_AS_SERIES_FOR_HOST(G); \ - SET_BUFFER_AS_SERIES_FOR_HOST(H); \ - SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(8); +#define RELEASE_BUFFER8(A, B, C, D, E, F, G, H) \ + RELEASE_BUFFER8_NO_ENSURE(A, B, C, D, E, F, G, H); \ + SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(8); \ No newline at end of file From 92ee47a4326e7389bd01cff2a74825f13c73532f Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Wed, 22 May 2024 19:39:38 +0200 Subject: [PATCH 3/7] WIP. Fixes In ItemsHistory and ItemsHistoryTfCandleProvider for Candle indicator and fixes in Indi_TickMt. Still requires work in the area where ticks are used to form candles. --- Candle.struct.h | 19 ++- Dict.enum.h | 8 +- DictBase.mqh | 39 ++++- DictStruct.mqh | 6 +- Indicator/IndicatorCandle.h | 25 +++- Indicator/IndicatorData.h | 32 +++- Indicator/IndicatorRenko.h | 148 ++++++++++--------- Indicator/IndicatorTf.provider.h | 18 ++- Indicator/tests/classes/IndicatorTfDummy.h | 14 +- Indicators/Account/Indi_AccountStats.mqh | 7 +- Indicators/Tick/Indi_TickMt.mqh | 162 ++++++++++++++------- Platform.h | 10 +- Storage/ItemsHistory.h | 18 ++- 13 files changed, 338 insertions(+), 168 deletions(-) diff --git a/Candle.struct.h b/Candle.struct.h index ebbe5b3ce..17b6f287d 100644 --- a/Candle.struct.h +++ b/Candle.struct.h @@ -26,8 +26,8 @@ */ #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Forward class declaration. @@ -252,9 +252,12 @@ struct CandleOCTOHLC : CandleOHLC { } /** - * Initializes candle with a given start time, lenght in seconds, first tick's timestamp and its price. + * Initializes candle with a given start time, length in seconds, first tick's timestamp and its price. */ void Init(int _start_time, int _length, long _timestamp_ms = -1, T _price = 0) { + if (_start_time < 0) { + Print("Error!"); + } is_complete = false; start_time = _start_time; length = _length; @@ -269,7 +272,14 @@ struct CandleOCTOHLC : CandleOHLC { */ void Update(long _timestamp_ms, T _price) { if (!ContainsTimeMs(_timestamp_ms)) { - Print("Error: Cannot update candle. Given time doesn't fit in candle's time-frame!"); + Print("Error: Cannot update candle. Given time doesn't fit in candle's time-frame! Given time ", _timestamp_ms, + ", but candle range is ", (long)start_time * 1000, " - ", (long)(start_time + length) * 1000, "."); + if (_timestamp_ms < (long)start_time * 1000) { + Print("Looks like given time is ", (long)start_time * 1000 - _timestamp_ms, " ms before the candle starts."); + } else { + Print("Looks like given time is ", _timestamp_ms - (long)(start_time + length) * 1000, + " ms after the candle ends."); + } DebugBreak(); } @@ -277,6 +287,7 @@ struct CandleOCTOHLC : CandleOHLC { if (_is_init || _timestamp_ms < open_timestamp_ms) { open_timestamp_ms = _timestamp_ms; + start_time = int(_timestamp_ms / 1000); open = _price; } if (_is_init || _timestamp_ms > close_timestamp_ms) { diff --git a/Dict.enum.h b/Dict.enum.h index 99474def2..99618ba5b 100644 --- a/Dict.enum.h +++ b/Dict.enum.h @@ -26,12 +26,12 @@ */ #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif -#define DICT_GROW_UP_PERCENT_DEFAULT 25 -#define DICT_PERFORMANCE_PROBLEM_AVG_CONFLICTS 10 +#define DICT_GROW_UP_PERCENT_DEFAULT 50 +#define DICT_PERFORMANCE_PROBLEM_AVG_CONFLICTS 20 /** * Whether Dict operates in yet uknown mode, as dict or as list. diff --git a/DictBase.mqh b/DictBase.mqh index 269f9ddd2..9e3bbd95b 100644 --- a/DictBase.mqh +++ b/DictBase.mqh @@ -361,17 +361,50 @@ class DictBase { /** * Specialization of hashing function. */ - unsigned int Hash(unsigned int x) { return x; } + unsigned int Hash(float x) { return (unsigned int)((unsigned long)x * 10000 % 10000); } /** * Specialization of hashing function. */ - unsigned int Hash(int x) { return (unsigned int)x; } + unsigned int Hash(int value) { + value ^= (value >> 8); + value ^= (value << 3); + value ^= (value >> 9); + value ^= (value >> 4); + value ^= (value << 6); + value ^= (value >> 14); + return value; + } /** * Specialization of hashing function. */ - unsigned int Hash(float x) { return (unsigned int)((unsigned long)x * 10000 % 10000); } + unsigned int Hash(unsigned int value) { return Hash((int)value); } + + /** + * Specialization of hashing function. + */ + unsigned int Hash(long value) { + value ^= (value >> 33); + value ^= (value << 21); + value ^= (value >> 17); + + // Step 2: Combine upper and lower 32 bits to form a 32-bit hash + long hash = (int)(value ^ (value >> 32)); + + // Step 3: Further bit manipulation to spread the bits + hash ^= (hash >> 16); + hash *= 0x85ebca6b; // A large prime number + hash ^= (hash >> 13); + hash *= 0xc2b2ae35; // Another large prime number + hash ^= (hash >> 16); + return int(value >> 32); + } + + /** + * Specialization of hashing function. + */ + unsigned int Hash(unsigned long value) { return Hash((unsigned long)value); } }; #endif diff --git a/DictStruct.mqh b/DictStruct.mqh index b7bb1a173..97ee6ea1f 100644 --- a/DictStruct.mqh +++ b/DictStruct.mqh @@ -22,7 +22,7 @@ // Prevents processing this includes file for the second time. #ifndef __MQL__ -#pragma once + #pragma once #endif // Includes. @@ -433,7 +433,9 @@ class DictStruct : public DictBase { DictSlotsRef new_DictSlots; - if (ArrayResize(new_DictSlots.DictSlots, new_size) == -1) return false; + if (ArrayResize(new_DictSlots.DictSlots, new_size) == -1) { + return false; + } int i; diff --git a/Indicator/IndicatorCandle.h b/Indicator/IndicatorCandle.h index f8b7fdd0c..4efb3e3e1 100644 --- a/Indicator/IndicatorCandle.h +++ b/Indicator/IndicatorCandle.h @@ -25,8 +25,8 @@ #define INDICATOR_CANDLE_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -46,7 +46,7 @@ #include "TickBarCounter.h" #ifndef INDI_CANDLE_HISTORY_SIZE -#define INDI_CANDLE_HISTORY_SIZE 86400 + #define INDI_CANDLE_HISTORY_SIZE 86400 #endif // Indicator modes. @@ -368,7 +368,14 @@ class IndicatorCandle : public Indicator { /** * Called when data source emits new entry (new one in ascending order). */ - void OnDataSourceEntry(IndicatorDataEntry& entry) override { + void OnDataSourceEntry(IndicatorDataEntry& entry, + ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT) override { + Indicator::OnDataSourceEntry(entry, type); + + if (type != INDI_EMITTED_ENTRY_TYPE_TICK) { + return; + } + // Parent indicator (e.g., Indi_TickMt) emitted an entry containing tick's // ask and bid price. As an abstract class, we really don't know how to // update/create candles so we just pass the entry into history's @@ -376,6 +383,16 @@ class IndicatorCandle : public Indicator { history.GetItemProvider() PTR_DEREF OnTick(&history, entry.timestamp * 1000, (float)entry[0], (float)entry[1]); }; + /** + * Called when data source expects to emit given number of entries for given type. + * + * Called e.g., from Tick indicator in order Candle indicator to enlarge + * possible history size by given number of entries. We have to do that, + * because otherwise, we could end up with OnCalculate() working on partial + * history candles. + */ + void OnDataSourceWillEmitEntries(ENUM_INDI_EMITTED_ENTRY_TYPE _type, int _num_entries) override {} + /** * Returns value storage of given kind. */ diff --git a/Indicator/IndicatorData.h b/Indicator/IndicatorData.h index 29fde9ff6..18e560613 100644 --- a/Indicator/IndicatorData.h +++ b/Indicator/IndicatorData.h @@ -25,8 +25,8 @@ #define INDICATOR_DATA_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Forward class declaration. @@ -1757,10 +1757,21 @@ class IndicatorData : public IndicatorBase { /** * Sends entry to listening indicators. */ - void EmitEntry(IndicatorDataEntry& entry) { + void EmitEntry(IndicatorDataEntry& _entry, ENUM_INDI_EMITTED_ENTRY_TYPE _type = INDI_EMITTED_ENTRY_TYPE_PARENT) { for (int i = 0; i < ArraySize(listeners); ++i) { if (listeners[i].ObjectExists()) { - listeners[i].Ptr().OnDataSourceEntry(entry); + listeners[i].Ptr().OnDataSourceEntry(_entry, _type); + } + } + } + + /** + * Sends information about expected number of emitted entries to listening indicators. + */ + void WillEmitEntries(ENUM_INDI_EMITTED_ENTRY_TYPE _type, int _num_entries) { + for (int i = 0; i < ArraySize(listeners); ++i) { + if (listeners[i].ObjectExists()) { + listeners[i].Ptr().OnDataSourceWillEmitEntries(_type, _num_entries); } } } @@ -1824,7 +1835,18 @@ class IndicatorData : public IndicatorBase { /** * Called when data source emits new entry (historic or future one). */ - virtual void OnDataSourceEntry(IndicatorDataEntry& entry){}; + virtual void OnDataSourceEntry(IndicatorDataEntry& entry, + ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT) {} + + /** + * Called when data source expects to emit given number of entries for given type. + * + * Called e.g., from Tick indicator in order Candle indicator to enlarge + * possible history size by given number of entries. We have to do that, + * because otherwise, we could end up with OnCalculate() working on partial + * history candles. + */ + virtual void OnDataSourceWillEmitEntries(ENUM_INDI_EMITTED_ENTRY_TYPE _type, int _num_entries) {} /** * Called when new tick is retrieved from attached data source. diff --git a/Indicator/IndicatorRenko.h b/Indicator/IndicatorRenko.h index df4a403da..be9888e3d 100644 --- a/Indicator/IndicatorRenko.h +++ b/Indicator/IndicatorRenko.h @@ -30,8 +30,8 @@ #define INDICATOR_RENKO_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -171,97 +171,103 @@ class IndicatorRenko : public IndicatorCandle>::OnDataSourceEntry(entry, type); - if (entry.timestamp < last_entry_ts) { - Print("Error: IndicatorRenko doesn't support sending entries in non-ascending order!"); - DebugBreak(); - } + if (type != INDI_EMITTED_ENTRY_TYPE_TICK) { + return; + } + /* + @todo Move logic into ItemsHistoryRenkoCandleProvider class. - // We'll be updating candle from bid price. - double _price = entry[1]; + if (entry.timestamp < last_entry_ts) { + Print("Error: IndicatorRenko doesn't support sending entries in non-ascending order!"); + DebugBreak(); + } - CandleOCTOHLC _candle; - CandleOCTOHLC _last_completed_candle; - ENUM_INDI_RENKO_CANDLE_TYPE _last_completed_candle_type; + // We'll be updating candle from bid price. + double _price = entry[1]; - if (last_completed_candle_ts != 0) { - _last_completed_candle = icdata.GetByKey(last_completed_candle_ts); - _last_completed_candle_type = GetCandleType(_last_completed_candle); - } else { - _last_completed_candle_type = INDI_RENKO_CANDLE_TYPE_NONE; - } + CandleOCTOHLC _candle; + CandleOCTOHLC _last_completed_candle; + ENUM_INDI_RENKO_CANDLE_TYPE _last_completed_candle_type; - if (last_incomplete_candle_ts != 0) { - // There is previous candle. Retrieving and updating it. - _candle = icdata.GetByKey(last_incomplete_candle_ts); - _candle.Update(entry.timestamp, _price); + if (last_completed_candle_ts != 0) { + _last_completed_candle = icdata.GetByKey(last_completed_candle_ts); + _last_completed_candle_type = GetCandleType(_last_completed_candle); + } else { + _last_completed_candle_type = INDI_RENKO_CANDLE_TYPE_NONE; + } - // Checking for close price difference. - if (RenkoConditionMet(_last_completed_candle_type, _candle, _price)) { - // Closing current candle. - _candle.is_complete = true; - } + if (last_incomplete_candle_ts != 0) { + // There is previous candle. Retrieving and updating it. + _candle = icdata.GetByKey(last_incomplete_candle_ts); + _candle.Update(entry.timestamp, _price); + + // Checking for close price difference. + if (RenkoConditionMet(_last_completed_candle_type, _candle, _price)) { + // Closing current candle. + _candle.is_complete = true; + } - // Updating candle. - icdata.Add(_candle, last_incomplete_candle_ts); + // Updating candle. + icdata.Add(_candle, last_incomplete_candle_ts); - Print("Updated Candle: ", _candle.ToString()); + Print("Updated Candle: ", _candle.ToString()); - if (_candle.is_complete) { - last_completed_candle_ts = last_incomplete_candle_ts; - last_incomplete_candle_ts = 0; - } + if (_candle.is_complete) { + last_completed_candle_ts = last_incomplete_candle_ts; + last_incomplete_candle_ts = 0; + } + } else { + // There is no incomplete candle, creating one. + if (last_completed_candle_ts != 0) { + // Price of the last candle will be used to initialize open price for new, incomplete candle. + double _last_close_price = _last_completed_candle.close; + _candle = CandleOCTOHLC(_last_close_price, _last_close_price, _last_close_price, _last_close_price, + entry.timestamp, entry.timestamp); + // Current price will be added to newly created incomplete candle. + _candle.Update(entry.timestamp, _price); } else { - // There is no incomplete candle, creating one. - if (last_completed_candle_ts != 0) { - // Price of the last candle will be used to initialize open price for new, incomplete candle. - double _last_close_price = _last_completed_candle.close; - _candle = CandleOCTOHLC(_last_close_price, _last_close_price, _last_close_price, _last_close_price, - entry.timestamp, entry.timestamp); - // Current price will be added to newly created incomplete candle. - _candle.Update(entry.timestamp, _price); - } else { - // There was no completed candle. Creating new, incomplete candle from current price. - _candle = CandleOCTOHLC(_price, _price, _price, _price, entry.timestamp, entry.timestamp); - } + // There was no completed candle. Creating new, incomplete candle from current price. + _candle = CandleOCTOHLC(_price, _price, _price, _price, entry.timestamp, entry.timestamp); + } - _candle.is_complete = false; + _candle.is_complete = false; - // Creating new candle. - icdata.Add(_candle, entry.timestamp); + // Creating new candle. + icdata.Add(_candle, entry.timestamp); - Print("Added candle: ", _candle.ToString(), " now there is ", icdata.Size(), " candles in the buffer."); + Print("Added candle: ", _candle.ToString(), " now there is ", icdata.Size(), " candles in the buffer."); - last_incomplete_candle_ts = entry.timestamp; - } + last_incomplete_candle_ts = entry.timestamp; + } - static int iteration = 0; + static int iteration = 0; - ++iteration; + ++iteration; - Print("Iteration: ", iteration); + Print("Iteration: ", iteration); - if (iteration > 1793) { - // Print(icdata.ToJSON()); - } + if (iteration > 1793) { + // Print(icdata.ToJSON()); + } - Print("Last Incomplete Time: ", TimeToString(last_incomplete_candle_ts, TIME_DATE | TIME_MINUTES | - TIME_SECONDS), " (", last_incomplete_candle_ts, ")"); Print("Last Incomplete Candle: ", - icdata.GetByKey(last_incomplete_candle_ts).ToString()); Print("Last Completed Time: ", - TimeToString(last_completed_candle_ts, TIME_DATE | TIME_MINUTES | TIME_SECONDS), " (", last_completed_candle_ts, - ")"); Print("Last Completed Candle: ", icdata.GetByKey(last_completed_candle_ts).ToString()); + Print("Last Incomplete Time: ", TimeToString(last_incomplete_candle_ts, TIME_DATE | TIME_MINUTES | + TIME_SECONDS), " (", last_incomplete_candle_ts, ")"); Print("Last Incomplete Candle: ", + icdata.GetByKey(last_incomplete_candle_ts).ToString()); Print("Last Completed Time: ", + TimeToString(last_completed_candle_ts, TIME_DATE | TIME_MINUTES | TIME_SECONDS), " (", last_completed_candle_ts, + ")"); Print("Last Completed Candle: ", icdata.GetByKey(last_completed_candle_ts).ToString()); - // Updating tick & bar indices. Bar time is time of the last completed candle. - // Print(last_completed_candle_ts); - counter.OnTick(last_completed_candle_ts); + // Updating tick & bar indices. Bar time is time of the last completed candle. + // Print(last_completed_candle_ts); + counter.OnTick(last_completed_candle_ts); - Print("---------"); + Print("---------"); - last_entry_ts = entry.timestamp; - */ + last_entry_ts = entry.timestamp; + */ }; /** diff --git a/Indicator/IndicatorTf.provider.h b/Indicator/IndicatorTf.provider.h index 3318b58da..bb7e75928 100644 --- a/Indicator/IndicatorTf.provider.h +++ b/Indicator/IndicatorTf.provider.h @@ -125,19 +125,33 @@ class ItemsHistoryTfCandleProvider : public ItemsHistoryCandleProvider { // In order to (re)generate candle, we need to fetch ticks within a fixed time-frame and then update that candle // with all fetched ticks. For IndicatorTf, there is no difference between (re)generating backwards or forwards, // as candles are time-framed. The only problem is that there may be missing candles in fetched time-frames. We - // just need to skip such time-frame and fetch ticks for next time-frame. In order to determine + // just need to skip such time-frame and fetch ticks for next time-frame. IndicatorData* _indi_tick = indi PTR_DEREF GetTick(); // Ticks to form a candle. static ARRAY(TickTAB, _ticks); + if (_dir == ITEMS_HISTORY_DIRECTION_BACKWARD) { + // For backward direction we need to start from the beginning of the candle as _from_time_ms determines it's + // ending. + //_ticks_to_ms = _ticks_from_ms + (_candle_length_ms - 1); + } else { + //_ticks_to_ms = _ticks_from_ms - (_candle_length_ms - 1); + } + while (_num_items > 0) { // Calculating time from which and to which we want to retrieve ticks to form a candle. int _ticks_from_s = GetCandleTimeFromTimeMs(_from_time_ms, spc); long _ticks_from_ms = (long)_ticks_from_s * 1000; long _candle_length_ms = (long)spc * 1000; - long _ticks_to_ms = _ticks_from_ms + _candle_length_ms - 1; + long _ticks_to_ms; + + if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { + _ticks_to_ms = _ticks_from_ms + (_candle_length_ms - 1); + } else { + _ticks_to_ms = _ticks_from_ms - (_candle_length_ms - 1); + } // We will try to fetch history by two methods. // 1. By time range if IndicatorTick supports that way. diff --git a/Indicator/tests/classes/IndicatorTfDummy.h b/Indicator/tests/classes/IndicatorTfDummy.h index 4b2a73c03..a292e6b08 100644 --- a/Indicator/tests/classes/IndicatorTfDummy.h +++ b/Indicator/tests/classes/IndicatorTfDummy.h @@ -25,8 +25,8 @@ */ #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -49,11 +49,17 @@ class IndicatorTfDummy : public IndicatorTf { string GetName() override { return "IndicatorTfDummy(" + IntegerToString(iparams.spc) + ")"; } - void OnDataSourceEntry(IndicatorDataEntry& entry) override { + void OnDataSourceEntry(IndicatorDataEntry& entry, + ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT) override { + IndicatorTf::OnDataSourceEntry(entry, type); + + if (type != INDI_EMITTED_ENTRY_TYPE_TICK) { + return; + } + // When overriding OnDataSourceEntry() we have to remember to call parent // method, because IndicatorCandle also need to invoke it in order to // create/update matching candle. - IndicatorTf::OnDataSourceEntry(entry); #ifdef __debug_indicator__ Print(GetFullName(), " got new tick at ", entry.timestamp, diff --git a/Indicators/Account/Indi_AccountStats.mqh b/Indicators/Account/Indi_AccountStats.mqh index ece3896dc..367fd1866 100644 --- a/Indicators/Account/Indi_AccountStats.mqh +++ b/Indicators/Account/Indi_AccountStats.mqh @@ -191,14 +191,13 @@ class Indi_AccountStats : public Indicator { /** * Called when data source emits new entry (historic or future one). */ - virtual void OnDataSourceEntry(IndicatorDataEntry &entry/*, ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT*/) { - /* - Indicator::OnDataSourceEntry(entry, type); + virtual void OnDataSourceEntry(IndicatorDataEntry &entry, + ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT) override { + Indicator::OnDataSourceEntry(entry, type); if (type != INDI_EMITTED_ENTRY_TYPE_CANDLE) { return; } - */ // New candle means that account stats for current index 0 will be that we // will now extract and store in the buffers. diff --git a/Indicators/Tick/Indi_TickMt.mqh b/Indicators/Tick/Indi_TickMt.mqh index 320c3f298..6001716ce 100644 --- a/Indicators/Tick/Indi_TickMt.mqh +++ b/Indicators/Tick/Indi_TickMt.mqh @@ -127,60 +127,11 @@ class Indi_TickMt : public IndicatorTick, _out_ticks)) { + bool FetchHistoryByTimeRange(long _from_ms, long _to_ms, ARRAY_REF(TickTAB, _out_ticks)) override { ArrayResize(_out_ticks, 0); -#ifdef __MQL4__ - // Searching from current bar to older ones. - int _shift; - - if (_to_ms <= _cache_fetch_history_shift_to_ms) { - _shift = _cache_fetch_history_shift_shift; - } else { - _shift = 0; - } - - string _symbol = GetSymbol(); - - while (true) { - double _time = (double)iTime(_symbol, PERIOD_M1, _shift); - - if (_time == 0) { - // Invalid time. - break; - } - - long _time_ms = (long)_time * 1000; - - if (_time_ms > _to_ms) { - // No yet get into valid time range. - ++_shift; - continue; - } - - if (_time_ms < _from_ms) { - // No more ticks. - break; - } - - TickTAB _tick_o(_time_ms, iOpen(_Symbol, PERIOD_M1, _shift)); - TickTAB _tick_h(_time_ms, iHigh(_Symbol, PERIOD_M1, _shift)); - TickTAB _tick_l(_time_ms, iLow(_Symbol, PERIOD_M1, _shift)); - TickTAB _tick_c(_time_ms, iClose(_Symbol, PERIOD_M1, _shift)); - ArrayPushObject(_out_ticks, _tick_o); - ArrayPushObject(_out_ticks, _tick_h); - ArrayPushObject(_out_ticks, _tick_l); - ArrayPushObject(_out_ticks, _tick_c); - ++_shift; - } - - if (_shift != -1) { - _cache_fetch_history_shift_to_ms = _to_ms; - _cache_fetch_history_shift_shift = _shift; - } - - return ArraySize(_out_ticks) != 0; -#else +#ifdef __MQL5__ + // In MQL5 we firstly try to fetch ticks by CopyTicksRange. static MqlTick _tmp_ticks[]; ArrayResize(_tmp_ticks, 0); @@ -203,13 +154,54 @@ class Indi_TickMt : public IndicatorTick 0) { + return true; + } else { + // No ticks found. We will try to create ticks from historic OHLC values. + break; + } } } + #endif - // To many tries. Probably no ticks at the given range. - return false; + // In MQL4 or MQL5 (if there's no history returned by CopyTicksRange) we + // try to create and return OHLC ticks by calling iOpen/iHigh/iLow/iClose + // for each bar in history. Candle indicator will form candles from those + // 4 ticks per bar. + + string _symbol = GetSymbol(); + ARRAY(MqlRates, _rates); + + if (!CopyRates(_symbol, PERIOD_M1, (datetime)(int)(_from_ms / 1000), (datetime)(int)(_to_ms / 1000), _rates)) { + // Failure. + return false; + } + + if (ArraySize(_rates) == 0) { + // 0 ticks is OK. + return true; + } + + ArrayResize(_out_ticks, ArraySize(_rates) * 4); // Number of ticks * OHLC(4). + + int _tick_idx = 0; + + for (int i = 0; i < ArraySize(_rates); ++i) { + long _time_ms = (long)_rates[i].time * 1000; + // Note that we modify the time in order to simulate real open/close time. + TickTAB _tick_o(_time_ms + 0, _rates[i].open, _rates[i].open); + TickTAB _tick_h(_time_ms + 1, _rates[i].high, _rates[i].high); + TickTAB _tick_l(_time_ms + 2, _rates[i].low, _rates[i].low); + TickTAB _tick_c(_time_ms + 3, _rates[i].close, _rates[i].close); + + _out_ticks[_tick_idx++] = _tick_o; + _out_ticks[_tick_idx++] = _tick_h; + _out_ticks[_tick_idx++] = _tick_l; + _out_ticks[_tick_idx++] = _tick_c; + } + + return ArraySize(_out_ticks) != 0; } /** @@ -222,7 +214,65 @@ class Indi_TickMt : public IndicatorTick, _ticks); + if (!FetchHistoryByTimeRange( + (long)iTime(GetSymbol(), PERIOD_M1, _num_bars) * 1000, + (long)iTime(GetSymbol(), PERIOD_M1, 0) * 1000, + _ticks + )) { + // No history at all. + Print("Indi_TickMt: FetchHistoryByTimeRange() also end up with no history. We will work without history."); + return; + } + + if (ArraySize(_ticks) != _num_bars * 4) { + Alert("Error: Inconsistent number of ticks in the history! There should be as many ticks as 4 times the + number of bars!"); DebugBreak(); + } + + history.ReserveAdditionalHistoryMaxSize(ArraySize(_ticks)); + // @todo Calculate number of emitted candles. + // WillEmitEntries(INDI_EMITTED_ENTRY_TYPE_CANDLE, _num_candles); + for (int i = 0; i < ArraySize(_ticks); ++i) { + TickAB _tick(_ticks[i].ask, _ticks[i].bid); + IndicatorDataEntry _entry(TickToEntry(_ticks[i].time_ms, _tick)); + EmitEntry(_entry, INDI_EMITTED_ENTRY_TYPE_TICK); + // Appending tick into the history. + AppendEntry(_entry); + #ifdef __debug__ + if (i % 10000 == 0 || i == ArraySize(_ticks) - 1) { + Print("Indi_TickMt: EmitHistory(): Done ", i, " / ", ArraySize(_ticks)); + } + #endif + } + } + else { + // We're good as CopyTicks() returned some history. + history.ReserveAdditionalHistoryMaxSize(ArraySize(_tmp_ticks)); + for (int i = 0; i < ArraySize(_tmp_ticks); ++i) { + TickAB _tick(_tmp_ticks[i].ask, _tmp_ticks[i].bid); + IndicatorDataEntry _entry(TickToEntry(_tmp_ticks[i].time_msc / 1000, _tick)); + EmitEntry(_entry); + // Appending tick into the history. + AppendEntry(_entry); + } + } + */ + } void OnTick(int _global_tick_index) override { #ifdef __MQL4__ diff --git a/Platform.h b/Platform.h index 780d4c116..5d67bd29c 100644 --- a/Platform.h +++ b/Platform.h @@ -32,11 +32,11 @@ #include "Std.h" #ifdef __MQLBUILD__ -#include "Indicators/Tick/Indi_TickMt.mqh" -#define PLATFORM_DEFAULT_INDICATOR_TICK Indi_TickMt + #include "Indicators/Tick/Indi_TickMt.mqh" + #define PLATFORM_DEFAULT_INDICATOR_TICK Indi_TickMt #else -#include "Indicators/Tick/Indi_TickRandom.mqh" -#define PLATFORM_DEFAULT_INDICATOR_TICK Indi_TickRandom + #include "Indicators/Tick/Indi_TickRandom.mqh" + #define PLATFORM_DEFAULT_INDICATOR_TICK Indi_TickRandom #endif #include "SymbolInfo.struct.static.h" @@ -121,7 +121,7 @@ class Platform { static void OnCalculate(const int rates_total, const int prev_calculated) { if (!emitted_history) { for (DictStructIterator> _iter = indis.Begin(); _iter.IsValid(); ++_iter) { - EmitHistory(_iter.Value().Ptr()); + // EmitHistory(_iter.Value().Ptr()); } emitted_history = true; } diff --git a/Storage/ItemsHistory.h b/Storage/ItemsHistory.h index d1edaed1e..fdff0cc8e 100644 --- a/Storage/ItemsHistory.h +++ b/Storage/ItemsHistory.h @@ -25,8 +25,8 @@ #define ITEMS_HISTORY_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif #include "../DictStruct.mqh" @@ -161,6 +161,16 @@ class ItemsHistory { */ unsigned int GetPeakSize() { return peak_size; } + /** + * Increments maximum size of historic items. + */ + void ReserveAdditionalHistoryMaxSize(unsigned int size) { history_max_size += size; } + + /** + * Changes maximum size of historic items. + */ + void SetHistoryMaxSize(unsigned int size) { history_max_size += size; } + /** * Will regenerate items from item provider. "_dir" indicates if we have to prepend or append items. */ @@ -352,11 +362,11 @@ class ItemsHistory { */ bool EnsureShiftExists(int _shift) { if (history.Size() == 0) { - return false; + // return false; } #ifdef __debug_items_history__ - Print("EnsureShiftExists(", _shift, ")"); + // Print("EnsureShiftExists(", _shift, ")"); #endif int _index = GetShiftIndex(_shift); From 6a38ad36af65acfb0d839eb0d62c10355356489d Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Wed, 26 Jun 2024 18:45:52 +0200 Subject: [PATCH 4/7] Committing the ItemsHistory-related code before merging `v3.008-dev-new`. This code is not yet ready to merge. --- Candle.struct.h | 7 +- Indicator/IndicatorCandle.provider.h | 13 +- Indicator/IndicatorTf.h | 17 +- Indicator/IndicatorTf.provider.h | 41 ++++- Indicator/IndicatorTf.struct.h | 11 +- Indicator/IndicatorTick.provider.h | 21 ++- Indicator/tests/classes/IndicatorTfDummy.h | 5 +- Indicators/Tick/Indi_TickMt.mqh | 4 +- Platform.h | 5 +- Storage/ItemsHistory.h | 171 +++++++++++++-------- 10 files changed, 189 insertions(+), 106 deletions(-) diff --git a/Candle.struct.h b/Candle.struct.h index 17b6f287d..2a05bc513 100644 --- a/Candle.struct.h +++ b/Candle.struct.h @@ -275,10 +275,11 @@ struct CandleOCTOHLC : CandleOHLC { Print("Error: Cannot update candle. Given time doesn't fit in candle's time-frame! Given time ", _timestamp_ms, ", but candle range is ", (long)start_time * 1000, " - ", (long)(start_time + length) * 1000, "."); if (_timestamp_ms < (long)start_time * 1000) { - Print("Looks like given time is ", (long)start_time * 1000 - _timestamp_ms, " ms before the candle starts."); + long _ms = (long)start_time * 1000 - _timestamp_ms; + Print("Looks like given time is ", _ms, " ms (", (double)_ms / 1000, " s) before the candle starts."); } else { - Print("Looks like given time is ", _timestamp_ms - (long)(start_time + length) * 1000, - " ms after the candle ends."); + long _ms = _timestamp_ms - (long)(start_time + length) * 1000; + Print("Looks like given time is ", _ms, " ms (", (double)_ms / 1000, "s) after the candle ends."); } DebugBreak(); } diff --git a/Indicator/IndicatorCandle.provider.h b/Indicator/IndicatorCandle.provider.h index b6737c945..f20b5f69e 100644 --- a/Indicator/IndicatorCandle.provider.h +++ b/Indicator/IndicatorCandle.provider.h @@ -25,8 +25,8 @@ #define INDICATOR_CANDLE_PROVIDER_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -62,6 +62,15 @@ class ItemsHistoryCandleProvider : public ItemsHistoryItemProvider, ItemsHistoryCandleProvider>* _history, int _start_index, + int _end_index, ARRAY_REF(CandleOCTOHLC, _out_arr)) { + return false; + } }; #endif diff --git a/Indicator/IndicatorTf.h b/Indicator/IndicatorTf.h index db9ab8980..f4b8c996d 100644 --- a/Indicator/IndicatorTf.h +++ b/Indicator/IndicatorTf.h @@ -25,11 +25,12 @@ #define INDICATOR_TF_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. +#include "../Chart.struct.tf.h" #include "IndicatorCandle.h" #include "IndicatorTf.provider.h" @@ -50,25 +51,16 @@ class IndicatorTf : public IndicatorCandle(iparams.GetSecsPerCandle(), THIS_PTR)); + history.SetItemProvider(new ItemsHistoryTfCandleProvider(ChartTf::TfToSeconds(GetTf()), THIS_PTR)); } public: /* Special methods */ - /** - * Class constructor with timeframe enum. - */ - IndicatorTf(unsigned int _spc) { - iparams.SetSecsPerCandle(_spc); - Init(); - } - /** * Class constructor with timeframe enum. */ IndicatorTf(ENUM_TIMEFRAMES _tf = PERIOD_CURRENT) { - iparams.SetSecsPerCandle(ChartTf::TfToSeconds(_tf)); tf = _tf; Init(); } @@ -77,7 +69,6 @@ class IndicatorTf : public IndicatorCandle { /** * Retrieves given number of items starting from the given microseconds or index (inclusive). "_dir" identifies if we - * want previous or next items from selected starting point. + * want previous or next items from selected starting point. Should return false if retrieving items by this method + * is not available. */ - void GetItems(ItemsHistory, ItemsHistoryTfCandleProvider>* _history, long _from_time_ms, + bool GetItems(ItemsHistory, ItemsHistoryTfCandleProvider>* _history, long _from_time_ms, ENUM_ITEMS_HISTORY_DIRECTION _dir, int _num_items, ARRAY_REF(CandleOCTOHLC, _out_arr)) { // Method is called if there is a missing item (candle) in the history. We need to regenerate it. if (_from_time_ms != 0) { @@ -140,6 +141,8 @@ class ItemsHistoryTfCandleProvider : public ItemsHistoryCandleProvider { //_ticks_to_ms = _ticks_from_ms - (_candle_length_ms - 1); } + bool _is_first_item = true; + while (_num_items > 0) { // Calculating time from which and to which we want to retrieve ticks to form a candle. int _ticks_from_s = GetCandleTimeFromTimeMs(_from_time_ms, spc); @@ -147,10 +150,23 @@ class ItemsHistoryTfCandleProvider : public ItemsHistoryCandleProvider { long _candle_length_ms = (long)spc * 1000; long _ticks_to_ms; + _ticks_to_ms = _ticks_from_ms + (_candle_length_ms - 1); + if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { - _ticks_to_ms = _ticks_from_ms + (_candle_length_ms - 1); - } else { - _ticks_to_ms = _ticks_from_ms - (_candle_length_ms - 1); + // Backwards. + if (_is_first_item) { + // As _from_time_ms in backward direction is next candle time - 1ms + // then we need to include it in our calculations. + long _new_start_ms = _from_time_ms - (_candle_length_ms - 1); + long _new_end_ms = _from_time_ms; + + _ticks_from_ms = _new_start_ms; + _ticks_from_s = int(_new_start_ms / 1000); + _from_time_ms = _new_start_ms; + _ticks_to_ms = _new_end_ms; + + _is_first_item = false; + } } // We will try to fetch history by two methods. @@ -178,6 +194,10 @@ class ItemsHistoryTfCandleProvider : public ItemsHistoryCandleProvider { // Even if we don't form an item (a candle), we assume we've done one item. --_num_items; + if (_num_items % 10000 == 0) { + Print(_num_items, " left to process..."); + } + if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { _from_time_ms += _candle_length_ms; } else { @@ -188,6 +208,17 @@ class ItemsHistoryTfCandleProvider : public ItemsHistoryCandleProvider { Print("Error: GetItems() for IndicatorTf can only work with given _from_time_ms!"); DebugBreak(); } + + return true; + } + + /** + * Retrieves items between given indices (both indices inclusive). Should return false if retrieving items by this + * method is not available. + */ + bool GetItems(ItemsHistory, ItemsHistoryTfCandleProvider>* _history, int _start_index, + int _end_index, ARRAY_REF(CandleOCTOHLC, _out_arr)) { + return false; } /** diff --git a/Indicator/IndicatorTf.struct.h b/Indicator/IndicatorTf.struct.h index 49b31ab0c..606023e17 100644 --- a/Indicator/IndicatorTf.struct.h +++ b/Indicator/IndicatorTf.struct.h @@ -26,8 +26,8 @@ */ #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -36,13 +36,8 @@ /* Structure for IndicatorTf class parameters. */ struct IndicatorTfParams : IndicatorParams { ChartTf tf; - unsigned int spc; // Seconds per candle. // Struct constructor. - IndicatorTfParams(string _name = "", unsigned int _spc = 60) : IndicatorParams(_name), spc(_spc) {} - // Getters. - unsigned int GetSecsPerCandle() { return spc; } - // Setters. - void SetSecsPerCandle(unsigned int _spc) { spc = _spc; } + IndicatorTfParams(string _name, ENUM_TIMEFRAMES _tf) : IndicatorParams(_name) { tf.SetTf(_tf); } // Copy constructor. IndicatorTfParams(const IndicatorTfParams &_params, ENUM_TIMEFRAMES _tf = PERIOD_CURRENT) { THIS_REF = _params; diff --git a/Indicator/IndicatorTick.provider.h b/Indicator/IndicatorTick.provider.h index 16659088a..40aef5054 100644 --- a/Indicator/IndicatorTick.provider.h +++ b/Indicator/IndicatorTick.provider.h @@ -25,8 +25,8 @@ #define INDICATOR_TICK_PROVIDER_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif // Includes. @@ -57,12 +57,21 @@ class ItemsHistoryTickProvider : public ItemsHistoryItemProvider> { /** * Retrieves given number of items starting from the given microseconds or index (inclusive). "_dir" identifies if we - * want previous or next items from selected starting point. + * want previous or next items from selected starting point. Should return false if retrieving items by this method + * is not available. */ - void GetItems(ItemsHistory, ItemsHistoryTickProvider>* _history, long _from_time_ms, + bool GetItems(ItemsHistory, ItemsHistoryTickProvider>* _history, long _from_time_ms, ENUM_ITEMS_HISTORY_DIRECTION _dir, int _num_items, ARRAY_REF(TickTAB, _out_arr)) { - // Method is called if there is a missing item (tick) in the history. We need to regenerate it. - indi PTR_DEREF FetchHistoryByStartTimeAndCount(_from_time_ms, _dir, _num_items, _out_arr); + return false; + } + + /** + * Retrieves items between given indices (both indices inclusive). Should return false if retrieving items by this + * method is not available. + */ + bool GetItems(ItemsHistory, ItemsHistoryTickProvider>* _history, int _start_index, int _end_index, + ARRAY_REF(TickTAB, _out_arr)) { + return false; } /** diff --git a/Indicator/tests/classes/IndicatorTfDummy.h b/Indicator/tests/classes/IndicatorTfDummy.h index a292e6b08..7ff6269c8 100644 --- a/Indicator/tests/classes/IndicatorTfDummy.h +++ b/Indicator/tests/classes/IndicatorTfDummy.h @@ -35,7 +35,7 @@ // Params for dummy candle-based indicator. struct IndicatorTfDummyParams : IndicatorTfParams { - IndicatorTfDummyParams(unsigned int _spc = 60) : IndicatorTfParams("IndicatorTf", _spc) {} + IndicatorTfDummyParams(ENUM_TIMEFRAMES _tf = PERIOD_CURRENT) : IndicatorTfParams("IndicatorTf", _tf) {} }; /** @@ -43,11 +43,10 @@ struct IndicatorTfDummyParams : IndicatorTfParams { */ class IndicatorTfDummy : public IndicatorTf { public: - IndicatorTfDummy(unsigned int _spc) : IndicatorTf(_spc) {} IndicatorTfDummy(ENUM_TIMEFRAMES _tf = PERIOD_CURRENT) : IndicatorTf(_tf) {} IndicatorTfDummy(ENUM_TIMEFRAMES_INDEX _tfi = 0) : IndicatorTf(_tfi) {} - string GetName() override { return "IndicatorTfDummy(" + IntegerToString(iparams.spc) + ")"; } + string GetName() override { return "IndicatorTfDummy(" + EnumToString(GetTf()) + ")"; } void OnDataSourceEntry(IndicatorDataEntry& entry, ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT) override { diff --git a/Indicators/Tick/Indi_TickMt.mqh b/Indicators/Tick/Indi_TickMt.mqh index 6001716ce..7de3b4050 100644 --- a/Indicators/Tick/Indi_TickMt.mqh +++ b/Indicators/Tick/Indi_TickMt.mqh @@ -295,7 +295,7 @@ class Indi_TickMt : public IndicatorTick _tick(0, 0); IndicatorDataEntry _entry(TickToEntry(TimeCurrent(), _tick)); - EmitEntry(_entry); + EmitEntry(_entry, INDI_EMITTED_ENTRY_TYPE_TICK); // Appending tick into the history. AppendEntry(_entry); return; @@ -324,7 +324,7 @@ class Indi_TickMt : public IndicatorTick _tick(_ask, _bid); IndicatorDataEntry _entry(TickToEntry(_time, _tick)); - EmitEntry(_entry); + EmitEntry(_entry, INDI_EMITTED_ENTRY_TYPE_TICK); // Appending tick into the history. AppendEntry(_entry); } diff --git a/Platform.h b/Platform.h index 5d67bd29c..d92d8dd76 100644 --- a/Platform.h +++ b/Platform.h @@ -32,11 +32,14 @@ #include "Std.h" #ifdef __MQLBUILD__ + #include "Indicators/Tf/Indi_TfMt.h" #include "Indicators/Tick/Indi_TickMt.mqh" #define PLATFORM_DEFAULT_INDICATOR_TICK Indi_TickMt + #define PLATFORM_DEFAULT_INDICATOR_TF Indi_TfMt #else #include "Indicators/Tick/Indi_TickRandom.mqh" #define PLATFORM_DEFAULT_INDICATOR_TICK Indi_TickRandom + #define PLATFORM_DEFAULT_INDICATOR_TF IndicatorTfDummy #endif #include "SymbolInfo.struct.static.h" @@ -274,7 +277,7 @@ class Platform { string _key = Util::MakeKey("PlatformIndicatorCandle", _symbol, (int)_tf); IndicatorData *_indi_candle; if (!Objects::TryGet(_key, _indi_candle)) { - _indi_candle = Objects::Set(_key, new IndicatorTfDummy(_tf)); + _indi_candle = Objects::Set(_key, new PLATFORM_DEFAULT_INDICATOR_TF(_tf)); // Adding indicator to list of default indicators in order to tick it on every Tick() call. Ref _ref = _indi_candle; diff --git a/Storage/ItemsHistory.h b/Storage/ItemsHistory.h index fdff0cc8e..3e10d25af 100644 --- a/Storage/ItemsHistory.h +++ b/Storage/ItemsHistory.h @@ -56,13 +56,21 @@ class ItemsHistoryItemProvider : public Dynamic { /** * Retrieves given number of items starting from the given microseconds or index (inclusive). "_dir" identifies if we - * want previous or next items from selected starting point. + * want previous or next items from selected starting point. Should return false if retrieving items by this method + * is not available. */ - void GetItems(ItemsHistory>* _history, long _from_time_ms, + bool GetItems(ItemsHistory>* _history, long _from_time_ms, ENUM_ITEMS_HISTORY_DIRECTION _dir, int _num_items, ARRAY_REF(IV, _out_arr)) { - // Method is called if there is a missing item (candle) in the history. We need to regenerate it. - Print("Error: Retrieving items by this item provider is not implemented!"); - DebugBreak(); + return false; + } + + /** + * Retrieves items between given indices (both indices inclusive). Should return false if retrieving items by this + * method is not available. + */ + bool GetItems(ItemsHistory>* _history, int _start_index, int _end_index, + ARRAY_REF(IV, _out_arr)) { + return false; } /** @@ -179,80 +187,111 @@ class ItemsHistory { Print("RegenerateHistory(", _from_index, ", ", _to_index, ", ", EnumToString(_dir), "), ", GetInfo()); #endif - static ARRAY(IV, _items); // Items generated by provider. - ArrayResize(_items, 0); - int _item_count = _to_index - _from_index + 1; - long _from_time_ms; - IV _item; - - // Calculating time to be passed to GetItems(). - if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { - if (history.Size() == 0) { -#ifdef __debug_items_history__ - Print("RegenerateHistory: Getting initial time from item provider"); -#endif - - // Time from we'll be getting items will be the time of the first possible item/candle/tick. - _from_time_ms = item_provider REF_DEREF GetInitialTimeMs(); - } else { -#ifdef __debug_items_history__ - Print("RegenerateHistory: Getting last valid item at index ", last_valid_index); -#endif - // Time will be the time of last valid item + item's length + 1ms. - // Note that ticks have length of 0ms, so next tick could be at least 1ms after the previous tick. - _item = GetItemByIndex(last_valid_index); - _from_time_ms = _item.GetTimeMs() + _item.GetLengthMs() + 1; - } - - long _current_time_ms = TimeCurrent() * 1000; - - if (_from_time_ms > (long)TimeCurrent() * 1000) { - // There won't be items in the future. - return; - } - } else if (_dir == ITEMS_HISTORY_DIRECTION_BACKWARD) { - if (history.Size() == 0) { - // Time from we'll be getting items will be the time of the first possible item/candle/tick - 1ms. - _from_time_ms = item_provider REF_DEREF GetInitialTimeMs() - 1; - } else { -#ifdef __debug_items_history__ - Print("RegenerateHistory: Getting first valid item at index ", first_valid_index); -#endif + // Static buffer for items generated by the item provider. + static ARRAY(IV, _items); + ArrayResize(_items, 0); - // Time will be the time of the first valid item - 1ms. - _item = GetItemByIndex(first_valid_index); - _from_time_ms = _item.GetTimeMs() - 1; - } - } else { - Print("Error: We shouldn't be here!"); - DebugBreak(); - return; - } + // Currently we only support getting items via start -> end indices. - item_provider REF_DEREF GetItems(THIS_PTR, _from_time_ms, _dir, _item_count, _items); + item_provider REF_DEREF GetItems(THIS_PTR, _from_index, _to_index, _items); if (ArraySize(_items) != _item_count) { -// There's really no problem if number of generated items are less than -// requested. -// @todo However, if there's too many calls for RegenerateHistory then we -// need to find a way to make it exit earlier. #ifdef __debug_items_history__ Print("RegenerateHistory: Notice: Requested ", _item_count, " historic items, got ", ArraySize(_items), ". from index = ", _from_index, ", to index = ", _to_index, ", dir = ", EnumToString(_dir), " (", GetInfo(), ")"); #endif - return; } - for (int i = 0; i < _item_count; ++i) { - if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { + int i; + + if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { + for (i = 0; i < _item_count; ++i) { Append(_items[i], false); - } else if (_dir == ITEMS_HISTORY_DIRECTION_BACKWARD) { + } + } else if (_dir == ITEMS_HISTORY_DIRECTION_BACKWARD) { + // Our _items array contain most recent items at the start, so we have to iterate from the end in order to prepend + // items. + if (peak_size <= 1) { + // It's the first time we're trying to retrieve historic items. We need to make sure all history will fit in the + // buffer. Note that we use <= 1, because Tick indicator could already have appeneded first candle. + history_max_size = MathMax(history_max_size, history.Size() + _item_count); + } + for (i = 0; i < _item_count; ++i) { Prepend(_items[i], false); } } + + /* + + int _item_count = _to_index - _from_index + 1; + long _from_time_ms; + IV _item; + + // Calculating time to be passed to GetItems(). + if (_dir == ITEMS_HISTORY_DIRECTION_FORWARD) { + if (history.Size() == 0) { + #ifdef __debug_items_history__ + Print("RegenerateHistory: Getting initial time from item provider"); + #endif + + // Time from we'll be getting items will be the time of the first possible item/candle/tick. + _from_time_ms = item_provider REF_DEREF GetInitialTimeMs(); + } else { + #ifdef __debug_items_history__ + Print("RegenerateHistory: Getting last valid item at index ", last_valid_index); + #endif + + // Time will be the time of last valid item + item's length + 1ms. + // Note that ticks have length of 0ms, so next tick could be at least 1ms after the previous tick. + _item = GetItemByIndex(last_valid_index); + _from_time_ms = _item.GetTimeMs() + _item.GetLengthMs() + 1; + } + + long _current_time_ms = TimeCurrent() * 1000; + + if (_from_time_ms > (long)TimeCurrent() * 1000) { + // There won't be items in the future. + return; + } + } else if (_dir == ITEMS_HISTORY_DIRECTION_BACKWARD) { + if (history.Size() == 0) { + // Time from we'll be getting items will be the time of the first possible item/candle/tick - 1ms. + _from_time_ms = item_provider REF_DEREF GetInitialTimeMs() - 1; + } else { + #ifdef __debug_items_history__ + Print("RegenerateHistory: Getting first valid item at index ", first_valid_index); + #endif + + // Time will be the time of the first valid item - 1ms. + _item = GetItemByIndex(first_valid_index); + _from_time_ms = _item.GetTimeMs() - 1; + } + } else { + Print("Error: We shouldn't be here!"); + DebugBreak(); + return; + } + + item_provider REF_DEREF GetItems(THIS_PTR, _from_time_ms, _dir, _item_count, _items); + + if (ArraySize(_items) != _item_count) { + // There's really no problem if number of generated items are less than + // requested. + // @todo However, if there's too many calls for RegenerateHistory then we + // need to find a way to make it exit earlier. + #ifdef __debug_items_history__ + Print("RegenerateHistory: Notice: Requested ", _item_count, " historic items, got ", ArraySize(_items), + ". from index = ", _from_index, ", to index = ", _to_index, ", dir = ", EnumToString(_dir), " (", + GetInfo(), + ")"); + #endif + return; + } + + */ } string GetInfo() { @@ -473,7 +512,13 @@ class ItemsHistory { return (datetime)0; } - return (datetime)(GetItemTimeByShiftMsc(_shift) / 1000); + datetime _dt = (datetime)(GetItemTimeByShiftMsc(_shift) / 1000); + +#ifdef __debug_items_history__ + Print("GetItemTimeByShift(", _shift, "), ", GetInfo(), " = ", _dt); +#endif + + return _dt; } /** From 63f1ddbfefd1252a360fe293ed9b5008369dce79 Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Thu, 27 Jun 2024 12:33:12 +0200 Subject: [PATCH 5/7] Fixes compilation errors in MT4 and MT5. --- Candle.struct.h | 5 +++-- Indicator/IndicatorCandle.provider.h | 5 ++--- Indicator/IndicatorRenko.provider.h | 18 +++++++++++++++--- Indicator/IndicatorTf.h | 2 +- Indicators/Account/Indi_AccountStats.mqh | 2 +- Indicators/Tick/Indi_TickMt.mqh | 2 +- Object.mqh | 5 ----- Platform.h | 16 ++++++++-------- Std.h | 23 ++++++++++++----------- 9 files changed, 43 insertions(+), 35 deletions(-) diff --git a/Candle.struct.h b/Candle.struct.h index 85f439323..74e9c9bab 100644 --- a/Candle.struct.h +++ b/Candle.struct.h @@ -284,11 +284,12 @@ struct CandleOCTOHLC : CandleOHLC { if (!ContainsTimeMs(_timestamp_ms)) { Print("Error: Cannot update candle. Given time doesn't fit in candle's time-frame! Given time ", _timestamp_ms, ", but candle range is ", (long)start_time * 1000, " - ", (long)(start_time + length) * 1000, "."); + long _ms; if (_timestamp_ms < (long)start_time * 1000) { - long _ms = (long)start_time * 1000 - _timestamp_ms; + _ms = (long)start_time * 1000 - _timestamp_ms; Print("Looks like given time is ", _ms, " ms (", (double)_ms / 1000, " s) before the candle starts."); } else { - long _ms = _timestamp_ms - (long)(start_time + length) * 1000; + _ms = _timestamp_ms - (long)(start_time + length) * 1000; Print("Looks like given time is ", _ms, " ms (", (double)_ms / 1000, "s) after the candle ends."); } DebugBreak(); diff --git a/Indicator/IndicatorCandle.provider.h b/Indicator/IndicatorCandle.provider.h index 2f57dcfbb..4cbccc76c 100644 --- a/Indicator/IndicatorCandle.provider.h +++ b/Indicator/IndicatorCandle.provider.h @@ -56,11 +56,10 @@ class ItemsHistoryCandleProvider : public ItemsHistoryItemProvider, ItemsHistoryCandleProvider>* _history, long _from_time_ms, + bool GetItems(ItemsHistory, ItemsHistoryCandleProvider>* _history, long _from_time_ms, ENUM_ITEMS_HISTORY_DIRECTION _dir, int _num_items, ARRAY_REF(CandleOCTOHLC, _out_arr)) { // Method is called if there is a missing item (candle) in the history. We need to regenerate it. - Print("Error: Retrieving items by this item provider is not implemented!"); - DebugBreak(); + return false; } /** diff --git a/Indicator/IndicatorRenko.provider.h b/Indicator/IndicatorRenko.provider.h index e987f12ba..63c483a80 100644 --- a/Indicator/IndicatorRenko.provider.h +++ b/Indicator/IndicatorRenko.provider.h @@ -25,8 +25,8 @@ #define INDICATOR_RENKO_PROVIDER_H #ifndef __MQL__ -// Allows the preprocessor to include a header file when it is needed. -#pragma once + // Allows the preprocessor to include a header file when it is needed. + #pragma once #endif /** @@ -68,12 +68,24 @@ class ItemsHistoryRenkoCandleProvider : public ItemsHistoryCandleProvider { * Retrieves given number of items starting from the given microseconds or index (inclusive). "_dir" identifies if we * want previous or next items from selected starting point. */ - void GetItems(ItemsHistory, ItemsHistoryRenkoCandleProvider>* _history, long _from_time_ms, + bool GetItems(ItemsHistory, ItemsHistoryRenkoCandleProvider>* _history, long _from_time_ms, ENUM_ITEMS_HISTORY_DIRECTION _dir, int _num_items, ARRAY_REF(CandleOCTOHLC, _out_arr)) { // Method is called if there is a missing item (candle) in the history. We need to regenerate it. Print("IndicatorRenko is not yet implemented!"); DebugBreak(); + return false; + } + + /** + * Retrieves items between given indices (both indices inclusive). Should return false if retrieving items by this + * method is not available. + */ + bool GetItems(ItemsHistory, ItemsHistoryRenkoCandleProvider>* _history, int _start_index, + int _end_index, ARRAY_REF(CandleOCTOHLC, _out_arr)) { + Print("IndicatorRenko is not yet implemented!"); + DebugBreak(); + return false; } }; diff --git a/Indicator/IndicatorTf.h b/Indicator/IndicatorTf.h index 5e44cf291..f00021569 100644 --- a/Indicator/IndicatorTf.h +++ b/Indicator/IndicatorTf.h @@ -87,7 +87,7 @@ class IndicatorTf : public IndicatorCandle { */ virtual void OnDataSourceEntry(IndicatorDataEntry &entry, ENUM_INDI_EMITTED_ENTRY_TYPE type = INDI_EMITTED_ENTRY_TYPE_PARENT) override { - Indicator::OnDataSourceEntry(entry, type); + Indicator::OnDataSourceEntry(entry, type); if (type != INDI_EMITTED_ENTRY_TYPE_CANDLE) { return; diff --git a/Indicators/Tick/Indi_TickMt.mqh b/Indicators/Tick/Indi_TickMt.mqh index 28b15a7cb..0bd2f29a9 100644 --- a/Indicators/Tick/Indi_TickMt.mqh +++ b/Indicators/Tick/Indi_TickMt.mqh @@ -264,7 +264,7 @@ class Indi_TickMt : public IndicatorTick @@ -122,18 +123,18 @@ */ #define CONST_REF_TO(T) const T& - /** +/** - * Reference to the array. - * - * @usage - * ARRAY_REF(, ) - */ -#define ARRAY_TYPE(T) _cpp_array -#define ARRAY_REF(T, N) ARRAY_TYPE(T) & N -#define FIXED_ARRAY_REF(T, N, S) T(&N)[S] +* Reference to the array. +* +* @usage +* ARRAY_REF(, ) +*/ + #define ARRAY_TYPE(T) _cpp_array + #define ARRAY_REF(T, N) ARRAY_TYPE(T) & N + #define FIXED_ARRAY_REF(T, N, S) T(&N)[S] -#define CONST_ARRAY_REF(T, N) const _cpp_array& N + #define CONST_ARRAY_REF(T, N) const _cpp_array& N #define CONST_ARRAY_REF(T, N) const _cpp_array& N /** @@ -568,4 +569,4 @@ struct AsSeriesReleaseEnsurer { SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(7); #define RELEASE_BUFFER8(A, B, C, D, E, F, G, H) \ RELEASE_BUFFER8_NO_ENSURE(A, B, C, D, E, F, G, H); \ - SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(8); \ No newline at end of file + SET_BUFFER_AS_SERIES_RELEASE_ENSURER_END(8); From c132e9b6f7194769a456b890f9a1ced3b8752734 Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Thu, 27 Jun 2024 13:08:17 +0200 Subject: [PATCH 6/7] Added missing files. --- Indicators/Tf/Indi_TfMt.h | 100 +++++++++++++++++ Indicators/Tf/Indi_TfMt.provider.h | 173 +++++++++++++++++++++++++++++ 2 files changed, 273 insertions(+) create mode 100644 Indicators/Tf/Indi_TfMt.h create mode 100644 Indicators/Tf/Indi_TfMt.provider.h diff --git a/Indicators/Tf/Indi_TfMt.h b/Indicators/Tf/Indi_TfMt.h new file mode 100644 index 000000000..ae2f16811 --- /dev/null +++ b/Indicators/Tf/Indi_TfMt.h @@ -0,0 +1,100 @@ +//+------------------------------------------------------------------+ +//| EA31337 framework | +//| Copyright 2016-2023, EA31337 Ltd | +//| https://github.com/EA31337 | +//+------------------------------------------------------------------+ + +/* + * This file is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +/** + * @file + * Tf-based candle indicator for MT. + */ + +#ifndef __MQL__ + // Allows the preprocessor to include a header file when it is needed. + #pragma once +#endif + +// Includes. +#include "../../Indicator/IndicatorCandle.h" +#include "Indi_TfMt.provider.h" + +// Params for MT Tf-based candle indicator. +struct Indi_TfMtParams : IndicatorTfParams { + Indi_TfMtParams(ENUM_TIMEFRAMES _tf = PERIOD_CURRENT) : IndicatorTfParams("IndicatorTf", _tf) {} +}; + +/** + * Tf-based candle indicator for MT. + */ +class Indi_TfMt : public IndicatorCandle> { + protected: + // Time-frame used to create candles. + ENUM_TIMEFRAMES tf; + + /* Protected methods */ + + /** + * Initialize class. + * + * Called on constructor. + */ + void Init() { history.SetItemProvider(new ItemsHistoryTfMtCandleProvider(THIS_PTR)); } + + public: + /* Special methods */ + + /** + * Class constructor with timeframe enum. + */ + Indi_TfMt(ENUM_TIMEFRAMES _tf = PERIOD_CURRENT) { + tf = _tf; + Init(); + } + + /** + * Class constructor with timeframe index. + */ + Indi_TfMt(ENUM_TIMEFRAMES_INDEX _tfi = 0) { + tf = ChartTf::IndexToTf(_tfi); + Init(); + } + + /** + * Class constructor with parameters. + */ + Indi_TfMt(Indi_TfMtParams& _icparams, const IndicatorDataParams& _idparams) { Init(); } + + /** + * Gets indicator's time-frame. + */ + ENUM_TIMEFRAMES GetTf() override { return tf; } + + /** + * Returns current tick index (incremented every OnTick()). + */ + int GetTickIndex() override { return history.GetItemProvider() PTR_DEREF GetTickIndex(); } + + /** + * Returns the number of bars on the chart decremented by iparams.shift. + */ + int GetBars() override { + // Will return number of bars prepended and appended to the history, + // even if those bars were cleaned up because of history's candle limit. + return ::Bars(GetSymbol(), GetTf()) - iparams.shift; + } +}; diff --git a/Indicators/Tf/Indi_TfMt.provider.h b/Indicators/Tf/Indi_TfMt.provider.h new file mode 100644 index 000000000..4af7095de --- /dev/null +++ b/Indicators/Tf/Indi_TfMt.provider.h @@ -0,0 +1,173 @@ +//+------------------------------------------------------------------+ +//| EA31337 framework | +//| Copyright 2016-2021, EA31337 Ltd | +//| https://github.com/EA31337 | +//+------------------------------------------------------------------+ + +/* + * This file is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + * + */ + +#ifndef __MQL__ + // Allows the preprocessor to include a header file when it is needed. + #pragma once +#endif + +/** + * Candle grouping and regeneration for time-frame based candles. + */ +template +class ItemsHistoryTfMtCandleProvider : public ItemsHistoryCandleProvider { + // Pointer to Tf Indicator, e.g., Indi_TfMt or IndicatorTfDummy. Used to fetch data by fixed timeframe candles. + IndicatorData* indi; + + // Current tick index. Effectively a number of ticks generated by attached Tick indicator. + int tick_index; + + public: + /** + * Constructor. + */ + ItemsHistoryTfMtCandleProvider(IndicatorData* _indi_tf) : indi(_indi_tf), tick_index(0) {} + + /** + * Called when new tick was emitted from IndicatorTick-based source. + */ + virtual void OnTick(ItemsHistory, ItemsHistoryTfMtCandleProvider>* _history, long _time_ms, + float _ask, float _bid) { + ++tick_index; + + // Seconds per candle calculated from TF. + int _spc = (int)ChartTf::TfToSeconds(indi PTR_DEREF GetTf()); + + Print("Indi_TfMt's history: New tick: ", TimeToString(_time_ms / 1000, TIME_DATE | TIME_MINUTES | TIME_SECONDS), + ", ", _ask, ", ", _bid); + + // We know that tick's timestamp will be ahead of the last tick and thus + // inside or ahead of the last created candle. In order to retrieve last + // valid candle, we need to use ItemsHistory::GetItemByShift(0) to check if + // we have to update last or create/append new candle. + CandleOCTOHLC _candle; + + // Will regenerate candles up to the last added candle ever. We have to + // call it, because some of the previous actions may have removed some of + // the recent candles. Note that OnTick() advances its _time_ms in + // ascending order, so all we need to most recent history. + // + // Note that EnsureShiftExists() may return false when there never been any + // candle added. + _history PTR_DEREF EnsureShiftExists(0); + + if (_history PTR_DEREF TryGetItemByShift(0, _candle, false) && _candle.ContainsTimeMs(_time_ms)) { + // Time given fits in the last added candle's time-frame, updating the candle with given price. + _candle.Update(_time_ms, _bid); + + // Storing candle in the history. + _history PTR_DEREF Update(_candle, _history PTR_DEREF GetShiftIndex(0)); + } else { + CandleOCTOHLC _candle_tmp; + + // We don't want to regenerate history, because at the start there will bo no candle however. + if (_history PTR_DEREF TryGetItemByShift(0, _candle_tmp, false)) { + // Print("Completed candle: ", _candle_tmp.ToString()); + // Print("Real candle: ", iOpen(NULL, Period(), 1), " ", iHigh(NULL, Period(), 1), " ", + // iLow(NULL, Period(), 1), " ", ChartStatic::iClose(NULL, (ENUM_TIMEFRAMES)Period(), 1)); + // Print("--"); + } + + // Either there is no candle at shift 0 or given time doesn't fit in the #0 candle's time-frame. + _candle.Init(GetCandleTimeFromTimeMs(_time_ms, _spc), _spc, _time_ms, _bid); + + // Adding candle as the most recent item in the history. It will now become the candle at shift 0. + _history PTR_DEREF Append(_candle); + } + } + + /** + * Returns current tick index. Effectively a number of ticks generated by + * attached IndicatorTick. + */ + int GetTickIndex() { return tick_index; } + + /** + * Returns start time of the candle (the place it's on the chart) for the given tick's time in milliseconds. + */ + int GetCandleTimeFromTimeMs(long _time_ms, int _length_in_secs) { + return (int)((_time_ms - _time_ms % ((long)_length_in_secs * 1000)) / 1000); + } + + /** + * Retrieves given number of items starting from the given microseconds or index (inclusive). "_dir" identifies if we + * want previous or next items from selected starting point. Should return false if retrieving items by this method + * is not available. + */ + bool GetItems(ItemsHistory, ItemsHistoryTfMtCandleProvider>* _history, long _from_time_ms, + ENUM_ITEMS_HISTORY_DIRECTION _dir, int _num_items, ARRAY_REF(CandleOCTOHLC, _out_arr)) { + return false; + } + + /** + * Retrieves items between given indices (both indices inclusive). Should return false if retrieving items by this + * method is not available. + */ + bool GetItems(ItemsHistory, ItemsHistoryTfMtCandleProvider>* _history, int _start_index, + int _end_index, ARRAY_REF(CandleOCTOHLC, _out_arr)) { + Print("Indi_TfMt::GetItems()"); + + // Converting absolute indices into MT shifts. + int _num_bars = _end_index - _start_index + 1; + // Current candle index. Could be 0 if no candles have been added or if there is only one candle added so far. + int _current_index = _history PTR_DEREF GetCurrentIndex(); + int _start_shift = _current_index - _start_index; + int _end_shift = _current_index - _end_index; + + Print("Indi_TfMt::GetItems(): Will fetch ", _num_bars, " bars between shift ", _start_shift, " and ", _end_shift); + + // Seconds per candle calculated from TF. + int _spc = (int)ChartTf::TfToSeconds(indi PTR_DEREF GetTf()); + + // Static, reusable array of rates. + static ARRAY(MqlRates, _rates); + int _count = _end_index - _start_index + 1; + ArrayResize(_rates, _count); + + Print("CopyRates(", indi PTR_DEREF GetSymbol(), ", ", EnumToString(indi PTR_DEREF GetTf()), ", ", _start_index, + ", ", _count, ", ...)"); + + // As GetItems() will only be called for missing candles, we can just ask MT for OHLCs and return them. + // Note that we have to specify most recent shift and count of history ticks to return. + // Also note that CopyRates() will insert oldest candle at the start of _rates array, so we have to inverse items in + // the array in order all candles be from most recent to the oldest ones. + int _num_copied = CopyRates(indi PTR_DEREF GetSymbol(), indi PTR_DEREF GetTf(), _end_shift, _count, _rates); + ArrayResize(_out_arr, _num_copied); + + for (int i = 0; i < _num_copied; ++i) { + MqlRates _rate = _rates[i]; + int _start_secs = (int)(long)_rate.time; + CandleOCTOHLC _candle(_rate.open, _rate.high, _rate.low, _rate.close, _start_secs, _spc, + (long)_start_secs * 1000, long(_start_secs + _spc) * 1000 - 1, (int)_rate.tick_volume); + // Reversing output, so most recent candle will be at start. + _out_arr[_num_copied - i - 1] = _candle; + } + + ArrayResize(_rates, 0); + return true; + } + + /** + * Returns information about item provider. + */ + string const ToString() override { return "Indi_TfMt candle provider on " + indi PTR_DEREF GetFullName(); } +}; From 2a805d1d560ff8c0130cdd783a15f72902952989 Mon Sep 17 00:00:00 2001 From: Adrian Kierzkowski Date: Thu, 27 Jun 2024 13:48:38 +0200 Subject: [PATCH 7/7] Removed unnecessary include. Fixes LinearWeightedMAOnBuffer() call in MT4. --- Indicators/Indi_MA.mqh | 95 +++++++++++++++++++++++++++++++----------- 1 file changed, 71 insertions(+), 24 deletions(-) diff --git a/Indicators/Indi_MA.mqh b/Indicators/Indi_MA.mqh index 0c7eb3634..f3f1801ab 100644 --- a/Indicators/Indi_MA.mqh +++ b/Indicators/Indi_MA.mqh @@ -24,9 +24,6 @@ #ifndef INDI_MA_MQH #define INDI_MA_MQH -// Includes. -#include - #include "../Dict.mqh" #include "../DictObject.mqh" #include "../Indicator/Indicator.h" @@ -35,27 +32,6 @@ #include "../Storage/ValueStorage.h" #include "../String.mqh" -#ifdef __MQL4__ -// MQL4 version of the method doesn't have last parameter. -int LinearWeightedMAOnBuffer(const int rates_total, const int prev_calculated, const int begin, const int period, - const double &price[], double &buffer[]) { - int _weight_sum; - return LinearWeightedMAOnBuffer(rates_total, prev_calculated, begin, period, price, buffer, _weight_sum); -} -#else // !__MQL__4 -// Defines global functions (for MQL4 backward compability). -double iMA(string _symbol, int _tf, int _ma_period, int _ma_shift, int _ma_method, int _ap, int _shift) { - ResetLastError(); - return Indi_MA::iMA(_symbol, (ENUM_TIMEFRAMES)_tf, _ma_period, _ma_shift, (ENUM_MA_METHOD)_ma_method, - (ENUM_APPLIED_PRICE)_ap, _shift); -} -double iMAOnArray(double &_arr[], int _total, int _period, int _ma_shift, int _ma_method, int _abs_shift, - IndicatorCalculateCache *_cache = NULL) { - ResetLastError(); - return Indi_MA::iMAOnArray(_arr, _total, _period, _ma_shift, _ma_method, _abs_shift, _cache); -} -#endif // __MQL4__ - // Structs. struct IndiMAParams : IndicatorParams { unsigned int period; @@ -552,6 +528,55 @@ class Indi_MA : public Indicator { return (rates_total); } + static int LinearWeightedMAOnBuffer(const int rates_total, const int prev_calculated, const int begin, + const int period, const ARRAY_REF(double, price), ARRAY_REF(double, buffer), + int &weight_sum) { + int i, k; + + // Check period. + if (period <= 1 || period > (rates_total - begin)) return (0); + // Save as_series flags. + bool as_series_price = ArrayGetAsSeries(price); + bool as_series_buffer = ArrayGetAsSeries(buffer); + + ArraySetAsSeries(price, false); + ArraySetAsSeries(buffer, false); + // Calculate start position. + int start_position; + + if (prev_calculated == 0) { + // First calculation or number of bars was changed. + // Set empty value for first bars. + start_position = period + begin; + + for (i = 0; i < start_position; i++) buffer[i] = 0.0; + // Calculate first visible value. + double first_value = 0; + int wsum = 0; + + for (i = begin, k = 1; i < start_position; i++, k++) { + first_value += k * price[i]; + wsum += k; + } + + buffer[start_position - 1] = first_value / wsum; + weight_sum = wsum; + } else + start_position = prev_calculated - 1; + // Main loop. + for (i = start_position; i < rates_total; i++) { + double sum = 0; + + for (int j = 0; j < period; j++) sum += (period - j) * price[i - j]; + + buffer[i] = sum / weight_sum; + } + // Restore as_series flags. + ArraySetAsSeries(price, as_series_price); + ArraySetAsSeries(buffer, as_series_buffer); + return (rates_total); + } + static int SmoothedMAOnBuffer(const int rates_total, const int prev_calculated, const int begin, const int period, ValueStorage &price, ValueStorage &buffer) { int i; @@ -785,4 +810,26 @@ class Indi_MA : public Indicator { iparams.applied_array = _applied_price; } }; + +#ifdef __MQL4__ +// MQL4 version of the method doesn't have last parameter. +int LinearWeightedMAOnBuffer(const int rates_total, const int prev_calculated, const int begin, const int period, + const double &price[], double &buffer[]) { + int _weight_sum; + return Indi_MA::LinearWeightedMAOnBuffer(rates_total, prev_calculated, begin, period, price, buffer, _weight_sum); +} +#else // !__MQL__4 +// Defines global functions (for MQL4 backward compability). +double iMA(string _symbol, int _tf, int _ma_period, int _ma_shift, int _ma_method, int _ap, int _shift) { + ResetLastError(); + return Indi_MA::iMA(_symbol, (ENUM_TIMEFRAMES)_tf, _ma_period, _ma_shift, (ENUM_MA_METHOD)_ma_method, + (ENUM_APPLIED_PRICE)_ap, _shift); +} +double iMAOnArray(double &_arr[], int _total, int _period, int _ma_shift, int _ma_method, int _abs_shift, + IndicatorCalculateCache *_cache = NULL) { + ResetLastError(); + return Indi_MA::iMAOnArray(_arr, _total, _period, _ma_shift, _ma_method, _abs_shift, _cache); +} +#endif // __MQL4__ + #endif // INDI_MA_MQH