2021-10-18 11:41:43 +02:00
|
|
|
//
|
|
|
|
// LineChartView.swift
|
|
|
|
// Mastodon
|
|
|
|
//
|
|
|
|
// Created by Cirno MainasuK on 2021-10-18.
|
|
|
|
//
|
|
|
|
|
|
|
|
import UIKit
|
|
|
|
import Accelerate
|
2022-01-27 14:23:39 +01:00
|
|
|
import MastodonAsset
|
2021-10-18 11:41:43 +02:00
|
|
|
|
2022-04-13 14:43:16 +02:00
|
|
|
public final class LineChartView: UIView {
|
2021-10-18 11:41:43 +02:00
|
|
|
|
2022-04-13 14:43:16 +02:00
|
|
|
public var data: [CGFloat] = [] {
|
2021-10-18 11:41:43 +02:00
|
|
|
didSet {
|
|
|
|
setNeedsLayout()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let lineShapeLayer = CAShapeLayer()
|
|
|
|
let gradientLayer = CAGradientLayer()
|
|
|
|
|
2022-04-13 14:43:16 +02:00
|
|
|
public override init(frame: CGRect) {
|
2021-10-18 11:41:43 +02:00
|
|
|
super.init(frame: frame)
|
|
|
|
_init()
|
|
|
|
}
|
|
|
|
|
2022-04-13 14:43:16 +02:00
|
|
|
public required init?(coder: NSCoder) {
|
2021-10-18 11:41:43 +02:00
|
|
|
super.init(coder: coder)
|
|
|
|
_init()
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension LineChartView {
|
|
|
|
private func _init() {
|
|
|
|
lineShapeLayer.frame = bounds
|
|
|
|
gradientLayer.frame = bounds
|
|
|
|
layer.addSublayer(lineShapeLayer)
|
|
|
|
layer.addSublayer(gradientLayer)
|
|
|
|
|
|
|
|
gradientLayer.colors = [
|
2022-06-02 11:31:23 +02:00
|
|
|
Asset.Colors.Primary._300.color.withAlphaComponent(0.5).cgColor, // set the same alpha to fill
|
|
|
|
Asset.Colors.Primary._300.color.withAlphaComponent(0.5).cgColor,
|
2021-10-18 11:41:43 +02:00
|
|
|
]
|
|
|
|
gradientLayer.startPoint = CGPoint(x: 0.5, y: 0)
|
|
|
|
gradientLayer.endPoint = CGPoint(x: 0.5, y: 1)
|
|
|
|
}
|
|
|
|
|
2022-04-13 14:43:16 +02:00
|
|
|
public override func layoutSubviews() {
|
2021-10-18 11:41:43 +02:00
|
|
|
super.layoutSubviews()
|
|
|
|
|
|
|
|
lineShapeLayer.frame = bounds
|
|
|
|
gradientLayer.frame = bounds
|
|
|
|
|
|
|
|
guard data.count > 1 else {
|
|
|
|
lineShapeLayer.path = nil
|
|
|
|
gradientLayer.isHidden = true
|
|
|
|
return
|
|
|
|
}
|
|
|
|
gradientLayer.isHidden = false
|
|
|
|
|
|
|
|
// Draw smooth chart
|
|
|
|
guard let maxDataPoint = data.max() else {
|
|
|
|
return
|
|
|
|
}
|
2021-10-18 12:35:19 +02:00
|
|
|
func calculateY(for point: CGFloat, in frame: CGRect) -> CGFloat {
|
2021-10-18 11:41:43 +02:00
|
|
|
guard maxDataPoint > 0 else { return .zero }
|
2021-10-18 12:35:19 +02:00
|
|
|
return (1 - point / maxDataPoint) * frame.height
|
2021-10-18 11:41:43 +02:00
|
|
|
}
|
|
|
|
|
2021-10-18 12:35:19 +02:00
|
|
|
let segmentCount = data.count - 1
|
2021-10-18 11:41:43 +02:00
|
|
|
let segmentWidth = bounds.width / CGFloat(segmentCount)
|
|
|
|
|
2021-10-18 12:35:19 +02:00
|
|
|
let points: [CGPoint] = {
|
|
|
|
var points: [CGPoint] = []
|
|
|
|
var x: CGFloat = 0
|
|
|
|
for value in data {
|
|
|
|
let point = CGPoint(x: x, y: calculateY(for: value, in: bounds))
|
|
|
|
points.append(point)
|
|
|
|
x += segmentWidth
|
|
|
|
}
|
|
|
|
return points
|
|
|
|
}()
|
2021-10-18 11:41:43 +02:00
|
|
|
|
2021-10-18 12:35:19 +02:00
|
|
|
guard let linePath = CurveAlgorithm.shared.createCurvedPath(points) else { return }
|
|
|
|
let dotPath = UIBezierPath()
|
2021-10-18 11:41:43 +02:00
|
|
|
|
|
|
|
if let last = points.last {
|
2021-10-18 12:35:19 +02:00
|
|
|
dotPath.addArc(withCenter: last, radius: 3, startAngle: 0, endAngle: 2 * .pi, clockwise: true)
|
2021-10-18 11:41:43 +02:00
|
|
|
}
|
2021-10-18 13:00:55 +02:00
|
|
|
|
2022-01-27 14:23:39 +01:00
|
|
|
lineShapeLayer.lineWidth = 1
|
2022-06-02 11:31:23 +02:00
|
|
|
lineShapeLayer.strokeColor = Asset.Colors.Primary._700.color.cgColor
|
2021-10-18 11:41:43 +02:00
|
|
|
lineShapeLayer.fillColor = UIColor.clear.cgColor
|
2021-10-18 13:00:55 +02:00
|
|
|
lineShapeLayer.lineJoin = .round
|
2021-10-18 11:41:43 +02:00
|
|
|
lineShapeLayer.lineCap = .round
|
|
|
|
lineShapeLayer.path = linePath.cgPath
|
|
|
|
|
|
|
|
let maskPath = UIBezierPath(cgPath: linePath.cgPath)
|
|
|
|
maskPath.addLine(to: CGPoint(x: bounds.maxX, y: bounds.maxY))
|
|
|
|
maskPath.addLine(to: CGPoint(x: bounds.minX, y: bounds.maxY))
|
|
|
|
maskPath.close()
|
|
|
|
let maskLayer = CAShapeLayer()
|
|
|
|
maskLayer.path = maskPath.cgPath
|
2023-06-02 09:52:12 +02:00
|
|
|
maskLayer.fillColor = Asset.Colors.Brand.blurple.color.cgColor
|
2021-10-18 11:41:43 +02:00
|
|
|
maskLayer.strokeColor = UIColor.clear.cgColor
|
|
|
|
maskLayer.lineWidth = 0.0
|
|
|
|
gradientLayer.mask = maskLayer
|
|
|
|
}
|
|
|
|
}
|