RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 1592629
Accepted
Pavel Lazarev
Pavel Lazarev
Asked:2024-09-02 19:40:45 +0000 UTC2024-09-02 19:40:45 +0000 UTC 2024-09-02 19:40:45 +0000 UTC

如何获取 UILabel 中最后一个字符的位置?

  • 772

我需要做这样的事情:

两个 UILabel 的示例 其中“左长文本”和“右文本”是 2 个独立的 UILabels,并且它们之间有一条虚线

为此,我需要知道如何获取最后一个字符的位置,但这是我的猜测)

请告诉我如何获取 UILabel 中最后一个字符的位置,或者也许还有另一种方法来实现这个?

ios
  • 1 1 个回答
  • 15 Views

1 个回答

  • Voted
  1. Best Answer
    schmidt9
    2024-09-03T03:28:47Z2024-09-03T03:28:47Z

    下面是获取所有标签行的边框并使用左右标签最后一行的边框的方法。行边界是通过迭代行中所有字符的坐标来确定的 - 如果两个字符在 Y 轴上的坐标不同,则它们位于不同的行上。然后我们将这些边界转换为屏幕坐标并绘制虚线

    限制:标签必须具有相同的字体大小和文本格式,在测试项目中它们彼此底部对齐

    线条边界是可见的,黑线是线条的完整高度,红线是沿着线条基线的边界(这正是绘制虚线所需的)

    在此输入图像描述

    最终结果

    在此输入图像描述

    测试项目:https://github.com/schmidt9/TwoLabelsWithLineBetween

    //
    //  GlyphPositionLabel.swift
    //  TwoLabelsWithLineBetween
    //
    
    import UIKit
    
    struct TextLineGeometry {
        var fullRect: CGRect // used for debug here
        var baselineRect: CGRect
    }
    
    class GlyphPositionLabel: UILabel {
        
        var debug = false
        
        /// See https://stackoverflow.com/a/77427472/3004003
        var linesGeometry: [TextLineGeometry] {
            guard let text, let attributedText, let font, !text.isEmpty else {
                return []
            }
            
            let textContainer = NSTextContainer(size: bounds.size)
            textContainer.lineFragmentPadding  = 0;
            textContainer.maximumNumberOfLines = self.numberOfLines;
            textContainer.lineBreakMode = self.lineBreakMode;
    
            let textStorage = NSTextStorage(attributedString: attributedText)
            let layoutManager = NSLayoutManager()
            textStorage.addLayoutManager(layoutManager)
            layoutManager.addTextContainer(textContainer)
            
            let baseline = font.ascender
            
            var geometries = [TextLineGeometry]()
            var index = text.startIndex
            var currentRect: CGRect?
            
            while index != text.endIndex {
                let range = index..<text.index(after: index)
                // ignore white spaces
                if text[range].trimmingCharacters(in: .whitespaces).isEmpty {
                    index = text.index(after: index)
                    continue
                }
    
                let rect = layoutManager.boundingRect(forGlyphRange: NSRange(range, in: text), in: textContainer)
                index = text.index(after: index)
                
                if currentRect == nil {
                    currentRect = rect
                } else if abs(currentRect!.minY - rect.minY) < 1 {
                    // glyph on the same line, resize line rect
                    currentRect!.size = CGSize(width: rect.maxX, height: currentRect!.height)
                } else {
                    let geometry = TextLineGeometry(
                        fullRect: currentRect!,
                        baselineRect: CGRect(origin: currentRect!.origin, size: CGSize(width: currentRect!.width, height: baseline))
                    )
                    geometries.append(geometry)
                    currentRect = rect
                }
            }
            
            let geometry = TextLineGeometry(
                fullRect: currentRect!,
                baselineRect: CGRect(origin: currentRect!.origin, size: CGSize(width: currentRect!.width, height: baseline))
            )
            
            geometries.append(geometry)
            
            return geometries
        }
    
        override func draw(_ rect: CGRect) {
            super.draw(rect)
            
            if !debug {
                return
            }
            
            linesGeometry.forEach {
                UIColor.black.setStroke()
                let fullPath = UIBezierPath(rect: $0.fullRect)
                fullPath.stroke()
                
                UIColor.red.setStroke()
                let baseLinePath = UIBezierPath(rect: $0.baselineRect)
                baseLinePath.stroke()
            }
        }
    
    }
    
    //
    //  ViewController.swift
    //  TwoLabelsWithLineBetween
    //
    
    import UIKit
    
    class ViewController: UIViewController {
    
        @IBOutlet var leftLabel: GlyphPositionLabel!
        @IBOutlet var rightLabel: GlyphPositionLabel!
        
        let lineLayerName = "lineLayerName"
        let linePadding: CGFloat = 4
        
        override func viewDidLayoutSubviews() {
            super.viewDidLayoutSubviews()
            drawLine()
        }
    
        func drawLine() {
            leftLabel.debug = false
            rightLabel.debug = false
            
            guard
                let leftLastLineGeometry = leftLabel.linesGeometry.last,
                let rightLastLineGeometry = rightLabel.linesGeometry.last else {
                return
            }
            
            let leftBaselineRect = leftLabel.convert(leftLastLineGeometry.baselineRect, to: view)
            let leftBaselineBottomRightPoint = CGPoint(x: leftBaselineRect.maxX + linePadding, y: leftBaselineRect.maxY)
            
            let rightBaselineRect = rightLabel.convert(rightLastLineGeometry.baselineRect, to: view)
            let rightBaselineBottomLeftPoint = CGPoint(x: rightBaselineRect.minX - linePadding, y: rightBaselineRect.maxY)
            
            for layer in view.layer.sublayers ?? [] {
                if layer.name == lineLayerName {
                    layer.removeFromSuperlayer()
                    break
                }
            }
            
            let lineLayer = CAShapeLayer()
            lineLayer.lineDashPattern = [4, 2]
            lineLayer.strokeColor = UIColor.gray.cgColor
            lineLayer.lineWidth = 1
            lineLayer.name = lineLayerName
            view.layer.addSublayer(lineLayer)
            
            let linePath = UIBezierPath()
            linePath.move(to: leftBaselineBottomRightPoint)
            linePath.addLine(to: rightBaselineBottomLeftPoint)
            lineLayer.path = linePath.cgPath
        }
    
    }
    
    • 1

相关问题

  • UIImageView 相对于 superview 缩放后的位置

  • 在密码中授权 type basic 时,符号#swift 4 被错误传输到服务器

  • 将 AppDelegate.h 添加到项目中

  • 调用共享表时出错

  • 接近传感器模拟

  • 帮助 ios 应用程序架构 (MVP)

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    我看不懂措辞

    • 1 个回答
  • Marko Smith

    请求的模块“del”不提供名为“default”的导出

    • 3 个回答
  • Marko Smith

    "!+tab" 在 HTML 的 vs 代码中不起作用

    • 5 个回答
  • Marko Smith

    我正在尝试解决“猜词”的问题。Python

    • 2 个回答
  • Marko Smith

    可以使用哪些命令将当前指针移动到指定的提交而不更改工作目录中的文件?

    • 1 个回答
  • Marko Smith

    Python解析野莓

    • 1 个回答
  • Marko Smith

    问题:“警告:检查最新版本的 pip 时出错。”

    • 2 个回答
  • Marko Smith

    帮助编写一个用值填充变量的循环。解决这个问题

    • 2 个回答
  • Marko Smith

    尽管依赖数组为空,但在渲染上调用了 2 次 useEffect

    • 2 个回答
  • Marko Smith

    数据不通过 Telegram.WebApp.sendData 发送

    • 1 个回答
  • Martin Hope
    Alexandr_TT 2020年新年大赛! 2020-12-20 18:20:21 +0000 UTC
  • Martin Hope
    Alexandr_TT 圣诞树动画 2020-12-23 00:38:08 +0000 UTC
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5