2

我想让一个 NSAttributedString 包含多条消息。如果一条消息有一个长文本并且它环绕,我希望有一个行间距,比如说,5。因为我有一个包含多条消息的 NSAttributedString,我希望每条消息之间有更大的行间距;比如说20。

我想要的是

“我明白了”是一条信息。“我认为两者都是……”是一条消息,尽管它包含两行,而“像单向聊天一样”是一条消息。

注意第 2 和第 3 之间的行距如何小于第 1 和第 2 以及第 3 和第 4 之间的行距。

我试过的

我在每条消息的末尾附加了一个 \n 并且我尝试使用 NSParagraphStyle 来控制行距,但它似乎是全部或全部:

        // index is the index of the group of messages as I iterate through them
        // contentText is an NSMutableAttributedString
        if index != messages.count - 1 {
            let style = NSMutableParagraphStyle()
            style.lineSpacing = 40.0

            let lineReturn = NSMutableAttributedString(string: "\n")
            contentText.appendAttributedString(lineReturn)

            if index == 0 {
                contentText.addAttribute(NSParagraphStyleAttributeName, value: style, range: NSMakeRange(contentText.length-lineReturn.length, lineReturn.length))
            } else {
                contentText.addAttribute(NSParagraphStyleAttributeName, value: style, range: NSMakeRange(contentText.length-lineReturn.length-1, lineReturn.length+1))
            }
        }

如果我将行距添加到开头,它将为整个标签设置行距。

        if index == 0 {
            let style = NSMutableParagraphStyle()
            style.lineSpacing = 40.0
            contentText.addAttribute(NSParagraphStyleAttributeName, value: style1, range: NSMakeRange(start, 1))
        }

(这真的只是我最近的尝试。)

谢谢你的帮助!:)

4

2 回答 2

1

细节

  • 在您的英语消息中有非常基本的自定义标记,以便您可以解析出不同的部分

  • 指示您的翻译人员保留标记并翻译
    其余部分 拥有一个可用作此消息容器的 UIView

  • 将您的英文信息分成几部分,以将常规文本与可点击文本分开

  • 为每一块在容器 UIView 上创建一个 UILabel

  • 对于可点击的部分,设置您的样式,允许用户交互
    并创建您的点击手势识别器

  • 做一些非常基本的簿记,将单词完美地
    放在线条上

    为了解。

在视图控制器的 viewDidLoad 我放置了这个:

[self buildAgreeTextViewFromString:NSLocalizedString(@"I agree to the #<ts>terms of service# and #<pp>privacy policy#",


                                                 @"PLEASE NOTE: please translate \"terms of service\" and \"privacy policy\" as well, and leave the #<ts># and #<pp># around your translations just as in the English version of this message.")];

我正在调用一个将构建消息的方法。注意我想出的标记。你当然可以自己发明,但关键是我还标记了每个可点击区域的末端,因为它们跨越多个单词。

这是将消息放在一起的方法——见下文。首先,我通过 # 字符(或者更确切地说是 @"#" 字符串)分解英文消息。这样我就得到了需要单独创建标签的每一件作品。我遍历它们并寻找我的基本标记<ts><pp>检测哪些片段是指向什么的链接。如果我正在使用的文本块是一个链接,那么我会设置一些样式并为它设置一个点击手势识别器。当然,我也去掉了标记字符。我认为这是一个非常简单的方法。

请注意一些细微之处,例如我如何处理空格:我只是从(本地化)字符串中获取空格。如果没有空格(中文,日文),那么块之间也不会有空格。如果有空格,则那些会根据需要自动将块隔开(例如英语)。但是,当我必须在下一行的开头放置一个单词时,我确实需要确保从该文本中删除任何空白前缀,否则它无法正确对齐。

- (void)buildAgreeTextViewFromString:(NSString *)localizedString
{
  // 1. Split the localized string on the # sign:
  NSArray *localizedStringPieces = [localizedString componentsSeparatedByString:@"#"];

  // 2. Loop through all the pieces:
  NSUInteger msgChunkCount = localizedStringPieces ? localizedStringPieces.count : 0;
  CGPoint wordLocation = CGPointMake(0.0, 0.0);
  for (NSUInteger i = 0; i < msgChunkCount; i++)
  {
    NSString *chunk = [localizedStringPieces objectAtIndex:i];
    if ([chunk isEqualToString:@""])
    {
      continue;     // skip this loop if the chunk is empty
    }

    // 3. Determine what type of word this is:
    BOOL isTermsOfServiceLink = [chunk hasPrefix:@"<ts>"];
    BOOL isPrivacyPolicyLink  = [chunk hasPrefix:@"<pp>"];
    BOOL isLink = (BOOL)(isTermsOfServiceLink || isPrivacyPolicyLink);

    // 4. Create label, styling dependent on whether it's a link:
    UILabel *label = [[UILabel alloc] init];
    label.font = [UIFont systemFontOfSize:15.0f];
    label.text = chunk;
    label.userInteractionEnabled = isLink;

    if (isLink)
    {
      label.textColor = [UIColor colorWithRed:110/255.0f green:181/255.0f blue:229/255.0f alpha:1.0];
      label.highlightedTextColor = [UIColor yellowColor];

      // 5. Set tap gesture for this clickable text:
      SEL selectorAction = isTermsOfServiceLink ? @selector(tapOnTermsOfServiceLink:) : @selector(tapOnPrivacyPolicyLink:);
      UITapGestureRecognizer *tapGesture = [[UITapGestureRecognizer alloc] initWithTarget:self
                                                                                   action:selectorAction];
      [label addGestureRecognizer:tapGesture];

      // Trim the markup characters from the label:
      if (isTermsOfServiceLink) 
        label.text = [label.text stringByReplacingOccurrencesOfString:@"<ts>" withString:@""];
      if (isPrivacyPolicyLink)  
        label.text = [label.text stringByReplacingOccurrencesOfString:@"<pp>" withString:@""];
    }
    else
    {
      label.textColor = [UIColor whiteColor];
    }

    // 6. Lay out the labels so it forms a complete sentence again:

    // If this word doesn't fit at end of this line, then move it to the next
    // line and make sure any leading spaces are stripped off so it aligns nicely:

    [label sizeToFit];

    if (self.agreeTextContainerView.frame.size.width < wordLocation.x + label.bounds.size.width)
    {
      wordLocation.x = 0.0;                       // move this word all the way to the left...
      wordLocation.y += label.frame.size.height;  // ...on the next line

      // And trim of any leading white space:
      NSRange startingWhiteSpaceRange = [label.text rangeOfString:@"^\\s*"
                                                          options:NSRegularExpressionSearch];
      if (startingWhiteSpaceRange.location == 0)
      {
        label.text = [label.text stringByReplacingCharactersInRange:startingWhiteSpaceRange
                                                         withString:@""];
        [label sizeToFit];
      }
    }

    // Set the location for this label:
    label.frame = CGRectMake(wordLocation.x,
                             wordLocation.y,
                             label.frame.size.width,
                             label.frame.size.height);
    // Show this label:
    [self.agreeTextContainerView addSubview:label];

    // Update the horizontal position for the next word:
    wordLocation.x += label.frame.size.width;
  }
}

如果您想使用手势,请使用此方法。

- (void)tapOnTermsOfServiceLink:(UITapGestureRecognizer *)tapGesture
{
  if (tapGesture.state == UIGestureRecognizerStateEnded)
  {
    NSLog(@"User tapped on the Terms of Service link");
  }
}


- (void)tapOnPrivacyPolicyLink:(UITapGestureRecognizer *)tapGesture
{
  if (tapGesture.state == UIGestureRecognizerStateEnded)
  {
    NSLog(@"User tapped on the Privacy Policy link");
  }
}

希望这可以帮助。我确信有很多更聪明、更优雅的方法可以做到这一点,但这是我能够想出的,而且效果很好。

这个答案显示输出如下屏幕截图......但你从这个答案中得到了想法。

在此处输入图像描述

于 2015-12-11T04:10:01.097 回答
1

明白了!

您需要使用baselineOffset属性:

let contentText = NSMutableAttributedString(
string: "I see\nI'd think it`d be both a notification and a\nplace to see past announcements\nLike a one way chat.")

contentText.addAttribute(.baselineOffset, value: 10, range: NSRange(location: 0, length: 5))
contentText.addAttribute(.baselineOffset, value: -10, range: NSRange(location: 85, length: 20))


结果:

“我知道

我认为它既是一个通知,也是一个
查看过去公告的地方,

就像单向聊天一样。”

于 2019-04-27T00:56:34.323 回答