//+------------------------------------------------------------------+ //| ZigZag_MACD.mq5 | //| Copyright © 2016, MetaQuotes Software Corp. | //| http://www.metaquotes.net/ | //+------------------------------------------------------------------+ //---- авторство индикатора #property copyright "Copyright © 2016, MetaQuotes Software Corp." //---- ссылка на сайт автора #property link "http://www.metaquotes.net/" //---- номер версии индикатора #property version "1.00" #property description "ZigZag" //---- отрисовка индикатора в отдельном окне #property indicator_separate_window //---- для расчёта и отрисовки индикатора использовано 4 буфера #property indicator_buffers 4 //---- использовано всего 2 графических построения #property indicator_plots 2 //+----------------------------------------------+ //| Параметры отрисовки осциллятора | //+----------------------------------------------+ //--- отрисовка индикатора в виде цветного облака #property indicator_type1 DRAW_FILLING //--- в качестве цветов индикатора использованы #property indicator_color1 clrAquamarine,clrMistyRose //---- отображение бычей метки индикатора #property indicator_label1 "MACD Oscillator" //+----------------------------------------------+ //| Параметры отрисовки зигзага | //+----------------------------------------------+ //---- в качестве индикатора использован ZIGZAG #property indicator_type2 DRAW_ZIGZAG //---- отображение метки индикатора #property indicator_label2 "ZigZag_MACD" //---- в качестве цвета линии индикатора использован DeepPink цвет #property indicator_color2 clrDeepPink //---- линия индикатора - непрерывная кривая #property indicator_style2 STYLE_SOLID //---- толщина линии индикатора равна 2 #property indicator_width2 2 //+----------------------------------------------+ //| объявление перечислений | //+----------------------------------------------+ enum ENUM_MODE //Тип константы { MODE_MAIN = 0, //Зигзаг по основной линии индикатора MODE_SIGNAL //Зигзаг по сигнальной линии индикатора }; //+----------------------------------------------+ //| объявление констант | //+----------------------------------------------+ #define RESET 0 // Константа для возврата терминалу команды на пересчёт индикатора //+----------------------------------------------+ //| Входные параметры индикатора | //+----------------------------------------------+ //---- входные параметры осциллятора MACD input ENUM_MODE Mode=MODE_MAIN; // Вариант построения //---- входные параметры осциллятора OsMA input uint fast_ma_period=12; // быстрый период input uint slow_ma_period=26; // медленный период input uint sign_ma_period=9; // период усреднения разности input ENUM_APPLIED_PRICE applied_price=PRICE_CLOSE; // тип цены //---- входные параметры зигзага input uint ExtDepth=3; input double ExtDeviation=10; input uint ExtBackstep=5; //+----------------------------------------------+ //---- объявление динамических массивов, которые будут в // дальнейшем использованы в качестве индикаторных буферов double LowestBuffer[],HighestBuffer[],MACDBuffer[],SignBuffer[]; //---- Объявление переменных памяти для пересчёта индикатора только на непосчитанных барах int LASTlowpos,LASThighpos,LASTColor; double LASTlow0,LASTlow1,LASThigh0,LASThigh1; //---- Объявление целых переменных начала отсчёта данных int min_rates_1,min_rates_total,MACD_Handle; //+------------------------------------------------------------------+ //| Custom indicator initialization function | //+------------------------------------------------------------------+ int OnInit() { //--- получение хендла индикатора iMACD MACD_Handle=iMACD(NULL,0,fast_ma_period,slow_ma_period,sign_ma_period,applied_price); if(MACD_Handle==INVALID_HANDLE) { Print(" Не удалось получить хендл индикатора iMACD"); return(INIT_FAILED); } //---- Инициализация переменных начала отсчёта данных min_rates_1=int(MathMax(fast_ma_period,slow_ma_period)+sign_ma_period); min_rates_total=int(min_rates_1+ExtDepth+ExtBackstep); //---- превращение динамических массивов в индикаторные буферы SetIndexBuffer(0,MACDBuffer,INDICATOR_DATA); SetIndexBuffer(1,SignBuffer,INDICATOR_DATA); SetIndexBuffer(2,LowestBuffer,INDICATOR_DATA); SetIndexBuffer(3,HighestBuffer,INDICATOR_DATA); //---- запрет на отрисовку индикатором пустых значений PlotIndexSetDouble(0,PLOT_EMPTY_VALUE,EMPTY_VALUE); PlotIndexSetDouble(1,PLOT_EMPTY_VALUE,EMPTY_VALUE); //---- индексация элементов в буферах как в таймсериях ArraySetAsSeries(LowestBuffer,true); ArraySetAsSeries(HighestBuffer,true); ArraySetAsSeries(MACDBuffer,true); ArraySetAsSeries(SignBuffer,true); //---- установка позиции, с которой начинается отрисовка PlotIndexSetInteger(0,PLOT_DRAW_BEGIN,min_rates_1); PlotIndexSetInteger(1,PLOT_DRAW_BEGIN,min_rates_total); //---- Установка формата точности отображения индикатора IndicatorSetInteger(INDICATOR_DIGITS,3); //---- имя для окон данных и лэйба для субъокон IndicatorSetString(INDICATOR_SHORTNAME,"ZigZag_MACD"); //--- return(INIT_SUCCEEDED); } //+------------------------------------------------------------------+ //| Custom indicator iteration function | //+------------------------------------------------------------------+ 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[]) { //---- проверка количества баров на достаточность для расчёта if(BarsCalculated(MACD_Handle)rates_total || prev_calculated<=0)// проверка на первый старт расчёта индикатора { limit=rates_total-min_rates_total; // стартовый номер для расчёта всех баров lastlow1=-1; lasthigh1=-1; lastlowpos=-1; lasthighpos=-1; } else { limit=rates_total-prev_calculated; // стартовый номер для расчёта новых баров climit=limit+min_rates_total; // стартовый номер для раскраски индикатора //---- восстанавливаем значения переменных lastlow0=LASTlow0; lasthigh0=LASThigh0; lastlow1=LASTlow1; lasthigh1=LASThigh1; lastlowpos=LASTlowpos+limit; lasthighpos=LASThighpos+limit; } to_copy=limit+1; //--- копируем вновь появившиеся данные в массивы if(CopyBuffer(MACD_Handle,0,0,to_copy,MACDBuffer)<=0) return(RESET); if(CopyBuffer(MACD_Handle,1,0,to_copy,SignBuffer)<=0) return(RESET); //---- Первый большой цикл расчёта индикатора if(Mode==MODE_MAIN)for(bar=limit; bar>=0 && !IsStopped(); bar--) { //---- запоминаем значения переменных перед прогонами на текущем баре if(rates_total!=prev_calculated && bar==0) { LASTlow0=lastlow0; LASThigh0=lasthigh0; } LowestBuffer[bar]=EMPTY_VALUE; HighestBuffer[bar]=EMPTY_VALUE; //--- low val=MACDBuffer[ArrayMinimum(MACDBuffer,bar,ExtDepth)]; if(val==lastlow0) val=EMPTY_VALUE; else { lastlow0=val; if(MACDBuffer[bar]-val>ExtDeviation*_Point) val=EMPTY_VALUE; else { for(back=1; back<=int(ExtBackstep); back++) { res=LowestBuffer[bar+back]; if(res!=EMPTY_VALUE && res>val) LowestBuffer[bar+back]=EMPTY_VALUE; } } } LowestBuffer[bar]=val; //--- high val=MACDBuffer[ArrayMaximum(MACDBuffer,bar,ExtDepth)]; if(val==lasthigh0) val=EMPTY_VALUE; else { lasthigh0=val; if(val-MACDBuffer[bar]>ExtDeviation*_Point) val=EMPTY_VALUE; else { for(back=1; back<=int(ExtBackstep); back++) { res=HighestBuffer[bar+back]; if(res!=EMPTY_VALUE && res=0 && !IsStopped(); bar--) { //---- запоминаем значения переменных перед прогонами на текущем баре if(rates_total!=prev_calculated && bar==0) { LASTlow0=lastlow0; LASThigh0=lasthigh0; } LowestBuffer[bar]=EMPTY_VALUE; HighestBuffer[bar]=EMPTY_VALUE; //--- low val=SignBuffer[ArrayMinimum(SignBuffer,bar,ExtDepth)]; if(val==lastlow0) val=EMPTY_VALUE; else { lastlow0=val; if(SignBuffer[bar]-val>ExtDeviation) val=EMPTY_VALUE; else { for(back=1; back<=int(ExtBackstep); back++) { res=LowestBuffer[bar+back]; if(res!=EMPTY_VALUE && res>val) LowestBuffer[bar+back]=EMPTY_VALUE; } } } LowestBuffer[bar]=val; //--- high val=SignBuffer[ArrayMaximum(SignBuffer,bar,ExtDepth)]; if(val==lasthigh0) val=EMPTY_VALUE; else { lasthigh0=val; if(val-SignBuffer[bar]>ExtDeviation) val=EMPTY_VALUE; else { for(back=1; back<=int(ExtBackstep); back++) { res=HighestBuffer[bar+back]; if(res!=EMPTY_VALUE && res=0 && !IsStopped(); bar--) { //---- запоминаем значения переменных перед прогонами на текущем баре if(rates_total!=prev_calculated && bar==0) { LASTlow1=lastlow1; LASThigh1=lasthigh1; //---- LASTlowpos=lastlowpos; LASThighpos=lasthighpos; } curlow=LowestBuffer[bar]; curhigh=HighestBuffer[bar]; //--- if(curlow==EMPTY_VALUE && curhigh==EMPTY_VALUE) continue; //--- if(curhigh!=EMPTY_VALUE) { if(lasthigh1>0) { if(lasthigh10) { if(lastlow1>curlow) LowestBuffer[lastlowpos]=EMPTY_VALUE; else LowestBuffer[bar]=EMPTY_VALUE; } //--- if(curlow