web-dev-qa-db-ja.com

UIScrollViewをプログラムで下にスクロール

コード内でUIScrollViewを一番下までスクロールするにはどうすればよいですか?または、より一般的な方法で、サブビューの任意のポイントに?

271
nico

UIScrollViewのsetContentOffset:animated:関数を使用して、コンテンツビューの任意の部分にスクロールできます。 scrollViewがself.scrollViewであると仮定して、下にスクロールするコードを次に示します。

CGPoint bottomOffset = CGPointMake(0, self.scrollView.contentSize.height - self.scrollView.bounds.size.height + self.scrollView.contentInset.bottom);
[self.scrollView setContentOffset:bottomOffset animated:YES];

お役に立てば幸いです!

595
Ben Gotow

簡単なコピーペーストのための受け入れられた回答の迅速なバージョン:

let bottomOffset = CGPoint(x: 0, y: scrollView.contentSize.height - scrollView.bounds.size.height)
scrollView.setContentOffset(bottomOffset, animated: true)
96
Esqarrouth

最も簡単なソリューション:

[scrollview scrollRectToVisible:CGRectMake(scrollview.contentSize.width - 1,scrollview.contentSize.height - 1, 1, 1) animated:YES];
50
Hai Hw

既存の回答の単なる強化。

CGPoint bottomOffset = CGPointMake(0, self.scrollView.contentSize.height - self.scrollView.bounds.size.height + self.scrollView.contentInset.bottom);
[self.scrollView setContentOffset:bottomOffset animated:YES];

下部の差し込みも処理します(キーボードが表示されているときにスクロールビューを調整するためにそれを使用している場合)

26
vivek241

コンテンツサイズの高さに対するコンテンツオフセットの設定は間違っています。コンテンツの下部をスクロールビューのtopにスクロールするため、見えなくなります。

正しい解決策は、次のようにコンテンツの下部をスクロールビューのbottomにスクロールすることです(svはUIScrollViewです):

CGSize csz = sv.contentSize;
CGSize bsz = sv.bounds.size;
if (sv.contentOffset.y + bsz.height > csz.height) {
    [sv setContentOffset:CGPointMake(sv.contentOffset.x, 
                                     csz.height - bsz.height) 
                animated:YES];
}
19
matt

contentInsetを考慮に入れたSwift 2.2ソリューション

let bottomOffset = CGPoint(x: 0, y: scrollView.contentSize.height - scrollView.bounds.size.height + scrollView.contentInset.bottom)
scrollView.setContentOffset(bottomOffset, animated: true)

これは拡張機能である必要があります

extension UIScrollView {

  func scrollToBottom() {
    let bottomOffset = CGPoint(x: 0, y: contentSize.height - bounds.size.height + contentInset.bottom)
    setContentOffset(bottomOffset, animated: true)
  }
}

スクロールする前にbottomOffset.y > 0をチェックする必要がある場合があることに注意してください

15
onmyway133

迅速な実装:

extension UIScrollView {
   func scrollToBottom(animated: Bool) {
     if self.contentSize.height < self.bounds.size.height { return }
     let bottomOffset = CGPoint(x: 0, y: self.contentSize.height - self.bounds.size.height)
     self.setContentOffset(bottomOffset, animated: animated)
  }
}

これを使って:

yourScrollview.scrollToBottom(animated: true)
13
duan

トップにスクロールします

- CGPoint topOffset = CGPointMake(0, 0);
- [scrollView setContentOffset:topOffset animated:YES];

下にスクロール

- CGPoint bottomOffset = CGPointMake(0, scrollView.contentSize.height - self.scrollView.bounds.size.height);
 - [scrollView setContentOffset:bottomOffset animated:YES];
13
Waseem

contentSizeboundsよりも小さい場合はどうなりますか?

Swiftの場合:

scrollView.setContentOffset(CGPointMake(0, max(scrollView.contentSize.height - scrollView.bounds.size.height, 0) ), animated: true)

また、UITableview(UIScrollViewのサブクラス)を使用している場合に、これを行う別の便利な方法を見つけました。

[(UITableView *)self.view scrollToRowAtIndexPath:scrollIndexPath atScrollPosition:UITableViewScrollPositionBottom animated:YES];
7
nico

UIScrollViewのsetContentOffset:animated:関数を使用して、Swiftで一番下までスクロールします。

let bottomOffset : CGPoint = CGPointMake(0, scrollView.contentSize.height - scrollView.bounds.size.height + scrollView.contentInset.bottom)
scrollView.setContentOffset(bottomOffset, animated: true)
6
Kim

(オプションの)footerViewとcontentInsetを使用したソリューションは次のとおりです。

CGPoint bottomOffset = CGPointMake(0, _tableView.contentSize.height - tableView.frame.size.height + _tableView.contentInset.bottom);
if (bottomOffset.y > 0) [_tableView setContentOffset: bottomOffset animated: YES];
5
Pieter

valdyr、これがあなたを助けることを願っています:

CGPoint bottomOffset = CGPointMake(0, [textView contentSize].height - textView.frame.size.height);

if (bottomOffset.y > 0)
 [textView setContentOffset: bottomOffset animated: YES];
3
Misha

救助するカテゴリ!

これを共有ユーティリティヘッダーにどこかに追加します。

@interface UIScrollView (ScrollToBottom)
- (void)scrollToBottomAnimated:(BOOL)animated;
@end

そして、そのユーティリティの実装に:

@implementation UIScrollView(ScrollToBottom)
- (void)scrollToBottomAnimated:(BOOL)animated
{
     CGPoint bottomOffset = CGPointMake(0, self.contentSize.height - self.bounds.size.height);
     [self setContentOffset:bottomOffset animated:animated];
}
@end

次に、たとえば次のように好きな場所に実装します。

[[myWebView scrollView] scrollToBottomAnimated:YES];
2
BadPirate

アニメーションが必要ない場合、これは動作します:

[self.scrollView setContentOffset:CGPointMake(0, CGFLOAT_MAX) animated:NO];
1
Krys Jurgowski

Mattソリューションは正しいように思えますが、セットアップされているコレクションビューがある場合は、コレクションビューのインセットも考慮する必要があります。

適合コードは次のとおりです。

CGSize csz = sv.contentSize;
CGSize bsz = sv.bounds.size;
NSInteger bottomInset = sv.contentInset.bottom;
if (sv.contentOffset.y + bsz.height + bottomInset > csz.height) {
    [sv setContentOffset:CGPointMake(sv.contentOffset.x, 
                                     csz.height - bsz.height + bottomInset) 
                animated:YES];
}
1
tiguero

コンテンツの下部が見えるようにする良い方法は、次の式を使用することです:

contentOffsetY = MIN(0, contentHeight - boundsHeight)

これにより、コンテンツの下端が常にビューの下端以上になります。 MIN(0, ...)が必要なのは、UITableView(およびおそらくUIScrollView)が、contentOffsetY >= 0を視覚的にスナップしてスクロールしようとしたときにcontentOffsetY = 0を保証するためです。これはユーザーにはかなり奇妙に見えます。

これを実装するコードは次のとおりです。

UIScrollView scrollView = ...;
CGSize contentSize = scrollView.contentSize;
CGSize boundsSize = scrollView.bounds.size;
if (contentSize.height > boundsSize.height)
{
    CGPoint contentOffset = scrollView.contentOffset;
    contentOffset.y = contentSize.height - boundsSize.height;
    [scrollView setContentOffset:contentOffset animated:YES];
}
1
jjrscott

テーブルビューの最後のアイテムにスクロールするソリューション:

スイフト3:

if self.items.count > 0 {
        self.tableView.scrollToRow(at:  IndexPath.init(row: self.items.count - 1, section: 0), at: UITableViewScrollPosition.bottom, animated: true)
}
1
MarionFlex

contentSizeは実際のテキストのサイズを実際に反映していないことがわかったので、一番下までスクロールしようとすると少しずれます。実際のコンテンツサイズを判断する最良の方法は、実際にNSLayoutManagerusedRectForTextContainer:メソッドを使用することです。

UITextView *textView;
CGSize textSize = [textView.layoutManager usedRectForTextContainer:textView.textContainer].size;

UITextViewに実際に表示されるテキストの量を判断するには、フレームの高さからテキストコンテナーのインセットを差し引くことで計算できます。

UITextView *textView;
UIEdgeInsets textInsets = textView.textContainerInset;
CGFloat textViewHeight = textView.frame.size.height - textInsets.top - textInsets.bottom;

その後、スクロールしやすくなります。

// if you want scroll animation, use contentOffset
UITextView *textView;
textView.contentOffset = CGPointMake(textView.contentOffset.x, textSize - textViewHeight);

// if you don't want scroll animation
CGRect scrollBounds = textView.bounds;
scrollBounds.Origin = CGPointMake(textView.contentOffset.x, textSize - textViewHeight);
textView.bounds = scrollBounds;

空のUITextViewのさまざまなサイズが表すものに関する参照用のいくつかの数値。

textView.frame.size = (width=246, height=50)
textSize = (width=10, height=16.701999999999998)
textView.contentSize = (width=246, height=33)
textView.textContainerInset = (top=8, left=0, bottom=8, right=0)
0
mikeho
CGFloat yOffset = scrollView.contentOffset.y;

CGFloat height = scrollView.frame.size.height;

CGFloat contentHeight = scrollView.contentSize.height;

CGFloat distance = (contentHeight  - height) - yOffset;

if(distance < 0)
{
    return ;
}

CGPoint offset = scrollView.contentOffset;

offset.y += distance;

[scrollView setContentOffset:offset animated:YES];
0
8suhas

Xamarin.iOS受け入れられた回答のバージョン

var bottomOffset = new CGPoint (0,
     scrollView.ContentSize.Height - scrollView.Frame.Size.Height
     + scrollView.ContentInset.Bottom);

scrollView.SetContentOffset (bottomOffset, false);
0
Abdur

ここでの回答はすべて、安全な領域を考慮していないようです。 iOS 11以降、iPhone Xには安全な領域が導入されました。これはscrollViewのcontentInsetに影響する場合があります。

IOS 11以降の場合、コンテンツのインセットを含めて適切に下にスクロールします。 adjustedContentInsetの代わりにcontentInsetを使用する必要があります。このコードを確認してください:

  • 迅速:
let bottomOffset = CGPoint(x: 0, y: scrollView.contentSize.height - scrollView.bounds.height + scrollView.adjustedContentInset.bottom)
scrollView.setContentOffset(bottomOffset, animated: true)
  • Objective-C
CGPoint bottomOffset = CGPointMake(0, self.scrollView.contentSize.height - self.scrollView.bounds.size.height + self.scrollView.adjustedContentInset.bottom);
[self.scrollView setContentOffset:bottomOffset animated:YES];
  • Swift拡張(これは元のcontentOffset.xを保持します):
extension UIScrollView {
    func scrollsToBottom(animated: Bool) {
        let bottomOffset = CGPoint(x: contentOffset.x,
                                   y: contentSize.height - bounds.height + adjustedContentInset.bottom)
        setContentOffset(bottomOffset, animated: animated)
    }
}

参照:

0
Honghao Zhang

Xamarin.iOSコピーアンドペーストを容易にするために受け入れられた回答のUICollectionViewのバージョン

var bottomOffset = new CGPoint (0, CollectionView.ContentSize.Height - CollectionView.Frame.Size.Height + CollectionView.ContentInset.Bottom);          
CollectionView.SetContentOffset (bottomOffset, false);
0
Abdur

UITableViewControllerを追加した後、self.tableView(iOS 4.1)footerViewで使用しようとすると、うまくいきませんでした。境界線からスクロールして、黒い画面を表示します。

代替ソリューション:

 CGFloat height = self.tableView.contentSize.height; 

 [self.tableView setTableFooterView: myFooterView];
 [self.tableView reloadData];

 CGFloat delta = self.tableView.contentSize.height - height;
 CGPoint offset = [self.tableView contentOffset];
 offset.y += delta;

 [self.tableView setContentOffset: offset animated: YES];
0
valdyr

水平スクロールビューの場合

私のように水平スクロールビューがあり、その最後までスクロールしたい場合(私の場合は右端まで)、受け入れられた回答の一部を変更する必要があります。

Objective-C

CGPoint rightOffset = CGPointMake(self.scrollView.contentSize.width - self.scrollView.bounds.size.width + self.scrollView.contentInset.right, 0 );
[self.scrollView setContentOffset:rightOffset animated:YES];

スイフト

let rightOffset: CGPoint = CGPoint(x: self.scrollView.contentSize.width - self.scrollView.bounds.size.width + self.scrollView.contentInset.right, y: 0)
self.scrollView.setContentOffset(rightOffset, animated: true)
0
Esmaeil

Swiftの場合:

   if self.mainScroll.contentSize.height > self.mainScroll.bounds.size.height {
        let bottomOffset = CGPointMake(0, self.mainScroll.contentSize.height - self.mainScroll.bounds.size.height);
        self.mainScroll.setContentOffset(bottomOffset, animated: true)
    }
0
Ponja