Please note, this blog entry is from a previous course. You might want to check out the current one.
Your graphing UIView must be generic and reusable (i.e. it should be a generic x-y graphing class and know nothing about a CalculatorBrain). Use a protocol to get the graphing view’s data because Views should not own their data.
Basically we need a way to calculate an y value for a given x value, and to know where we want to place the graph and the size of the graph, which we define as protocol:
@protocol GraphViewDataSource - (id)calculateYValueFromXValue:(double)xValue; // NSNumber or string with error @property (nonatomic) CGFloat scale; // 1 = 100% @property (nonatomic) CGPoint origin; // point to place in the middle of the screen @end ... @property (nonatomic, weak) IBOutlet id <GraphViewDataSource> dataSource; ....
and make our controller the data source:
@interface GraphViewController () <GraphViewDataSource>
and
- (void)setGraphView:(GraphView *)graphView { if (graphView == _graphView) return; _graphView = graphView; graphView.dataSource = self; }
As origin and scale are already implemented, we have only to provide the calculation by creating a new dictionary with the given x value and provide it to the model:
- (id)calculateYValueFromXValue:(double)xValue { return [CalculatorBrain runProgram:self.program usingVariableValues:[NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithDouble:xValue], @"x", nil]]; }
Finally we use our data source in the drawing function of the view:
CGFloat scale = self.dataSource.scale; CGPoint origin = self.dataSource.origin; [AxesDrawer drawAxesInRect:area originAtPoint:origin scale:scale]; [[UIColor blackColor] setStroke]; NSInteger widthInPixel = area.size.width * self.contentScaleFactor; CGContextSetLineWidth(context, 2); CGContextBeginPath(context); for (NSInteger xPixel = 0; xPixel <= widthInPixel; xPixel++) { id result = [self.dataSource calculateYValueFromXValue: [self xValueFromPixel:xPixel inRect:area originAtPoint:origin scale:scale]]; if (![result isKindOfClass:[NSNumber class]]) { start = YES; continue; } double y = [result doubleValue]; if (start) { CGContextMoveToPoint(context, [self xPointFromPixel:xPixel inRect:area], [self yPointFromValue:y inRect:area originAtPoint:origin scale:scale] ); start = NO; } else CGContextAddLineToPoint(context, [self xPointFromPixel:xPixel inRect:area], [self yPointFromValue:y inRect:area originAtPoint:origin scale:scale] ); CGContextStrokePath(context);
where we just connect the dots with lines and use a couple of helper functions for the adjustment from/to pixel/points/values:
- (CGFloat)xPointFromPixel:(NSInteger)xPixel inRect:(CGRect)bounds { return xPixel / self.contentScaleFactor + bounds.origin.x; } - (double)xValueFromPixel:(NSInteger)xPixel inRect:(CGRect)bounds originAtPoint:(CGPoint)origin scale:(CGFloat)pointsPerUnit { return ([self xPointFromPixel:xPixel inRect:bounds] - origin.x) / pointsPerUnit; } - (CGFloat)yPointFromValue:(double)y inRect:(CGRect)bounds originAtPoint:(CGPoint)origin scale:(CGFloat)pointsPerUnit { return origin.y - y * pointsPerUnit; }
data:image/s3,"s3://crabby-images/0463e/0463e68a5d77bbec630deb1cc37d4d49d1933e77" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/93960/93960774b56bccf62f288efe326475d585239d8b" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/0c920/0c920341ab12cbcb5a3b1d2fb88f22374f2db577" alt="Share on Reddit reddit"
data:image/s3,"s3://crabby-images/d89b8/d89b80500f1626baf2ffa1e4369c261e01677d36" alt="Pin it with Pinterest pinterest"
data:image/s3,"s3://crabby-images/e9270/e92707689a9d88fc53f343370506e7f2aaedb7d1" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/f159f/f159fb333858fef1148f2a558ee7fa6a6881800a" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/c7ab2/c7ab2f6c43fffe17aac7f185fd64ee95565487b9" alt="Share by email mail"
data:image/s3,"s3://crabby-images/6d85a/6d85a390df03e0c75624ac0579180ad35a62eeac" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/e5f0c/e5f0cd4639d7b3d76b66c05981935f6c2bb06c71" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/40b77/40b7752481674a140035def4b6339fa4accc753f" alt="Share on Reddit reddit"
data:image/s3,"s3://crabby-images/1393f/1393f0291ffd80beac5e1f70b986671c6635d8b7" alt="Pin it with Pinterest pinterest"
data:image/s3,"s3://crabby-images/aa22d/aa22de91a0cd30772c623a5c7979ee81fe426013" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/5dc32/5dc32b39b30592e11870503bb4e8e715d50b21a0" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/56655/56655b42634351b14f9d7507dacc4e33f64b90af" alt="Share by email mail"