Заменить подстроку NSAttributedString на другую NSAttributedString
Я хочу заменить подстроку (например, @"replace"
) из NSAttributedString
с другим NSAttributedString
,
Я ищу эквивалентный метод NSString
"s stringByReplacingOccurrencesOfString:withString:
за NSAttributedString
,
10 ответов
Преобразуйте вашу приписанную строку в экземпляр
NSMutableAttributedString
,Изменяемая атрибутивная строка имеет
mutableString
имущество. Согласно документации:"Получатель отслеживает изменения в этой строке и постоянно обновляет сопоставления атрибутов".
Таким образом, вы можете использовать результирующую изменяемую строку для выполнения замены с
replaceOccurrencesOfString:withString:options:range:
,
В моем случае единственным был следующий способ (протестирован на iOS9):
NSAttributedString *attributedString = ...;
NSAttributedString *anotherAttributedString = ...; //the string which will replace
while ([attributedString.mutableString containsString:@"replace"]) {
NSRange range = [attributedString.mutableString rangeOfString:@"replace"];
[attributedString replaceCharactersInRange:range withAttributedString:anotherAttributedString];
}
Конечно, было бы неплохо найти другой лучший способ.
Вот как вы можете изменить строку NSMutableAttributedString, сохранив ее атрибуты:
Swift:
// first we create a mutable copy of attributed text
let originalAttributedText = nameLabel.attributedText?.mutableCopy() as! NSMutableAttributedString
// then we replace text so easily
let newAttributedText = originalAttributedText.mutableString.setString("new text to replace")
Objective-C:
NSMutableAttributedString *newAttrStr = [attribtedTxt.mutableString setString:@"new string"];
Swift 4: обновлено отличное решение Sunkas для Swift 4 и заключено в "расширение". Просто закрепите это в вашем ViewController (вне класса) и используйте его.
extension NSAttributedString {
func stringWithString(stringToReplace: String, replacedWithString newStringPart: String) -> NSMutableAttributedString
{
let mutableAttributedString = mutableCopy() as! NSMutableAttributedString
let mutableString = mutableAttributedString.mutableString
while mutableString.contains(stringToReplace) {
let rangeOfStringToBeReplaced = mutableString.range(of: stringToReplace)
mutableAttributedString.replaceCharacters(in: rangeOfStringToBeReplaced, with: newStringPart)
}
return mutableAttributedString
}
}
С Swift 4 и iOS 11 вы можете использовать один из двух следующих способов, чтобы решить вашу проблему.
# 1. С помощью NSMutableAttributedString
replaceCharacters(in:with:)
метод
NSMutableAttributedString
имеет метод, называемый replaceCharacters(in:with:)
, replaceCharacters(in:with:)
имеет следующую декларацию:
Заменяет символы и атрибуты в заданном диапазоне на символы и атрибуты заданной приписанной строки.
func replaceCharacters(in range: NSRange, with attrString: NSAttributedString)
Код Playground ниже показывает, как использовать replaceCharacters(in:with:)
чтобы заменить подстроку NSMutableAttributedString
экземпляр с новым NSMutableAttributedString
пример:
import UIKit
// Set initial attributed string
let initialString = "This is the initial string"
let attributes = [NSAttributedStringKey.foregroundColor : UIColor.red]
let mutableAttributedString = NSMutableAttributedString(string: initialString, attributes: attributes)
// Set new attributed string
let newString = "new"
let newAttributes = [NSAttributedStringKey.underlineStyle : NSUnderlineStyle.styleSingle.rawValue]
let newAttributedString = NSMutableAttributedString(string: newString, attributes: newAttributes)
// Get range of text to replace
guard let range = mutableAttributedString.string.range(of: "initial") else { exit(0) }
let nsRange = NSRange(range, in: mutableAttributedString.string)
// Replace content in range with the new content
mutableAttributedString.replaceCharacters(in: nsRange, with: newAttributedString)
# 2. С помощью NSMutableString
replaceOccurrences(of:with:options:range:)
метод
NSMutableString
имеет метод, называемый replaceOccurrences(of:with:options:range:)
, replaceOccurrences(of:with:options:range:)
имеет следующую декларацию:
Заменяет все вхождения данной строки в данном диапазоне на другую данную строку, возвращая количество замен.
func replaceOccurrences(of target: String, with replacement: String, options: NSString.CompareOptions = [], range searchRange: NSRange) -> Int
Код Playground ниже показывает, как использовать replaceOccurrences(of:with:options:range:)
чтобы заменить подстроку NSMutableAttributedString
экземпляр с новым NSMutableAttributedString
пример:
import UIKit
// Set initial attributed string
let initialString = "This is the initial string"
let attributes = [NSAttributedStringKey.foregroundColor : UIColor.red]
let mutableAttributedString = NSMutableAttributedString(string: initialString, attributes: attributes)
// Set new string
let newString = "new"
// Replace replaceable content in mutableAttributedString with new content
let totalRange = NSRange(location: 0, length: mutableAttributedString.string.count)
_ = mutableAttributedString.mutableString.replaceOccurrences(of: "initial", with: newString, options: [], range: totalRange)
// Get range of text that requires new attributes
guard let range = mutableAttributedString.string.range(of: newString) else { exit(0) }
let nsRange = NSRange(range, in: mutableAttributedString.string)
// Apply new attributes to the text matching the range
let newAttributes = [NSAttributedStringKey.underlineStyle : NSUnderlineStyle.styleSingle.rawValue]
mutableAttributedString.setAttributes(newAttributes, range: nsRange)
Мне пришлось жирным шрифтом <b>
теги, вот что я сделал:
- (NSAttributedString *)boldString:(NSString *)string {
UIFont *boldFont = [UIFont boldSystemFontOfSize:14];
NSMutableAttributedString *attributedDescription = [[NSMutableAttributedString alloc] initWithString:string];
NSRegularExpression *regex = [NSRegularExpression regularExpressionWithPattern:@".*?<b>(.*?)<\\/b>.*?" options:NSRegularExpressionCaseInsensitive error:NULL];
NSArray *myArray = [regex matchesInString:string options:0 range:NSMakeRange(0, string.length)] ;
for (NSTextCheckingResult *match in myArray) {
NSRange matchRange = [match rangeAtIndex:1];
[attributedDescription addAttribute:NSFontAttributeName value:boldFont range:matchRange];
}
while ([attributedDescription.string containsString:@"<b>"] || [attributedDescription.string containsString:@"</b>"]) {
NSRange rangeOfTag = [attributedDescription.string rangeOfString:@"<b>"];
[attributedDescription replaceCharactersInRange:rangeOfTag withString:@""];
rangeOfTag = [attributedDescription.string rangeOfString:@"</b>"];
[attributedDescription replaceCharactersInRange:rangeOfTag withString:@""];
}
return attributedDescription;
}
У меня есть конкретные требования и исправлены, как показано ниже. Это может кому-то помочь.
Требование: в раскадровке расширенный текст непосредственно добавляется в атрибут UITextView, который содержит слово "Версия приложения: 1.0". Теперь я должен динамизировать номер версии, читая его из информационного списка.
Решение: удалил версию 1.0 из раскадровки, просто сохранил "Версия приложения:" и добавил код ниже.
NSAttributedString *attribute = self.firsttextView.attributedText;
NSMutableAttributedString *mutableAttri = [[NSMutableAttributedString alloc] initWithAttributedString:attribute];
NSString *appVersionText = @"App Version:";
if ([[mutableAttri mutableString] containsString:appVersionText]) {
NSDictionary* infoDict = [[NSBundle mainBundle] infoDictionary];
NSString* version = [infoDict objectForKey:@"CFBundleShortVersionString"];
NSString *newappversion = [NSString stringWithFormat:@"%@ %@",appVersionText,version] ;
[[mutableAttri mutableString] replaceOccurrencesOfString:appVersionText withString:newappversion options:NSCaseInsensitiveSearch range:NSMakeRange(0, mutableAttri.length)];
self.firsttextView.attributedText = mutableAttri;
}
Готово!! Обновлен / изменен приписанный текст.
NSMutableAttributedString *result = [[NSMutableAttributedString alloc] initWithString:@"I am a boy."];
[result addAttribute:NSForegroundColorAttributeName value:[UIColor blackColor] range:NSMakeRange(0, [result length])];
NSMutableAttributedString *replace = [[NSMutableAttributedString alloc] initWithString:@"a"];
[replace addAttribute:NSForegroundColorAttributeName value:[UIColor redColor] range:NSMakeRange(0, [replace length])];
[result replaceCharactersInRange:NSMakeRange(5, [replace length]) withAttributedString:replace];
Я считаю, что все остальные ответы не работают. Вот как я заменил содержимое строки NSAttributed в расширении категории:
func stringWithString(stringToReplace:String, replacedWithString newStringPart:String) -> NSMutableAttributedString
{
let mutableAttributedString = mutableCopy() as! NSMutableAttributedString
let mutableString = mutableAttributedString.mutableString
while mutableString.containsString(stringToReplace) {
let rangeOfStringToBeReplaced = mutableString.rangeOfString(stringToReplace)
mutableAttributedString.replaceCharactersInRange(rangeOfStringToBeReplaced, withString: newStringPart)
}
return mutableAttributedString
}
я создал расширение Swift 5 для этого
extension NSMutableAttributedString {
func replace(_ findString: String, with replacement: String, attributes: [NSAttributedString.Key : Any]) {
let ms = mutableString
var range = ms.range(of: findString)
while range.location != NSNotFound {
addAttributes(attributes, range: range)
ms.replaceCharacters(in: range, with: replacement)
range = ms.range(of: findString)
}
}
}
вариант использования
attributedString.replace("%EMAIL%", with: email, attributes: [.font:boldFont])