Indicadores: Reversal - página 2

 
Christian:

A criação e a ordem prevalecem aqui! :-)

... estava macio como manteiga e com charme pré-natalino... pelo menos foi o que eu pensei

;)
 

Peguei esse Indi e o reprogramei.

De 286 linhas confusas, passou para apenas 151 linhas claras, e também pode fazer mais.

Agora ele é um indicador muito simples que tem potencial em combinação com, por exemplo, uma EMA 50 (linha de compra/venda de Steven Primo) e uma saída adequada.

//+------------------------------------------------------------------+
//|Reversão_V2.0.mq5 |
//| Copyright © 2019, Ing. Otto Pauser
//| https://www.mql5.com/pt/users/kronenchakra |
//| encontrado em https://www.mql5.com/pt/code/19605 |
//+------------------------------------------------------------------+
#property copyright     "Copyright © 2019, Ing. Otto Pauser"
#property link          "https://www.mql5.com/pt/users/kronenchakra"
#property version       "2.00"

#property indicator_chart_window
#property indicator_buffers 2
#property indicator_plots   2

//+------------------------------------------------------------------+
//| parâmetros de entrada|
//+------------------------------------------------------------------+
input int   inp_PeriodBack    =      10;  // Retrospectiva de períodos
input int   inp_ArrowCodeDN   =     242;  // Venda de ArrowCode
input int   inp_ArrowCodeUP   =     241;  // Compra de ArrowCode
input color inp_ArrowColorDN  =  clrRed;  // Venda de cor de seta
input color inp_ArrowColorUP  = clrLime;  // Compra de ArrowColor
input int   inp_ArrowSize     =       3;  // ArrowSize
input int   inp_ArrowOffs     =      30;  // ArrowOffset
input bool  inp_DrawRange     =    true;  // Desenhar o intervalo verificado
input bool  inp_SendMail      =   false;  // Enviar e-mail no sinal

//+------------------------------------------------------------------+
//| Buffers de indicadores|
//+------------------------------------------------------------------+
double buDN[];    // buffer Setas para baixo
double buUP[];    // buffer Setas para cima

//+------------------------------------------------------------------+
//| Função de inicialização do indicador personalizado
//+------------------------------------------------------------------+
int OnInit()
{
   InitArrows(buDN,inp_ArrowCodeDN,inp_ArrowColorDN,inp_ArrowSize,-inp_ArrowOffs,"Sell");
   InitArrows(buUP,inp_ArrowCodeUP,inp_ArrowColorUP,inp_ArrowSize,+inp_ArrowOffs,"Buy" );
   return(INIT_SUCCEEDED);
}

//+------------------------------------------------------------------+
//| Função de iteração de indicador personalizado
//+------------------------------------------------------------------+
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[])
{
   ArraySetAsSeries(high ,true);
   ArraySetAsSeries(close,true);
   ArraySetAsSeries(open ,true);
   ArraySetAsSeries(low  ,true);
   ArraySetAsSeries(time ,true);

   int limit=(prev_calculated==0)?0:rates_total-prev_calculated-1;

   for(int i=limit; i>=0; i--)
      {
         buDN[i]=NULL;
         buUP[i]=NULL;
         
         if(i>=MathMin(5000-1,rates_total-1-50)) continue; //com algumas taxas antigas para evitar "Array out of range" ou cálculo lento

                                                      /Role para baixo
         if(high[1+i]==HH(high,inp_PeriodBack))       //A alta do candlestick é igual à resistência
            if(close[1+i]<open[1+i])                  //Fechamento do candelabro < Abertura do candelabro (barra de baixa)
               {
                  buDN[i]=high[i];                    /Definir o valor do indicador em Candlestick High
                  if(i==1) 
                     myAlert(time[1],"Sell");         //Alerta sobre a abertura da próxima barra
               }
                                                      /Aumenta o braço
         if(low[1+i]==LL(low,inp_PeriodBack))         /A mínima do candlestick é igual ao suporte
            if(close[1+i]>open[1+i])                  //Fechamento do candelabro > Abertura do candelabro (barra de alta)
               {
                  buUP[i]=low[i];                     /Definir o valor do indicador em Candlestick Low
                  if(i==1) 
                     myAlert(time[1],"Buy");          //Alerta sobre a abertura da próxima barra
               }
      }

   if(inp_DrawRange)
      DrawRange("Range",time[inp_PeriodBack],HH(high,inp_PeriodBack),time[0],LL(low,inp_PeriodBack));

   return(rates_total);
}

//+------------------------------------------------------------------+
void myAlert(datetime aTime, string aMsg)    // Função de alerta
{
   static datetime lastTime=NULL;
   if(aTime!=lastTime)
      {
         aMsg="Reversal_V2.9 @ "+_Symbol+","+PeriodShortStr(_Period)+" -> "+aMsg;
         Print(aMsg);
         if(inp_SendMail) 
            SendMail("Reversal_V2.9", aMsg);
         lastTime=aTime;
      }
}

string PeriodShortStr(ENUM_TIMEFRAMES _tf)            // cortes do 'PERIOD_' inicial / retorna 'M1', 'M10', .... 
{
   return(StringSubstr(EnumToString(_tf),7));
}

//+------------------------------------------------------------------+
double HH(const double &aBuffer[], int aPeriBack)     // maior alta de um período para trás
{
   return(aBuffer[ArrayMaximum(aBuffer,0,aPeriBack)]);
}

double LL(const double &aBuffer[], int aPeriBack)     // mínimo mais baixo de um período para trás
{
   return(aBuffer[ArrayMinimum(aBuffer,0,aPeriBack)]);
}

void DrawRange(string aName, datetime aBegin, double aHigh, datetime aEnd, double aLow)
{
   if(ObjectFind   (0,aName)==0) 
      ObjectDelete (0,aName);
   ObjectCreate    (0,aName,OBJ_RECTANGLE,0,aBegin,aHigh,aEnd,aLow);
   ObjectSetInteger(0,aName,OBJPROP_STYLE,STYLE_DASHDOT);
}
//+------------------------------------------------------------------+
//| init indicatorbuffer with arrows|
//+------------------------------------------------------------------+
void InitArrows(double &aBuffer[], int aArrowCode, color aArrowColor, int aArrowSize, int aArrowOffs, string aPlotLabel)
{
   static int idx=0;
   SetIndexBuffer(idx,aBuffer,INDICATOR_DATA);
   PlotIndexSetInteger(idx,PLOT_DRAW_TYPE  ,DRAW_ARROW);
   PlotIndexSetInteger(idx,PLOT_ARROW      ,aArrowCode);
   PlotIndexSetInteger(idx,PLOT_LINE_COLOR ,aArrowColor);
   PlotIndexSetInteger(idx,PLOT_LINE_WIDTH ,aArrowSize);
   PlotIndexSetInteger(idx,PLOT_ARROW_SHIFT,aArrowOffs);
   PlotIndexSetString (idx,PLOT_LABEL      ,aPlotLabel);
   PlotIndexSetDouble (idx,PLOT_EMPTY_VALUE,NULL);
   ArrayInitialize    (aBuffer,NULL);
   ArraySetAsSeries   (aBuffer,true);
   idx++;
}

Esse indicador é um bom candidato para minha homepage, que estará on-line no início do próximo ano.

Arquivos anexados:
 
Otto Pauser:

Peguei esse Indi e o reprogramei.

Reduzi de 286 linhas confusas para apenas 151 linhas claras e posso fazer mais.

Agora ele é um indicador muito simples que, em conjunto com uma EMA 50 (linha de compra/venda de Steven Primo) e uma saída adequada, por exemplo, definitivamente tem potencial.

Esse indicador é um bom candidato para minha homepage, que estará on-line no início do próximo ano.

Muito gentil de sua parte! Eu o testei imediatamente e ele funciona perfeitamente. Muito obrigado!

 
não está funcionando
 
não funciona
 
há erros na compilação no meta editor. se possível, corrija-os. obrigado pela oferta gratuita de um conceito de indicador potencialmente excelente
 
Joseph Kisakye #:
há erros na compilação no meta editor. se possível, corrija-os. obrigado pela oferta gratuita de um conceito de indicador potencialmente excelente

este é o código modificado para que não haja erro...


//+------------------------------------------------------------------+
//|Indicador: reversal.mq5
//||
//||
//+------------------------------------------------------------------+
#property copyright "Sergey Vradiy"
#property version   "1.00"
#property description ""

//--- configurações do indicador
#property indicator_chart_window
#property indicator_buffers 2
#property indicator_plots 2

#property indicator_type1 DRAW_ARROW
#property indicator_width1 3
#property indicator_color1 0x0000FF
#property indicator_label1 "Sell"

#property indicator_type2 DRAW_ARROW
#property indicator_width2 3
#property indicator_color2 0xFFAA00
#property indicator_label2 "Buy"

//--- buffers de indicadores
double Buffer1[];
double Buffer2[];

input int Interval=10;
datetime time_alert;     //usado ao enviar um alerta
input bool Send_Email=true;
double myPoint; //inicializado em OnInit

//+------------------------------------------------------------------+
//||
//+------------------------------------------------------------------+
void myAlert(string type,string message)
{
   if(type=="print")
      Print(message);
   else if(type=="error")
   {
      Print(type+" | reversal @ "+Symbol()+","+(string)Period()+" | "+message);
   }
   else if(type=="indicator")
   {
      if(Send_Email)
         SendMail("reversal",type+" | reversal @ "+Symbol()+","+(string)Period()+" | "+message);
   }
}
//+------------------------------------------------------------------+
//||
//+------------------------------------------------------------------+
void DrawLine(string objname,double price,int count,int start_index) //cria ou modifica o objeto existente, se necessário
{
   if((price<0) && ObjectFind(0,objname)>=0)
   {
      ObjectDelete(0,objname);
   }
   else if(ObjectFind(0,objname)>=0 && ObjectGetInteger(0,objname,OBJPROP_TYPE)==OBJ_TREND)
   {
      datetime Time[];
      ArraySetAsSeries(Time,true);
      CopyTime(Symbol(),Period(),0,start_index+count,Time);
      ObjectSetInteger(0,objname,OBJPROP_TIME,Time[start_index]);
      ObjectSetDouble(0,objname,OBJPROP_PRICE,price);
      ObjectSetInteger(0,objname,OBJPROP_TIME,1,Time[start_index+count-1]);
      ObjectSetDouble(0,objname,OBJPROP_PRICE,1,price);
   }
   else
   {
      datetime Time[];
      ArraySetAsSeries(Time,true);
      CopyTime(Symbol(),Period(),0,start_index+count,Time);
      ObjectCreate(0,objname,OBJ_TREND,0,Time[start_index],price,Time[start_index+count-1],price);
      ObjectSetInteger(0,objname,OBJPROP_RAY_LEFT,0);
      ObjectSetInteger(0,objname,OBJPROP_RAY_RIGHT,0);
      ObjectSetInteger(0,objname,OBJPROP_COLOR,C'0x00,0x00,0xFF');
      ObjectSetInteger(0,objname,OBJPROP_STYLE,STYLE_SOLID);
      ObjectSetInteger(0,objname,OBJPROP_WIDTH,2);
   }
}
//+------------------------------------------------------------------+
//||
//+------------------------------------------------------------------+
double Support(int time_interval,bool fixed_tod,int hh,int mm,bool draw,int shift)
{
   int start_index=shift;
   int count=time_interval/PeriodSeconds();
   if(fixed_tod)
   {
      datetime start_time;
      datetime Time[];
      ArraySetAsSeries(Time,true);
      CopyTime(Symbol(),Period(),0,Bars(Symbol(),Period())-count,Time);
      if(shift==0)
         start_time=TimeCurrent();
      else
         start_time=Time[shift-1];
      datetime dt=StringToTime(TimeToString(start_time,TIME_DATE)+" "+(string)hh+":"+(string)mm); //horário mais próximo hh:mm
      if(dt>start_time)
         dt-=86400; //voltar 24 horas atrás
      int dt_index = iBarShift(Symbol(), Period(), dt, true);
      datetime dt2 = dt;
      while(dt_index<0 && dt>Time[Bars(Symbol(),Period())-1-count]) //barra não encontrada => procure alguns dias atrás
      {
         dt-=86400; //voltar 24 horas atrás
         dt_index=iBarShift(Symbol(),Period(),dt,true);
      }
      if(dt_index<0) /ainda não foi encontrado => localizar o bar mais próximo
         dt_index =iBarShift(Symbol(), Period(), dt2, false);
      start_index=dt_index+1; //barra após S/R abrir em dt
   }
   double _Low[];
   ArraySetAsSeries(_Low,true);
   CopyLow(Symbol(),Period(),start_index,count,_Low);
   double ret=_Low[ArrayMinimum(_Low,0,count)];
   if(draw) DrawLine("Support",ret,count,start_index);
   return(ret);
}
//+------------------------------------------------------------------+
//||
//+------------------------------------------------------------------+
double Resistance(int time_interval,bool fixed_tod,int hh,int mm,bool draw,int shift)
{
   int start_index=shift;
   int count=time_interval/PeriodSeconds();
   if(fixed_tod)
   {
      datetime start_time;
      datetime Time[];
      ArraySetAsSeries(Time,true);
      CopyTime(Symbol(),Period(),0,Bars(Symbol(),Period())-count,Time);
      if(shift==0)
         start_time=TimeCurrent();
      else
         start_time=Time[shift-1];
      datetime dt=StringToTime(TimeToString(start_time,TIME_DATE)+" "+(string)hh+":"+(string)mm); //horário mais próximo hh:mm
      if(dt>start_time)
         dt-=86400; //voltar 24 horas atrás
      int dt_index = iBarShift(Symbol(), Period(), dt, true);
      datetime dt2 = dt;
      while(dt_index<0 && dt>Time[Bars(Symbol(),Period())-1-count]) //barra não encontrada => procure alguns dias atrás
      {
         dt-=86400; //voltar 24 horas atrás
         dt_index=iBarShift(Symbol(),Period(),dt,true);
      }
      if(dt_index<0) /ainda não foi encontrado => localizar o bar mais próximo
         dt_index =iBarShift(Symbol(), Period(), dt2, false);
      start_index=dt_index+1; //barra após S/R abrir em dt
   }
   double _High[];
   ArraySetAsSeries(_High,true);
   CopyHigh(Symbol(),Period(),start_index,count,_High);
   double ret=_High[ArrayMaximum(_High,0,count)];
   if(draw) DrawLine("Resistance",ret,count,start_index);
   return(ret);
}
//+------------------------------------------------------------------+
//| Função de inicialização do indicador personalizado
//+------------------------------------------------------------------+
int OnInit()
{
   SetIndexBuffer(0,Buffer1);
   PlotIndexSetDouble(0,PLOT_EMPTY_VALUE,0);
   PlotIndexSetInteger(0,PLOT_ARROW,242);
   SetIndexBuffer(1,Buffer2);
   PlotIndexSetDouble(1,PLOT_EMPTY_VALUE,0);
   PlotIndexSetInteger(1,PLOT_ARROW,241);

   //inicializar myPoint
   myPoint=Point();
   if(Digits()==5 || Digits()==3)
   {
      myPoint*=10;
   }
   return(INIT_SUCCEEDED);
}
//+------------------------------------------------------------------+
//| Função de iteração de indicador personalizado|
//+------------------------------------------------------------------+
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 limit=rates_total-prev_calculated;
   if(limit<=0) return(rates_total);

//--- contando de 0 a rates_total
   ArraySetAsSeries(Buffer1,true);
   ArraySetAsSeries(Buffer2,true);

   //--- zero inicial
   if(prev_calculated<1)
   {
      ArrayInitialize(Buffer1,0);
      ArrayInitialize(Buffer2,0);
   }
   else
      limit++;

   //--- loop principal
   for(int i=limit-1; i>=0; i--)
   {
      if(i>=MathMin(5000-1,rates_total-1-50)) continue; //omitir taxas antigas

      //Buffer do indicador 1 (Venda)
      if(high[i]==Resistance(Interval*PeriodSeconds(),false,00,00,false,i)
         && close[i]<open[i]) //Condição para venda
      {
         Buffer1[i]=high[i]; /Definir o valor do indicador em Candlestick High
         if(i==1 && time[1]!=time_alert)
            myAlert("indicator","Sell"); //Alerta sobre a abertura da próxima barra
         time_alert=time[1];
      }
      else
      {
         Buffer1[i]=0;
      }

      //Buffer do indicador 2 (Buy)
      if(low[i]==Support(Interval*PeriodSeconds(),false,00,00,false,i)
         && close[i]>open[i]) //Condição para compra
      {
         Buffer2[i]=low[i]; /Definir o valor do indicador em Candlestick Low
         if(i==1 && time[1]!=time_alert)
            myAlert("indicator","Buy"); //Alerta sobre a abertura da próxima barra
         time_alert=time[1];
      }
      else
      {
         Buffer2[i]=0;
      }
   }
   return(rates_total);
}
//+------------------------------------------------------------------+



 
bitcoin razak #:

Este é o código modificado para não haver erro.




Obrigado, Razak, o código atualizado foi compilado com sucesso. Vou testá-lo e ver se ele funciona conforme explicado em sua introdução.

Fiz o backtesting do indicador usando as configurações padrão, mas não vejo nenhuma seta de compra e venda no gráfico. Por favor, revise o código se esse erro for confirmado. Obrigado, senhor