25

以下代码在 iOS < 7.0 中可以正常工作。在 iOS 7 中,当 UITextView 更新时,滚动会断断续续且不稳定。我不确定这是否是 iOS 7 中的错误,或者我做错了什么。

测试控制器.h

//TODO: Add UITextView in storyboard and tie to textView outlet

#define MAX_TEXT_VIEW_CHARACTERS 1000
@interface TestController : UIViewController  {
    NSMutableString *_outputText;
    NSTimer *_outputTimer;
}

@property (strong, nonatomic) IBOutlet UITextView *textView;

@end

测试控制器.m

@implementation TestController
@synthesize textView;

- (void)viewWillAppear:(BOOL)animated {
    [super viewWillAppear:animated];
    _outputText = [NSMutableString stringWithCapacity:MAX_TEXT_VIEW_CHARACTERS];
    _outputTimer =  [NSTimer scheduledTimerWithTimeInterval:0.01 target:self selector:@selector(outputLine:) userInfo:nil repeats:YES];
}

-(void)outputLine:(NSTimer *) theTimer {
    static int i = 0;
    //Run this 100 times
    if (i > 99) {
        [_outputTimer invalidate];
        return;
    }
    [self outputToScreen:[NSString stringWithFormat:@"Some string %d\r", ++i]];
}

-(void)outputToScreen:(NSString *)str {
    if (!str || !str.length) return;  //Nothing to output

    NSInteger outputTextSize = _outputText.length;
    [_outputText appendString:str];
    if (outputTextSize > MAX_TEXT_VIEW_CHARACTERS)
        [_outputText deleteCharactersInRange:NSMakeRange(0, outputTextSize - MAX_TEXT_VIEW_CHARACTERS)];
    self.textView.text = _outputText;

    [self scrollOutputToBottom];
}

-(void)scrollOutputToBottom {
    CGPoint p = [textView contentOffset];
    [textView setContentOffset:p animated:NO];
    [textView scrollRangeToVisible:NSMakeRange([textView.text length], 0)];
}

@end
4

9 回答 9

56

这适用于我在 iOS7 中。

-(void) scrollToBottom {
  [textView scrollRangeToVisible:NSMakeRange([textView.text length], 0)];
  [textView setScrollEnabled:NO];
  [textView setScrollEnabled:YES];
}
于 2014-01-08T08:01:06.933 回答
27

这显然是 iOS 7 的错误。这是一个解决方法,直到苹果修复它。解决方法基本上是UITextView通过创建NSTextStorageNSLayoutManager从头开始实例化 a。苹果一定是忘记在UITextView初始化方法中初始化一些东西了。我提交了一个错误报告,我希望你也这样做。

// ios7 bug fix
// check if the device is running iOS 7.0 or later
NSString *reqSysVer = @"7.0";
NSString *currSysVer = [[UIDevice currentDevice] systemVersion];
BOOL osVersionSupported = ([currSysVer compare:reqSysVer  options:NSNumericSearch] != NSOrderedAscending);

if (osVersionSupported) {
    NSTextStorage* textStorage = [[NSTextStorage alloc] init];
    NSLayoutManager* layoutManager = [NSLayoutManager new];
    [textStorage addLayoutManager:layoutManager];
    NSTextContainer *textContainer = [[NSTextContainer alloc] initWithSize:self.view.bounds.size];
    [layoutManager addTextContainer:textContainer];
    yourTextView = [[UITextView alloc] initWithFrame:someFrameForYourTextView
                                       textContainer:textContainer];
    // if using ARC, remove these 3 lines
    [textContainer release];
    [layoutManager release];
    [textStorage release];
}
else {
    yourTextView = [[UITextView alloc] initWithFrame:someFrameForYourTextView];
}
于 2013-10-12T22:08:09.510 回答
4

iOS 7 中有两个问题可以解释您的问题:

  • 在 iOS 7 中 contentOffset 并不总是最新的。
  • scrollRangeToVisible:不会滚动到文本视图末尾的空行。

解决方案可能是:

-(void)scrollOutputToBottom {
    CGRect caretRect = [textView caretRectForPosition:textView.endOfDocument];
    [textView scrollRectToVisible:caretRect animated:NO];
}
于 2013-10-09T16:09:19.327 回答
2

尝试这个:

// Don't forget to set textView's delegate 
-(void)textViewDidChangeSelection:(UITextView *)textView {
    [textView scrollRangeToVisible:NSMakeRange([textView.text length], 0)];
}
于 2015-04-01T06:27:13.693 回答
1

对于那些使用 Swift 的人,我在这里发布与 RawMean 相同的答案(再次感谢!)。在我写这篇文章时(2014 年 12 月),iOS 8.1 中仍然存在问题,他的解决方案完美运行......

var textView: UITextView!
var textStorage: NSTextStorage!
var layoutManager: NSLayoutManager!
var textContainer: NSTextContainer!


override func viewDidLoad() {
    textStorage = NSTextStorage()
    layoutManager = NSLayoutManager()
    textStorage.addLayoutManager(layoutManager)

    let newTextViewRect = view.bounds
    let containerSize = CGSize(width: newTextViewRect.width, height: CGFloat.max)

    textContainer = NSTextContainer(size: containerSize)
    layoutManager.addTextContainer(textContainer)

    textView = UITextView(frame: newTextViewRect, textContainer: textContainer)

    textView.delegate = self
    view.addSubview(textView)

}

override func viewDidLayoutSubviews() {
    textView.frame = view.bounds
}

我使用 scrollRangeToVisible 方法在添加文本时在底部平滑滚动...

let length = countElements(textView.text)
let range:NSRange = NSMakeRange(length - 1, 1)
textView.scrollRangeToVisible(range)
于 2014-12-16T00:58:16.890 回答
0

基本上 setScrollEnabled = YES 需要在 layoutSubviews 被调用之前设置。它对我有用。

于 2014-01-10T16:01:08.443 回答
0

这对我行得通。参考:UITextView setText 不应该在 ios8 中跳转到顶部

self.textView.layoutManager.allowsNonContiguousLayout = NO;
self.textView.text = fileContent;
if(fileContent.length > 1)
{
    NSRange range = NSMakeRange(self.textView.text.length - 1, 1);
    [self.textView scrollRangeToVisible:range];
}
于 2015-08-09T15:05:10.580 回答
0

斯威夫特 2.0 - IOS 8

这基本上是上面dklt 答案的 Swift 2.0 版本。以前我使用相同的方法,没有scrollEnabled. 大多数时候它工作正常。但是,当scrollToBottom()几乎同时快速连续调用时,它有时不起作用。

的 2 行scrollEnabled没有多大意义,但在添加它们之后,该方法始终有效!

注意:我试图将 2 行放在scrollEnabled之前或之后的不同位置scrollRangeTovisible,正如 dklt 的回答评论中所建议的那样......

只有 dklt 的原始解决方案对我有用。其余的没有。

func scrollToBottom()
{
    let range:NSRange = NSMakeRange(self.textView.text.characters.count - 1, 1)

    self.textView.scrollRangeToVisible(range)
    self.textView.scrollEnabled = false
    self.textView.scrollEnabled = true
}
于 2015-12-01T15:50:06.473 回答
-1

请尝试此解决方案

-(void) scrollToBottom {
    [textView setContentOffset:CGPointMake(0.0, textView.contentSize.height) animated:YES];
}
于 2014-04-22T08:05:35.783 回答