diff --git a/iphone/Chart/Chart.xcodeproj/project.pbxproj b/iphone/Chart/Chart.xcodeproj/project.pbxproj index 2866a49a3c..483955d281 100644 --- a/iphone/Chart/Chart.xcodeproj/project.pbxproj +++ b/iphone/Chart/Chart.xcodeproj/project.pbxproj @@ -15,7 +15,7 @@ 47375E4D2420E97100FFCC49 /* ChartView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E412420E96D00FFCC49 /* ChartView.swift */; }; 47375E4E2420E97100FFCC49 /* ChartInfoView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E432420E96F00FFCC49 /* ChartInfoView.swift */; }; 47375E4F2420E97100FFCC49 /* ChartPointInfoView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E442420E96F00FFCC49 /* ChartPointInfoView.swift */; }; - 47375E502420E97100FFCC49 /* ChartPointIntersectionsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E452420E96F00FFCC49 /* ChartPointIntersectionsView.swift */; }; + 47375E502420E97100FFCC49 /* ChartPointIntersectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E452420E96F00FFCC49 /* ChartPointIntersectionView.swift */; }; 47375E512420E97100FFCC49 /* ChartMyPositionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E462420E96F00FFCC49 /* ChartMyPositionView.swift */; }; 47375E522420E97100FFCC49 /* ChartYAxisView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E472420E96F00FFCC49 /* ChartYAxisView.swift */; }; 47375E542420E97100FFCC49 /* ChartPreviewView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 47375E492420E97100FFCC49 /* ChartPreviewView.swift */; }; @@ -32,7 +32,7 @@ 47375E412420E96D00FFCC49 /* ChartView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartView.swift; sourceTree = ""; }; 47375E432420E96F00FFCC49 /* ChartInfoView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartInfoView.swift; sourceTree = ""; }; 47375E442420E96F00FFCC49 /* ChartPointInfoView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartPointInfoView.swift; sourceTree = ""; }; - 47375E452420E96F00FFCC49 /* ChartPointIntersectionsView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartPointIntersectionsView.swift; sourceTree = ""; }; + 47375E452420E96F00FFCC49 /* ChartPointIntersectionView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartPointIntersectionView.swift; sourceTree = ""; }; 47375E462420E96F00FFCC49 /* ChartMyPositionView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartMyPositionView.swift; sourceTree = ""; }; 47375E472420E96F00FFCC49 /* ChartYAxisView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartYAxisView.swift; sourceTree = ""; }; 47375E492420E97100FFCC49 /* ChartPreviewView.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = ChartPreviewView.swift; sourceTree = ""; }; @@ -103,7 +103,7 @@ children = ( 47375E432420E96F00FFCC49 /* ChartInfoView.swift */, 47375E442420E96F00FFCC49 /* ChartPointInfoView.swift */, - 47375E452420E96F00FFCC49 /* ChartPointIntersectionsView.swift */, + 47375E452420E96F00FFCC49 /* ChartPointIntersectionView.swift */, 47375E462420E96F00FFCC49 /* ChartMyPositionView.swift */, ); path = ChartInfo; @@ -192,7 +192,7 @@ 47375E512420E97100FFCC49 /* ChartMyPositionView.swift in Sources */, 47375E3E2420E94E00FFCC49 /* ChartData.swift in Sources */, 47375E4C2420E97100FFCC49 /* ChartLineView.swift in Sources */, - 47375E502420E97100FFCC49 /* ChartPointIntersectionsView.swift in Sources */, + 47375E502420E97100FFCC49 /* ChartPointIntersectionView.swift in Sources */, 47375E522420E97100FFCC49 /* ChartYAxisView.swift in Sources */, 47375E4B2420E97100FFCC49 /* ChartXAxisView.swift in Sources */, 47375E542420E97100FFCC49 /* ChartPreviewView.swift in Sources */, diff --git a/iphone/Chart/Chart/ChartData/ChartPresentationData.swift b/iphone/Chart/Chart/ChartData/ChartPresentationData.swift index e885008153..a0b67f9a88 100644 --- a/iphone/Chart/Chart/ChartData/ChartPresentationData.swift +++ b/iphone/Chart/Chart/ChartData/ChartPresentationData.swift @@ -25,16 +25,19 @@ public class ChartPresentationData { var upper = CGFloat(Int.min) func labelAt(_ point: CGFloat) -> String { + formatter.distanceString(from: xAxisValueAt(point)) + } + + func xAxisValueAt(_ point: CGFloat) -> Double { let p1 = Int(floor(point)) let p2 = Int(ceil(point)) let v1 = chartData.xAxisValues[p1] let v2 = chartData.xAxisValues[p2] - let value = v1 + (v2 - v1) * Double(point.truncatingRemainder(dividingBy: 1)) - return formatter.distanceString(from: value) + return v1 + (v2 - v1) * Double(point.truncatingRemainder(dividingBy: 1)) } func lineAt(_ index: Int) -> ChartPresentationLine { - return presentationLines[index] + presentationLines[index] } private func recalcBounds() { diff --git a/iphone/Chart/Chart/Views/ChartInfo/ChartInfoView.swift b/iphone/Chart/Chart/Views/ChartInfo/ChartInfoView.swift index 1e6dcc0006..a18123e49b 100644 --- a/iphone/Chart/Chart/Views/ChartInfo/ChartInfoView.swift +++ b/iphone/Chart/Chart/Views/ChartInfo/ChartInfoView.swift @@ -16,7 +16,7 @@ class ChartInfoView: UIView { weak var delegate: ChartInfoViewDelegate? private let pointInfoView = ChartPointInfoView() - private let pointsView = ChartPointIntersectionsView(frame: CGRect(x: 0, y: 0, width: 2, height: 0)) + private let pointsView = ChartPointIntersectionView(frame: CGRect(x: 0, y: 0, width: 2, height: 0)) private let myPositionView = ChartMyPositionView(frame: CGRect(x: 50, y: 0, width: 2, height: 0)) private var lineInfo: ChartLineInfo? @@ -26,13 +26,12 @@ class ChartInfoView: UIView { } } - var infoX: CGFloat = 0 { - didSet { - if bounds.width > 0 { - let x = bounds.width * infoX - update(x) - updateViews(point: lineInfo!.point) - } + private var _infoX: CGFloat = 0 + var infoX: CGFloat { + get { _infoX } + set { + _infoX = newValue + update(bounds.width * _infoX) } } @@ -68,18 +67,15 @@ class ChartInfoView: UIView { } func update(_ x: CGFloat? = nil) { + guard bounds.width > 0 else { return } let x = x ?? pointsView.center.x + _infoX = x / bounds.width guard let delegate = delegate, let (label, intersectionPoints) = delegate.chartInfoView(self, infoAtPointX: x) else { return } lineInfo = intersectionPoints[0] - self.pointsView.updatePoints(intersectionPoints) + pointsView.updatePoint(lineInfo!) pointInfoView.update(x: x, label: label, points: intersectionPoints) updateViews(point: lineInfo!.point) - let y = max(pointInfoView.frame.height / 2 + 5, - min(bounds.height - pointInfoView.frame.height / 2 - 5, bounds.height - lineInfo!.point.y)); - pointInfoView.center = CGPoint(x: pointInfoView.center.x, y: y) - let arrowPoint = convert(CGPoint(x: 0, y: bounds.height - lineInfo!.point.y), to: pointInfoView) - pointInfoView.arrowY = arrowPoint.y } @objc func onPan(_ sender: UIPanGestureRecognizer) { @@ -96,7 +92,6 @@ class ChartInfoView: UIView { return } update(x) - updateViews(point: lineInfo!.point) } case .ended, .cancelled, .failed: captured = false @@ -105,19 +100,21 @@ class ChartInfoView: UIView { } } - func updateViews(point: CGPoint) { + private func updateViews(point: CGPoint) { pointsView.alpha = 1 pointsView.center = CGPoint(x: point.x, y: bounds.midY) let s = pointInfoView.systemLayoutSizeFitting(UIView.layoutFittingCompressedSize) pointInfoView.frame.size = s + let y = max(pointInfoView.frame.height / 2 + 5, + min(bounds.height - pointInfoView.frame.height / 2 - 5, bounds.height - lineInfo!.point.y)); let orientationChangeX = pointInfoView.alignment == .left ? s.width + 40 : bounds.width - s.width - 40 if point.x > orientationChangeX { pointInfoView.alignment = .left - pointInfoView.center = CGPoint(x: point.x - s.width / 2 - 20, y: pointInfoView.center.y) + pointInfoView.center = CGPoint(x: point.x - s.width / 2 - 20, y: y) } else { pointInfoView.alignment = .right - pointInfoView.center = CGPoint(x: point.x + s.width / 2 + 20, y: pointInfoView.center.y) + pointInfoView.center = CGPoint(x: point.x + s.width / 2 + 20, y: y) } var f = pointInfoView.frame if f.minX < 0 { @@ -127,6 +124,8 @@ class ChartInfoView: UIView { f.origin.x = bounds.width - f.width pointInfoView.frame = f } + let arrowPoint = convert(CGPoint(x: 0, y: bounds.height - lineInfo!.point.y), to: pointInfoView) + pointInfoView.arrowY = arrowPoint.y } override func layoutSubviews() { @@ -141,14 +140,7 @@ class ChartInfoView: UIView { mf.size.height = bounds.height myPositionView.frame = mf - if lineInfo == nil, bounds.width > 0 { - let x = bounds.width * infoX - guard let (date, intersectionPoints) = delegate?.chartInfoView(self, infoAtPointX: x) else { return } - lineInfo = intersectionPoints[0] - pointsView.setPoints(intersectionPoints) - pointInfoView.set(x: x, label: date, points: intersectionPoints) - updateViews(point: lineInfo!.point) - } + update(bounds.width * infoX) } } diff --git a/iphone/Chart/Chart/Views/ChartInfo/ChartPointIntersectionsView.swift b/iphone/Chart/Chart/Views/ChartInfo/ChartPointIntersectionView.swift similarity index 67% rename from iphone/Chart/Chart/Views/ChartInfo/ChartPointIntersectionsView.swift rename to iphone/Chart/Chart/Views/ChartInfo/ChartPointIntersectionView.swift index ae9db6b642..7aabaea11f 100644 --- a/iphone/Chart/Chart/Views/ChartInfo/ChartPointIntersectionsView.swift +++ b/iphone/Chart/Chart/Views/ChartInfo/ChartPointIntersectionView.swift @@ -46,39 +46,33 @@ fileprivate class CircleView: UIView { } } -class ChartPointIntersectionsView: UIView { - fileprivate var intersectionViews: [CircleView] = [] +class ChartPointIntersectionView: UIView { + fileprivate var intersectionView = CircleView() + + var color: UIColor = UIColor(red: 0.14, green: 0.61, blue: 0.95, alpha: 1) { + didSet { + intersectionView.color = color + backgroundColor = color.withAlphaComponent(0.5) + } + } override init(frame: CGRect) { super.init(frame: frame) - backgroundColor = UIColor(red: 0.14, green: 0.61, blue: 0.95, alpha: 0.5) + backgroundColor = color.withAlphaComponent(0.5) transform = CGAffineTransform.identity.scaledBy(x: 1, y: -1) + + intersectionView.color = color + intersectionView.frame = CGRect(x: 0, y: 0, width: 24, height: 24) + intersectionView.center = CGPoint(x: bounds.midX, y: bounds.midY) + addSubview(intersectionView) } required init?(coder aDecoder: NSCoder) { fatalError() } - func setPoints(_ points: [ChartLineInfo]) { - intersectionViews.forEach { $0.removeFromSuperview() } - intersectionViews.removeAll() - - for point in points { - let v = CircleView() - v.color = point.color - v.frame = CGRect(x: 0, y: 0, width: 24, height: 24) - v.center = CGPoint(x: bounds.midX, y: point.point.y) - intersectionViews.append(v) - addSubview(v) - } - } - - func updatePoints(_ points: [ChartLineInfo]) { - for i in 0.. upper { + let current = Double(chartInfoView.infoX) * rangeLength + lower + let dx = x - current + let dIdx = Int(dx / routeLength * Double(chartData.pointsCount)) + var lowerIdx = chartPreviewView.minX + dIdx + var upperIdx = chartPreviewView.maxX + dIdx + if lowerIdx < 0 { + upperIdx -= lowerIdx + lowerIdx = 0 + } else if upperIdx >= chartData.pointsCount { + lowerIdx -= upperIdx - chartData.pointsCount - 1 + upperIdx = chartData.pointsCount - 1 + } + chartPreviewView.setX(min: lowerIdx, max: upperIdx) + lower = chartData.xAxisValueAt(CGFloat(chartPreviewView.minX)) + } + chartInfoView.infoX = CGFloat((x - lower) / rangeLength) + } override public func layoutSubviews() {