Metatrader Forum | Forex Expert-Advisor | Broker & Forex Tools

Metatrader Forum | Forex Expert-Advisor | Broker & Forex Tools (http://www.expert-advisor.com/forum/index.php)
-   Programmierung MQL4 (http://www.expert-advisor.com/forum/forumdisplay.php?f=220)
-   -   Trendlinien (http://www.expert-advisor.com/forum/showthread.php?t=5798)

Lena 05.08.17 02:17

Trendlinien
 
Mal wieder am basteln und an einem Hindernis angekommen. Diesmal hapert es am automatischen platzieren einer Trendlinie.

Erst einmal den Code der dafür zuständigen Funktion:
Code:

void Func_Trend_Lines () {
  int      Pos1  =iHighest      (NULL,PERIOD_H1,MODE_HIGH,24,1);                                                            //Ermitteln welcher Bar den höchsten Preis aufweist
  double  Price1=iHigh        (NULL,0,Pos1);                                                                              //Den höchsten Preis abfragen
  datetime Time1 =StringToTime  ((string)TimeYear(Time[0])+"."+(string)TimeMonth(Time[0])+"."+(string)TimeDay(Time[Pos1]));  //Ermitteln zu welcher Zeit der höchste Preis war
  int      Pos2  =iHighest      (NULL,PERIOD_H1,MODE_HIGH,48,25);
  double  Price2=iHigh        (NULL,0,Pos2);
  datetime Time2 =StringToTime  ((string)TimeYear(Time[0])+"."+(string)TimeMonth(Time[0])+"."+(string)TimeDay(Time[Pos2])); 

  int      Pos3  =iLowest      (NULL,PERIOD_H1,MODE_LOW,24,1);
  double  Price3=iLow          (NULL,0,Pos3);
  datetime Time3 =StringToTime  ((string)TimeYear(Time[0])+"."+(string)TimeMonth(Time[0])+"."+(string)TimeDay(Time[Pos3])); 
  int      Pos4  =iLowest      (NULL,PERIOD_H1,MODE_LOW,48,25);
  double  Price4=iLow          (NULL,0,Pos4);
  datetime Time4 =StringToTime  ((string)TimeYear(Time[0])+"."+(string)TimeMonth(Time[0])+"."+(string)TimeDay(Time[Pos4])); 

  if(Price2<Price1){
      ObjectCreate(0,"High_Trendline",OBJ_TREND,0,Time1,Price1,Time2,Price2);
      ObjectSetInteger(0,"High_Trendline",OBJPROP_COLOR,clrAqua);
      ObjectSetInteger(0,"High_Trendline",OBJPROP_WIDTH,1);
      ObjectSetInteger(0,"High_Trendline",OBJPROP_STYLE,STYLE_DOT);
  }
  if(Price4>Price3){
      ObjectCreate(0,"Low_Trendline",OBJ_TREND,0,Time3,Price3,Time4,Price4);
      ObjectSetInteger(0,"Low_Trendline",OBJPROP_COLOR,clrAqua);
      ObjectSetInteger(0,"Low_Trendline",OBJPROP_WIDTH,1);
      ObjectSetInteger(0,"Low_Trendline",OBJPROP_STYLE,STYLE_DOT);
  }
}

Kurz und Knapp, die Funktion soll zwischen dem höchsten Punkt von vorgestern und gestern eine Linie zeichnen. Ebenso zwischen den beiden tiefsten Punkten. Dafür sollen/werden in den ersten beiden Abschnitten die entsprechenden Werte ermittelt.
In den letzten beiden Absätzen werden dann die Linien gesetzt und formatiert. Ein Fehler oder Warnung wird nicht ausgegeben. Ich denke es ist ein Problem beim ermitteln der benötigten Werte.

next user 05.08.17 05:21

Die ermittelten Werte sind soweit richtig. Dein Fehler liegt im Folgenden:
Code:

int      Pos1  =iHighest      (NULL,PERIOD_H1,MODE_HIGH,24,1);
double  Price1=iHigh        (NULL,0,Pos1);
datetime Time1 =StringToTime  ((string)TimeYear(Time[0])+"."+(string)TimeMonth(Time[0])+"."+(string)TimeDay(Time[Pos1]));

Du willst eine Trendlinie an den höchsten bzw. tiefsten Punkten im H1-Chart zeichnen, ermittelst die Zeitkoordinate aber nur bis zum Tag.
Dein Code muss von
Code:

... (string)TimeDay(Time[Pos1])
zu
Code:

... (string)TimeDay(Time[0])+" "+(string)TimeHour(Time[Pos1])
geändert werden. Für den Vortag nimmste dann z.B TimeDay(Time[24]). Für die anderen drei natürlich genau so.

Außerdem gehen deine Bedingungen
Code:

if(Price2<Price1)
...
if(Price4>Price3)

nur bei "aufwärts gerichtet" (High) und "abwärts gerichtet" (Low) auf. Auf umgekehrter Reihenfolge nicht. Hoffe das ist so gewollt.

Beim Vortag müsste folgender Code
Code:

(NULL,PERIOD_H1,MODE_LOW,48,25);
(NULL,PERIOD_H1,MODE_HIGH,48,25);

so heißen
Code:

(NULL,PERIOD_H1,MODE_LOW,24,25);
(NULL,PERIOD_H1,MODE_HIGH,24,25);

denn der vorletzte Parameter gibt die Anzahl zu überprüfender Kerzen an, beginnend vom letzten Parameter.
Mit deiner Angabe überprüfst du die letzten BEIDEN Vortage.

Noch ein kleiner Tipp.
Deine Zeilen zur Ermittlung der Zeitkoordinate sind unnötig. Du kannst auch einfach bei der Objekterstellung anstatt
Code:

ObjectCreate(0,"High_Trendline",OBJ_TREND,0,Time1,Price1,Time2,Price2)
folgendes schreiben
Code:

ObjectCreate(0,"High_Trendline",OBJ_TREND,0,Time[Pos1],Price1,Time[Pos2],Price2)
So sparst du dir 4 Variablen und 16 (bzw. mit der Korrektur 20) Funktionsaufrufe.

Lena 05.08.17 10:44

Ich danke dir für deine Hilfe, besonders für den kleinen Tip am Ende. Mir war bewusst dass man den Code noch kompakter schreiben konnte. Da es das erste Mal war, dass ich eine Trendlinie erstellt habe wollte ich es aber lieber in kleinen Schritten abarbeiten. So hatte ich die Möglichkeit Zwischenkontrollen einzubauen.
Da hatten sich ja doch 1-2 Flüchtigkeitsfehler eingeschlichen. Trotzdem bin ich recht zufrieden, dass keine größeren Bugs enthalten waren. Obwohl, grobe Schnitzer lassen sich leichter finden.

Lena 05.08.17 16:19

Verdammt... gibt es eine Möglichkeit den zweiten Ankerpunkt inclusive Preis zu verschieben?
Ich vermute mal es geht mit:
ObjectGetDouble(0,"Name",OBJPROP_PRICE,Variable_Pr eis)
Das Problem ist aber dann doch, das Programm weiß nicht an welcher zeitlichen Stelle der Preis geändert werden muss. Dadurch ändert sich ja auch die Steigung.

next user 05.08.17 18:28

Da nutzt du einfach
Code:

ObjectSetInteger(0,"OBJEKT",OBJPROP_TIME,0,ZEIT_ANKERPUNKT_1);
ObjectSetDouble(0,"OBJECT",OBJPROP_PRICE,0,PREIS_ANKERPUNKT_1);

ObjectSetInteger(0,"OBJEKT",OBJPROP_TIME,1,ZEIT_ANKERPUNKT_2);
ObjectSetDouble(0,"OBJECT",OBJPROP_PRICE,1,PREIS_ANKERPUNKT_2);

Und nein, ObjectGetDouble(...) liefert den Preis zurück. Zum Setzen musst du ObjectSetDouble(...) verwenden.

Lena 05.08.17 21:40

Natürlich muss es ObjectSet... und NICHT ObjectGet... heißen. Da war ich mal wieder zu schnell auf der Tastatur. Vielen Dank für deine Hilfe. Bei den vielen Befehlen und Möglichkeiten verliere ich noch häufig den Überblick. Oftmals weiß ich auch garnicht wonach ich suchen soll.

Habe mir jetzt aber erfolgreich eine Funktion gebastelt welche eine Trendlinie zwischen dem höchsten Punkt des vorgestrigen und gestrigen Tages einzeichnet. Diese wird auch in die "Zukunft" weiter projeziert. Zudem wird die Trendlinie so verschoben, dass es keine Schneidung am Vortag gibt. Der Code ist beigefügt. Evtl kann es ja jemand gebrauchen. Die verwendeten Zeiträume ließen sich auch schnell abändern.

Code:

//+------------------------------------------------------------------+
//| Funktion TRENDLINIEN                                            |
//+------------------------------------------------------------------+
void Func_Trend_Lines () {
  int      Pos1  =iHighest      (NULL,PERIOD_H1,MODE_HIGH,24,1);                                                //Ermitteln welcher Bar den höchsten Preis im laufe des gestrigen Tages aufweist
  double  Price1=iHigh        (NULL,0,Pos1);                                                                  //Den höchsten Preis abfragen
  int      Pos2  =iHighest      (NULL,PERIOD_H1,MODE_HIGH,24,25);                                                //Vorherige beiden Anweisungen wiederholen
  double  Price2=iHigh        (NULL,0,Pos2);

  int      Pos3  =iLowest      (NULL,PERIOD_H1,MODE_LOW,24,1);
  double  Price3=iLow          (NULL,0,Pos3);
  int      Pos4  =iLowest      (NULL,PERIOD_H1,MODE_LOW,24,25);
  double  Price4=iLow          (NULL,0,Pos4);
  if(Price2>Price1){                                                                                            //Abfrage ob das Hoch des vorgestrigen Tages größer dem Hoch des gestrigen Tages
      ObjectCreate      (0,"High_Trendline",OBJ_TREND,0,Time[Pos2],Price2,Time[Pos1],Price1);                    //Erstelle Trendlinie mit entsprechenden Koordinaten und Preisen
      for(i=Pos1-1;i>=1;i--){                                                                                    //Schleife (Zählervariable beginnt mit Position des Bars, welches Tageshoch bildet)
        if(iHigh(NULL,0,i)>ObjectGetValueByShift("High_Trendline",i)){                                          //Abfrage  (Ist Bar Tageshoch größer Trendlinie an gleicher Position)
            ObjectSetInteger  (0,"High_Trendline",OBJPROP_TIME2, 0,Time[i]);                                      //Ankerpunkt des gestrigen Tages wird auf Position des abgefragten Bars gesetzt
            ObjectSetDouble  (0,"High_Trendline",OBJPROP_PRICE2,0,iHigh(NULL,0,i));                              //Preis an zugehörigem Punkt wird auf Tageshoch des entsprechenden Bars gesetzt
            }
      }                       
      ObjectSetInteger  (0,"High_Trendline",OBJPROP_COLOR,clrAqua);                                              //Formatierung der Trendlinie
      ObjectSetInteger  (0,"High_Trendline",OBJPROP_WIDTH,1);
      ObjectSetInteger  (0,"High_Trendline",OBJPROP_STYLE,STYLE_DOT);
      ObjectSet        (  "High_Trendline",OBJPROP_RAY_RIGHT,true);
  }

  if(Price3>Price4){                                                                                            //siehe Kommentare zu "High_Trendline"
      ObjectCreate    (0,"Low_Trendline",OBJ_TREND,0,Time[Pos4],Price4,Time[Pos3],Price3);
      for(i=Pos3;i>=1;i--){
        if(iLow(NULL,0,i)<ObjectGetValueByShift("Low_Trendline",i)){
            ObjectSetInteger  (0,"Low_Trendline",OBJPROP_TIME2, 0,Time[i]);
            ObjectSetDouble  (0,"Low_Trendline",OBJPROP_PRICE2,0,iLow(NULL,0,i));
            }
      }                       
      ObjectSetInteger(0,"Low_Trendline",OBJPROP_COLOR,clrAqua);
      ObjectSetInteger(0,"Low_Trendline",OBJPROP_WIDTH,1);
      ObjectSetInteger(0,"Low_Trendline",OBJPROP_STYLE,STYLE_DOT);
      ObjectSet      (  "Low_Trendline",OBJPROP_RAY_RIGHT,true);
  }
}


next user 05.08.17 23:14

Lena, wieso nutzt du jetzt ObjectGetValueByShift() ?
Das ist in deinem Fall doch vollkommen unnötig.

Du erstellst eine Trendlinie mit 2 Punkten. ObjectGetValueByShift() macht nichts anderes, als bei der angegebenen Kerze den Preis zu kalkulieren, indem
die beiden Koordinaten und eine lineare Gleichung genutzt werden.

Die beiden For-Schleifen sind somit sinnlos.

Lena 06.08.17 08:42

Liste der Anhänge anzeigen (Anzahl: 2)
Ich denke die Funktionsweise lässt sich besser anhand von erklären.


http://www.expert-advisor.com/forum/...8&d=1502001582

http://www.expert-advisor.com/forum/...9&d=1502001582

next user 06.08.17 19:00

Ok, war gestern wohl doch schon Zeit schlafen zu gehen :)

So ist es verständlich, muss allerdings sagen, das meine Berechnungen für eine Trendlinie und deren Punkte generell anders aussehen würden.

Aber es geht ja darum, was du willst, also schon ok.

Lena 06.08.17 20:37

Mir ging es hier einfach nur darum, wie ich eine "Trendlinie" automatisch erstelle und an diverse Faktoren anpassen kann.


Alle Zeitangaben in WEZ +2. Es ist jetzt 17:54 Uhr.

Powered by vBulletin® Version 3.8.5 (Deutsch)
Copyright ©2000 - 2024, Jelsoft Enterprises Ltd.
SEO by vBSEO 3.6.1
Powered by vBCMS® 2.7.0 ©2002 - 2024 vbdesigns.de
Copyright ©2009 - 2023 by Expert-Advisor.com - Das Metatrader Forum