CoreText(四) - 长按文字划线功能

在掌阅APP阅读小说时,会出现长按文字划线的功能,自己琢磨了一下,总结一下。

0.效果图

Dec-20-2018 14-34-01.gif

1.添加长按手势

    override init(frame: CGRect) {
    super.init(frame: frame)
    
    backgroundColor = UIColor.white
    
    if longPress == nil {
        longPress = UILongPressGestureRecognizer(target: self, action: #selector(longPressAction(longGesture:)))
        addGestureRecognizer(longPress)
    }
    
    let tapGesture = UITapGestureRecognizer(target: self, action: #selector(tapAction))
    addGestureRecognizer(tapGesture)
}

2.在长按手势的方法里面获取到选中的range以及对应的rect,并刷新界面UI

  • 首先根据点击位置,获取到点击位置及其相邻位置的两个点的range;
  • 将获取到的range转换成对应的frame坐标;
  • 手势滑动时,更新range的位置和对应的frame;
  • 保存手势最初位置的rang,以便于手势移动的更新range的位置;
  • 根据frame绘制界面;
    @objc func longPressAction(longGesture: UILongPressGestureRecognizer) {
    var originPoint = CGPoint.zero
    switch longPress.state {
    case .began:
        print("longPress-Began")
        
        originPoint = longGesture.location(in: self)
        originRange = getTouchLocationRange(point: originPoint)
        rects = getRangeRects(range: selectedRange, ctframe: ctFrame)
        selectedRange = originRange
        setNeedsDisplay()
        
    case .changed:
        
        let finalRange = getTouchLocationRange(point: longGesture.location(in: self))
        if finalRange.location == 0 || finalRange.location == NSNotFound {
            return
        }
        var range = NSRange(location: 0, length: 0)
        range.location = min(finalRange.location, originRange.location)
        if finalRange.location > originRange.location {
            range.length = finalRange.location - originRange.location + finalRange.length
        } else {
            range.length = originRange.location - finalRange.location + originRange.length
        }
 
        selectedRange = range
        rects = getRangeRects(range: selectedRange, ctframe: ctFrame)
        setNeedsDisplay()
    case .ended:
        print("longPress-Ended")
    case .cancelled:
        print("longPress-Cancelled")
    default:
        break
    }
    }

3.其中根据点击位置获取range和根据range获取rect的两个方法如下

 //MARK: - 获取点击位置的两个字符的range
private func getTouchLocationRange(point: CGPoint) -> NSRange {
    var resultRange = NSRange(location: 0, length: 0)
    guard let ctFrame = ctFrame else { return resultRange }
    
    var lines = CTFrameGetLines(ctFrame) as Array
    var origins = [CGPoint](repeating: CGPoint.zero, count: lines.count)
    CTFrameGetLineOrigins(ctFrame, CFRange(location: 0, length: 0), &origins)
    
    for i in 0..<lines.count {
        let line = lines[i] as! CTLine
        let origin = origins[I]
        
        var ascent: CGFloat = 0
        var descent: CGFloat = 0
        
        CTLineGetTypographicBounds(line, &ascent, &descent, nil)
        
        let lineRect = CGRect(x: origin.x, y: self.frame.height - origin.y - (ascent + descent), width: CTLineGetOffsetForStringIndex(line, 100000, nil), height: ascent + descent)
        
        if lineRect.contains(point) {
            
            let lineRange = CTLineGetStringRange(line)
            for j in 0..<lineRange.length {
                
                let index = lineRange.location + j
                
                var offsetX = CTLineGetOffsetForStringIndex(line, index, nil)
                var offsetX2 = CTLineGetOffsetForStringIndex(line, index + 1, nil)
                
                offsetX += origin.x
                offsetX2 += origin.x
                
                let runs = CTLineGetGlyphRuns(line) as Array
                
                for k in 0..<runs.count {
                    let run = runs[k] as! CTRun
                    let runRange = CTRunGetStringRange(run)
                
                    if runRange.location <= index && index <= (runRange.location + runRange.length - 1) {
                        
                        // 说明在当前的run中
                        var ascent: CGFloat = 0
                        var descent: CGFloat = 0
                        
                        CTRunGetTypographicBounds(run, CFRange(location: 0, length: 0), &ascent, &descent, nil)
                        
                        let frame = CGRect(x: offsetX, y: self.frame.height - origin.y - (ascent + descent), width: (offsetX2 - offsetX) * 2, height: ascent + descent)
                        
                        if frame.contains(point) {
                            // 每次获取两个字符的长度
                            resultRange = NSRange(location: index, length: 2)
                        }

                    }
                    
                }
        }
        
    }
    }
    
    return resultRange
}


//MARK: - 获取range所占用的rects
private func getRangeRects(range: NSRange, ctframe: CTFrame?) -> [CGRect] {
    var rects = [CGRect]()
    guard let ctframe = ctframe else { return rects }
    guard range.location != NSNotFound else { return rects }
    
    var lines = CTFrameGetLines(ctframe) as Array
    var origins = [CGPoint](repeating: CGPoint.zero, count: lines.count)
    CTFrameGetLineOrigins(ctframe, CFRange(location: 0, length: 0), &origins)
    
    for i in 0..<lines.count {
        let line = lines[i] as! CTLine
        let origin = origins[I]
        let lineCFRange = CTLineGetStringRange(line)
        
        if lineCFRange.location != NSNotFound {
            let lineRange = NSRange(location: lineCFRange.location, length: lineCFRange.length)
            
            if lineRange.location + lineRange.length > range.location && lineRange.location < (range.location + range.length) {
                
                var ascent: CGFloat = 0
                var descent: CGFloat = 0
                var startX: CGFloat = 0
                
                var contentRange = NSRange(location: range.location, length: 0)
                let end = min(lineRange.location + lineRange.length, range.location + range.length)
                contentRange.length = end - contentRange.location
                
                CTLineGetTypographicBounds(line, &ascent, &descent, nil)
                
                let y = origin.y - descent
                
                startX = CTLineGetOffsetForStringIndex(line, contentRange.location, nil)
                
                let endX = CTLineGetOffsetForStringIndex(line, contentRange.location + contentRange.length, nil)
                
                let rect = CGRect(x: origin.x + startX, y: y, width: endX - startX, height: ascent + descent)
                
                rects.append(rect)
                
            }
            
        }
        
    }

    return rects
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 14,206评论 4 61
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 13,801评论 1 32
  • 画面故事:有一只可怕的怪兽吃掉了热气球星球上所有的颜色。小兔子是这个星球上唯一的一个生物,他要代表这个星球去找回原...
    芒果兔子阅读 1,865评论 0 1
  • 有人说,没有关系不要从政,不要进入体制内。去公司干,人际关系更单纯。 我想说的,当今社会,不管你是干什么的,你在什...
    当道阅读 5,031评论 0 2
  • 【环境】公历2018.6.21 农历2018.5.8 星期四 多云转晴 最高温度33.4℃ 最低温度19℃ 东风3...
    贠大师阅读 1,319评论 0 0