generated from EA31337/Strategy-Meta_Conditions
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathStg_Meta_Delegate.mqh
231 lines (215 loc) · 9.24 KB
/
Stg_Meta_Delegate.mqh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
/**
* @file
* Implements Delegate meta strategy.
*/
// Prevents processing this includes file multiple times.
#ifndef STG_META_DELEGATE_MQH
#define STG_META_DELEGATE_MQH
// Trade conditions.
enum ENUM_STG_META_DELEGATE_CONDITION {
STG_META_DELEGATE_COND_0_NONE = 0, // None
STG_META_DELEGATE_COND_ORDER_DATETIME_EOD, // End of day
STG_META_DELEGATE_COND_ORDER_LIFETIME_GT_1D, // Order opened over a day
};
// User input params.
INPUT2_GROUP("Meta Delegate strategy: main params");
INPUT2 uint Meta_Delegate_MagicNo_Min = 30000; // Magic number range (min) to monitor
INPUT2 uint Meta_Delegate_MagicNo_Max = 40000; // Magic number range (max) to monitor
INPUT2 ENUM_STRATEGY Meta_Delegate_Strategy_Main = STRAT_AMA; // Main strategy
INPUT2 ENUM_STG_META_DELEGATE_CONDITION Meta_Delegate_Condition =
STG_META_DELEGATE_COND_ORDER_LIFETIME_GT_1D; // Order condition to delegate
INPUT2 ENUM_STRATEGY Meta_Delegate_Strategy_Delegate = STRAT_AMA; // Strategy to delegate
INPUT3_GROUP("Meta Delegate strategy: common params");
INPUT3 float Meta_Delegate_LotSize = 0; // Lot size
INPUT3 int Meta_Delegate_SignalOpenMethod = 0; // Signal open method
INPUT3 float Meta_Delegate_SignalOpenLevel = 0; // Signal open level
INPUT3 int Meta_Delegate_SignalOpenFilterMethod = 32; // Signal open filter method
INPUT3 int Meta_Delegate_SignalOpenFilterTime = 3; // Signal open filter time (0-31)
INPUT3 int Meta_Delegate_SignalOpenBoostMethod = 0; // Signal open boost method
INPUT3 int Meta_Delegate_SignalCloseMethod = 0; // Signal close method
INPUT3 int Meta_Delegate_SignalCloseFilter = 32; // Signal close filter (-127-127)
INPUT3 float Meta_Delegate_SignalCloseLevel = 0; // Signal close level
INPUT3 int Meta_Delegate_PriceStopMethod = 1; // Price limit method
INPUT3 float Meta_Delegate_PriceStopLevel = 2; // Price limit level
INPUT3 int Meta_Delegate_TickFilterMethod = 32; // Tick filter method (0-255)
INPUT3 float Meta_Delegate_MaxSpread = 4.0; // Max spread to trade (in pips)
INPUT3 short Meta_Delegate_Shift = 0; // Shift
INPUT3 float Meta_Delegate_OrderCloseLoss = 200; // Order close loss
INPUT3 float Meta_Delegate_OrderCloseProfit = 200; // Order close profit
INPUT3 int Meta_Delegate_OrderCloseTime = 2880; // Order close time in mins (>0) or bars (<0)
// Structs.
// Defines struct with default user strategy values.
struct Stg_Meta_Delegate_Params_Defaults : StgParams {
Stg_Meta_Delegate_Params_Defaults()
: StgParams(::Meta_Delegate_SignalOpenMethod, ::Meta_Delegate_SignalOpenFilterMethod,
::Meta_Delegate_SignalOpenLevel, ::Meta_Delegate_SignalOpenBoostMethod,
::Meta_Delegate_SignalCloseMethod, ::Meta_Delegate_SignalCloseFilter,
::Meta_Delegate_SignalCloseLevel, ::Meta_Delegate_PriceStopMethod, ::Meta_Delegate_PriceStopLevel,
::Meta_Delegate_TickFilterMethod, ::Meta_Delegate_MaxSpread, ::Meta_Delegate_Shift) {
Set(STRAT_PARAM_LS, ::Meta_Delegate_LotSize);
Set(STRAT_PARAM_OCL, ::Meta_Delegate_OrderCloseLoss);
Set(STRAT_PARAM_OCP, ::Meta_Delegate_OrderCloseProfit);
Set(STRAT_PARAM_OCT, ::Meta_Delegate_OrderCloseTime);
Set(STRAT_PARAM_SOFT, ::Meta_Delegate_SignalOpenFilterTime);
}
};
class Stg_Meta_Delegate : public Strategy {
protected:
DictStruct<long, Ref<Order>> orders_active, orders_delegated;
DictStruct<long, Ref<Strategy>> strats;
Trade strade; // Trade instance.
public:
Stg_Meta_Delegate(StgParams &_sparams, TradeParams &_tparams, ChartParams &_cparams, string _name = "")
: Strategy(_sparams, _tparams, _cparams, _name), strade(_tparams, _cparams) {}
static Stg_Meta_Delegate *Init(ENUM_TIMEFRAMES _tf = NULL, EA *_ea = NULL) {
// Initialize strategy initial values.
Stg_Meta_Delegate_Params_Defaults stg_meta_delegate_defaults;
StgParams _stg_params(stg_meta_delegate_defaults);
// Initialize Strategy instance.
ChartParams _cparams(_tf, _Symbol);
TradeParams _tparams;
Strategy *_strat = new Stg_Meta_Delegate(_stg_params, _tparams, _cparams, "(Meta) Delegate");
return _strat;
}
/**
* Event on strategy's init.
*/
void OnInit() {
StrategyAdd(Meta_Delegate_Strategy_Main, 1);
StrategyAdd(Meta_Delegate_Strategy_Delegate, 2);
}
/**
* Sets strategy.
*/
bool StrategyAdd(ENUM_STRATEGY _sid, long _index = -1) {
ENUM_TIMEFRAMES _tf = Get<ENUM_TIMEFRAMES>(STRAT_PARAM_TF);
Ref<Strategy> _strat = StrategiesManager::StrategyInitByEnum(_sid, _tf);
#ifdef __strategies_meta__
if (!_strat.IsSet()) {
_strat = StrategiesMetaManager::StrategyInitByEnum((ENUM_STRATEGY_META)_sid, _tf);
}
#endif
if (_strat.IsSet()) {
_strat.Ptr().Set<long>(STRAT_PARAM_ID, Get<long>(STRAT_PARAM_ID));
_strat.Ptr().Set<ENUM_TIMEFRAMES>(STRAT_PARAM_TF, _tf);
_strat.Ptr().Set<int>(STRAT_PARAM_TYPE, _sid);
_strat.Ptr().OnInit();
if (_index >= 0) {
strats.Set(_index, _strat);
} else {
strats.Push(_strat);
}
}
return _strat.IsSet();
}
/**
* Find new active orders (to monitor) by magic number.
*/
bool OrdersFindNewByMagic() {
ResetLastError();
unsigned long _smagic = Get<unsigned long>(TRADE_PARAM_MAGIC_NO);
int _total_active = TradeStatic::TotalActive();
for (int pos = 0; pos < _total_active; pos++) {
if (OrderStatic::SelectByPosition(pos)) {
unsigned long _omagic = OrderStatic::MagicNumber();
unsigned long _ticket = OrderStatic::Ticket();
if (orders_active.KeyExists(_omagic) || orders_delegated.KeyExists(_omagic)) {
// Ignore finding orders which were already added.
continue;
}
if (_omagic > ::Meta_Delegate_MagicNo_Min && _omagic < ::Meta_Delegate_MagicNo_Max) {
// if (_omagic != _smagic) {
Ref<Order> _order = new Order(_ticket);
orders_active.Set(_ticket, _order);
// }
}
}
}
return GetLastError() == ERR_NO_ERROR;
}
/**
* Check conditions for active orders.
*/
void OrdersActiveProcessConditions() {
Ref<Order> _order;
for (DictStructIterator<long, Ref<Order>> iter = orders_active.Begin(); iter.IsValid(); ++iter) {
_order = iter.Value();
long _order_time_opened = _order.Ptr().Get<long>(ORDER_TIME_SETUP);
DateTime _order_datetime(_order_time_opened);
switch (::Meta_Delegate_Condition) {
case STG_META_DELEGATE_COND_ORDER_DATETIME_EOD:
// _result |= _shift > 0 && _time_opened < GetChart().GetBarTime(_shift - 1);
// _result |= _order_time_opened >= GetChart().GetBarTime(_shift);
break;
case STG_META_DELEGATE_COND_ORDER_LIFETIME_GT_1D:
if (_order_time_opened > _order_time_opened + 86400) {
MqlTradeRequest _request = {(ENUM_TRADE_REQUEST_ACTIONS)0};
MqlTradeCheckResult _result_check = {0};
MqlTradeResult _result = {0};
_request.action = TRADE_ACTION_MODIFY;
// _order.OrderModify(_request);
// OrderSend(_request, _result, _result_check);
}
break;
}
}
}
/**
* Event on new time periods.
*/
virtual void OnPeriod(unsigned int _periods = DATETIME_NONE) {
if ((_periods & DATETIME_MINUTE) != 0) {
// New minute started.
OrdersFindNewByMagic();
OrdersActiveProcessConditions();
}
}
/**
* Gets price stop value.
*/
float PriceStop(ENUM_ORDER_TYPE _cmd, ENUM_ORDER_TYPE_VALUE _mode, int _method = 0, float _level = 0.0f,
short _bars = 4) {
float _result = 0;
if (_method == 0) {
// Ignores calculation when method is 0.
return (float)_result;
}
Ref<Strategy> _strat_ref = strats.GetByKey(2);
if (!_strat_ref.IsSet()) {
// Returns false when strategy is not set.
return false;
}
_level = _level == 0.0f ? _strat_ref.Ptr().Get<float>(STRAT_PARAM_SOL) : _level;
_method = _strat_ref.Ptr().Get<int>(STRAT_PARAM_SOM);
//_shift = _shift == 0 ? _strat_ref.Ptr().Get<int>(STRAT_PARAM_SHIFT) : _shift;
_result = _strat_ref.Ptr().PriceStop(_cmd, _mode, _method, _level /*, _shift*/);
return (float)_result;
}
/**
* Check strategy's opening signal.
*/
bool SignalOpen(ENUM_ORDER_TYPE _cmd, int _method, float _level = 0.0f, int _shift = 0) {
bool _result = true;
// uint _ishift = _indi.GetShift();
uint _ishift = _shift;
Ref<Strategy> _strat_ref = strats.GetByKey(2);
if (!_strat_ref.IsSet()) {
// Returns false when strategy is not set.
return false;
}
_level = _level == 0.0f ? _strat_ref.Ptr().Get<float>(STRAT_PARAM_SOL) : _level;
_method = _method == 0 ? _strat_ref.Ptr().Get<int>(STRAT_PARAM_SOM) : _method;
_shift = _shift == 0 ? _strat_ref.Ptr().Get<int>(STRAT_PARAM_SHIFT) : _shift;
_result &= _strat_ref.Ptr().SignalOpen(_cmd, _method, _level, _shift);
return _result;
}
/**
* Check strategy's closing signal.
*/
bool SignalClose(ENUM_ORDER_TYPE _cmd, int _method, float _level = 0.0f, int _shift = 0) {
bool _result = true;
_result &= SignalOpen(Order::NegateOrderType(_cmd), _method, _level, _shift);
return _result;
}
};
#endif // STG_META_DELEGATE_MQH