在swift for iOS中用普通(Android格式)文本创建一个属性字符串

时间:2014-11-27 07:07:06

标签: ios xcode swift nsattributedstring nsmutableattributedstring

我正在从Localizable.strings中读取字符串,其中包含类似于Android应用程序的strings.xml中的内容

"testShort" = "A <b>short</b>\ntest with another<b>bold text</b>";

粗体和换行符是我文本中唯一的两个格式属性。我试图开发这样的方法好几天没有成功:

func ConvertText(inputText: String) -> NSAttributedString {
    // here comes the conversion to a representation with Helvetica 14pt and Helvetica-Bold 14pt including line feeds.
}

我的最终目标是在UITextView的attributesText中显示文本。

在不知道Objective-C的情况下成为Swift和iOS的新手我发现很难做String操作,因为它们完全不同且复杂,所有示例都在Objective-C中。更令人困难的是,大多数API方法在Swift中不可用或与Objective-C中不同......

以下是我到目前为止在方法主体上尝试过的许多其他帖子的帮助:

var test = inputText.dataUsingEncoding(NSUnicodeStringEncoding, allowLossyConversion: true)!
attrStr = NSAttributedString(
        data: test,
        options: [NSDocumentTypeDocumentAttribute: NSHTMLTextDocumentType],
        documentAttributes: nil,
        error: nil)!
return attrStr

这里的主要问题是\n未被转换,字体非常小而且不同。

接下来,我尝试手动加粗文本的一部分。它看起来像那样:

var newText = NSMutableAttributedString(string: inputText)
newText.addAttribute(NSFontAttributeName, value: UIFont(name: "Helvetica-Bold", size: 14.0)!, range: NSRange(location:2,length:4))

现在我尝试在文本中搜索属性,删除它们并使用类似的addAttribute

// Start of bold text
var range = newText.rangeOfString("<b>")!
// End of bold text
var range2 = newText.rangeOfString("</b>")!

// replacing attributes
newText = newText.stringByReplacingCharactersInRange(range, withString: "")
newText = newText.stringByReplacingCharactersInRange(range2, withString: "")

// creating a range for bold => error "'String.Index' is not convertible to 'int'"
// how to do such a thing
var boldRange = NSMakeRange(range.startIndex, range2.endIndex -3)

// setting bold
newText.addAttribute(NSFontAttributeName, value: UIFont(name: "Helvetica-Bold", size: 14.0)!, range: boldRange)

这一整个范围是我目前的主要问题,因为它与字符串中的简单位置完全不同。

这个问题是缺乏(或隐藏的)文档的一个很好的例子:

addAttribute需要一个NSRange,rangeOfString似乎根据我得到的错误消息提供了一个通用范围 - 但是没有关于它的信息。 rangeOfString()上的Xcode中的搜索文档按钮指向NSString。 在那里搜索rangeOfString()表示它返回NSRange。单击它会导致_NSRange的类型别名信息,而location又有两个名为length和{{1}}的NSUInteger属性。我在XCode中看到的startIndex和endIndex属性在哪里?很混乱...

如果你能给我一些片段或提示,我在这里错误甚至是方法体,那会很棒,因为如果你对iOS和Swift有所了解,我仍然希望它不会太难。我的目标是支持iOS 7.1,但如果使用iOS 8更容易,那么它也很好。

1 个答案:

答案 0 :(得分:8)

关于NSAttributedString的第一个方法:

这给出了(现在为Swift 2更新):

func convertText(inputText: String) -> NSAttributedString {

    var html = inputText

    // Replace newline character by HTML line break
    while let range = html.rangeOfString("\n") {
        html.replaceRange(range, with: "<br />")
    }

    // Embed in a <span> for font attributes:
    html = "<span style=\"font-family: Helvetica; font-size:14pt;\">" + html + "</span>"

    let data = html.dataUsingEncoding(NSUnicodeStringEncoding, allowLossyConversion: true)!
    let attrStr = try? NSAttributedString(
        data: data,
        options: [NSDocumentTypeDocumentAttribute: NSHTMLTextDocumentType],
        documentAttributes: nil)
    return attrStr!
}

关于你的第二种方法:

有两种不同的rangeOfString()方法,一种用于(Swift)String和一种 for(Foundation)NSStringString方法返回Range<String.Index> 并且NSString方法返回NSRange

这两者之间的转换是可能的,但很复杂。原因在于 一个String每个&#34;扩展的字形集群&#34;算作一个字符,而在 NSString计算每个UTF-16单位。扩展的字形集群可以是 一个或多个UTF-16单元(&#34;&#34;两个UTF-16单元,&#34;&#34;四个)。

addAttribute()方法只接受NSRange。最简单的解决方法 这个问题是将Swift字符串转换为NSString并使用NSRange 只要。然后您的方法可能如下所示:

func convertText(inputText: String) -> NSAttributedString {

    let attrString = NSMutableAttributedString(string: inputText)
    let boldFont = UIFont(name: "Helvetica-Bold", size: 14.0)!

    var r1 = (attrString.string as NSString).rangeOfString("<b>")
    while r1.location != NSNotFound {
        let r2 = (attrString.string as NSString).rangeOfString("</b>")
        if r2.location != NSNotFound  && r2.location > r1.location {
            let r3 = NSMakeRange(r1.location + r1.length, r2.location - r1.location - r1.length)
            attrString.addAttribute(NSFontAttributeName, value: boldFont, range: r3)
            attrString.replaceCharactersInRange(r2, withString: "")
            attrString.replaceCharactersInRange(r1, withString: "")
        } else {
            break
        }
        r1 = (attrString.string as NSString).rangeOfString("<b>")
    }

    return attrString
}