Compare commits

..

No commits in common. "393551a66b1abd7ca5579c8eb906e0e4efe577f9" and "e2f25cbbaf75349a5885028c85e9adb3e5d20da4" have entirely different histories.

6 changed files with 217 additions and 1659 deletions

View File

@ -1,216 +0,0 @@
#region Using declarations
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media;
using System.Xml.Serialization;
using NinjaTrader.Cbi;
using NinjaTrader.Gui;
using NinjaTrader.Gui.Chart;
using NinjaTrader.Gui.SuperDom;
using NinjaTrader.Gui.Tools;
using NinjaTrader.Data;
using NinjaTrader.NinjaScript;
using NinjaTrader.Core.FloatingPoint;
using NinjaTrader.NinjaScript.DrawingTools;
using SharpDX.DirectWrite;
#endregion
//This namespace holds Indicators in this folder and is required. Do not change it.
namespace NinjaTrader.NinjaScript.Indicators
{
public class DailyCOL : Indicator
{
private static int DAILY_BARS = 1;
private static int TEXT_PADDING = 5; // Pixels
private static string DATE = "{date}";
private static string LEVEL = "{level}";
private Dictionary<double, string> DailyOpens = new Dictionary<double, string>(); // Open -> Date
protected override void OnStateChange()
{
if (State == State.SetDefaults)
{
Description = @"Plots the open of daily candles.";
Name = "Daily COL";
Calculate = Calculate.OnPriceChange;
IsOverlay = true;
DisplayInDataBox = true;
DrawOnPricePanel = true;
DrawHorizontalGridLines = true;
DrawVerticalGridLines = true;
PaintPriceMarkers = true;
ScaleJustification = NinjaTrader.Gui.Chart.ScaleJustification.Right;
//Disable this property if your indicator requires custom values that cumulate with each new market data event.
//See Help Guide for additional information.
IsSuspendedWhileInactive = true;
NumberOfDays = 3;
DailyOpenStroke = new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3);
DrawLabels = true;
LabelText = DATE + " Daily COL @ " + LEVEL;
DailyOpenFont = new SimpleFont("Arial", 13);
DailyOpenFontColor = Brushes.LightGray;
}
else if (State == State.Configure)
{
// TODO: Force ETH?
AddDataSeries(Instrument.FullName, new BarsPeriod { BarsPeriodType = BarsPeriodType.Minute, Value = 1440 },
NumberOfDays, Bars.TradingHours.Name, Bars.IsResetOnNewTradingDay);
}
}
protected override void OnBarUpdate()
{
if (BarsInProgress == DAILY_BARS && IsFirstTickOfBar)
{
double dailyOpen = Open[0];
string day = Time[0].ToString("MM/dd");
DailyOpens[dailyOpen] = day;
}
}
protected override void OnRender(ChartControl chartControl, ChartScale chartScale)
{
base.OnRender(chartControl, chartScale);
SharpDX.Direct2D1.Brush dailyOpenBrush = DailyOpenStroke.Brush.ToDxBrush(RenderTarget);
TextFormat textFormat = DailyOpenFont.ToDirectWriteTextFormat();
SharpDX.Direct2D1.Brush textBrush = DailyOpenFontColor.ToDxBrush(RenderTarget);
foreach (double dailyOpen in DailyOpens.Keys)
{
// TODO: Only need to render daily opens visible on the chart.
int dailyOpenY = chartScale.GetYByValue(dailyOpen);
RenderTarget.DrawLine(new SharpDX.Vector2(ChartPanel.X, dailyOpenY),
new SharpDX.Vector2(ChartPanel.X + ChartPanel.W, dailyOpenY),
dailyOpenBrush, DailyOpenStroke.Width, DailyOpenStroke.StrokeStyle);
if (DrawLabels)
{
string dailyOpenText = LabelText.Replace(DATE, DailyOpens[dailyOpen]).Replace(LEVEL, dailyOpen.ToString(dailyOpen % 1 == 0 ? "F0" : "F2"));
// TODO: Extract text rendering into a separate utility.
TextLayout textLayout = new TextLayout(Core.Globals.DirectWriteFactory, dailyOpenText, textFormat, 500, textFormat.FontSize);
SharpDX.Vector2 textOrigin = new SharpDX.Vector2(ChartPanel.W - textLayout.Metrics.Width - TEXT_PADDING,
ChartPanel.Y + (float)chartScale.GetYByValue(dailyOpen) - (float)textLayout.Metrics.Height - TEXT_PADDING);
RenderTarget.DrawTextLayout(textOrigin, textLayout, textBrush, SharpDX.Direct2D1.DrawTextOptions.NoSnap);
textLayout.Dispose();
}
}
dailyOpenBrush.Dispose();
textFormat.Dispose();
textBrush.Dispose();
}
public override string DisplayName
{
get { return Name; }
}
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "Number of Days", Description = "Number of daily opens to consider", Order = 1, GroupName = "Daily COL")]
public int NumberOfDays
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Daily Open", Description = "Daily open level drawn on the chart", Order = 2, GroupName = "Daily COL")]
public Stroke DailyOpenStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Draw Labels", Order = 1, GroupName = "Labels")]
public bool DrawLabels
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Label Text", Order = 2, GroupName = "Labels")]
public string LabelText
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Font", Description = "Font used to display daily COLs", Order = 3, GroupName = "Labels")]
public SimpleFont DailyOpenFont
{ get; set; }
[XmlIgnore]
[NinjaScriptProperty]
[Display(Name = "Font Color", Description = "Color of the text used to label the daily COLs", Order = 4, GroupName = "Labels")]
public Brush DailyOpenFontColor
{ get; set; }
[Browsable(false)]
public string DailyOpenFontColorSerialization
{
get { return Serialize.BrushToString(DailyOpenFontColor); }
set { DailyOpenFontColor = Serialize.StringToBrush(value); }
}
}
}
#region NinjaScript generated code. Neither change nor remove.
namespace NinjaTrader.NinjaScript.Indicators
{
public partial class Indicator : NinjaTrader.Gui.NinjaScript.IndicatorRenderBase
{
private DailyCOL[] cacheDailyCOL;
public DailyCOL DailyCOL(int numberOfDays, Stroke dailyOpenStroke, bool drawLabels, string labelText, SimpleFont dailyOpenFont, Brush dailyOpenFontColor)
{
return DailyCOL(Input, numberOfDays, dailyOpenStroke, drawLabels, labelText, dailyOpenFont, dailyOpenFontColor);
}
public DailyCOL DailyCOL(ISeries<double> input, int numberOfDays, Stroke dailyOpenStroke, bool drawLabels, string labelText, SimpleFont dailyOpenFont, Brush dailyOpenFontColor)
{
if (cacheDailyCOL != null)
for (int idx = 0; idx < cacheDailyCOL.Length; idx++)
if (cacheDailyCOL[idx] != null && cacheDailyCOL[idx].NumberOfDays == numberOfDays && cacheDailyCOL[idx].DailyOpenStroke == dailyOpenStroke && cacheDailyCOL[idx].DrawLabels == drawLabels && cacheDailyCOL[idx].LabelText == labelText && cacheDailyCOL[idx].DailyOpenFont == dailyOpenFont && cacheDailyCOL[idx].DailyOpenFontColor == dailyOpenFontColor && cacheDailyCOL[idx].EqualsInput(input))
return cacheDailyCOL[idx];
return CacheIndicator<DailyCOL>(new DailyCOL(){ NumberOfDays = numberOfDays, DailyOpenStroke = dailyOpenStroke, DrawLabels = drawLabels, LabelText = labelText, DailyOpenFont = dailyOpenFont, DailyOpenFontColor = dailyOpenFontColor }, input, ref cacheDailyCOL);
}
}
}
namespace NinjaTrader.NinjaScript.MarketAnalyzerColumns
{
public partial class MarketAnalyzerColumn : MarketAnalyzerColumnBase
{
public Indicators.DailyCOL DailyCOL(int numberOfDays, Stroke dailyOpenStroke, bool drawLabels, string labelText, SimpleFont dailyOpenFont, Brush dailyOpenFontColor)
{
return indicator.DailyCOL(Input, numberOfDays, dailyOpenStroke, drawLabels, labelText, dailyOpenFont, dailyOpenFontColor);
}
public Indicators.DailyCOL DailyCOL(ISeries<double> input , int numberOfDays, Stroke dailyOpenStroke, bool drawLabels, string labelText, SimpleFont dailyOpenFont, Brush dailyOpenFontColor)
{
return indicator.DailyCOL(input, numberOfDays, dailyOpenStroke, drawLabels, labelText, dailyOpenFont, dailyOpenFontColor);
}
}
}
namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.DailyCOL DailyCOL(int numberOfDays, Stroke dailyOpenStroke, bool drawLabels, string labelText, SimpleFont dailyOpenFont, Brush dailyOpenFontColor)
{
return indicator.DailyCOL(Input, numberOfDays, dailyOpenStroke, drawLabels, labelText, dailyOpenFont, dailyOpenFontColor);
}
public Indicators.DailyCOL DailyCOL(ISeries<double> input , int numberOfDays, Stroke dailyOpenStroke, bool drawLabels, string labelText, SimpleFont dailyOpenFont, Brush dailyOpenFontColor)
{
return indicator.DailyCOL(input, numberOfDays, dailyOpenStroke, drawLabels, labelText, dailyOpenFont, dailyOpenFontColor);
}
}
}
#endregion

View File

@ -0,0 +1,143 @@
#region Using declarations
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media;
using System.Xml.Serialization;
using NinjaTrader.Cbi;
using NinjaTrader.Gui;
using NinjaTrader.Gui.Chart;
using NinjaTrader.Gui.SuperDom;
using NinjaTrader.Gui.Tools;
using NinjaTrader.Data;
using NinjaTrader.NinjaScript;
using NinjaTrader.Core.FloatingPoint;
using NinjaTrader.NinjaScript.DrawingTools;
#endregion
//This namespace holds Indicators in this folder and is required. Do not change it.
namespace NinjaTrader.NinjaScript.Indicators
{
public class DailyOpenLevels : Indicator
{
private static int DAILY_BARS = 1;
private HashSet<double> DailyOpens = new HashSet<double>();
protected override void OnStateChange()
{
if (State == State.SetDefaults)
{
Description = @"Plots the open of daily candles.";
Name = "Daily COL";
Calculate = Calculate.OnPriceChange;
IsOverlay = true;
DisplayInDataBox = true;
DrawOnPricePanel = true;
DrawHorizontalGridLines = true;
DrawVerticalGridLines = true;
PaintPriceMarkers = true;
ScaleJustification = NinjaTrader.Gui.Chart.ScaleJustification.Right;
//Disable this property if your indicator requires custom values that cumulate with each new market data event.
//See Help Guide for additional information.
IsSuspendedWhileInactive = true;
LookbackPeriod = 3;
DailyOpenStroke = new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3);
}
else if (State == State.Configure)
{
AddDataSeries(Instrument.FullName, new BarsPeriod { BarsPeriodType = BarsPeriodType.Day, Value = 1 }, LookbackPeriod, Bars.TradingHours.Name, Bars.IsResetOnNewTradingDay);
}
}
protected override void OnBarUpdate()
{
if (BarsInProgress == DAILY_BARS)
{
DailyOpens.Add(Opens[DAILY_BARS][0]);
}
else
{
foreach (double dailyOpen in DailyOpens)
{
if (DrawObjects["Open@" + dailyOpen] == null)
Draw.HorizontalLine(this, "Open@" + dailyOpen, dailyOpen, DailyOpenStroke.Brush, DailyOpenStroke.DashStyleHelper, (int)DailyOpenStroke.Width);
}
}
}
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "Lookback Period", Description = "Number of daily opens to consider", Order = 1, GroupName = "Parameters")]
public int LookbackPeriod
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Daily Open", Description = "Daily open level drawn on the chart", Order = 2, GroupName = "Parameters")]
public Stroke DailyOpenStroke
{ get; set; }
}
}
#region NinjaScript generated code. Neither change nor remove.
namespace NinjaTrader.NinjaScript.Indicators
{
public partial class Indicator : NinjaTrader.Gui.NinjaScript.IndicatorRenderBase
{
private DailyOpenLevels[] cacheDailyOpenLevels;
public DailyOpenLevels DailyOpenLevels(int lookbackPeriod, Stroke dailyOpenStroke)
{
return DailyOpenLevels(Input, lookbackPeriod, dailyOpenStroke);
}
public DailyOpenLevels DailyOpenLevels(ISeries<double> input, int lookbackPeriod, Stroke dailyOpenStroke)
{
if (cacheDailyOpenLevels != null)
for (int idx = 0; idx < cacheDailyOpenLevels.Length; idx++)
if (cacheDailyOpenLevels[idx] != null && cacheDailyOpenLevels[idx].LookbackPeriod == lookbackPeriod && cacheDailyOpenLevels[idx].DailyOpenStroke == dailyOpenStroke && cacheDailyOpenLevels[idx].EqualsInput(input))
return cacheDailyOpenLevels[idx];
return CacheIndicator<DailyOpenLevels>(new DailyOpenLevels(){ LookbackPeriod = lookbackPeriod, DailyOpenStroke = dailyOpenStroke }, input, ref cacheDailyOpenLevels);
}
}
}
namespace NinjaTrader.NinjaScript.MarketAnalyzerColumns
{
public partial class MarketAnalyzerColumn : MarketAnalyzerColumnBase
{
public Indicators.DailyOpenLevels DailyOpenLevels(int lookbackPeriod, Stroke dailyOpenStroke)
{
return indicator.DailyOpenLevels(Input, lookbackPeriod, dailyOpenStroke);
}
public Indicators.DailyOpenLevels DailyOpenLevels(ISeries<double> input , int lookbackPeriod, Stroke dailyOpenStroke)
{
return indicator.DailyOpenLevels(input, lookbackPeriod, dailyOpenStroke);
}
}
}
namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.DailyOpenLevels DailyOpenLevels(int lookbackPeriod, Stroke dailyOpenStroke)
{
return indicator.DailyOpenLevels(Input, lookbackPeriod, dailyOpenStroke);
}
public Indicators.DailyOpenLevels DailyOpenLevels(ISeries<double> input , int lookbackPeriod, Stroke dailyOpenStroke)
{
return indicator.DailyOpenLevels(input, lookbackPeriod, dailyOpenStroke);
}
}
}
#endregion

View File

@ -1,237 +0,0 @@
#region Using declarations
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media;
using System.Xml.Serialization;
using NinjaTrader.Cbi;
using NinjaTrader.Gui;
using NinjaTrader.Gui.Chart;
using NinjaTrader.Gui.SuperDom;
using NinjaTrader.Gui.Tools;
using NinjaTrader.Data;
using NinjaTrader.NinjaScript;
using NinjaTrader.Core.FloatingPoint;
using NinjaTrader.NinjaScript.DrawingTools;
#endregion
//This namespace holds Indicators in this folder and is required. Do not change it.
namespace NinjaTrader.NinjaScript.Indicators
{
public class HighsAndLows : Indicator
{
private static string HIGHER_HIGH = "HH";
private static string HIGHER_LOW = "HL";
private static string LOWER_HIGH = "LH";
private static string LOWER_LOW = "LL";
private double PivotHigh = double.MinValue;
private int PivotHighCandle = int.MinValue;
private string PivotHighLabel = string.Empty;
private double PivotLow = double.MaxValue;
private int PivotLowCandle = int.MinValue;
private string PivotLowLabel = string.Empty;
protected override void OnStateChange()
{
if (State == State.SetDefaults)
{
Description = @"Plots higher highs, lower highs, higher lows, and lower lows";
Name = "Highs and Lows";
Calculate = Calculate.OnBarClose;
IsOverlay = true;
DisplayInDataBox = true;
DrawOnPricePanel = true;
DrawHorizontalGridLines = true;
DrawVerticalGridLines = true;
PaintPriceMarkers = true;
ScaleJustification = NinjaTrader.Gui.Chart.ScaleJustification.Right;
//Disable this property if your indicator requires custom values that cumulate with each new market data event.
//See Help Guide for additional information.
IsSuspendedWhileInactive = true;
BarsToLookBack = 10;
BarsToLookAhead = 10;
TextColor = Brushes.Yellow;
}
}
protected override void OnBarUpdate()
{
if (CurrentBar < (BarsToLookBack + BarsToLookAhead) + 1)
return;
double currentHigh = High[BarsToLookAhead];
bool isPivotHigh = true;
for (int i = 0; i < BarsToLookAhead; i++)
{
if (High[i] > currentHigh)
{
isPivotHigh = false;
break;
}
}
if (isPivotHigh)
{
for (int i = 0; i < BarsToLookBack; i++)
{
if (High[BarsToLookAhead + (i + 1)] > currentHigh)
{
isPivotHigh = false;
break;
}
}
}
if (isPivotHigh)
{
if ((CurrentBar - PivotHighCandle) < 5 && currentHigh == PivotHigh)
{
RemoveDrawObject(PivotHighLabel + "@" + PivotHighCandle);
}
if (currentHigh > PivotHigh)
{
PivotHighLabel = HIGHER_HIGH;
}
else if (currentHigh < PivotHigh)
{
PivotHighLabel = LOWER_HIGH;
}
PivotHigh = currentHigh;
PivotHighCandle = CurrentBar;
Draw.Text(this, PivotHighLabel + "@" + PivotHighCandle, PivotHighLabel, BarsToLookAhead, PivotHigh + TickSize, TextColor);
}
double currentLow = Low[BarsToLookAhead];
bool isPivotLow = true;
for (int i = 0; i < BarsToLookAhead; i++)
{
if (Low[i] < currentLow)
{
isPivotLow = false;
break;
}
}
if (isPivotLow)
{
for (int i = 0; i < BarsToLookBack; i++)
{
if (Low[BarsToLookAhead + (i + 1)] < currentLow)
{
isPivotLow = false;
break;
}
}
}
if (isPivotLow)
{
if ((CurrentBar - PivotLowCandle) < 5 && currentLow == PivotLow)
{
RemoveDrawObject(PivotLowLabel + "@" + PivotLowCandle);
}
if (currentLow < PivotLow)
{
PivotLowLabel = LOWER_LOW;
}
else if (currentLow > PivotLow)
{
PivotLowLabel = HIGHER_LOW;
}
PivotLow = currentLow;
PivotLowCandle = CurrentBar;
Draw.Text(this, PivotLowLabel + "@" + PivotLowCandle, PivotLowLabel, BarsToLookAhead, PivotLow - TickSize, TextColor);
}
}
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "Look Back Period", Description = "Number of bars to look backward when determining pivots", Order = 1, GroupName = "Parameters")]
public int BarsToLookBack
{ get; set; }
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "Look Ahead Period", Description = "Number of bars to look forward when determining pivots", Order = 2, GroupName = "Parameters")]
public int BarsToLookAhead
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Text Color", Description = "Color of the text used to label the pivots", Order = 3, GroupName = "Parameters")]
public Brush TextColor
{ get; set; }
}
}
#region NinjaScript generated code. Neither change nor remove.
namespace NinjaTrader.NinjaScript.Indicators
{
public partial class Indicator : NinjaTrader.Gui.NinjaScript.IndicatorRenderBase
{
private HighsAndLows[] cacheHighsAndLows;
public HighsAndLows HighsAndLows(int barsToLookBack, int barsToLookAhead, Brush textColor)
{
return HighsAndLows(Input, barsToLookBack, barsToLookAhead, textColor);
}
public HighsAndLows HighsAndLows(ISeries<double> input, int barsToLookBack, int barsToLookAhead, Brush textColor)
{
if (cacheHighsAndLows != null)
for (int idx = 0; idx < cacheHighsAndLows.Length; idx++)
if (cacheHighsAndLows[idx] != null && cacheHighsAndLows[idx].BarsToLookBack == barsToLookBack && cacheHighsAndLows[idx].BarsToLookAhead == barsToLookAhead && cacheHighsAndLows[idx].TextColor == textColor && cacheHighsAndLows[idx].EqualsInput(input))
return cacheHighsAndLows[idx];
return CacheIndicator<HighsAndLows>(new HighsAndLows(){ BarsToLookBack = barsToLookBack, BarsToLookAhead = barsToLookAhead, TextColor = textColor }, input, ref cacheHighsAndLows);
}
}
}
namespace NinjaTrader.NinjaScript.MarketAnalyzerColumns
{
public partial class MarketAnalyzerColumn : MarketAnalyzerColumnBase
{
public Indicators.HighsAndLows HighsAndLows(int barsToLookBack, int barsToLookAhead, Brush textColor)
{
return indicator.HighsAndLows(Input, barsToLookBack, barsToLookAhead, textColor);
}
public Indicators.HighsAndLows HighsAndLows(ISeries<double> input , int barsToLookBack, int barsToLookAhead, Brush textColor)
{
return indicator.HighsAndLows(input, barsToLookBack, barsToLookAhead, textColor);
}
}
}
namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.HighsAndLows HighsAndLows(int barsToLookBack, int barsToLookAhead, Brush textColor)
{
return indicator.HighsAndLows(Input, barsToLookBack, barsToLookAhead, textColor);
}
public Indicators.HighsAndLows HighsAndLows(ISeries<double> input , int barsToLookBack, int barsToLookAhead, Brush textColor)
{
return indicator.HighsAndLows(input, barsToLookBack, barsToLookAhead, textColor);
}
}
}
#endregion

View File

@ -19,94 +19,29 @@ using NinjaTrader.Data;
using NinjaTrader.NinjaScript;
using NinjaTrader.Core.FloatingPoint;
using NinjaTrader.NinjaScript.DrawingTools;
using SharpDX.DirectWrite;
using NinjaTrader.NinjaScript.Indicators;
using System.Globalization;
#endregion
public enum TimeFrame
{
Seconds,
Minutes,
Hours
}
//This namespace holds Indicators in this folder and is required. Do not change it.
namespace NinjaTrader.NinjaScript.Indicators
{
public class OR
{
public double High { get; set; }
public double Low { get; set; }
public double Mid { get; set; }
public double LatestPrice { get; set; }
public DateTime StartTime { get; set; }
public DateTime EndTime { get; set; }
}
[CategoryOrder("Opening Range", 1)]
[CategoryOrder("Appearance", 2)]
[CategoryOrder("Labels", 3)]
[TypeConverter("NinjaTrader.NinjaScript.Indicators.OpeningRangePropertyConverter")]
public class OpeningRange : Indicator
{
public static int DefaultOpeningRangePeriod = 30;
public static OpeningRangeBarType DefaultOpeningRangeType =
OpeningRangeBarType.Minutes;
public static OpeningRangeStartTime DefaultOpeningRangeStartTime =
OpeningRangeStartTime.Default;
public static DateTime DefaultCustomOpeningRangeStartTime =
DateTime.Parse("09:30", CultureInfo.InvariantCulture);
public static OpeningRangeColorScheme DefaultOpeningRangeColorScheme =
OpeningRangeColorScheme.Default;
public static Stroke DefaultOpeningRangeStroke =
new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3);
public static Stroke DefaultOpeningRangeMidStroke =
new Stroke(Brushes.Gray, DashStyleHelper.Dash, 2);
public static Stroke DefaultPriceAboveStroke =
new Stroke(Brushes.LimeGreen, DashStyleHelper.Solid, 3);
public static Stroke DefaultPriceBelowStroke =
new Stroke(Brushes.Red, DashStyleHelper.Solid, 3);
public static Stroke DefaultPriceInsideStroke =
new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3);
public static SimpleFont DefaultOpeningRangeFont =
new SimpleFont("Arial", 12);
public static Brush DefaultOpeningRangeFontColor = Brushes.LightGray;
public static bool DefaultShowLabels = true;
public static string DefaultOpeningRangeHighLabel = "ORH @ {level}";
public static string DefaultOpeningRangeLowLabel = "ORL @ {level}";
public static string DefaultOpeningRangeMidLabel = "ORM @ {level}";
public static OpeningRangeLabelPosition DefaultOpeningRangeLabelPosition =
OpeningRangeLabelPosition.Above;
private const int PrimaryBars = 0;
private int OpeningRangeBars;
private int RegularTradingHoursBars;
private const string RegularTradingHours = "US Equities RTH";
private TimeSpan RegularTradingHoursOpen;
private TimeSpan SessionClose;
private List<OR> OpeningRanges;
private OR CurrentOpeningRange;
private double OpeningRangeHigh;
private double OpeningRangeLow;
private double OpeningRangeMid;
private double LastPrice;
private const int LabelPadding = 5;
private const string LevelFormatString = "{level}";
private int OpeningRangeInSeconds;
protected override void OnStateChange()
{
if (State == State.SetDefaults)
{
Description = @"Opening Range Indicator";
Description = @"Plots the user-defined opening range";
Name = "Opening Range";
Calculate = Calculate.OnPriceChange;
IsOverlay = true;
@ -115,447 +50,102 @@ namespace NinjaTrader.NinjaScript.Indicators
DrawHorizontalGridLines = true;
DrawVerticalGridLines = true;
PaintPriceMarkers = true;
ScaleJustification = ScaleJustification.Right;
ScaleJustification = NinjaTrader.Gui.Chart.ScaleJustification.Right;
//Disable this property if your indicator requires custom values that cumulate with each new market data event.
//See Help Guide for additional information.
IsSuspendedWhileInactive = true;
OpeningRangePeriod = DefaultOpeningRangePeriod;
OpeningRangeType = DefaultOpeningRangeType;
StartTime = DefaultOpeningRangeStartTime;
CustomOpeningRangeStartTime = DefaultCustomOpeningRangeStartTime;
ColorScheme = DefaultOpeningRangeColorScheme;
OpeningRangeHighStroke = DefaultOpeningRangeStroke;
OpeningRangeLowStroke = DefaultOpeningRangeStroke;
OpeningRangeMidStroke = DefaultOpeningRangeMidStroke;
PriceAboveStroke = DefaultPriceAboveStroke;
PriceBelowStroke = DefaultPriceBelowStroke;
PriceInsideStroke = DefaultPriceInsideStroke;
ShowLabels = DefaultShowLabels;
OpeningRangeFont = DefaultOpeningRangeFont;
OpeningRangeFontColor = DefaultOpeningRangeFontColor;
OpeningRangeHighLabel = DefaultOpeningRangeHighLabel;
OpeningRangeHighLabelPosition = DefaultOpeningRangeLabelPosition;
OpeningRangeLowLabel = DefaultOpeningRangeLowLabel;
OpeningRangeLowLabelPosition = DefaultOpeningRangeLabelPosition;
OpeningRangeMidLabel = DefaultOpeningRangeMidLabel;
OpeningRangeMidLabelPosition = DefaultOpeningRangeLabelPosition;
ArePlotsConfigurable = false;
OpeningRangeLength = 30;
OpeningRangeTimeFrame = TimeFrame.Minutes;
OpeningRangeHighStroke = new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3);
OpeningRangeLowStroke = new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3);
AddPlot(Brushes.Transparent, "ORH");
AddPlot(Brushes.Transparent, "ORL");
AddPlot(Brushes.Transparent, "ORM");
}
else if (State == State.Configure)
{
ResetOpeningRange(DateTime.MinValue);
OpeningRanges = new List<OR>();
AddDataSeries(Instrument.FullName, new BarsPeriod {
BarsPeriodType = BarsPeriodType.Second, Value = 1 },
Instrument.MasterInstrument.TradingHours.Name);
OpeningRangeBars = 1;
RegularTradingHoursBars = 0;
if (OpeningRangeStartTime.Default == StartTime && RegularTradingHours != Bars.TradingHours.Name)
OpeningRangeHigh = 0.0;
OpeningRangeLow = 0.0;
if (OpeningRangeTimeFrame == TimeFrame.Seconds)
{
AddDataSeries(Instrument.FullName, new BarsPeriod {
BarsPeriodType = BarsPeriodType.Second, Value = 1 }, RegularTradingHours);
RegularTradingHoursBars = 2;
OpeningRangeBars = 2;
AddDataSeries(Data.BarsPeriodType.Second, 1);
OpeningRangeInSeconds = OpeningRangeLength;
}
}
else if (State == State.DataLoaded)
else if (OpeningRangeTimeFrame == TimeFrame.Hours)
{
SessionIterator regularTradingHoursSession = new SessionIterator(BarsArray[RegularTradingHoursBars]);
RegularTradingHoursOpen = regularTradingHoursSession
.GetTradingDayBeginLocal(regularTradingHoursSession.ActualTradingDayExchange).TimeOfDay;
SessionIterator chartSession = new SessionIterator(BarsArray[PrimaryBars]);
SessionClose = chartSession.GetTradingDayEndLocal(chartSession.ActualTradingDayExchange).TimeOfDay;
AddDataSeries(Data.BarsPeriodType.Minute, 30);
OpeningRangeInSeconds = OpeningRangeLength * 60 * 60;
}
else if (State == State.Historical)
else
{
SetZOrder(-1); // Display behind bars on chart.
AddDataSeries(Data.BarsPeriodType.Minute, 1);
OpeningRangeInSeconds = OpeningRangeLength * 60;
}
}
}
protected override void OnBarUpdate()
{
DateTime now = Times[BarsInProgress][0];
// Converting time to UTC in order to simplify opening range calculation.
DateTime now = Times[BarsInProgress][0].ToUniversalTime();
DateTime marketOpen = now.Date.AddHours(14).AddMinutes(30);
DateTime marketClose = now.Date.AddHours(21);
DateTime openingRangeStartTime = GetOpeningRangeStartTime(now);
if (OpeningRangeBars == BarsInProgress)
{
if (now > openingRangeStartTime && now <= GetOpeningRangeEndTime(openingRangeStartTime))
{
if (CurrentOpeningRange == null)
{
CurrentOpeningRange = new OR
{
High = OpeningRangeHigh,
Low = OpeningRangeLow,
Mid = OpeningRangeMid,
LatestPrice = LastPrice,
StartTime = now
};
OpeningRanges.Add(CurrentOpeningRange);
}
if (Highs[BarsInProgress][0] > OpeningRangeHigh || OpeningRangeHigh == 0.0)
OpeningRangeHigh = Highs[BarsInProgress][0];
if (Lows[BarsInProgress][0] < OpeningRangeLow || OpeningRangeLow == 0.0)
OpeningRangeLow = Lows[BarsInProgress][0];
}
}
if (PrimaryBars == BarsInProgress)
{
if (Bars.IsFirstBarOfSession && IsFirstTickOfBar)
ResetOpeningRange(now);
ORH[0] = OpeningRangeHigh;
ORL[0] = OpeningRangeLow;
OpeningRangeMid = Instrument.MasterInstrument
.RoundToTickSize((OpeningRangeLow + OpeningRangeHigh) / 2.0);
ORM[0] = OpeningRangeMid;
LastPrice = Close[0];
UpdateOpeningRange(now);
}
}
protected override void OnRender(ChartControl chartControl, ChartScale chartScale)
{
base.OnRender(chartControl, chartScale);
SharpDX.Direct2D1.Brush openingRangeMidBrush = OpeningRangeMidStroke.Brush.ToDxBrush(RenderTarget);
TextFormat textFormat = OpeningRangeFont.ToDirectWriteTextFormat();
SharpDX.Direct2D1.Brush textBrush = OpeningRangeFontColor.ToDxBrush(RenderTarget);
foreach (OR openingRange in OpeningRanges)
{
int openingRangeStartX = chartControl.GetXByTime(openingRange.StartTime);
int openingRangeEndX;
if (openingRange.EndTime == default(DateTime))
openingRangeEndX = ChartPanel.X + ChartPanel.W;
else
openingRangeEndX = chartControl.GetXByTime(openingRange.EndTime);
if (openingRange.High > 0.0)
{
double openingRangeHigh = openingRange.High;
float openingRangeHighEndX = openingRangeEndX;
if (ShowLabels && openingRange.EndTime == default(DateTime)
&& openingRangeStartX < ChartPanel.X + ChartPanel.W)
{
SharpDX.Vector2 labelOrigin = DrawLabel(OpeningRangeHighLabel, openingRangeHigh,
OpeningRangeHighLabelPosition, textFormat, textBrush, chartScale);
if (OpeningRangeLabelPosition.Center == OpeningRangeHighLabelPosition)
openingRangeHighEndX = labelOrigin.X - LabelPadding;
}
int openingRangeHighY = chartScale.GetYByValue(openingRangeHigh);
if (openingRangeStartX < openingRangeHighEndX)
{
Stroke openingRangeHighStroke = GetStroke(openingRange, OpeningRangeHighStroke);
SharpDX.Direct2D1.Brush openingRangeHighBrush = openingRangeHighStroke.Brush.ToDxBrush(RenderTarget);
RenderTarget.DrawLine(new SharpDX.Vector2(openingRangeStartX, openingRangeHighY),
new SharpDX.Vector2(openingRangeHighEndX, openingRangeHighY),
openingRangeHighBrush, openingRangeHighStroke.Width, openingRangeHighStroke.StrokeStyle);
openingRangeHighBrush.Dispose();
}
}
if (openingRange.Low > 0.0)
{
double openingRangeLow = openingRange.Low;
float openingRangeLowEndX = openingRangeEndX;
if (ShowLabels && openingRange.EndTime == default(DateTime)
&& openingRangeStartX < ChartPanel.X + ChartPanel.W)
{
SharpDX.Vector2 labelOrigin = DrawLabel(OpeningRangeLowLabel, openingRangeLow,
OpeningRangeLowLabelPosition, textFormat, textBrush, chartScale);
if (OpeningRangeLabelPosition.Center == OpeningRangeLowLabelPosition)
openingRangeLowEndX = labelOrigin.X - LabelPadding;
}
int openingRangeLowY = chartScale.GetYByValue(openingRangeLow);
if (openingRangeStartX < openingRangeLowEndX)
{
Stroke openingRangeLowStroke = GetStroke(openingRange, OpeningRangeLowStroke);
SharpDX.Direct2D1.Brush openingRangeLowBrush = openingRangeLowStroke.Brush.ToDxBrush(RenderTarget);
RenderTarget.DrawLine(new SharpDX.Vector2(openingRangeStartX, openingRangeLowY),
new SharpDX.Vector2(openingRangeLowEndX, openingRangeLowY),
openingRangeLowBrush, openingRangeLowStroke.Width, openingRangeLowStroke.StrokeStyle);
openingRangeLowBrush.Dispose();
}
}
if (openingRange.Mid > 0.0)
{
double openingRangeMid = openingRange.Mid;
float openingRangeMidEndX = openingRangeEndX;
if (ShowLabels && openingRange.EndTime == default(DateTime)
&& openingRangeStartX < ChartPanel.X + ChartPanel.W)
{
SharpDX.Vector2 labelOrigin = DrawLabel(OpeningRangeMidLabel, openingRangeMid,
OpeningRangeMidLabelPosition, textFormat, textBrush, chartScale);
if (OpeningRangeLabelPosition.Center == OpeningRangeMidLabelPosition)
openingRangeMidEndX = labelOrigin.X - LabelPadding;
}
int openingRangeMidY = chartScale.GetYByValue(openingRangeMid);
if (openingRangeStartX < openingRangeMidEndX)
RenderTarget.DrawLine(new SharpDX.Vector2(openingRangeStartX, openingRangeMidY),
new SharpDX.Vector2(openingRangeMidEndX, openingRangeMidY),
openingRangeMidBrush, OpeningRangeMidStroke.Width, OpeningRangeMidStroke.StrokeStyle);
}
}
openingRangeMidBrush.Dispose();
textFormat.Dispose();
textBrush.Dispose();
}
private Stroke GetStroke(OR openingRange, Stroke defaultStroke)
{
if (ColorScheme == OpeningRangeColorScheme.PriceBased)
{
if (openingRange.LatestPrice > openingRange.High)
return PriceAboveStroke;
else if (openingRange.LatestPrice < openingRange.Low)
return PriceBelowStroke;
else
return PriceInsideStroke;
}
return defaultStroke;
}
private SharpDX.Vector2 DrawLabel(string label, double level, OpeningRangeLabelPosition position,
TextFormat textFormat, SharpDX.Direct2D1.Brush textBrush, ChartScale chartScale)
{
string labelText = label.Replace(LevelFormatString, level.ToString("F2"));
TextLayout textLayout = new TextLayout(Core.Globals.DirectWriteFactory,
labelText, textFormat, 500, textFormat.FontSize);
int levelY = chartScale.GetYByValue(level);
float labelY;
switch (position)
{
case OpeningRangeLabelPosition.Below:
labelY = ChartPanel.Y + (float)levelY + LabelPadding;
break;
case OpeningRangeLabelPosition.Center:
labelY = ChartPanel.Y + (float)levelY - (textLayout.Metrics.Height / 2.0f);
break;
case OpeningRangeLabelPosition.Above:
default:
labelY = ChartPanel.Y + (float)levelY - textLayout.Metrics.Height - LabelPadding;
break;
}
SharpDX.Vector2 textOrigin = new SharpDX.Vector2(
ChartPanel.W - textLayout.Metrics.Width - LabelPadding, labelY);
RenderTarget.DrawTextLayout(textOrigin, textLayout, textBrush, SharpDX.Direct2D1.DrawTextOptions.NoSnap);
textLayout.Dispose();
return textOrigin;
}
private DateTime GetOpeningRangeStartTime(DateTime now)
{
if (OpeningRangeStartTime.Custom == StartTime)
return now.Date + CustomOpeningRangeStartTime.TimeOfDay;
return now.Date + RegularTradingHoursOpen;
}
private DateTime GetOpeningRangeEndTime(DateTime openingRangeStartTime)
{
switch (OpeningRangeType)
{
case OpeningRangeBarType.Seconds:
return openingRangeStartTime.AddSeconds(OpeningRangePeriod);
case OpeningRangeBarType.Hours:
return openingRangeStartTime.AddHours(OpeningRangePeriod);
default:
return openingRangeStartTime.AddMinutes(OpeningRangePeriod);
}
}
private void UpdateOpeningRange(DateTime now)
{
if (CurrentOpeningRange != null)
{
CurrentOpeningRange.High = OpeningRangeHigh;
CurrentOpeningRange.Low = OpeningRangeLow;
CurrentOpeningRange.Mid = OpeningRangeMid;
DateTime sessionClose = now.Date + SessionClose;
if (GetOpeningRangeStartTime(now) > sessionClose)
sessionClose = sessionClose.AddDays(1);
if (CurrentOpeningRange.EndTime == default(DateTime))
{
CurrentOpeningRange.LatestPrice = LastPrice;
if (now >= sessionClose)
CurrentOpeningRange.EndTime = sessionClose;
}
}
}
private void ResetOpeningRange(DateTime now)
if (now > marketClose)
{
OpeningRangeHigh = 0.0;
OpeningRangeLow = 0.0;
OpeningRangeMid = 0.0;
LastPrice = 0.0;
if (CurrentOpeningRange != null && CurrentOpeningRange.EndTime == default(DateTime))
CurrentOpeningRange.EndTime = now;
CurrentOpeningRange = null;
}
public override string DisplayName
if (BarsInProgress == 0)
{
get { return Name; }
ORH[0] = OpeningRangeHigh;
ORL[0] = OpeningRangeLow;
return;
}
if (now > marketOpen && now <= marketOpen.AddSeconds(OpeningRangeInSeconds))
{
// TODO: The OR would ideally start on the first candle of the market session.
// As it is, the OR begins plotting on the candle just prior to open.
if (Highs[1][0] > OpeningRangeHigh || OpeningRangeHigh == 0.0)
{
RemoveDrawObject("ORH@" + OpeningRangeHigh);
OpeningRangeHigh = Highs[1][0];
Draw.Line(this, "ORH@" + OpeningRangeHigh, true, marketOpen.ToLocalTime(), OpeningRangeHigh, marketClose.ToLocalTime(), OpeningRangeHigh,
OpeningRangeHighStroke.Brush, OpeningRangeHighStroke.DashStyleHelper, (int)OpeningRangeHighStroke.Width);
}
if (Lows[1][0] < OpeningRangeLow || OpeningRangeLow == 0.0)
{
RemoveDrawObject("ORL@" + OpeningRangeLow);
OpeningRangeLow = Lows[1][0];
Draw.Line(this, "ORL@" + OpeningRangeLow, true, marketOpen.ToLocalTime(), OpeningRangeLow, marketClose.ToLocalTime(), OpeningRangeLow,
OpeningRangeLowStroke.Brush, OpeningRangeLowStroke.DashStyleHelper, (int)OpeningRangeLowStroke.Width);
}
}
}
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "Period", Description = "Opening range period", Order = 1, GroupName = "Opening Range")]
public int OpeningRangePeriod
[Display(Name="Length", Description="Length of opening range", Order=1, GroupName="Parameters")]
public int OpeningRangeLength
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Type", Description = "Type of opening range being calculated", Order = 2, GroupName = "Opening Range")]
public OpeningRangeBarType OpeningRangeType
[Display(Name="Time Frame", Description="Time frame on which opening range is being calculated", Order=2, GroupName="Parameters")]
public TimeFrame OpeningRangeTimeFrame
{ get; set; }
[NinjaScriptProperty]
[PropertyEditor("NinjaTrader.Gui.Tools.StringStandardValuesEditorKey")]
[TypeConverter(typeof(OpeningRangeStartTypeConverter))]
[RefreshProperties(RefreshProperties.All)]
[Display(Name = "Start Time", Description = "Start time of opening range", Order = 3, GroupName = "Opening Range")]
public OpeningRangeStartTime StartTime
{ get; set; }
[NinjaScriptProperty]
[PropertyEditor("NinjaTrader.Gui.Tools.AutoCloseTimeEditorKey")]
[Display(Name = "", Description = "Opening range start in local time", Order = 4, GroupName = "Opening Range")]
public DateTime CustomOpeningRangeStartTime
{ get; set; }
[NinjaScriptProperty]
[PropertyEditor("NinjaTrader.Gui.Tools.StringStandardValuesEditorKey")]
[TypeConverter(typeof(OpeningRangeColorSchemeConverter))]
[RefreshProperties(RefreshProperties.All)]
[Display(Name = "Color Scheme", Description = "Opening range coloring scheme", Order = 1, GroupName = "Appearance")]
public OpeningRangeColorScheme ColorScheme
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Price Above", Description = "Opening range lines drawn on chart when price is > ORH", Order = 2, GroupName = "Appearance")]
public Stroke PriceAboveStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Price Below", Description = "Opening range lines drawn on chart when price is < ORL", Order = 3, GroupName = "Appearance")]
public Stroke PriceBelowStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Price Inside", Description = "Opening range lines drawn on chart when price is inside OR", Order = 4, GroupName = "Appearance")]
public Stroke PriceInsideStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Opening Range High", Description = "Opening range high line drawn on chart", Order = 5, GroupName = "Appearance")]
[Display(Name = "Opening Range High", Description = "Opening range high line drawn on chart", Order = 3, GroupName = "Parameters")]
public Stroke OpeningRangeHighStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Opening Range Mid", Description = "Opening range mid line drawn on chart", Order = 6, GroupName = "Appearance")]
public Stroke OpeningRangeMidStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Opening Range Low", Description = "Opening range low line drawn on chart", Order = 7, GroupName = "Appearance")]
[Display(Name = "Opening Range Low", Description = "Opening range low line drawn on chart", Order = 4, GroupName = "Parameters")]
public Stroke OpeningRangeLowStroke
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Show Labels", Order = 1, GroupName = "Labels")]
public bool ShowLabels
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Font", Description = "Font used to display the opening range labels", Order = 2, GroupName = "Labels")]
public SimpleFont OpeningRangeFont
{ get; set; }
[XmlIgnore]
[NinjaScriptProperty]
[Display(Name = "Font Color", Description = "Color of the text used to label the opening range levels", Order = 3, GroupName = "Labels")]
public Brush OpeningRangeFontColor
{ get; set; }
[Browsable(false)]
public string OpeningRangeFontColorSerialization
{
get { return Serialize.BrushToString(OpeningRangeFontColor); }
set { OpeningRangeFontColor = Serialize.StringToBrush(value); }
}
[NinjaScriptProperty]
[Display(Name = "Opening Range High", Order = 4, GroupName = "Labels")]
public string OpeningRangeHighLabel
{ get; set; }
[NinjaScriptProperty]
[Display(Name = " Label Position", Order = 5, GroupName = "Labels")]
public OpeningRangeLabelPosition OpeningRangeHighLabelPosition
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Opening Range Mid", Order = 6, GroupName = "Labels")]
public string OpeningRangeMidLabel
{ get; set; }
[NinjaScriptProperty]
[Display(Name = " Label Position", Order = 7, GroupName = "Labels")]
public OpeningRangeLabelPosition OpeningRangeMidLabelPosition
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Opening Range Low", Order = 8, GroupName = "Labels")]
public string OpeningRangeLowLabel
{ get; set; }
[NinjaScriptProperty]
[Display(Name = " Label Position", Order = 9, GroupName = "Labels")]
public OpeningRangeLabelPosition OpeningRangeLowLabelPosition
{ get; set; }
[Browsable(false)]
[XmlIgnore]
public Series<double> ORH
@ -569,218 +159,6 @@ namespace NinjaTrader.NinjaScript.Indicators
{
get { return Values[1]; }
}
[Browsable(false)]
[XmlIgnore]
public Series<double> ORM
{
get { return Values[2]; }
}
}
public class OpeningRangePropertyConverter : IndicatorBaseConverter
{
public override PropertyDescriptorCollection GetProperties(ITypeDescriptorContext context, object component, Attribute[] attrs)
{
OpeningRange indicator = component as OpeningRange;
PropertyDescriptorCollection properties = base.GetPropertiesSupported(context) ?
base.GetProperties(context, component, attrs) : TypeDescriptor.GetProperties(component, attrs);
if (indicator == null || properties == null)
return properties;
PropertyDescriptor customOpeningRangeStartTime = properties["CustomOpeningRangeStartTime"];
properties.Remove(customOpeningRangeStartTime);
if (indicator.StartTime == OpeningRangeStartTime.Custom)
properties.Add(customOpeningRangeStartTime);
PropertyDescriptor priceAboveStroke = properties["PriceAboveStroke"];
PropertyDescriptor priceBelowStroke = properties["PriceBelowStroke"];
PropertyDescriptor priceInsideStroke = properties["PriceInsideStroke"];
properties.Remove(priceAboveStroke);
properties.Remove(priceBelowStroke);
properties.Remove(priceInsideStroke);
if (indicator.ColorScheme == OpeningRangeColorScheme.PriceBased)
{
PropertyDescriptor openingRangeHighStroke = properties["OpeningRangeHighStroke"];
PropertyDescriptor openingRangeLowStroke = properties["OpeningRangeLowStroke"];
properties.Remove(openingRangeHighStroke);
properties.Remove(openingRangeLowStroke);
properties.Add(priceAboveStroke);
properties.Add(priceBelowStroke);
properties.Add(priceInsideStroke);
}
return properties;
}
public override bool GetPropertiesSupported(ITypeDescriptorContext context)
{ return true; }
}
}
public class OpeningRangeStartTypeConverter : TypeConverter
{
private const string DEFAULT = "US RTH Open";
private const string CUSTOM = "Custom";
public override StandardValuesCollection GetStandardValues(ITypeDescriptorContext context)
{
List<string> values = new List<string>() { DEFAULT, CUSTOM };
return new StandardValuesCollection(values);
}
public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, object value)
{
switch (value.ToString())
{
case DEFAULT:
return OpeningRangeStartTime.Default;
case CUSTOM:
return OpeningRangeStartTime.Custom;
}
return OpeningRangeStartTime.Default;
}
public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType)
{
OpeningRangeStartTime enumValue = (OpeningRangeStartTime)Enum.Parse(typeof(OpeningRangeStartTime), value.ToString());
switch (enumValue)
{
case OpeningRangeStartTime.Default:
return DEFAULT;
case OpeningRangeStartTime.Custom:
return CUSTOM;
}
return DEFAULT;
}
public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
{ return true; }
public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
{ return true; }
public override bool GetStandardValuesExclusive(ITypeDescriptorContext context)
{ return true; }
public override bool GetStandardValuesSupported(ITypeDescriptorContext context)
{ return true; }
}
public class OpeningRangeColorSchemeConverter : TypeConverter
{
private const string DEFAULT = "Default";
private const string PRICE_BASED = "Price Based";
public override StandardValuesCollection GetStandardValues(ITypeDescriptorContext context)
{
List<string> values = new List<string>() { DEFAULT, PRICE_BASED };
return new StandardValuesCollection(values);
}
public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, object value)
{
switch (value.ToString())
{
case DEFAULT:
return OpeningRangeColorScheme.Default;
case PRICE_BASED:
return OpeningRangeColorScheme.PriceBased;
}
return OpeningRangeColorScheme.Default;
}
public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType)
{
OpeningRangeColorScheme enumValue = (OpeningRangeColorScheme)Enum.Parse(typeof(OpeningRangeColorScheme), value.ToString());
switch (enumValue)
{
case OpeningRangeColorScheme.Default:
return DEFAULT;
case OpeningRangeColorScheme.PriceBased:
return PRICE_BASED;
}
return DEFAULT;
}
public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
{ return true; }
public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
{ return true; }
public override bool GetStandardValuesExclusive(ITypeDescriptorContext context)
{ return true; }
public override bool GetStandardValuesSupported(ITypeDescriptorContext context)
{ return true; }
}
public enum OpeningRangeBarType
{
Seconds,
Minutes,
Hours
}
public enum OpeningRangeStartTime
{
Default,
Custom
}
public enum OpeningRangeColorScheme
{
Default,
PriceBased
}
public enum OpeningRangeLabelPosition
{
Above,
Below,
Center
}
// Creates an easy to use constructor for use in strategies.
namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.OpeningRange OpeningRange(int openingRangePeriod, OpeningRangeBarType openingRangeType)
{
return indicator.OpeningRange(
Input,
openingRangePeriod,
openingRangeType,
Indicators.OpeningRange.DefaultOpeningRangeStartTime,
Indicators.OpeningRange.DefaultCustomOpeningRangeStartTime,
Indicators.OpeningRange.DefaultOpeningRangeColorScheme,
Indicators.OpeningRange.DefaultPriceAboveStroke,
Indicators.OpeningRange.DefaultPriceBelowStroke,
Indicators.OpeningRange.DefaultPriceInsideStroke,
Indicators.OpeningRange.DefaultOpeningRangeStroke,
Indicators.OpeningRange.DefaultOpeningRangeMidStroke,
Indicators.OpeningRange.DefaultOpeningRangeStroke,
Indicators.OpeningRange.DefaultShowLabels,
Indicators.OpeningRange.DefaultOpeningRangeFont,
Indicators.OpeningRange.DefaultOpeningRangeFontColor,
Indicators.OpeningRange.DefaultOpeningRangeHighLabel,
Indicators.OpeningRange.DefaultOpeningRangeLabelPosition,
Indicators.OpeningRange.DefaultOpeningRangeMidLabel,
Indicators.OpeningRange.DefaultOpeningRangeLabelPosition,
Indicators.OpeningRange.DefaultOpeningRangeLowLabel,
Indicators.OpeningRange.DefaultOpeningRangeLabelPosition
);
}
}
}
@ -791,18 +169,18 @@ namespace NinjaTrader.NinjaScript.Indicators
public partial class Indicator : NinjaTrader.Gui.NinjaScript.IndicatorRenderBase
{
private OpeningRange[] cacheOpeningRange;
public OpeningRange OpeningRange(int openingRangePeriod, OpeningRangeBarType openingRangeType, OpeningRangeStartTime startTime, DateTime customOpeningRangeStartTime, OpeningRangeColorScheme colorScheme, Stroke priceAboveStroke, Stroke priceBelowStroke, Stroke priceInsideStroke, Stroke openingRangeHighStroke, Stroke openingRangeMidStroke, Stroke openingRangeLowStroke, bool showLabels, SimpleFont openingRangeFont, Brush openingRangeFontColor, string openingRangeHighLabel, OpeningRangeLabelPosition openingRangeHighLabelPosition, string openingRangeMidLabel, OpeningRangeLabelPosition openingRangeMidLabelPosition, string openingRangeLowLabel, OpeningRangeLabelPosition openingRangeLowLabelPosition)
public OpeningRange OpeningRange(int openingRangeLength, TimeFrame openingRangeTimeFrame, Stroke openingRangeHighStroke, Stroke openingRangeLowStroke)
{
return OpeningRange(Input, openingRangePeriod, openingRangeType, startTime, customOpeningRangeStartTime, colorScheme, priceAboveStroke, priceBelowStroke, priceInsideStroke, openingRangeHighStroke, openingRangeMidStroke, openingRangeLowStroke, showLabels, openingRangeFont, openingRangeFontColor, openingRangeHighLabel, openingRangeHighLabelPosition, openingRangeMidLabel, openingRangeMidLabelPosition, openingRangeLowLabel, openingRangeLowLabelPosition);
return OpeningRange(Input, openingRangeLength, openingRangeTimeFrame, openingRangeHighStroke, openingRangeLowStroke);
}
public OpeningRange OpeningRange(ISeries<double> input, int openingRangePeriod, OpeningRangeBarType openingRangeType, OpeningRangeStartTime startTime, DateTime customOpeningRangeStartTime, OpeningRangeColorScheme colorScheme, Stroke priceAboveStroke, Stroke priceBelowStroke, Stroke priceInsideStroke, Stroke openingRangeHighStroke, Stroke openingRangeMidStroke, Stroke openingRangeLowStroke, bool showLabels, SimpleFont openingRangeFont, Brush openingRangeFontColor, string openingRangeHighLabel, OpeningRangeLabelPosition openingRangeHighLabelPosition, string openingRangeMidLabel, OpeningRangeLabelPosition openingRangeMidLabelPosition, string openingRangeLowLabel, OpeningRangeLabelPosition openingRangeLowLabelPosition)
public OpeningRange OpeningRange(ISeries<double> input, int openingRangeLength, TimeFrame openingRangeTimeFrame, Stroke openingRangeHighStroke, Stroke openingRangeLowStroke)
{
if (cacheOpeningRange != null)
for (int idx = 0; idx < cacheOpeningRange.Length; idx++)
if (cacheOpeningRange[idx] != null && cacheOpeningRange[idx].OpeningRangePeriod == openingRangePeriod && cacheOpeningRange[idx].OpeningRangeType == openingRangeType && cacheOpeningRange[idx].StartTime == startTime && cacheOpeningRange[idx].CustomOpeningRangeStartTime == customOpeningRangeStartTime && cacheOpeningRange[idx].ColorScheme == colorScheme && cacheOpeningRange[idx].PriceAboveStroke == priceAboveStroke && cacheOpeningRange[idx].PriceBelowStroke == priceBelowStroke && cacheOpeningRange[idx].PriceInsideStroke == priceInsideStroke && cacheOpeningRange[idx].OpeningRangeHighStroke == openingRangeHighStroke && cacheOpeningRange[idx].OpeningRangeMidStroke == openingRangeMidStroke && cacheOpeningRange[idx].OpeningRangeLowStroke == openingRangeLowStroke && cacheOpeningRange[idx].ShowLabels == showLabels && cacheOpeningRange[idx].OpeningRangeFont == openingRangeFont && cacheOpeningRange[idx].OpeningRangeFontColor == openingRangeFontColor && cacheOpeningRange[idx].OpeningRangeHighLabel == openingRangeHighLabel && cacheOpeningRange[idx].OpeningRangeHighLabelPosition == openingRangeHighLabelPosition && cacheOpeningRange[idx].OpeningRangeMidLabel == openingRangeMidLabel && cacheOpeningRange[idx].OpeningRangeMidLabelPosition == openingRangeMidLabelPosition && cacheOpeningRange[idx].OpeningRangeLowLabel == openingRangeLowLabel && cacheOpeningRange[idx].OpeningRangeLowLabelPosition == openingRangeLowLabelPosition && cacheOpeningRange[idx].EqualsInput(input))
if (cacheOpeningRange[idx] != null && cacheOpeningRange[idx].OpeningRangeLength == openingRangeLength && cacheOpeningRange[idx].OpeningRangeTimeFrame == openingRangeTimeFrame && cacheOpeningRange[idx].OpeningRangeHighStroke == openingRangeHighStroke && cacheOpeningRange[idx].OpeningRangeLowStroke == openingRangeLowStroke && cacheOpeningRange[idx].EqualsInput(input))
return cacheOpeningRange[idx];
return CacheIndicator<OpeningRange>(new OpeningRange(){ OpeningRangePeriod = openingRangePeriod, OpeningRangeType = openingRangeType, StartTime = startTime, CustomOpeningRangeStartTime = customOpeningRangeStartTime, ColorScheme = colorScheme, PriceAboveStroke = priceAboveStroke, PriceBelowStroke = priceBelowStroke, PriceInsideStroke = priceInsideStroke, OpeningRangeHighStroke = openingRangeHighStroke, OpeningRangeMidStroke = openingRangeMidStroke, OpeningRangeLowStroke = openingRangeLowStroke, ShowLabels = showLabels, OpeningRangeFont = openingRangeFont, OpeningRangeFontColor = openingRangeFontColor, OpeningRangeHighLabel = openingRangeHighLabel, OpeningRangeHighLabelPosition = openingRangeHighLabelPosition, OpeningRangeMidLabel = openingRangeMidLabel, OpeningRangeMidLabelPosition = openingRangeMidLabelPosition, OpeningRangeLowLabel = openingRangeLowLabel, OpeningRangeLowLabelPosition = openingRangeLowLabelPosition }, input, ref cacheOpeningRange);
return CacheIndicator<OpeningRange>(new OpeningRange(){ OpeningRangeLength = openingRangeLength, OpeningRangeTimeFrame = openingRangeTimeFrame, OpeningRangeHighStroke = openingRangeHighStroke, OpeningRangeLowStroke = openingRangeLowStroke }, input, ref cacheOpeningRange);
}
}
}
@ -811,14 +189,14 @@ namespace NinjaTrader.NinjaScript.MarketAnalyzerColumns
{
public partial class MarketAnalyzerColumn : MarketAnalyzerColumnBase
{
public Indicators.OpeningRange OpeningRange(int openingRangePeriod, OpeningRangeBarType openingRangeType, OpeningRangeStartTime startTime, DateTime customOpeningRangeStartTime, OpeningRangeColorScheme colorScheme, Stroke priceAboveStroke, Stroke priceBelowStroke, Stroke priceInsideStroke, Stroke openingRangeHighStroke, Stroke openingRangeMidStroke, Stroke openingRangeLowStroke, bool showLabels, SimpleFont openingRangeFont, Brush openingRangeFontColor, string openingRangeHighLabel, OpeningRangeLabelPosition openingRangeHighLabelPosition, string openingRangeMidLabel, OpeningRangeLabelPosition openingRangeMidLabelPosition, string openingRangeLowLabel, OpeningRangeLabelPosition openingRangeLowLabelPosition)
public Indicators.OpeningRange OpeningRange(int openingRangeLength, TimeFrame openingRangeTimeFrame, Stroke openingRangeHighStroke, Stroke openingRangeLowStroke)
{
return indicator.OpeningRange(Input, openingRangePeriod, openingRangeType, startTime, customOpeningRangeStartTime, colorScheme, priceAboveStroke, priceBelowStroke, priceInsideStroke, openingRangeHighStroke, openingRangeMidStroke, openingRangeLowStroke, showLabels, openingRangeFont, openingRangeFontColor, openingRangeHighLabel, openingRangeHighLabelPosition, openingRangeMidLabel, openingRangeMidLabelPosition, openingRangeLowLabel, openingRangeLowLabelPosition);
return indicator.OpeningRange(Input, openingRangeLength, openingRangeTimeFrame, openingRangeHighStroke, openingRangeLowStroke);
}
public Indicators.OpeningRange OpeningRange(ISeries<double> input , int openingRangePeriod, OpeningRangeBarType openingRangeType, OpeningRangeStartTime startTime, DateTime customOpeningRangeStartTime, OpeningRangeColorScheme colorScheme, Stroke priceAboveStroke, Stroke priceBelowStroke, Stroke priceInsideStroke, Stroke openingRangeHighStroke, Stroke openingRangeMidStroke, Stroke openingRangeLowStroke, bool showLabels, SimpleFont openingRangeFont, Brush openingRangeFontColor, string openingRangeHighLabel, OpeningRangeLabelPosition openingRangeHighLabelPosition, string openingRangeMidLabel, OpeningRangeLabelPosition openingRangeMidLabelPosition, string openingRangeLowLabel, OpeningRangeLabelPosition openingRangeLowLabelPosition)
public Indicators.OpeningRange OpeningRange(ISeries<double> input , int openingRangeLength, TimeFrame openingRangeTimeFrame, Stroke openingRangeHighStroke, Stroke openingRangeLowStroke)
{
return indicator.OpeningRange(input, openingRangePeriod, openingRangeType, startTime, customOpeningRangeStartTime, colorScheme, priceAboveStroke, priceBelowStroke, priceInsideStroke, openingRangeHighStroke, openingRangeMidStroke, openingRangeLowStroke, showLabels, openingRangeFont, openingRangeFontColor, openingRangeHighLabel, openingRangeHighLabelPosition, openingRangeMidLabel, openingRangeMidLabelPosition, openingRangeLowLabel, openingRangeLowLabelPosition);
return indicator.OpeningRange(input, openingRangeLength, openingRangeTimeFrame, openingRangeHighStroke, openingRangeLowStroke);
}
}
}
@ -827,14 +205,14 @@ namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.OpeningRange OpeningRange(int openingRangePeriod, OpeningRangeBarType openingRangeType, OpeningRangeStartTime startTime, DateTime customOpeningRangeStartTime, OpeningRangeColorScheme colorScheme, Stroke priceAboveStroke, Stroke priceBelowStroke, Stroke priceInsideStroke, Stroke openingRangeHighStroke, Stroke openingRangeMidStroke, Stroke openingRangeLowStroke, bool showLabels, SimpleFont openingRangeFont, Brush openingRangeFontColor, string openingRangeHighLabel, OpeningRangeLabelPosition openingRangeHighLabelPosition, string openingRangeMidLabel, OpeningRangeLabelPosition openingRangeMidLabelPosition, string openingRangeLowLabel, OpeningRangeLabelPosition openingRangeLowLabelPosition)
public Indicators.OpeningRange OpeningRange(int openingRangeLength, TimeFrame openingRangeTimeFrame, Stroke openingRangeHighStroke, Stroke openingRangeLowStroke)
{
return indicator.OpeningRange(Input, openingRangePeriod, openingRangeType, startTime, customOpeningRangeStartTime, colorScheme, priceAboveStroke, priceBelowStroke, priceInsideStroke, openingRangeHighStroke, openingRangeMidStroke, openingRangeLowStroke, showLabels, openingRangeFont, openingRangeFontColor, openingRangeHighLabel, openingRangeHighLabelPosition, openingRangeMidLabel, openingRangeMidLabelPosition, openingRangeLowLabel, openingRangeLowLabelPosition);
return indicator.OpeningRange(Input, openingRangeLength, openingRangeTimeFrame, openingRangeHighStroke, openingRangeLowStroke);
}
public Indicators.OpeningRange OpeningRange(ISeries<double> input , int openingRangePeriod, OpeningRangeBarType openingRangeType, OpeningRangeStartTime startTime, DateTime customOpeningRangeStartTime, OpeningRangeColorScheme colorScheme, Stroke priceAboveStroke, Stroke priceBelowStroke, Stroke priceInsideStroke, Stroke openingRangeHighStroke, Stroke openingRangeMidStroke, Stroke openingRangeLowStroke, bool showLabels, SimpleFont openingRangeFont, Brush openingRangeFontColor, string openingRangeHighLabel, OpeningRangeLabelPosition openingRangeHighLabelPosition, string openingRangeMidLabel, OpeningRangeLabelPosition openingRangeMidLabelPosition, string openingRangeLowLabel, OpeningRangeLabelPosition openingRangeLowLabelPosition)
public Indicators.OpeningRange OpeningRange(ISeries<double> input , int openingRangeLength, TimeFrame openingRangeTimeFrame, Stroke openingRangeHighStroke, Stroke openingRangeLowStroke)
{
return indicator.OpeningRange(input, openingRangePeriod, openingRangeType, startTime, customOpeningRangeStartTime, colorScheme, priceAboveStroke, priceBelowStroke, priceInsideStroke, openingRangeHighStroke, openingRangeMidStroke, openingRangeLowStroke, showLabels, openingRangeFont, openingRangeFontColor, openingRangeHighLabel, openingRangeHighLabelPosition, openingRangeMidLabel, openingRangeMidLabelPosition, openingRangeLowLabel, openingRangeLowLabelPosition);
return indicator.OpeningRange(input, openingRangeLength, openingRangeTimeFrame, openingRangeHighStroke, openingRangeLowStroke);
}
}
}

View File

@ -1,366 +0,0 @@
#region Using declarations
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media;
using System.Xml.Serialization;
using NinjaTrader.Cbi;
using NinjaTrader.Gui;
using NinjaTrader.Gui.Chart;
using NinjaTrader.Gui.SuperDom;
using NinjaTrader.Gui.Tools;
using NinjaTrader.Data;
using NinjaTrader.NinjaScript;
using NinjaTrader.Core.FloatingPoint;
using NinjaTrader.NinjaScript.DrawingTools;
using System.Globalization;
#endregion
//This namespace holds Indicators in this folder and is required. Do not change it.
namespace NinjaTrader.NinjaScript.Indicators
{
[CategoryOrder("RSI Histogram", 1)]
[CategoryOrder("Appearance", 2)]
public class RSIHistogram : Indicator
{
public static RSIHistogramColorScheme DefaultColorScheme = RSIHistogramColorScheme.PreviousBar;
public static Brush DefaultPositiveBarColor = Brushes.LimeGreen;
public static Brush DefaultNegativeBarColor = Brushes.Red;
public static double DefaultUpperLevel = 20.0;
public static double DefaultLowerLevel = -20.0;
public static bool DefaultShowLevels = true;
public static Stroke DefaultLevelStroke = new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 2);
private RSI RSIValue;
protected override void OnStateChange()
{
if (State == State.SetDefaults)
{
Description = @"Credit to @BestForexMethod on Twitter";
Name = "RSI Histogram";
Calculate = Calculate.OnPriceChange;
IsOverlay = false;
DisplayInDataBox = true;
DrawOnPricePanel = true;
DrawHorizontalGridLines = true;
DrawVerticalGridLines = true;
PaintPriceMarkers = true;
ScaleJustification = ScaleJustification.Right;
IsSuspendedWhileInactive = true;
ArePlotsConfigurable = false;
AreLinesConfigurable = false;
RSIPeriod = 8;
RSISmoothing = 3;
RSIModifier = 1.5;
ColorScheme = DefaultColorScheme;
PositiveBarColor = DefaultPositiveBarColor;
NegativeBarColor = DefaultNegativeBarColor;
ShowLevels = DefaultShowLevels;
UpperLevel = DefaultUpperLevel;
LowerLevel = DefaultLowerLevel;
UpperLevelStroke = DefaultLevelStroke;
LowerLevelStroke = DefaultLevelStroke;
AddPlot(new Stroke(Brushes.Transparent, 0), PlotStyle.Bar, "Histogram");
}
else if (State == State.Configure)
{
if (ShowLevels)
{
AddLine(UpperLevelStroke, UpperLevel, "Upper Level");
AddLine(LowerLevelStroke, LowerLevel, "Lower Level");
}
}
else if (State == State.DataLoaded)
{
RSIValue = RSI(RSIPeriod, RSISmoothing);
}
}
protected override void OnBarUpdate()
{
if (CurrentBar < 2)
return;
double rsiHistogramValue = (RSIValue[0] - 50) * RSIModifier;
// If no change in value, use previous bar color, regardless of color scheme.
if (rsiHistogramValue == Histogram[1])
PlotBrushes[0][0] = PlotBrushes[0][1];
else
{
if ((ColorScheme == RSIHistogramColorScheme.PositiveNegative && rsiHistogramValue > 0.0) ||
(ColorScheme == RSIHistogramColorScheme.PreviousBar && rsiHistogramValue > Histogram[1]))
PlotBrushes[0][0] = PositiveBarColor;
else
PlotBrushes[0][0] = NegativeBarColor;
}
Histogram[0] = rsiHistogramValue;
}
protected override void OnRender(ChartControl chartControl, ChartScale chartScale)
{
base.OnRender(chartControl, chartScale);
foreach (Plot plot in Plots)
{
plot.Width = (float)chartControl.GetBarPaintWidth(ChartBars);
}
}
public override string DisplayName
{
get { return Name; }
}
#region Properties
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "RSI Period", Description = "Period used in RSI calculation", Order = 1, GroupName = "RSI Histogram")]
public int RSIPeriod
{ get; set; }
[NinjaScriptProperty]
[Range(1, int.MaxValue)]
[Display(Name = "RSI Smoothing", Description = "Smoothing used in RSI calculation", Order = 2, GroupName = "RSI Histogram")]
public int RSISmoothing
{ get; set; }
[NinjaScriptProperty]
[Range(0.1, double.MaxValue)]
[Display(Name = "RSI Modifier", Description = "Modifier used to scale RSI calculation", Order = 3, GroupName = "RSI Histogram")]
public double RSIModifier
{ get; set; }
[NinjaScriptProperty]
[PropertyEditor("NinjaTrader.Gui.Tools.StringStandardValuesEditorKey")]
[TypeConverter(typeof(RSIHistogramColorSchemeConverter))]
[Display(Name = "Color Scheme", Description = "RSI histogram coloring scheme", Order = 4, GroupName = "RSI Histogram")]
public RSIHistogramColorScheme ColorScheme
{ get; set; }
[NinjaScriptProperty]
[XmlIgnore]
[Display(Name = "Positive Bar Color", Description = "Color of bars with a positive value", Order = 5, GroupName = "RSI Histogram")]
public Brush PositiveBarColor
{ get; set; }
[NinjaScriptProperty]
[XmlIgnore]
[Display(Name = "Negative Bar Color", Description = "Color of bars with a negative value", Order = 6, GroupName = "RSI Histogram")]
public Brush NegativeBarColor
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Show Levels", Order = 1, GroupName = "Levels")]
public bool ShowLevels
{ get; set; }
[NinjaScriptProperty]
[Range(double.MinValue, double.MaxValue)]
[Display(Name = "Upper Level", Description = "Upper level in histogram", Order = 2, GroupName = "Levels")]
public double UpperLevel
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Upper Level Style", Description = "Upper level displayed in histogram", Order = 3, GroupName = "Levels")]
public Stroke UpperLevelStroke
{ get; set; }
[NinjaScriptProperty]
[Range(double.MinValue, double.MaxValue)]
[Display(Name = "Lower Level", Description = "Lower level in histogram", Order = 4, GroupName = "Levels")]
public double LowerLevel
{ get; set; }
[NinjaScriptProperty]
[Display(Name = "Lower Level Style", Description = "Lower level displayed in histogram", Order = 5, GroupName = "Levels")]
public Stroke LowerLevelStroke
{ get; set; }
[Browsable(false)]
[XmlIgnore]
public Series<double> Histogram
{
get { return Values[0]; }
}
#endregion
}
}
public enum RSIHistogramColorScheme
{
PositiveNegative,
PreviousBar
}
public class RSIHistogramColorSchemeConverter : TypeConverter
{
private const string POSITIVE_NEGATIVE = "Above / Below 0";
private const string PREVIOUS_BAR = "Based on Previous Bar";
public override StandardValuesCollection GetStandardValues(ITypeDescriptorContext context)
{
List<string> values = new List<string>() { POSITIVE_NEGATIVE, PREVIOUS_BAR };
return new StandardValuesCollection(values);
}
public override object ConvertFrom(ITypeDescriptorContext context, CultureInfo culture, object value)
{
switch (value.ToString())
{
case POSITIVE_NEGATIVE:
return RSIHistogramColorScheme.PositiveNegative;
case PREVIOUS_BAR:
return RSIHistogramColorScheme.PreviousBar;
}
return RSIHistogramColorScheme.PreviousBar;
}
public override object ConvertTo(ITypeDescriptorContext context, CultureInfo culture, object value, Type destinationType)
{
RSIHistogramColorScheme enumValue = (RSIHistogramColorScheme)Enum.Parse(typeof(RSIHistogramColorScheme), value.ToString());
switch (enumValue)
{
case RSIHistogramColorScheme.PositiveNegative:
return POSITIVE_NEGATIVE;
case RSIHistogramColorScheme.PreviousBar:
return PREVIOUS_BAR;
}
return PREVIOUS_BAR;
}
public override bool CanConvertFrom(ITypeDescriptorContext context, Type sourceType)
{ return true; }
public override bool CanConvertTo(ITypeDescriptorContext context, Type destinationType)
{ return true; }
public override bool GetStandardValuesExclusive(ITypeDescriptorContext context)
{ return true; }
public override bool GetStandardValuesSupported(ITypeDescriptorContext context)
{ return true; }
}
namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.RSIHistogram RSIHistogram(int rsiPeriod, int rsiSmoothing, double rsiModifier)
{
return indicator.RSIHistogram(
Input,
rsiPeriod,
rsiSmoothing,
rsiModifier,
Indicators.RSIHistogram.DefaultColorScheme,
Indicators.RSIHistogram.DefaultPositiveBarColor,
Indicators.RSIHistogram.DefaultNegativeBarColor,
Indicators.RSIHistogram.DefaultShowLevels,
Indicators.RSIHistogram.DefaultUpperLevel,
Indicators.RSIHistogram.DefaultLevelStroke,
Indicators.RSIHistogram.DefaultLowerLevel,
Indicators.RSIHistogram.DefaultLevelStroke
);
}
}
}
namespace NinjaTrader.NinjaScript.Indicators
{
public partial class Indicator : NinjaTrader.Gui.NinjaScript.IndicatorRenderBase
{
public RSIHistogram RSIHistogram(int rsiPeriod, int rsiSmoothing, double rsiModifier)
{
return RSIHistogram(
Input,
rsiPeriod,
rsiSmoothing,
rsiModifier,
Indicators.RSIHistogram.DefaultColorScheme,
Indicators.RSIHistogram.DefaultPositiveBarColor,
Indicators.RSIHistogram.DefaultNegativeBarColor,
Indicators.RSIHistogram.DefaultShowLevels,
Indicators.RSIHistogram.DefaultUpperLevel,
Indicators.RSIHistogram.DefaultLevelStroke,
Indicators.RSIHistogram.DefaultLowerLevel,
Indicators.RSIHistogram.DefaultLevelStroke
);
}
}
}
#region NinjaScript generated code. Neither change nor remove.
namespace NinjaTrader.NinjaScript.Indicators
{
public partial class Indicator : NinjaTrader.Gui.NinjaScript.IndicatorRenderBase
{
private RSIHistogram[] cacheRSIHistogram;
public RSIHistogram RSIHistogram(int rSIPeriod, int rSISmoothing, double rSIModifier, RSIHistogramColorScheme colorScheme, Brush positiveBarColor, Brush negativeBarColor, bool showLevels, double upperLevel, Stroke upperLevelStroke, double lowerLevel, Stroke lowerLevelStroke)
{
return RSIHistogram(Input, rSIPeriod, rSISmoothing, rSIModifier, colorScheme, positiveBarColor, negativeBarColor, showLevels, upperLevel, upperLevelStroke, lowerLevel, lowerLevelStroke);
}
public RSIHistogram RSIHistogram(ISeries<double> input, int rSIPeriod, int rSISmoothing, double rSIModifier, RSIHistogramColorScheme colorScheme, Brush positiveBarColor, Brush negativeBarColor, bool showLevels, double upperLevel, Stroke upperLevelStroke, double lowerLevel, Stroke lowerLevelStroke)
{
if (cacheRSIHistogram != null)
for (int idx = 0; idx < cacheRSIHistogram.Length; idx++)
if (cacheRSIHistogram[idx] != null && cacheRSIHistogram[idx].RSIPeriod == rSIPeriod && cacheRSIHistogram[idx].RSISmoothing == rSISmoothing && cacheRSIHistogram[idx].RSIModifier == rSIModifier && cacheRSIHistogram[idx].ColorScheme == colorScheme && cacheRSIHistogram[idx].PositiveBarColor == positiveBarColor && cacheRSIHistogram[idx].NegativeBarColor == negativeBarColor && cacheRSIHistogram[idx].ShowLevels == showLevels && cacheRSIHistogram[idx].UpperLevel == upperLevel && cacheRSIHistogram[idx].UpperLevelStroke == upperLevelStroke && cacheRSIHistogram[idx].LowerLevel == lowerLevel && cacheRSIHistogram[idx].LowerLevelStroke == lowerLevelStroke && cacheRSIHistogram[idx].EqualsInput(input))
return cacheRSIHistogram[idx];
return CacheIndicator<RSIHistogram>(new RSIHistogram(){ RSIPeriod = rSIPeriod, RSISmoothing = rSISmoothing, RSIModifier = rSIModifier, ColorScheme = colorScheme, PositiveBarColor = positiveBarColor, NegativeBarColor = negativeBarColor, ShowLevels = showLevels, UpperLevel = upperLevel, UpperLevelStroke = upperLevelStroke, LowerLevel = lowerLevel, LowerLevelStroke = lowerLevelStroke }, input, ref cacheRSIHistogram);
}
}
}
namespace NinjaTrader.NinjaScript.MarketAnalyzerColumns
{
public partial class MarketAnalyzerColumn : MarketAnalyzerColumnBase
{
public Indicators.RSIHistogram RSIHistogram(int rSIPeriod, int rSISmoothing, double rSIModifier, RSIHistogramColorScheme colorScheme, Brush positiveBarColor, Brush negativeBarColor, bool showLevels, double upperLevel, Stroke upperLevelStroke, double lowerLevel, Stroke lowerLevelStroke)
{
return indicator.RSIHistogram(Input, rSIPeriod, rSISmoothing, rSIModifier, colorScheme, positiveBarColor, negativeBarColor, showLevels, upperLevel, upperLevelStroke, lowerLevel, lowerLevelStroke);
}
public Indicators.RSIHistogram RSIHistogram(ISeries<double> input , int rSIPeriod, int rSISmoothing, double rSIModifier, RSIHistogramColorScheme colorScheme, Brush positiveBarColor, Brush negativeBarColor, bool showLevels, double upperLevel, Stroke upperLevelStroke, double lowerLevel, Stroke lowerLevelStroke)
{
return indicator.RSIHistogram(input, rSIPeriod, rSISmoothing, rSIModifier, colorScheme, positiveBarColor, negativeBarColor, showLevels, upperLevel, upperLevelStroke, lowerLevel, lowerLevelStroke);
}
}
}
namespace NinjaTrader.NinjaScript.Strategies
{
public partial class Strategy : NinjaTrader.Gui.NinjaScript.StrategyRenderBase
{
public Indicators.RSIHistogram RSIHistogram(int rSIPeriod, int rSISmoothing, double rSIModifier, RSIHistogramColorScheme colorScheme, Brush positiveBarColor, Brush negativeBarColor, bool showLevels, double upperLevel, Stroke upperLevelStroke, double lowerLevel, Stroke lowerLevelStroke)
{
return indicator.RSIHistogram(Input, rSIPeriod, rSISmoothing, rSIModifier, colorScheme, positiveBarColor, negativeBarColor, showLevels, upperLevel, upperLevelStroke, lowerLevel, lowerLevelStroke);
}
public Indicators.RSIHistogram RSIHistogram(ISeries<double> input , int rSIPeriod, int rSISmoothing, double rSIModifier, RSIHistogramColorScheme colorScheme, Brush positiveBarColor, Brush negativeBarColor, bool showLevels, double upperLevel, Stroke upperLevelStroke, double lowerLevel, Stroke lowerLevelStroke)
{
return indicator.RSIHistogram(input, rSIPeriod, rSISmoothing, rSIModifier, colorScheme, positiveBarColor, negativeBarColor, showLevels, upperLevel, upperLevelStroke, lowerLevel, lowerLevelStroke);
}
}
}
#endregion

View File

@ -1,144 +0,0 @@
#region Using declarations
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Input;
using System.Windows.Media;
using System.Xml.Serialization;
using NinjaTrader.Cbi;
using NinjaTrader.Gui;
using NinjaTrader.Gui.Chart;
using NinjaTrader.Gui.SuperDom;
using NinjaTrader.Gui.Tools;
using NinjaTrader.Data;
using NinjaTrader.NinjaScript;
using NinjaTrader.Core.FloatingPoint;
using NinjaTrader.NinjaScript.Indicators;
using NinjaTrader.NinjaScript.DrawingTools;
#endregion
//This namespace holds Strategies in this folder and is required. Do not change it.
namespace NinjaTrader.NinjaScript.Strategies
{
public class OpeningRangeBreakout : Strategy
{
private OpeningRange OR;
private PivotHighLow Pivots;
private DateTime TradeDate = DateTime.MinValue;
protected override void OnStateChange()
{
if (State == State.SetDefaults)
{
Description = @"Opening range breakout strategy";
Name = "OpeningRangeBreakout";
Calculate = Calculate.OnPriceChange;
EntriesPerDirection = 2;
EntryHandling = EntryHandling.AllEntries;
IsExitOnSessionCloseStrategy = true;
ExitOnSessionCloseSeconds = 30;
IsFillLimitOnTouch = false;
MaximumBarsLookBack = MaximumBarsLookBack.TwoHundredFiftySix;
OrderFillResolution = OrderFillResolution.Standard;
Slippage = 0;
StartBehavior = StartBehavior.WaitUntilFlat;
TimeInForce = TimeInForce.Gtc;
TraceOrders = false;
RealtimeErrorHandling = RealtimeErrorHandling.StopCancelClose;
StopTargetHandling = StopTargetHandling.PerEntryExecution;
BarsRequiredToTrade = 20;
// Disable this property for performance gains in Strategy Analyzer optimizations
// See the Help Guide for additional information
IsInstantiatedOnEachOptimizationIteration = true;
}
else if (State == State.Configure)
{
AddDataSeries(BarsPeriodType.Tick, 1);
AddDataSeries(Instrument.FullName, new BarsPeriod
{
BarsPeriodType = BarsPeriodType.Minute,
Value = 30
}, "US Equities RTH");
}
else if (State == State.DataLoaded)
{
OR = OpeningRange(
30,
OpeningRangeBarType.Minutes,
new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3),
new Stroke(Brushes.Yellow, DashStyleHelper.Solid, 3),
new Stroke(Brushes.Gray, DashStyleHelper.Dot, 1)
);
Pivots = PivotHighLow(10, 10, 3,
new Stroke(Brushes.LimeGreen, DashStyleHelper.Solid, 3),
new Stroke(Brushes.Red, DashStyleHelper.Solid, 3));
}
}
protected override void OnBarUpdate()
{
if (CurrentBar <= BarsRequiredToTrade)
return;
// Ignore tick bars. They are used only for entering trades.
// Also ignore data series used for opening range calculation.
if (BarsInProgress == 1 || BarsInProgress == 2)
return;
if (Position.MarketPosition != MarketPosition.Flat)
{
if (Position.MarketPosition == MarketPosition.Long)
{
SetStopLoss("Long", CalculationMode.Price, Math.Max(OR.ORM[0], Pivots.PivotLow[0]), false);
SetStopLoss("LongRunner", CalculationMode.Price, Math.Max(OR.ORM[0], Pivots.PivotLow[0]), false);
}
else
{
SetStopLoss("Short", CalculationMode.Price, Math.Min(OR.ORM[0], Pivots.PivotHigh[0]), false);
SetStopLoss("ShortRunner", CalculationMode.Price, Math.Min(OR.ORM[0], Pivots.PivotHigh[0]), false);
}
return;
}
if (TradeDate == Time[0].Date)
return;
if (!IsFirstTickOfBar)
return;
// TODO: Change opening range end time to be timezone-agnostic.
if (ToTime(Time[0]) <= 70000)
return;
if (Close[1] > OR.ORH[0])
{
SetStopLoss("Long", CalculationMode.Price, Math.Max(OR.ORM[0], Pivots.PivotLow[0]), false);
SetProfitTarget("Long", CalculationMode.Ticks, (OR.ORH[0] - OR.ORM[0]) / TickSize * 1.5);
EnterLong(1, DefaultQuantity, "Long");
SetStopLoss("LongRunner", CalculationMode.Price, OR.ORM[0], false);
EnterLong(1, DefaultQuantity, "LongRunner");
TradeDate = Time[0].Date;
}
if (Close[1] < OR.ORL[0])
{
SetStopLoss("Short", CalculationMode.Price, Math.Min(OR.ORM[0], Pivots.PivotHigh[0]), false);
SetProfitTarget("Short", CalculationMode.Ticks, (OR.ORM[0] - OR.ORL[0]) / TickSize * 1.5);
EnterShort(1, DefaultQuantity, "Short");
SetStopLoss("ShortRunner", CalculationMode.Price, OR.ORM[0], false);
EnterShort(1, DefaultQuantity, "ShortRunner");
TradeDate = Time[0].Date;
}
}
}
}