web-dev-qa-db-ja.com

iOS 8でNSMutableAttributedStringを使用した文字列の一部に下線が機能しない

文字列の一部に下線を引こうとしています。たとえば、「テスト文字列」文字列の「文字列」部分です。私はNSMutableAttributedStringを使用していますが、私のソリューションはiOS7

NSMutableAttributedString *attributedString = [[NSMutableAttributedString alloc]
        initWithString:@"test string"];
[attributedString addAttribute:NSUnderlineStyleAttributeName
                         value:@(NSUnderlineStyleSingle)
                         range:NSMakeRange(5, 6)];
myLabel.attributedText = attributedString;

問題は、私のソリューションがiOS8もう。 NSMutableAttributedStringの複数のバリアントのテストに1時間を費やした後、このソリューションは範囲が0で始まる場合にのみ機能することがわかりました(長さは異なる場合があります)。その理由は何ですか?これを回避するにはどうすればよいですか?

59
KlimczakM

更新:この質問を調査することにより: iOS 8でNSMutableAttributedStringを表示する 私は最終的に解決策を見つけました!

文字列の先頭にNSUnderlineStyleNoneを追加する必要があります。

Swift 4.2(noneは削除されました):

let attributedString = NSMutableAttributedString()
attributedString.append(NSAttributedString(string: "test ",
                                           attributes: [.underlineStyle: 0]))
attributedString.append(NSAttributedString(string: "s",
                                           attributes: [.underlineStyle: NSUnderlineStyle.single.rawValue]))
attributedString.append(NSAttributedString(string: "tring",
                                           attributes: [.underlineStyle: 0]))

Objective-C:

 NSMutableAttributedString *attributedString = [[NSMutableAttributedString alloc] init];
 [attributedString appendAttributedString:[[NSAttributedString alloc] initWithString:@"test "
                                                                          attributes:@{NSUnderlineStyleAttributeName: @(NSUnderlineStyleNone)}]];
 [attributedString appendAttributedString:[[NSAttributedString alloc] initWithString:@"s"
                                                                         attributes:@{NSUnderlineStyleAttributeName: @(NSUnderlineStyleSingle),
                                                                                      NSBackgroundColorAttributeName: [UIColor clearColor]}]];
 [attributedString appendAttributedString:[[NSAttributedString alloc] initWithString:@"tring"]];

そのようなアプローチの別のボーナスは、範囲がないことです。ローカライズされた文字列にはとてもいい。

Apple bug :(

129
kelin

UnderlineStyleNoneを文字列全体に適用すると、中央から始まる部分に選択的に下線を適用できることがわかりました。

func underlinedString(string: NSString, term: NSString) -> NSAttributedString {
    let output = NSMutableAttributedString(string: string)
    let underlineRange = string.rangeOfString(term)
    output.addAttribute(NSUnderlineStyleAttributeName, value: NSUnderlineStyle.StyleNone.rawValue, range: NSMakeRange(0, string.length))
    output.addAttribute(NSUnderlineStyleAttributeName, value: NSUnderlineStyle.StyleSingle.rawValue, range: underlineRange)

    return output
}
17
exidy
NSMutableAttributedString *signUpString = [[NSMutableAttributedString alloc] initWithString:@"Not a member yet?Sign Up now"];

[signUpString appendAttributedString:[[NSAttributedString alloc] initWithString:@" "
                                                                         attributes:@{NSUnderlineStyleAttributeName: @(NSUnderlineStyleNone)}]];

[signUpString addAttributes: @{NSForegroundColorAttributeName:UIColorFromRGB(0x43484B),NSUnderlineStyleAttributeName:[NSNumber numberWithInteger:NSUnderlineStyleSingle]} range:NSMakeRange(17,11)];

signUpLbl.attributedText = [signUpString copy];

それは私のために働いた

6
Tanuj Jagoori

2018年9月ですので、この回答はiOS8に関するものではありませんが、文字列の一部に下線を引くことに関連しています。

すでに構成されているmyAttributedString内の特定の用語に下線を引くSwift 4拡張機能

_extension NSMutableAttributedString {

    func underline(term: String) {

        guard let underlineRange = string.range(of: term) else {

            return
        }

        let startPosition = string.distance(from: term.startIndex, to: underlineRange.lowerBound)
        let nsrange = NSRange(location: startPosition, length: term.count)

        addAttribute(
            .underlineStyle,
            value: NSUnderlineStyle.styleSingle.rawValue,
            range: nsrange)
    }
}
_

使用法:myAttributedString.underline(term: "some term")

3
Joss

Swift 5バージョン返されたNSMutableAttributedStringを追加することにより、@ Jossの回答を少し修正しました。これなしでは元のソリューションを使用できなかったためです。

extension NSMutableAttributedString {

func underline(term: String) -> NSMutableAttributedString {
    guard let underlineRange = string.range(of: term) else {
        return NSMutableAttributedString()
    }
    let startPosition = string.distance(from: term.startIndex, to: underlineRange.lowerBound)
    let nsrange = NSRange(location: startPosition, length: term.count)
    addAttribute(
        .underlineStyle,
        value: NSUnderlineStyle.single.rawValue,
        range: nsrange)
     return self
}

}

使用法:

 let myUnderLinedText = "Hello World"
 let underLinedMutableString =  NSMutableAttributedString(string: myUnderLinedText, attributes: titleAttributes).underline(term: myUnderLinedText)
0
Mussa Charles

私は遊び場/シミュレータで次の拡張機能を使用しました(exidyの機能を使用)、それはうまく機能しました、あなたはあなたのニーズに応じて属性を変更/追加することができます

 extension NSMutableAttributedString
{


    func changeWordsColour(terms:[NSString])
{
    let string = self.string as NSString
    self.addAttribute(NSForegroundColorAttributeName, value: UIColor.brownColor(), range: NSMakeRange(0, self.length))
    for term in terms
    {
        let underlineRange = string.rangeOfString(term as String)
        self.addAttribute(NSForegroundColorAttributeName, value: UIColor.redColor(), range: underlineRange)

    }
}
}

 let myStr = NSMutableAttributedString(string: "Change Words Colour")
 myStr.changeWordsColour(["change","Colour"])
0
Atka