scrollToItem 在我的 CollectionView 中不起作用?
scrollToItem isn't working in my CollectionView?
为什么 scrollToItem 在我的 CollectionView 中不起作用?我想在 EnabledID
上使用 scrollToItem。那么问题是什么以及如何解决?
代码:
var dataArray = NSMutableArray() // dataArray is has a data from Database
var EnabledID = Int()
EnabledID = UserDefaults.standard.integer(forKey: "EnabledID")
if EnabledID == 0 {
EnabledID = 1
UserDefaults.standard.set(1, forKey: "EnabledID")
} else {
if EnabledID < dataArray.count {
let index = NSIndexPath(item: EnabledID, section: 0)
self.myCollectionView.scrollToItem(at: index as IndexPath, at: .centeredVertically, animated: true)
}
}
使用这个
self.messageCV.reloadData()
let indexPath = IndexPath(row: 0, section: 0)
self.messageCV.scrollToItem(at: indexPath, at: .bottom, animated: true) // .top
为水平集合视图试试这个
var isViewDidLayoutCallFirstTime = true
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
guard self.isViewDidLayoutCallFirstTime else {return}
self.isViewDidLayoutCallFirstTime = false
self.collectionView.collectionViewLayout.collectionViewContentSize // It will required to re calculate collectionViewContentSize in internal method
let indexpath = IndexPath(item: self.currentIndex, section: 0)
self.collectionView.scrollToItem(at: indexpath, at: UICollectionViewScrollPosition.centeredHorizontally, animated: false)
}
试试这个代码...
[collectionView setDelegate:self];
[collectionView reloadData];
[collectionView layoutIfNeeded];
[collectionView scrollToItemAtIndexPath:[NSIndexPath indexPathForItem:0 inSection:0] atScrollPosition:UICollectionViewScrollPositionTop animated:YES];
Swift - 4.x
collectionView.delegate = self
collectionView.reloadData()
collectionView.layoutIfNeeded()
collectionView.scrollToItem(at: IndexPath(item: 0, section: 0), at: .top, animated: true)
滚动方向为水平时需要使用left,right或centeredHorizontally .
top为垂直方向
collectionView.scrollToItem(at: index, at: .top, animated: true)
我的问题有点奇怪。我的 iPhone 6/7/8 工作正常,但 iPhone X 或 11 无法滚动到我预期的位置。
我的原始代码是:
collectionView.reloadData()
collectionView.scrollToItem(at: IndexPath(item: 10, section: 0), at: .top, animated: true)
在我添加collectionView.layoutIfNeeded()之后。问题在我的 iPhone X 和 11 上解决。
collectionView.reloadData()
collectionView.layoutIfNeeded()
collectionView.scrollToItem(at: IndexPath(item: 10, section: 0), at: .top, animated: true)
这个答案的灵感来自于@ram。他应该得到荣誉。添加此答案只是为了强调不同 iPhone 之间的区别。这样人们可以更快地搜索他们的答案。
它不会工作,因为 contentSize 可能是 (0,0)。
如果您不想使用 layoutIfNeeded()(因为您可能会在那里执行一些不需要的或不必要的事情),您应该自己计算(假设您的数据已加载,并且视图已布局)。
Swift 5.2
collection.contentSize = CGSize(width: collection.bounds.width * CGFloat(items.count), height: collection.bounds.height)
collection.scrollToItem(at: IndexPath(row: startingIndex, section: 0), at: .left, animated: false)
我发现最好的方法是不使用 scrollToItem,而是获取索引的 CGRect,然后使其可见。
let rect = self.collectionView.layoutAttributesForItem(at: IndexPath(row: 5, section: 0))?.frame
self.collectionView.scrollRectToVisible(rect!, animated: false)
在我的例子中,.centeredVertically 没有任何效果,并且悄悄地失败了。
.centeredHorizontally 修复了它
我在 iOS 14.x 和 Xcode 12.x 上遇到过这个问题。
对我有用的解决方案是使用:
collectionView.scrollRectToVisible(rect, animated: true)
尝试 this.it 对我有用
func scrollToIndex(index:Int) {
let rect = self.collectionView.layoutAttributesForItem(at:IndexPath(row: index, section: 0))?.frame
self.collectionView.scrollRectToVisible(rect!, animated: true)
}
var scrollImg: Int = 0
override func viewDidLoad() {
super.viewDidLoad()
_ = Timer.scheduledTimer(timeInterval: 2, target: self, selector: #selector(autoScroll), userInfo: nil, repeats: true)
}
@objc func autoScroll() {
let totalCount = imageArray.count-1
if scrollImg == totalCount{
scrollImg = 0
}else {
scrollImg += 1
}
DispatchQueue.main.async {
let rect = self.listOfViewsCollectionViews.layoutAttributesForItem(at: IndexPath(row: self.scrollImg, section: 0))?.frame
if self.scrollImg == 0{
self.listOfViewsCollectionViews.scrollRectToVisible(rect!, animated: false)
}else {
self.listOfViewsCollectionViews.scrollRectToVisible(rect!, animated: true)
}
}
}
collectionView.collectionViewLayout.invalidateLayout()
collectionView.scrollToItem(at: indexPath, at: .centeredHorizontally, animated: true)
collectionView.collectionViewLayout.invalidateLayout()
为什么 scrollToItem 在我的 CollectionView 中不起作用?我想在 EnabledID
上使用 scrollToItem。那么问题是什么以及如何解决?
代码:
var dataArray = NSMutableArray() // dataArray is has a data from Database
var EnabledID = Int()
EnabledID = UserDefaults.standard.integer(forKey: "EnabledID")
if EnabledID == 0 {
EnabledID = 1
UserDefaults.standard.set(1, forKey: "EnabledID")
} else {
if EnabledID < dataArray.count {
let index = NSIndexPath(item: EnabledID, section: 0)
self.myCollectionView.scrollToItem(at: index as IndexPath, at: .centeredVertically, animated: true)
}
}
使用这个
self.messageCV.reloadData()
let indexPath = IndexPath(row: 0, section: 0)
self.messageCV.scrollToItem(at: indexPath, at: .bottom, animated: true) // .top
为水平集合视图试试这个
var isViewDidLayoutCallFirstTime = true
override func viewDidLayoutSubviews() {
super.viewDidLayoutSubviews()
guard self.isViewDidLayoutCallFirstTime else {return}
self.isViewDidLayoutCallFirstTime = false
self.collectionView.collectionViewLayout.collectionViewContentSize // It will required to re calculate collectionViewContentSize in internal method
let indexpath = IndexPath(item: self.currentIndex, section: 0)
self.collectionView.scrollToItem(at: indexpath, at: UICollectionViewScrollPosition.centeredHorizontally, animated: false)
}
试试这个代码...
[collectionView setDelegate:self];
[collectionView reloadData];
[collectionView layoutIfNeeded];
[collectionView scrollToItemAtIndexPath:[NSIndexPath indexPathForItem:0 inSection:0] atScrollPosition:UICollectionViewScrollPositionTop animated:YES];
Swift - 4.x
collectionView.delegate = self
collectionView.reloadData()
collectionView.layoutIfNeeded()
collectionView.scrollToItem(at: IndexPath(item: 0, section: 0), at: .top, animated: true)
滚动方向为水平时需要使用left,right或centeredHorizontally .
top为垂直方向
collectionView.scrollToItem(at: index, at: .top, animated: true)
我的问题有点奇怪。我的 iPhone 6/7/8 工作正常,但 iPhone X 或 11 无法滚动到我预期的位置。 我的原始代码是:
collectionView.reloadData()
collectionView.scrollToItem(at: IndexPath(item: 10, section: 0), at: .top, animated: true)
在我添加collectionView.layoutIfNeeded()之后。问题在我的 iPhone X 和 11 上解决。
collectionView.reloadData()
collectionView.layoutIfNeeded()
collectionView.scrollToItem(at: IndexPath(item: 10, section: 0), at: .top, animated: true)
这个答案的灵感来自于@ram。他应该得到荣誉。添加此答案只是为了强调不同 iPhone 之间的区别。这样人们可以更快地搜索他们的答案。
它不会工作,因为 contentSize 可能是 (0,0)。 如果您不想使用 layoutIfNeeded()(因为您可能会在那里执行一些不需要的或不必要的事情),您应该自己计算(假设您的数据已加载,并且视图已布局)。
Swift 5.2
collection.contentSize = CGSize(width: collection.bounds.width * CGFloat(items.count), height: collection.bounds.height)
collection.scrollToItem(at: IndexPath(row: startingIndex, section: 0), at: .left, animated: false)
我发现最好的方法是不使用 scrollToItem,而是获取索引的 CGRect,然后使其可见。
let rect = self.collectionView.layoutAttributesForItem(at: IndexPath(row: 5, section: 0))?.frame
self.collectionView.scrollRectToVisible(rect!, animated: false)
在我的例子中,.centeredVertically 没有任何效果,并且悄悄地失败了。 .centeredHorizontally 修复了它
我在 iOS 14.x 和 Xcode 12.x 上遇到过这个问题。
对我有用的解决方案是使用:
collectionView.scrollRectToVisible(rect, animated: true)
尝试 this.it 对我有用
func scrollToIndex(index:Int) {
let rect = self.collectionView.layoutAttributesForItem(at:IndexPath(row: index, section: 0))?.frame
self.collectionView.scrollRectToVisible(rect!, animated: true)
}
var scrollImg: Int = 0
override func viewDidLoad() {
super.viewDidLoad()
_ = Timer.scheduledTimer(timeInterval: 2, target: self, selector: #selector(autoScroll), userInfo: nil, repeats: true)
}
@objc func autoScroll() {
let totalCount = imageArray.count-1
if scrollImg == totalCount{
scrollImg = 0
}else {
scrollImg += 1
}
DispatchQueue.main.async {
let rect = self.listOfViewsCollectionViews.layoutAttributesForItem(at: IndexPath(row: self.scrollImg, section: 0))?.frame
if self.scrollImg == 0{
self.listOfViewsCollectionViews.scrollRectToVisible(rect!, animated: false)
}else {
self.listOfViewsCollectionViews.scrollRectToVisible(rect!, animated: true)
}
}
}
collectionView.collectionViewLayout.invalidateLayout()
collectionView.scrollToItem(at: indexPath, at: .centeredHorizontally, animated: true)
collectionView.collectionViewLayout.invalidateLayout()