利用Swift如何計算文本的size示例詳解
前言
對于swift 還處于摸索階段很多語法還不熟悉,本文主要給大家介紹的是關(guān)于利用Swift計算文本size的相關(guān)內(nèi)容,分享出來供大家參考學(xué)習(xí),下面話不多說了,來一起看看詳細(xì)的介紹吧。
iOS 11之前限制寬高計算字符串的size用的是UILabel的textRect(forBounds bounds: CGRect, limitedToNumberOfLines numberOfLines: Int) -> CGRect
方法,當(dāng)時也沒考慮線程安全問題(low爆了),Xcode也沒提示,用了好幾個版本,所幸一直都沒問題。
貼下方法(當(dāng)時為什么選這個方法就不解釋了):
func textSize(font: UIFont, constrainedSize: CGSize, lineSpacing: CGFloat?, lines: Int) -> CGSize { if self.isEmpty || lines < 0 { return CGSize.zero } let attributedString = NSMutableAttributedString(string: self) let range = NSRange(location: 0, length: attributedString.length) attributedString.addAttributes([NSFontAttributeName: font], range: range) if lineSpacing != nil { let paragraphStyle = NSMutableParagraphStyle() paragraphStyle.lineBreakMode = .byTruncatingTail paragraphStyle.lineSpacing = lineSpacing! attributedString.addAttribute(NSParagraphStyleAttributeName, value: paragraphStyle, range: range) } let calculatedLabel = UILabel() calculatedLabel.font = font calculatedLabel.attributedText = attributedString calculatedLabel.numberOfLines = lines let rect = calculatedLabel.textRect(forBounds: CGRect(x: 0, y: 0, width: constrainedSize.width, height: constrainedSize.height), limitedToNumberOfLines: lines) return rect.size }
最近升級了Xcode 9,運(yùn)行時警告我let calculatedLabel = UILabel()
要在主線程執(zhí)行,這時才意識到問題的嚴(yán)重性,馬上進(jìn)行了修改:
extension String { func boundingRect(with constrainedSize: CGSize, font: UIFont, lineSpacing: CGFloat? = nil) -> CGSize { let attritube = NSMutableAttributedString(string: self) let range = NSRange(location: 0, length: attritube.length) attritube.addAttributes([NSAttributedStringKey.font: font], range: range) if lineSpacing != nil { let paragraphStyle = NSMutableParagraphStyle() paragraphStyle.lineSpacing = lineSpacing! attritube.addAttribute(NSAttributedStringKey.paragraphStyle, value: paragraphStyle, range: range) } let rect = attritube.boundingRect(with: constrainedSize, options: [.usesLineFragmentOrigin, .usesFontLeading], context: nil) var size = rect.size if let currentLineSpacing = lineSpacing { // 文本的高度減去字體高度小于等于行間距,判斷為當(dāng)前只有1行 let spacing = size.height - font.lineHeight if spacing <= currentLineSpacing && spacing > 0 { size = CGSize(width: size.width, height: font.lineHeight) } } return size } func boundingRect(with constrainedSize: CGSize, font: UIFont, lineSpacing: CGFloat? = nil, lines: Int) -> CGSize { if lines < 0 { return .zero } let size = boundingRect(with: constrainedSize, font: font, lineSpacing: lineSpacing) if lines == 0 { return size } let currentLineSpacing = (lineSpacing == nil) ? (font.lineHeight - font.pointSize) : lineSpacing! let maximumHeight = font.lineHeight*CGFloat(lines) + currentLineSpacing*CGFloat(lines - 1) if size.height >= maximumHeight { return CGSize(width: size.width, height: maximumHeight) } return size } }
參數(shù)解釋
- constrainedSize:限制的size
- font:字號
- lineSpacing:默認(rèn)為nil,使用系統(tǒng)默認(rèn)的行間距
- lines:限制的行數(shù)
注:代碼版本為Swift 4.0
上面的兩個方法分別取代名:方法1和方法2。
方法1:限制寬高,可設(shè)置行間距,計算準(zhǔn)確
方法2:比方法1多了限制行數(shù)功能。
配合使用UILabel的擴(kuò)展方法:
extension UILabel { // 設(shè)置`numberOfLines = 0`的原因: // 配合方法`func boundingRect(with constrainedSize: CGSize, font: UIFont, lineSpacing: CGFloat? = nil, lines: Int) -> CGSize`使用,可以很好的解決不能正常顯示限制行數(shù)的問題; // 如果為label設(shè)置了限制行數(shù)(大于0的前提),使用上面的計算方法(帶行間距),同時字符串的實際行數(shù)大于限制行數(shù),這時候的高度會使label不能正常顯示。 func setText(with normalString: String, lineSpacing: CGFloat?, frame: CGRect) { self.frame = frame self.numberOfLines = 0 let paragraphStyle = NSMutableParagraphStyle() paragraphStyle.lineBreakMode = .byTruncatingTail if lineSpacing != nil { if (frame.height - font.lineHeight) <= lineSpacing! { paragraphStyle.lineSpacing = 0 } else { paragraphStyle.lineSpacing = lineSpacing! } } let attributedString = NSMutableAttributedString(string: normalString) let range = NSRange(location: 0, length: attributedString.length) attributedString.addAttributes([NSAttributedStringKey.font: font], range: range) attributedString.addAttribute(NSAttributedStringKey.paragraphStyle, value: paragraphStyle, range: range) self.attributedText = attributedString } }
在此感謝倉鼠:iOS 行距全攻略 和https://github.com/zhengwenming/WeChat
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
相關(guān)文章
Swift中循環(huán)語句中的轉(zhuǎn)移語句 break 和 continue
這篇文章主要介紹了Swift中循環(huán)語句中的轉(zhuǎn)移語句 break 和 continue,本文通過實例代碼給大家介紹的非常詳細(xì),需要的的朋友參考下2016-12-12Swift利用Decodable解析JSON的一個小問題詳解
這篇文章主要給大家介紹了關(guān)于Swift利用Decodable解析JSON的一個小問題的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧。2018-04-04