美文网首页iOS备忘录
Charts绘制K线图研究过程

Charts绘制K线图研究过程

作者: 程守斌 | 来源:发表于2018-09-11 19:06 被阅读436次

    示例代码

    研究过程

    Charts图表全貌 Charts-Source 蜡烛图详解 组合图详解 K线图表需求分析 Charts问题解决方案

    Charts 源码需改动 CandleStickChartRenderer

    由于_xBounds为internal修饰,所以需将Charts源码手动拖到项目内,再新增改动小部分源码即可。

    //1.在该方法中增加绘制最大值最小值
    open override func drawValues(context: CGContext)
        {
            guard
                let dataProvider = dataProvider,
                let candleData = dataProvider.candleData
                else { return }
            
            // if values are drawn
            if isDrawingValuesAllowed(dataProvider: dataProvider)
            {
                var dataSets = candleData.dataSets
                
                let phaseY = animator.phaseY
                
                var pt = CGPoint()
                
                for i in 0 ..< dataSets.count
                {
                    guard let dataSet = dataSets[i] as? IBarLineScatterCandleBubbleChartDataSet
                        else { continue }
                    
                    if !shouldDrawValues(forDataSet: dataSet)
                    {
                        continue
                    }
                    
                    let valueFont = dataSet.valueFont
                    
                    guard let formatter = dataSet.valueFormatter else { continue }
                    
                    let trans = dataProvider.getTransformer(forAxis: dataSet.axisDependency)
                    let valueToPixelMatrix = trans.valueToPixelMatrix
                    
                    let iconsOffset = dataSet.iconsOffset
                    
                    _xBounds.set(chart: dataProvider, dataSet: dataSet, animator: animator)
                    
                    let lineHeight = valueFont.lineHeight
                    let yOffset: CGFloat = lineHeight + 5.0
                    
                    for j in stride(from: _xBounds.min, through: _xBounds.range + _xBounds.min, by: 1)
                    {
                        guard let e = dataSet.entryForIndex(j) as? CandleChartDataEntry else { break }
                        
                        pt.x = CGFloat(e.x)
                        pt.y = CGFloat(e.high * phaseY)
                        pt = pt.applying(valueToPixelMatrix)
                        
                        if (!viewPortHandler.isInBoundsRight(pt.x))
                        {
                            break
                        }
                        
                        if (!viewPortHandler.isInBoundsLeft(pt.x) || !viewPortHandler.isInBoundsY(pt.y))
                        {
                            continue
                        }
                        
                        if dataSet.isDrawValuesEnabled
                        {
                            ChartUtils.drawText(
                                context: context,
                                text: formatter.stringForValue(
                                    e.high,
                                    entry: e,
                                    dataSetIndex: i,
                                    viewPortHandler: viewPortHandler),
                                point: CGPoint(
                                    x: pt.x,
                                    y: pt.y - yOffset),
                                align: .center,
                                attributes: [NSAttributedStringKey.font: valueFont, NSAttributedStringKey.foregroundColor: dataSet.valueTextColorAt(j)])
                        }
                        
                        if let icon = e.icon, dataSet.isDrawIconsEnabled
                        {
                            ChartUtils.drawImage(context: context,
                                                 image: icon,
                                                 x: pt.x + iconsOffset.x,
                                                 y: pt.y + iconsOffset.y,
                                                 size: icon.size)
                        }
                    }
                }
            }
            
            //TODO:新增绘制最大值最小值
            var dataSets = candleData.dataSets
            
            let phaseY = animator.phaseY
            
            for i in 0 ..< dataSets.count {
                
                guard let dataSet = dataSets[i] as? ICandleChartDataSet , dataSets.count > 0
                    else { continue }
                
                let valueFont = dataSet.valueFont
                
                guard let formatter = dataSet.valueFormatter else
                { continue }
                
                let trans = dataProvider.getTransformer(forAxis: dataSet.axisDependency)
                
                let valueToPixelMatrix = trans.valueToPixelMatrix
                
                _xBounds.set(chart: dataProvider, dataSet: dataSet, animator: animator)
                
                let lineHeight = valueFont.lineHeight
                
                let yOffset:CGFloat = lineHeight
                
                //找出所有画布内的点
                var values = [CandleChartDataEntry]()
                
                for j in stride(from: _xBounds.min, to: _xBounds.range + _xBounds.min, by: 1){
                    guard let e = dataSet.entryForIndex(j) as? CandleChartDataEntry else { break }
                    
                    var pt = CGPoint()
                    pt.x = CGFloat(e.x)
                    pt.y = CGFloat(e.high * phaseY)
                    pt = pt.applying(valueToPixelMatrix)
                    if (!viewPortHandler.isInBoundsRight(pt.x)){
                        break
                    }
                    if (!viewPortHandler.isInBoundsLeft(pt.x) || !viewPortHandler.isInBoundsY(pt.y)){
                        continue
                    }
                    values.append(e)
                }
                
                guard values.count > 0 else { continue }
                
                //计算最大值和最小值
                var maxEntry = values[0]
                var minEntry = values[0]
                for entry in values {
                    if entry.high > maxEntry.high {
                        maxEntry = entry
                    }
                    if entry.low < minEntry.low {
                        minEntry = entry
                    }
                }
                
                var maxPt = CGPoint(x: maxEntry.x, y: maxEntry.high)
                maxPt = maxPt.applying(valueToPixelMatrix)
                var minPt = CGPoint(x: minEntry.x, y: minEntry.low)
                minPt = minPt.applying(valueToPixelMatrix)
                
                if (maxPt.x > minPt.x){
                    //大值画左向,小值画右向
                    
                    var maxTest = formatter.stringForValue(maxEntry.high, entry: maxEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                    maxTest = "\(maxTest) →"
                    
                    let maxPoint = CGPoint(x: maxPt.x, y: maxPt.y - yOffset)
                    ChartUtils.drawText(context: context, text: maxTest, point: maxPoint, align: .right, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColorAt(Int(maxEntry.x))])
                    
                    var minTest = formatter.stringForValue(minEntry.low, entry: minEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                    minTest = "← \(minTest)"
                    let minPoint = CGPoint(x: minPt.x, y: minPt.y)
                    ChartUtils.drawText(context: context, text: minTest, point: minPoint, align: .left, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColorAt(Int(minEntry.x))])
                }else{
                    //大值画右向,小值画左向
                    var maxTest = formatter.stringForValue(maxEntry.high, entry: maxEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                    maxTest = "← \(maxTest)"
                    let maxPoint = CGPoint(x: maxPt.x, y: maxPt.y - yOffset)
                    ChartUtils.drawText(context: context, text: maxTest, point: maxPoint, align: .left, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColor])
                    
                    var minTest = formatter.stringForValue(minEntry.low, entry: minEntry, dataSetIndex: i, viewPortHandler: viewPortHandler)
                    minTest = "\(minTest) →"
                    let minPoint = CGPoint(x: minPt.x, y: minPt.y)
                    ChartUtils.drawText(context: context, text: minTest, point: minPoint, align: .right, attributes: [NSAttributedStringKey.font:valueFont,NSAttributedStringKey.foregroundColor:dataSet.valueTextColorAt(Int(minEntry.x))])
                }
            }
            
        }
    
    //2.该方法中修改高亮原点为收盘价
    open override func drawHighlighted(context: CGContext, indices: [Highlight])
        {
            guard
                let dataProvider = dataProvider,
                let candleData = dataProvider.candleData
                else { return }
            
            context.saveGState()
            
            for high in indices
            {
                guard
                    let set = candleData.getDataSetByIndex(high.dataSetIndex) as? ICandleChartDataSet,
                    set.isHighlightEnabled
                    else { continue }
                
                guard let e = set.entryForXValue(high.x, closestToY: high.y) as? CandleChartDataEntry else { continue }
                
                if !isInBoundsX(entry: e, dataSet: set)
                {
                    continue
                }
                
                let trans = dataProvider.getTransformer(forAxis: set.axisDependency)
                
                context.setStrokeColor(set.highlightColor.cgColor)
                context.setLineWidth(set.highlightLineWidth)
                
                if set.highlightLineDashLengths != nil
                {
                    context.setLineDash(phase: set.highlightLineDashPhase, lengths: set.highlightLineDashLengths!)
                }
                else
                {
                    context.setLineDash(phase: 0.0, lengths: [])
                }
                
    //            let lowValue = e.low * Double(animator.phaseY)
    //            let highValue = e.high * Double(animator.phaseY)
    //            let y = (lowValue + highValue) / 2.0
                //TODO:修改高亮原点为收盘价
                let y = e.close
                
                let pt = trans.pixelForValues(x: e.x, y: y)
                
                high.setDraw(pt: pt)
                
                // draw the lines
                drawHighlightLines(context: context, point: pt, set: set)
            }
            
            context.restoreGState()
        }
    

    各指标算法

    // 格指标算法
    class UtilAlgorithm: NSObject {
        
        /// 计算结果并赋值到model的计算属性
        ///
        /// - Parameter models: 数据集
        static func calculationResults(models:[KLineModel]){
            
            guard models.count > 0 else {
                return
            }
            /// 分时均线
            let aveLineArray   = calculateAvePrice(datas: models)
            for i in 0 ..< models.count {
                models[i].avePrice = aveLineArray[i]
            }
            
            /// SMA
            let smaLine1Array  = calculateSMA(dayCount: KLineConst.kSMALine1Days, datas: models)
            let smaLine2Array  = calculateSMA(dayCount: KLineConst.kSMALine2Days, datas: models)
            for i in 0 ..< models.count {
                models[i].smaLine1 = smaLine1Array[i]
                models[i].smaLine2 = smaLine2Array[i]
            }
            
            /// EMA
            let emaLine1Array  = calculateEMA(dayCount: KLineConst.kEMALine1Days, datas: models)
            let emaLine2Array  = calculateEMA(dayCount: KLineConst.kEMALine2Days, datas: models)
            for i in 0 ..< models.count {
                models[i].emaLine1 = emaLine1Array[i]
                models[i].emaLine2 = emaLine2Array[i]
            }
            
            /// BOLL
            let bollSet = calculateBOLL(dayCount: KLineConst.kBOLLDayCount, k: KLineConst.kBOLL_KValue, datas: models)
            let bollArray = bollSet.0
            let ubArray   = bollSet.1
            let lbArray   = bollSet.2
            for i in 0 ..< models.count {
                models[i].boll = bollArray[i]
                models[i].ub   = ubArray[i]
                models[i].lb   = lbArray[i]
            }
            
            /// MACD
            let macdSet = calculateMACD(p1: KLineConst.kMACD_P1, p2: KLineConst.kMACD_P2, p3: KLineConst.kMACD_P3, datas: models)
            let difArray = macdSet.0
            let deaArray = macdSet.1
            let barArray = macdSet.2
            for i in 0 ..< models.count {
                models[i].dif  = difArray[i]
                models[i].dea  = deaArray[i]
                models[i].bar  = barArray[i]
            }
            
            /// KDJ
            let kdjSet = calculateKDJ(p1: KLineConst.kKDJ_P1, p2: KLineConst.kKDJ_P2, p3: KLineConst.kKDJ_p3, datas: models)
            let kArray = kdjSet.0
            let dArray = kdjSet.1
            let jArray = kdjSet.2
            for i in 0 ..< models.count {
                models[i].k = kArray[i]
                models[i].d = dArray[i]
                models[i].j = jArray[i]
            }
            
            /// RSI
            let line1Array = calculateRSI(dayCount: KLineConst.kRSILine1DayCount, datas: models)
            let line2Array = calculateRSI(dayCount: KLineConst.kRSILine2DayCount, datas: models)
            let line3Array = calculateRSI(dayCount: KLineConst.kRSILine3DayCount, datas: models)
            for i in 0 ..< models.count {
                models[i].rsiLine1 = line1Array[i]
                models[i].rsiLine2 = line2Array[i]
                models[i].rsiLine3 = line3Array[i]
            }
        }
    }
    
    //MARK: - 算法 (fileprivate)
    extension UtilAlgorithm {
        
        /// 分时均线
        ///
        /// - Parameter datas: 数据集
        /// - Returns: 均值数据
        fileprivate static func calculateAvePrice(datas:[KLineModel]) -> [Double] {
            var result = [Double]()
            var totalAmount = 0.0
            var totalVolume = 0.0
            for i in 0 ..< datas.count {
                let model = datas[i]
                totalVolume += model.volume
                totalAmount += model.close * model.volume
                let avePrice = totalAmount / totalVolume
                result.append(avePrice)
            }
            return result
        }
        
        /// SMA(简单均线)
        ///
        /// - Parameters:
        ///   - dayCount: 天数
        ///   - data: 数据集
        /// - Returns: 均值数据
        fileprivate static func calculateSMA(dayCount: Int, datas: [KLineModel]) -> [Double] {
            let dayCount = dayCount - 1
            
            var result = [Double]()
            for i in 0 ..< datas.count {
                if (i < dayCount) {
                    result.append(Double.nan)
                    continue
                }
                var sum: Double = 0.0
                for j in 0 ..< dayCount {
                    sum = sum + datas[i - j].close
                }
                result.append(abs(sum / Double(dayCount)))
            }
            return result
        }
        
        
        /// EMA(指数移动平均数)
        /// EMA(N)=2/(N+1)*(close-昨日EMA)+昨日EMA
        ///
        /// - Parameters:
        ///   - dayCount: 天数
        ///   - datas: 数据集
        /// - Returns: 均值数据
        fileprivate static func calculateEMA(dayCount: Int, datas: [KLineModel]) -> [Double] {
            var yValues = [Double]()
            var prevEma:Double = 0.0    //前一个ema
            for (index, model) in datas.enumerated() {
                let close = model.close
                var ema: Double = 0.0
                if index > 0 {
                    ema = prevEma + (close - prevEma) * 2 / (Double(dayCount) + 1)
                } else {
                    ema = close
                }
                yValues.append(ema)
                prevEma = ema
            }
            return yValues
        }
        
        
        /// BOLL(布林轨道算法)
        /// 计算公式
        /// 中轨线=N日的移动平均线
        /// 上轨线=中轨线+两倍的标准差
        /// 下轨线=中轨线-两倍的标准差
        /// 计算过程
        /// (1)计算MA
        /// MA=N日内的收盘价之和÷N
        /// (2)计算标准差MD
        /// MD=平方根(N)日的(C-MA)的两次方之和除以N
        /// (3)计算MB、UP、DN线
        /// MB=(N)日的MA
        /// UP=MB+k×MD
        /// DN=MB-k×MD
        /// (K为参数,可根据股票的特性来做相应的调整,一般默认为2)
        /// - Parameters:
        ///   - dayCount: 天数
        ///   - k: 参数值
        ///   - datas: 数据集
        /// - Returns: (BOLL,UB,LB)
        fileprivate static func calculateBOLL(dayCount:Int, k:Int=2 ,datas:[KLineModel]) -> ([Double],[Double],[Double]) {
            
            var bollValues = [Double]()
            var ubValues = [Double]()
            var lbValues = [Double]()
            
            // n天标准差
            var mdArray = calculateBOLLSTD(dayCount: dayCount, datas: datas)
            // n天均值
            var mbArray = calculateSMA(dayCount: dayCount, datas: datas)
            
            for i in 0 ..< datas.count {
                let md = mdArray[i]
                let mb = mbArray[i]
                let ub = mb + Double(k) * md
                let lb = mb - Double(k) * md
                bollValues.append(mbArray[i])
                ubValues.append(ub)
                lbValues.append(lb)
            }
            return (bollValues,ubValues,lbValues)
        }
        
        /// 计算布林线中的MA平方差
        ///
        /// - Parameters:
        ///   - dayCount: 天数
        ///   - datas: 数据集
        /// - Returns: 结果
        fileprivate static func calculateBOLLSTD(dayCount:Int, datas:[KLineModel]) -> [Double] {
            
            var mdArray = [Double]()
            
            // n天均值
            var maArray = calculateSMA(dayCount: dayCount, datas: datas)
            
            for index in 0 ..< datas.count {
                if index + 1 >= dayCount {
                    var dx:Double = 0.0
                    for i in stride(from: index, through: index + 1 - dayCount, by: -1) {
                        dx += pow(datas[i].close - maArray[i], 2)
                    }
                    var md = dx / Double(dayCount)
                    md = pow(md, 0.5)
                    mdArray.append(md)
                }else{
                    var dx:Double = 0.0
                    for i in stride(from: index, through: 0, by: -1) {
                        dx += pow(datas[i].close - maArray[i], 2)
                    }
                    var md = dx / Double(dayCount)
                    md = pow(md, 0.5)
                    mdArray.append(md)
                }
            }
            return mdArray
        }
        
    
        /// MACD(平滑异同移动平均线)
        ///
        /// - Parameters:
        ///   - p1: 天数1 (12)
        ///   - p2: 天数2 (26)
        ///   - p3: 天数3 (9)
        ///   - datas: 数据集
        /// - Returns: ([DIF],[DEA],[BAR])
        fileprivate static func calculateMACD(p1:Int, p2:Int, p3:Int, datas:[KLineModel]) -> ([Double],[Double],[Double]){
            var difArray = [Double]()
            var deaArray = [Double]()
            var barArray = [Double]()
            
            //EMA(p1)=2/(p1+1)*(C-昨日EMA)+昨日EMA;
            let p1EmaArray = calculateEMA(dayCount: p1, datas: datas)
            //EMA(p2)=2/(p2+1)*(C-昨日EMA)+昨日EMA;
            let p2EmaArray = calculateEMA(dayCount: p2, datas: datas)
            //昨日dea
            var prevDea:Double = 0.0
            
            for i in 0 ..< datas.count {
                //DIF=今日EMA(p1)- 今日EMA(p2)
                let dif = p1EmaArray[i] - p2EmaArray[i]
                //dea(p3)=2/(p3+1)*(dif-昨日dea)+昨日dea;
                let dea = prevDea + (dif - prevDea) * 2 / (Double(p3) + 1)
                //BAR=2×(DIF-DEA)
                let bar = 2 * (dif - dea)
                prevDea = dea
                
                difArray.append(dif)
                deaArray.append(dea)
                barArray.append(bar)
            }
            return (difArray,deaArray,barArray)
        }
        
        
        /// MDJ()
        ///
        /// - Parameters:
        ///   - p1: k指标周期(9)
        ///   - p2: d指标周期(3)
        ///   - p3: j指标周期(3)
        ///   - datas: 数据集
        /// - Returns: ([K],[D],[J])
        fileprivate static func calculateKDJ(p1:Int, p2:Int, p3:Int, datas:[KLineModel]) -> ([Double],[Double],[Double]) {
            var kArray = [Double]()
            var dArray = [Double]()
            var jArray = [Double]()
            
            var prevK:Double = 50.0
            var prevD:Double = 50.0
            
            let rsvArray = calculateRSV(dayCount: p1, datas: datas)
            
            for i in 0 ..< datas.count {
                let rsv = rsvArray[i]
                let k = (2 * prevK + rsv) / 3
                let d = (2 * prevD + k) / 3
                let j = 3 * k - 2 * d
                prevK = k
                prevD = d
                kArray.append(k)
                dArray.append(d)
                jArray.append(j)
            }
            return (kArray,dArray,jArray)
        }
        
        
        /// RSV(未成熟随机值)
        ///
        /// - Parameters:
        ///   - dayCount: 计算天数范围
        ///   - index: 当前的索引位
        ///   - datas: 数据集
        /// - Returns: RSV集
        fileprivate static func calculateRSV(dayCount:Int, datas:[KLineModel]) -> [Double] {
            var rsvArray = [Double]()
            for i in (0 ..< datas.count) {
                var rsv   = 0.0
                let close = datas[i].close
                var high  = datas[i].high
                var low   = datas[i].low
                
                let startIndex = i < dayCount ? 0 : i - dayCount + 1
                //计算dayCount天内最高价最低价
                for j in (startIndex ..< i){
                    high = datas[j].high > high ? datas[j].high : high
                    low  = datas[j].low  < low  ? datas[j].low  : low
                }
                if high != low {
                    rsv = (close - low) / (high - low) * 100
                }
                rsvArray.append(rsv)
            }
            return rsvArray
        }
        
        
        /// RSI(相对强弱指标)
        /// 算法:N日RSI =N日内收盘涨幅的平均值/(N日内收盘涨幅均值+N日内收盘跌幅均值) ×100
        ///
        /// - Parameters:
        ///   - dayCount: 周期天数
        ///   - datas: 数据源
        /// - Returns: 结果集
        fileprivate static func calculateRSI(dayCount:Int, datas:[KLineModel]) -> [Double]{
            
            var rsiArray = [Double]()   //rsi值
            var ratioArray = [Double]() //涨跌幅
            for model in datas {
                if model.open == 0 {
                    ratioArray.append(0)
                } else {
                    let ratio = (model.close - model.open)/model.open
                    ratioArray.append(ratio)
                }
            }
            for i in 0 ..< ratioArray.count {
                let startIndex = i <= dayCount ? 0 : i-dayCount
                var upSum:Double   = 0.0
                var downSum:Double = 0.0
                for j in startIndex ... i {
                    if ratioArray[j] >= 0 {
                        upSum += ratioArray[j] //n日收盘涨幅总和
                    }else{
                        downSum += ratioArray[j] //n日收盘跌幅总和
                    }
                }
                let upAve = upSum / Double(dayCount)
                let downAve = downSum / Double(dayCount)
                let rsi = upAve / (upAve - downAve) * 100
                rsiArray.append(rsi)
            }
            return rsiArray
        }
    }
    

    相关文章

      网友评论

      • 9a9eb6bfde22:我照着你的demo写了一个,蜡烛图无法显示,其他显示正常

      本文标题:Charts绘制K线图研究过程

      本文链接:https://www.haomeiwen.com/subject/ymtpgftx.html