How to detect link in UILabel in swift 4?
Asked Answered
I

2

7

I was using ActiveLabel as third party library to make link in a label for particular words.The code works fine for Swift 3 & 3.2. but does not work for swift 4.

Below code i used

let customType1 = ActiveType.custom(pattern: "\\sTerms & Conditions\\b") //Looks for "are"
            labelTc.enabledTypes.append(customType1)
            labelTc.customize { (label) in
                labelTc.text = "UserAgreement".localized
                label.numberOfLines = 0
                label.lineSpacing = 4
                label.textColor = UIColor(red: 131 / 255, green: 147 / 255, blue: 168 / 255, alpha: 1)
                //Custom types
                label.customColor[customType1] = Constant.AppColor.greenMeadow
                label.customSelectedColor[customType1] = Constant.AppColor.greenMeadow
                label.configureLinkAttribute = { (type, attributes, isSelected) in
                    var atts = attributes
                    switch type {
                    case customType1:
                        atts[NSAttributedStringKey.font._rawValue as String] = UIFont(name: self.labelTc.font.fontName, size: 15.0)
                        atts[NSAttributedStringKey.underlineStyle.rawValue] = NSUnderlineStyle.styleSingle
                        break


                    case .mention:
                        break
                    case .hashtag:
                        break
                    case .url:
                        break
                    case .custom(let pattern):
                        break

                    default :
                        break
                    }

                    return atts
                }

Can anyone give me solution using native code instead of using third party library.

Inductee answered 16/11, 2017 at 14:40 Comment(5)
https://mcmap.net/q/1625904/-detect-uiwebview-links-without-clickFantail
In Swift 4 your attributes should have type [NSAttributedStringKey: Any] therefore assignment to them should be atts[.font] = UIFont(...). "Does not work" is not a very good problem description. The library is open source, look into the code, everything is there.Anklet
#21630284Oestriol
hackingwithswift.com/example-code/system/…Oestriol
for native code I had used UITextView instead of UILabel please check my answer if it helps you https://mcmap.net/q/1625905/-how-to-find-the-frame-of-substring-inside-a-uilabel-in-swift-4Demonstrator
I
3

I was able to find out the solution for swift 4 as well.

label.configureLinkAttribute = { (type, attributes, isSelected) in
                var atts = attributes
                switch type {
                case customType1:
                    atts[NSAttributedStringKey.font.rawValue] = UIFont(name: self.labelTc.font.fontName, size: 15.0)
                    atts[NSAttributedStringKey.underlineStyle.rawValue] = NSUnderlineStyle.styleSingle.rawValue
                    break

              default: ()
                }
                return atts
            }
Inductee answered 21/11, 2017 at 4:35 Comment(1)
If you pretty sure that your answer is the right solution, please make sure to accept (green tick) it.Peta
F
1

Another solution. For me this better.

  1. Get your text from the UILabel.
  2. Unwrape this text
  3. Do what you want

For example, in my case:

struct DetectedLinkData {

    var link: URL
    var range: Range<String.Index>
    init(link: URL, range: Range<String.Index>) {
        self.link = link
        self.range = range
    }

}

class LinkDetecter {

    static func getLinks(in string: String) -> [DetectedLinkData] {
        guard let detector = try? NSDataDetector(types: NSTextCheckingResult.CheckingType.link.rawValue) else {
            return []
        }

        var result: [DetectedLinkData] = []
        let matches = detector.matches(in: string, options: [], range: NSRange(location: 0, length: string.utf16.count) )
        for match in matches {
            guard let range = Range(match.range, in: string),
                let url = URL(string: String(string[range]) )
                else {
                    continue
            }
            result.append(DetectedLinkData(link: url, range: range))
        }
        return result
    }

}

let temp = LinkDetecter.getLinks(in: message["text"] as? String ?? "")
Fluecure answered 25/12, 2019 at 9:41 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.