From c2cb5fc66661645ac4d7c40bcaebcae48d7c6da9 Mon Sep 17 00:00:00 2001 From: Alexis Schultz Date: Fri, 15 Nov 2024 14:17:07 +0100 Subject: [PATCH 1/4] add dynamic type and weight support --- .../Core/MarkupStyle/MarkupStyleFont.swift | 115 ++++++++++-------- 1 file changed, 63 insertions(+), 52 deletions(-) diff --git a/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift b/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift index bb506d1..902d663 100644 --- a/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift +++ b/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift @@ -24,7 +24,7 @@ public struct MarkupStyleFont: MarkupStyleItem { } public enum FontFamily { case familyNames([String]) - + #if canImport(UIKit) func getFont(size: CGFloat) -> UIFont? { switch self { @@ -53,30 +53,27 @@ public struct MarkupStyleFont: MarkupStyleItem { } public enum FontWeightStyle: String, CaseIterable { case ultraLight, light, thin, regular, medium, semibold, bold, heavy, black - + public init?(rawValue: String) { guard let matchedStyle = FontWeightStyle.allCases.first(where: { style in return rawValue.lowercased().contains(style.rawValue) }) else { return nil } - + self = matchedStyle } - + #if canImport(UIKit) init?(font: UIFont) { if let traits = font.fontDescriptor.fontAttributes[.traits] as? [UIFontDescriptor.TraitKey: Any], let weight = traits[.weight] as? UIFont.Weight { self = weight.convertFontWeightStyle() return - } else if font.fontDescriptor.symbolicTraits.contains(.traitBold) { - self = .bold - return } else if let weightName = font.fontDescriptor.object(forKey: .face) as? String, let weight = Self.init(rawValue: weightName) { self = weight return } - + return nil } #elseif canImport(AppKit) @@ -91,43 +88,47 @@ public struct MarkupStyleFont: MarkupStyleItem { self = weight return } - + return nil } #endif } - + public var size: CGFloat? public var weight: FontWeight? public var italic: Bool? + public var bold: Bool? public var familyName: FontFamily? - - public init(size: CGFloat? = nil, weight: FontWeight? = nil, italic: Bool? = nil, familyName: FontFamily? = nil) { + + public init(size: CGFloat? = nil, weight: FontWeight? = nil, bold: Bool? = nil, italic: Bool? = nil, familyName: FontFamily? = nil) { self.size = size self.weight = weight + self.bold = bold self.italic = italic self.familyName = familyName } - + mutating func fillIfNil(from: MarkupStyleFont?) { self.size = self.size ?? from?.size self.weight = self.weight ?? from?.weight self.italic = self.italic ?? from?.italic + self.bold = self.bold ?? from?.bold self.familyName = self.familyName ?? from?.familyName } - + func isNil() -> Bool { return !([size, - weight, - italic, - familyName] as [Any?]).contains(where: { $0 != nil}) + weight, + italic, + bold, + familyName] as [Any?]).contains(where: { $0 != nil}) } - + func sizeOf(string: String) -> CGSize? { guard let font = getFont() else { return nil } - + return (string as NSString).size(withAttributes: [.font: font]) } } @@ -135,24 +136,25 @@ public struct MarkupStyleFont: MarkupStyleItem { #if canImport(UIKit) extension MarkupStyleFont { - + public init(_ font: UIFont) { self.size = font.pointSize self.italic = font.fontDescriptor.symbolicTraits.contains(.traitItalic) + self.bold = font.fontDescriptor.symbolicTraits.contains(.traitBold) if let fontWeight = FontWeightStyle.init(font: font) { self.weight = FontWeight.style(fontWeight) } self.familyName = .familyNames([font.familyName]) } - + func getFont() -> UIFont? { guard !isNil() else { return nil } - - var traits: [UIFontDescriptor.SymbolicTraits] = [] - + + var traits: UIFontDescriptor.SymbolicTraits = [] + let size = (self.size ?? MarkupStyle.default.font.size) ?? UIFont.systemFontSize let weight = self.weight?.convertToUIFontWeight() ?? .regular - + // There is no direct method in UIFont to specify the font family, italic and weight together. let font: UIFont @@ -163,28 +165,16 @@ extension MarkupStyleFont { // System Font font = UIFont.systemFont(ofSize: size, weight: weight) } - - if weight.rawValue >= UIFont.Weight.medium.rawValue { - traits.append(.traitBold) + + if bold == true { + traits.insert(.traitBold) } - + if let italic = self.italic, italic { - traits.append(.traitItalic) - } - - if traits.isEmpty { - return font - } else { - return withTraits(font: font, traits: traits) - } - } - - private func withTraits(font: UIFont, traits: [UIFontDescriptor.SymbolicTraits]) -> UIFont { - guard let descriptor = font.fontDescriptor - .withSymbolicTraits(UIFontDescriptor.SymbolicTraits(traits)) else { - return font + traits.insert(.traitItalic) } - return UIFont(descriptor: descriptor, size: font.pointSize) + + return font.with(weight: weight, symbolicTraits: traits) } } @@ -245,6 +235,27 @@ private extension UIFont.Weight { } } +public extension UIFont { + + /// Returns a font object that is the same as the receiver but which has the specified weight and symbolic traits + func with(weight: Weight, symbolicTraits: UIFontDescriptor.SymbolicTraits) -> UIFont { + + var mergedsymbolicTraits = fontDescriptor.symbolicTraits + mergedsymbolicTraits.formUnion(symbolicTraits) + + var traits = fontDescriptor.fontAttributes[.traits] as? [UIFontDescriptor.TraitKey: Any] ?? [:] + traits[.weight] = weight + traits[.symbolic] = mergedsymbolicTraits.rawValue + + var fontAttributes: [UIFontDescriptor.AttributeName: Any] = [:] + fontAttributes[.family] = familyName + fontAttributes[.traits] = traits + + let font = UIFont(descriptor: UIFontDescriptor(fontAttributes: fontAttributes), size: pointSize) + return UIFontMetrics.default.scaledFont(for: font) + } +} + #elseif canImport(AppKit) extension MarkupStyleFont { @@ -258,15 +269,15 @@ extension MarkupStyleFont { self.familyName = .familyNames([familyName]) } } - + func getFont() -> NSFont? { guard !isNil() else { return nil } - + var traits: [NSFontDescriptor.SymbolicTraits] = [] - + let size = (self.size ?? MarkupStyle.default.font.size) ?? NSFont.systemFontSize let weight = self.weight?.convertToUIFontWeight() ?? .regular - + // There is no direct method in UIFont to specify the font family, italic and weight together. let font: NSFont @@ -277,22 +288,22 @@ extension MarkupStyleFont { // System Font font = NSFont.systemFont(ofSize: size, weight: weight) } - + if weight.rawValue >= NSFont.Weight.medium.rawValue { traits.append(.bold) } - + if let italic = self.italic, italic { traits.append(.italic) } - + if traits.isEmpty { return font } else { return withTraits(font: font, traits: traits) } } - + private func withTraits(font: NSFont, traits: [NSFontDescriptor.SymbolicTraits]) -> NSFont { let descriptor = font.fontDescriptor.withSymbolicTraits(NSFontDescriptor.SymbolicTraits(traits)) return NSFont(descriptor: descriptor, size: font.pointSize) ?? font From 2fc3187df879e47efda99e90c1da4384828b2d2b Mon Sep 17 00:00:00 2001 From: ZhgChgLi <33706588+zhgchgli0718@users.noreply.github.com> Date: Sun, 24 Nov 2024 17:14:03 +0800 Subject: [PATCH 2/4] Update MarkupStyleFont.swift --- Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift b/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift index 902d663..648ebab 100644 --- a/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift +++ b/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift @@ -235,7 +235,7 @@ private extension UIFont.Weight { } } -public extension UIFont { +private extension UIFont { /// Returns a font object that is the same as the receiver but which has the specified weight and symbolic traits func with(weight: Weight, symbolicTraits: UIFontDescriptor.SymbolicTraits) -> UIFont { From bcea69a8fb54025abbd8a81fc6ef5497dd0d17d5 Mon Sep 17 00:00:00 2001 From: ZhgChgLi <33706588+zhgchgli0718@users.noreply.github.com> Date: Sun, 24 Nov 2024 17:15:50 +0800 Subject: [PATCH 3/4] Update MarkupStyleFontTests.swift --- Tests/ZMarkupParserTests/Core/MarkupStyleFontTests.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Tests/ZMarkupParserTests/Core/MarkupStyleFontTests.swift b/Tests/ZMarkupParserTests/Core/MarkupStyleFontTests.swift index 19c68e1..7342675 100644 --- a/Tests/ZMarkupParserTests/Core/MarkupStyleFontTests.swift +++ b/Tests/ZMarkupParserTests/Core/MarkupStyleFontTests.swift @@ -17,7 +17,7 @@ import AppKit final class MarkupStyleFontTests: XCTestCase { func testInit() { #if canImport(UIKit) - let markupStyleFont = MarkupStyleFont(UIFont.boldSystemFont(ofSize: 16)) + let markupStyleFont = MarkupStyleFont(UIFont.systemFont(ofSize: 16, weight: .bold)) if case let .style(weight) = markupStyleFont.weight, weight == .bold { // Success XCTAssertEqual(markupStyleFont.size, 16) @@ -161,7 +161,7 @@ final class MarkupStyleFontTests: XCTestCase { XCTAssertEqual(resultMarkupStyleFont.size, markupStyleFont.size!) if case let .style(weight) = resultMarkupStyleFont.weight { - XCTAssertEqual(weight, .bold) + XCTAssertEqual(weight, .heavy) } else { XCTFail() } From 820ec747615650b7a9ac7cf7d23e6d4aae1c56f6 Mon Sep 17 00:00:00 2001 From: ZhgChgLi <33706588+zhgchgli0718@users.noreply.github.com> Date: Sun, 24 Nov 2024 17:38:24 +0800 Subject: [PATCH 4/4] Update MarkupStyleFont.swift --- .../Core/MarkupStyle/MarkupStyleFont.swift | 44 ++++++++++++------- 1 file changed, 27 insertions(+), 17 deletions(-) diff --git a/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift b/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift index 648ebab..f3cb9d5 100644 --- a/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift +++ b/Sources/ZMarkupParser/Core/MarkupStyle/MarkupStyleFont.swift @@ -81,9 +81,6 @@ public struct MarkupStyleFont: MarkupStyleItem { if let traits = font.fontDescriptor.fontAttributes[.traits] as? [NSFontDescriptor.TraitKey: Any], let weight = traits[.weight] as? NSFont.Weight { self = weight.convertFontWeightStyle() return - } else if font.fontDescriptor.symbolicTraits.contains(.bold) { - self = .bold - return } else if let weightName = font.fontDescriptor.object(forKey: .face) as? String, let weight = Self.init(rawValue: weightName) { self = weight return @@ -273,7 +270,7 @@ extension MarkupStyleFont { func getFont() -> NSFont? { guard !isNil() else { return nil } - var traits: [NSFontDescriptor.SymbolicTraits] = [] + var traits: NSFontDescriptor.SymbolicTraits = [] let size = (self.size ?? MarkupStyle.default.font.size) ?? NSFont.systemFontSize let weight = self.weight?.convertToUIFontWeight() ?? .regular @@ -289,24 +286,15 @@ extension MarkupStyleFont { font = NSFont.systemFont(ofSize: size, weight: weight) } - if weight.rawValue >= NSFont.Weight.medium.rawValue { - traits.append(.bold) + if bold == true { + traits.insert(.bold) } if let italic = self.italic, italic { - traits.append(.italic) - } - - if traits.isEmpty { - return font - } else { - return withTraits(font: font, traits: traits) + traits.insert(.italic) } - } - private func withTraits(font: NSFont, traits: [NSFontDescriptor.SymbolicTraits]) -> NSFont { - let descriptor = font.fontDescriptor.withSymbolicTraits(NSFontDescriptor.SymbolicTraits(traits)) - return NSFont(descriptor: descriptor, size: font.pointSize) ?? font + return font.with(weight: weight, symbolicTraits: traits) } } @@ -367,4 +355,26 @@ private extension NSFont.Weight { } } +private extension NSFont { + + /// Returns a font object that is the same as the receiver but which has the specified weight and symbolic traits + func with(weight: Weight, symbolicTraits: NSFontDescriptor.SymbolicTraits) -> NSFont { + + var mergedsymbolicTraits = fontDescriptor.symbolicTraits + mergedsymbolicTraits.formUnion(symbolicTraits) + + var traits = fontDescriptor.fontAttributes[.traits] as? [NSFontDescriptor.TraitKey: Any] ?? [:] + traits[.weight] = weight + traits[.symbolic] = mergedsymbolicTraits.rawValue + + var fontAttributes: [NSFontDescriptor.AttributeName: Any] = [:] + fontAttributes[.family] = familyName + fontAttributes[.traits] = traits + + let font = NSFont(descriptor: NSFontDescriptor(fontAttributes: fontAttributes), size: pointSize) + // return UIFontMetrics.default.scaledFont(for: font) + return font ?? self + } +} + #endif