/* * Breakout Sample. */ // a link to the company website #property link "" // the company name #property copyright "" // チャートウィンドウ上に指標を表示する. #property indicator_chart_window // 表示する指標の数. #property indicator_buffers 3 // 線の色 #property indicator_color1 Green #property indicator_color2 Blue #property indicator_color3 Red /* * 外部変数. */ // output file name. extern string OUTPUT_FILE_NAME = "SampleBreakoutMA.log"; // debug extern bool DEBUG = false; // Averaging period for calculation. extern int MA_PERIOD = 21; // MA shift. Indicators line offset relate to the chart by timeframe. extern int MA_SHIFT = 0; // Moving Average methods. extern int MA_METHOD = MODE_SMA; // Price constants. extern int MA_APPLIED_PRICE = PRICE_CLOSE; // Timeframes. extern int MA_TIMEFRAME = PERIOD_H1; // Timeframe used on the chart. // チャート表示用のバッファ. double BUFFER_MA[]; double BUFFER_HIGH_BREAKOUT[]; double BUFFER_LOW_BREAKOUT[]; /* * 初期化処理. * チャートを表示するときに実行する. */ int init() { output("start 'init' function"); SetIndexBuffer(0, BUFFER_MA); SetIndexLabel(0, "moving average"); SetIndexBuffer(1, BUFFER_HIGH_BREAKOUT); SetIndexLabel(1, "highest breakout"); SetIndexStyle(1, DRAW_ARROW); SetIndexArrow(1, SYMBOL_ARROWUP); SetIndexBuffer(2, BUFFER_LOW_BREAKOUT); SetIndexLabel(2, "lowest breakout"); SetIndexStyle(2, DRAW_ARROW); SetIndexArrow(2, SYMBOL_ARROWDOWN); output("finish 'init' function"); return(0); } /* * 後処理. * チャートから削除するときに実行する. */ int deinit() { output("start 'deinit' function"); output("finish 'deinit' function"); return(0); } /* * 基本処理. * 次の tick が決まるたびに実行する. */ int start() { output("start 'start' function"); int countingBars = getCountingBars(); string message; datetime currentPeriodTime; int specifiedPeriodIndex; // 過去から現在に向かってバーのインデックスを移動する. for (int i = countingBars - 1; i >= 0; i--) { currentPeriodTime = currentPeriodIndex2Time(i); specifiedPeriodIndex = time2SpecifiedPeriodIndex(currentPeriodTime); BUFFER_MA[i] = getMA(specifiedPeriodIndex + 1); BUFFER_HIGH_BREAKOUT[i] = getHighestBreakout( BUFFER_MA[i + 1], getCloseValue(i + 1), getOpenValue(i)); BUFFER_LOW_BREAKOUT[i] = getLowestBreakout( BUFFER_MA[i + 1], getCloseValue(i + 1), getOpenValue(i)); message = createMessage(i, BUFFER_MA[i]); output(message); message = createMessage(i, BUFFER_HIGH_BREAKOUT[i]); output(message); message = createMessage(i, BUFFER_LOW_BREAKOUT[i]); output(message); } output("finish 'start' function"); return(0); } double getMA(int index) { double value = iMA( Symbol(), // 通貨ペア名 MA_TIMEFRAME, // 時間枠 MA_PERIOD, // 移動平均線を計算する期間 MA_SHIFT, // MA_METHOD, // 移動平均線を計算する方式 MA_APPLIED_PRICE, // 移動平均線を計算する際に使う値の種類 index); // 設定するインデックス return(value); } double getCloseValue(int index) { double value = iClose(Symbol(), 0, index); return(value); } double getOpenValue(int index) { double value = iOpen(Symbol(), 0, index); return(value); } double getHighestBreakout(double highestValue, double closeValue, double currentValue) { if (highestValue <= closeValue) { return(currentValue); } else { return(EMPTY_VALUE); } } double getLowestBreakout(double lowestValue, double closeValue, double currentValue) { if (lowestValue >= closeValue) { return(currentValue); } else { return(EMPTY_VALUE); } } /* * 指定した時刻、時間枠からインデックスを返す. */ int time2SpecifiedPeriodIndex(datetime time) { int index = iBarShift( Symbol(), // Symbol the data of which should be used to calculate indicator. NULL means the current symbol. MA_TIMEFRAME, // Timeframe. It can be any of Timeframe enumeration values. 0 means the current chart timeframe. time, // value to find (bar's open time). false); // Return mode when bar not found. false - iBarShift returns nearest. true - iBarShift returns -1. datetime specifiedPeriodTime = iTime(Symbol(), MA_TIMEFRAME, index); string message = StringConcatenate( "specified period time:", datetime2String(specifiedPeriodTime)); output(message); return(index); } /* * 現在の時間枠でのインデックスから対応する時刻を返す. */ datetime currentPeriodIndex2Time(int index) { datetime currentTime = iTime(Symbol(), 0, index); string currentTimeString = datetime2String(currentTime); string message = StringConcatenate( "current period time:", currentTimeString); output(message); return(currentTime); } /* * メッセージを作る. * format * symbol, datetime, price */ string createMessage(int index, double price) { string symbolString = Symbol(); string datetimeString = datetime2String( iTime(symbolString, 0, index)); string priceString = DoubleToStr( price, Digits); string message = StringConcatenate( symbolString, ",", datetimeString, ",", priceString); return(message); } /* * 日時から文字列に変換する */ string datetime2String(datetime dt) { string value = TimeToStr(dt, TIME_DATE | TIME_SECONDS); return(value); } /* * ファイルに出力する. */ void output(string message) { if (!DEBUG) { return; } int handle = openFile(); if (handle == -1) { return; } writeFile(handle, "info", message); closeFile(handle); } /* * 今回計算するバーの数を返す. */ int getCountingBars() { // チャート上のバーの数. int bars = Bars; // 変化のなかったバーの数. int countedBars = IndicatorCounted(); if (countedBars < 0) { return(0); } // 今回計算するバーの数. int countingBars = bars - countedBars - 1; return(countingBars); } /* * ファイルを開く. * return file handle. * not -1 : opened file * -1 : fail */ int openFile() { int handle = FileOpen(OUTPUT_FILE_NAME, FILE_CSV|FILE_READ|FILE_WRITE, ","); if (handle != -1) { FileSeek(handle, 0, SEEK_END); } else { int errorCode = GetLastError(); Print("Cannot open file. [name:", OUTPUT_FILE_NAME, "][errorCode:", errorCode, "]"); } return(handle); } /* * ファイルを閉じる. */ void closeFile(int handle) { FileClose(handle); } /* * ファイルに文字列を書く. */ void writeFile(int handle, string typeName, string message) { datetime current = TimeCurrent(); int result = FileWrite( handle, datetime2String(current), typeName, message); if (result >= 0) { FileFlush(handle); return; } int errorCode = GetLastError(); Print("Cannot write file. [name:", OUTPUT_FILE_NAME, "][errorCode:", errorCode, "]"); return; }