MQL5Book/Include/SymbolFilter.mqh
super.admin 1c8e83ce31 convert
2025-05-30 16:09:41 +02:00

236 lines
7 KiB
MQL5

//+------------------------------------------------------------------+
//| SymbolFilter.mqh |
//| Copyright 2022, MetaQuotes Ltd. |
//| https://www.mql5.com |
//+------------------------------------------------------------------+
#include "MapArray.mqh"
#include "SymbolMonitor.mqh"
#include "QuickSortTm.mqh"
#include "IS.mqh"
#include "Defines.mqh"
//+------------------------------------------------------------------+
//| Class for collecting symbols and their properties if they match |
//| specific conditions |
//+------------------------------------------------------------------+
class SymbolFilter
{
protected:
MapArray<ENUM_ANY,long> longs;
MapArray<ENUM_ANY,double> doubles;
MapArray<ENUM_ANY,string> strings;
MapArray<ENUM_ANY,IS> conditions;
static bool eps(const double v1, const double v2)
{
return fabs(v1 - v2) < DBL_EPSILON * fmax(v1, v2);
}
template<typename V>
static bool eps(const V v1, const V v2)
{
return false;
}
template<typename V>
static bool equal(const V v1, const V v2)
{
return v1 == v2 || eps(v1, v2);
}
template<typename V>
static bool greater(const V v1, const V v2)
{
return v1 > v2;
}
static bool equal(const string v1, const string v2)
{
if(StringFind(v2, "*") > -1)
{
int previous = 0;
string words[];
const int n = StringSplit(v2, '*', words);
for(int i = 0; i < n; ++i)
{
if(StringLen(words[i]) == 0) continue;
int index = StringFind(v1, words[i], previous);
if(index == -1)
{
return false;
}
previous = index + StringLen(words[i]);
}
return true;
}
return v1 == v2;
}
template<typename V>
bool match(const SymbolMonitor &m, const MapArray<ENUM_ANY,V> &data) const
{
static const V type = (V)NULL;
for(int i = 0; i < data.getSize(); ++i)
{
const ENUM_ANY key = data.getKey(i);
switch(conditions[key])
{
case EQUAL:
if(!equal(m.get(key, type), data.getValue(i)))
{
return false;
}
break;
case NOT_EQUAL:
if(equal(m.get(key, type), data.getValue(i)))
{
return false;
}
break;
case GREATER:
if(!greater(m.get(key, type), data.getValue(i)))
{
return false;
}
break;
case LESS:
if(greater(m.get(key, type), data.getValue(i)))
{
return false;
}
break;
}
}
return true;
}
public:
SymbolFilter *let(const ENUM_SYMBOL_INFO_INTEGER property, const long value, const IS cmp = EQUAL)
{
longs.put((ENUM_ANY)property, value);
conditions.put((ENUM_ANY)property, cmp);
return &this;
}
SymbolFilter *let(const ENUM_SYMBOL_INFO_DOUBLE property, const double value, const IS cmp = EQUAL)
{
doubles.put((ENUM_ANY)property, value);
conditions.put((ENUM_ANY)property, cmp);
return &this;
}
SymbolFilter *let(const ENUM_SYMBOL_INFO_STRING property, const string value, const IS cmp = EQUAL)
{
strings.put((ENUM_ANY)property, value);
conditions.put((ENUM_ANY)property, cmp);
return &this;
}
template<typename E,typename V>
bool select(const bool watch, const E property, string &symbols[], V &data[], const bool sort = false) const
{
E properties[1] = {property};
V tuples[][1];
const bool result = select(watch, properties, symbols, tuples, sort);
// NB: MQL5 supports copying arrays with different dimensions,
// here we copy (and unfold) 2D array into 1D array
ArrayCopy(data, tuples);
/* // otherwise we could rewrite it like this:
const int n = ArraySize(tuples);
ArrayResize(data, n);
for(int i = 0; i < n; ++i)
{
data[i] = tuples[i][0];
}
*/
return result;
}
// we need this overload because built-in ArraySort
// does not support arrays of strings
void ArraySort(string &s[][]) const
{
QuickSortTm<string> qt(s);
}
template<typename E,typename V>
bool select(const bool watch, const E &property[], string &symbols[], V &data[][], const bool sort = false) const
{
// size of array with properties must match output tuple size
const int q = ArrayRange(data, 1);
if(ArraySize(property) != q) return false; // error
const int n = SymbolsTotal(watch);
// loop through symbols
for(int i = 0; i < n; ++i)
{
const string s = SymbolName(i, watch);
// access all symbol properties via monitor
SymbolMonitor m(s);
// check all filtering conditions
if(match(m, longs)
&& match(m, doubles)
&& match(m, strings))
{
// for a matching symbol feed its name and properties into output arrays
const int k = EXPAND(data);
for(int j = 0; j < q; ++j)
{
data[k][j] = m.get(property[j]);
}
PUSH(symbols, s);
}
}
if(sort)
{
// we need aux array to keep track of original positions
// to reorder symbols/records according to new order
V array[][2];
const int p = ArrayRange(data, 0);
ArrayResize(array, p);
for(int i = 0; i < p; ++i)
{
array[i][0] = data[i][0];
array[i][1] = (V)i;
}
ArraySort(array);
string temp[];
V array2d[];
// make a flat copy of original array and then
// place elements from it into original array on new positions
ArrayCopy(array2d, data);
for(int i = 0; i < p; ++i)
{
const int k = (int)array[i][1];
PUSH(temp, symbols[k]);
for(int j = 0; j < q; ++j)
{
data[i][j] = array2d[k * q + j];
}
}
ArraySwap(symbols, temp);
}
return true;
}
void select(const bool watch, string &symbols[]) const
{
const int n = SymbolsTotal(watch);
for(int i = 0; i < n; ++i)
{
const string s = SymbolName(i, watch);
SymbolMonitor m(s);
if(match(m, longs)
&& match(m, doubles)
&& match(m, strings))
{
PUSH(symbols, s);
}
}
}
};
//+------------------------------------------------------------------+