From ec0a34503301b5181a0be02a4fb7cef09ff00312 Mon Sep 17 00:00:00 2001 From: pjadhav Date: Fri, 7 Jun 2019 12:03:34 +0530 Subject: [PATCH] Fixed Potential truncation issue in LabelLayout - Added line break mode support in LabelLayout to calculate correct size based on line break mode. - Added LabelLayout unit test to verify that size calculation logic considers line break mode while calculating LabelLayout size. --- LayoutKitTests/LabelLayoutTests.swift | 87 ++++++++++++++++++- .../NSAttributedStringExtension.swift | 18 ++-- Sources/Layouts/LabelLayout.swift | 15 +++- .../Builders/LOKLabelLayoutBuilder.h | 1 + .../Builders/LOKLabelLayoutBuilder.m | 12 +++ Sources/ObjCSupport/LOKLabelLayout.swift | 7 ++ Sources/Text.swift | 34 ++++++-- 7 files changed, 155 insertions(+), 19 deletions(-) diff --git a/LayoutKitTests/LabelLayoutTests.swift b/LayoutKitTests/LabelLayoutTests.swift index f1c20321..91b754c9 100644 --- a/LayoutKitTests/LabelLayoutTests.swift +++ b/LayoutKitTests/LabelLayoutTests.swift @@ -11,6 +11,14 @@ import LayoutKit class LabelLayoutTests: XCTestCase { + // For the defined `sampleText` and `labelLayoutMaxWidth` combination, `LabelLayout` requires 2 line of + // text for char-wrapping and 3 lines for word-wrapping/truncating-tail. + // So don't change this combination as the following tests are based on these values: + // - testSizeCalculationWithDifferentLineBreakMode + // - testAttributedTextSizeCalculationWithDifferentLineBreakMode + private static let sampleText = "Lorem ipsum dolor sit amet, consectetuer adipiscing elit. Aenean comm" + private static let labelLayoutMaxWidth = 305 + func testNeedsView() { let l = LabelLayout(text: "hi").arrangement().makeViews() XCTAssertNotNil(l as? UILabel) @@ -200,10 +208,78 @@ class LabelLayoutTests: XCTestCase { let maxSize = CGSize(width: 17, height: .max) XCTAssertEqual(layout.measurement(within: maxSize).size, label.sizeThatFits(maxSize)) } + + func testSizeCalculationWithDifferentLineBreakMode() { + // Use different line break mode for `LabelLayout` and dummy label. + // So in this case, size calculation should not match with dummy label's size calculation. + let label = UILabel(text: LabelLayoutTests.sampleText, numberOfLines: 0, lineBreakMode: .byCharWrapping) + let layout = LabelLayout(text: LabelLayoutTests.sampleText) + let maxSize = CGSize(width: LabelLayoutTests.labelLayoutMaxWidth, height: .max) + XCTAssertNotEqual(layout.measurement(within: maxSize).size, label.sizeThatFits(maxSize)) + } + + func testAttributedTextSizeCalculationWithDifferentLineBreakMode() { + // Use different line break mode for `LabelLayout` and dummy label. + // So in this case, size calculation should not match with dummy label's size calculation. + let attributedText = NSAttributedString(string: LabelLayoutTests.sampleText) + let label = UILabel(attributedText: attributedText, numberOfLines: 0, lineBreakMode: .byCharWrapping) + let layout = LabelLayout(attributedText: attributedText) + let maxSize = CGSize(width: LabelLayoutTests.labelLayoutMaxWidth, height: .max) + XCTAssertNotEqual(layout.measurement(within: maxSize).size, label.sizeThatFits(maxSize)) + } + + func testTextSizeCalculationWithSameLineBreakMode() { + // Use same line break mode for `LabelLayout` and dummy label and then match LabelLayout's size with dummy label's size calculation. + let lineBreakingModes: [NSLineBreakMode] = [ + .byWordWrapping, + .byCharWrapping, + .byClipping, + .byTruncatingHead, + .byTruncatingTail, + .byTruncatingMiddle + ] + + lineBreakingModes.forEach { (lineBreakMode) in + verifyTextSizeCalculation(with: LabelLayoutTests.sampleText, lineBreakMode: lineBreakMode) + } + } + + func testAttributedTextSizeCalculationWithSameLineBreakMode() { + // Use same line break mode for `LabelLayout` and dummy label and then match LabelLayout's size with dummy label's size calculation. + let lineBreakingModes: [NSLineBreakMode] = [ + .byWordWrapping, + .byCharWrapping, + .byClipping, + .byTruncatingHead, + .byTruncatingTail, + .byTruncatingMiddle + ] + + let attributedText = NSAttributedString(string: LabelLayoutTests.sampleText) + lineBreakingModes.forEach { (lineBreakMode) in + verifyAttributedTextSizeCalculation(with: attributedText, lineBreakMode: lineBreakMode) + } + } + + // MARK: Private Helpers + + private func verifyTextSizeCalculation(with text: String, lineBreakMode: NSLineBreakMode) { + let label = UILabel(text: text, numberOfLines: 0, lineBreakMode: lineBreakMode) + let layout = LabelLayout(text: text, lineBreakMode: lineBreakMode) + let maxSize = CGSize(width: LabelLayoutTests.labelLayoutMaxWidth, height: .max) + XCTAssertEqual(layout.measurement(within: maxSize).size, label.sizeThatFits(maxSize)) + } + + private func verifyAttributedTextSizeCalculation(with attributedText: NSAttributedString, lineBreakMode: NSLineBreakMode) { + let label = UILabel(attributedText: attributedText, numberOfLines: 0, lineBreakMode: lineBreakMode) + let layout = LabelLayout(attributedText: attributedText, lineBreakMode: lineBreakMode) + let maxSize = CGSize(width: LabelLayoutTests.labelLayoutMaxWidth, height: .max) + XCTAssertEqual(layout.measurement(within: maxSize).size, label.sizeThatFits(maxSize)) + } } extension UILabel { - convenience init(text: String, font: UIFont? = nil, numberOfLines: Int? = nil) { + convenience init(text: String, font: UIFont? = nil, numberOfLines: Int? = nil, lineBreakMode: NSLineBreakMode? = nil) { self.init() self.text = text if let font = font { @@ -212,9 +288,12 @@ extension UILabel { if let numberOfLines = numberOfLines { self.numberOfLines = numberOfLines } + if let lineBreakMode = lineBreakMode { + self.lineBreakMode = lineBreakMode + } } - convenience init(attributedText: NSAttributedString, font: UIFont? = nil, numberOfLines: Int? = nil) { + convenience init(attributedText: NSAttributedString, font: UIFont? = nil, numberOfLines: Int? = nil, lineBreakMode: NSLineBreakMode? = nil) { self.init() if let font = font { self.font = font @@ -222,6 +301,10 @@ extension UILabel { if let numberOfLines = numberOfLines { self.numberOfLines = numberOfLines } + if let lineBreakMode = lineBreakMode { + self.lineBreakMode = lineBreakMode + } + // Want to set attributed text AFTER font it set, otherwise the font seems to take precedence. self.attributedText = attributedText } diff --git a/Sources/Internal/NSAttributedStringExtension.swift b/Sources/Internal/NSAttributedStringExtension.swift index 673f789e..5a1ab0ee 100644 --- a/Sources/Internal/NSAttributedStringExtension.swift +++ b/Sources/Internal/NSAttributedStringExtension.swift @@ -12,16 +12,18 @@ extension NSAttributedString { /// Returns a new NSAttributedString with a given font and the same attributes. func with(font: UIFont) -> NSAttributedString { - let fontAttribute = [NSAttributedString.Key.font: font] - let attributedTextWithFont = NSMutableAttributedString(string: string, attributes: fontAttribute) + return with(additionalAttributes: [NSAttributedString.Key.font: font]) + } + + /// Returns a new NSAttributedString with previous as well as additional attributes. + func with(additionalAttributes: [NSAttributedString.Key : Any]?) -> NSAttributedString { + let attributedTextWithAdditionalAttributes = NSMutableAttributedString(string: string, attributes: additionalAttributes) let fullRange = NSMakeRange(0, (string as NSString).length) - attributedTextWithFont.beginEditing() + attributedTextWithAdditionalAttributes.beginEditing() self.enumerateAttributes(in: fullRange, options: .longestEffectiveRangeNotRequired, using: { (attributes, range, _) in - attributedTextWithFont.addAttributes(attributes, range: range) + attributedTextWithAdditionalAttributes.addAttributes(attributes, range: range) }) - attributedTextWithFont.endEditing() - - return attributedTextWithFont + attributedTextWithAdditionalAttributes.endEditing() + return attributedTextWithAdditionalAttributes } - } diff --git a/Sources/Layouts/LabelLayout.swift b/Sources/Layouts/LabelLayout.swift index c59a274f..d684c6da 100644 --- a/Sources/Layouts/LabelLayout.swift +++ b/Sources/Layouts/LabelLayout.swift @@ -17,11 +17,13 @@ open class LabelLayout: BaseLayout