誰かがcell
を選択する_Collection View
_を実行しようとしていますが、選択するたびに、選択内容を保持する別の_View Controller
_に移動します。ただし、このコード行をdidSelectItemAtIndexPath
に適用すると、問題が発生します。
_self.performSegueWithIdentifer("showDetail", sender: self)
_
シミュレータで実行すると、cell
の選択はindexPath
に従って動作しますが、新しいcell
を選択するたびに選択が記憶されます。たとえば、各cell
には写真とラベルがあり、cell
で最初のindexPath
を選択すると、segue
は最初に空白のビューになり、次に選択したcell
になります。 cell
で別のindexPath
の番号3を選択すると、空のビューは前の選択からの最初のcell
になり、その後、選択した3番目のcell
になります。そのたびにそれを行う。 performSegueWithIdentifer
コードを削除すると(Xcode 6.2(6.1.1ではランダム))選択は以前の選択であり、「selectedCell」ではありませんが、view
を取得するために少なくとも2回ではなく1回のみ選択します。 indexPath
に何か問題があります。これは私のprepareForSegue
のコードです
_override func prepareForSegue(segue: UIStoryBoardSegue, sender: AnyObject?) {
if segue.identifer == "showDetail" {
let detailVC:DetailViewController = segue.destinationViewController as DetailViewController
detailVC.selectedImageName = selectedImage
detailVC.selectedLabel = selectedLabels
}
}
_
私は何をすべきか、どの解決策を適用すべきかにこだわっています。 performSegueWithIdentifer
コードを保持し、Equatable
にfind(array, selection)
を実装するindexPath
を作成しますか?または、選択に基づいてindexPath
を実行し、選択されなくなったセルを削除するループを書くことができます(はるかに簡単だと思われます)。ただし、 'selectedCell'のプロパティの値はオプションであるためわからないため、ループにどの条件を記述するかはわかりません。
_for (index, value) in enumerate(cellItems) {
//something here to remove 'selectedItem' in the indexPath
}
_
performSegueWithIdentifer
からdidSelectItemAtIndexPath
コードを削除すると、正しいindexPathで選択を取得するためにprepareForSegue
で何ができますか?
didSelectItemAtIndexPath
で完全なコードを編集します
_override func collectionView(collectionView: UICollectionView, didSelectItemAtIndexPath indexPath: NSIndexPath) {
selectedImage = cellImages[indexPath.row] as String
selectedLabels = cellLabels[indexPath.row] as String
self.performSegueWithIdentifer("showDetail", sender: self)
}
_
performSegueWithIdentifer
の送信者をindexPath
に変更しようとしましたが、問題はまだ残っています。
EDIT 2 CollectionViewControllerの完全なコード
_class CollectionViewController: UIViewController, UICollectionViewDataSource, UICollectionViewDelegate {
@IBOutlet weak var collectionView: UICollectionView!
var selectedImage = String()
var selectedLabels = String()
var cellImages:[String] = ["1.jpg", "2.jpg", "3.jpg", "4.jpg", "5.jpg", "6.jpg", "7.jpg", "8.jpg", "9.jpg", "10.jpg", "11.jpg", "13.jpg", "14jpg"]
var cellLabels:[String] = ["Photo 1", "Photo 2", "Photo 3", "Photo 4", "Photo 5", "Photo 6", "Photo 7", "Photo 8", "Photo 9", "Photo 10", "Photo 11", "Photo 12", "Photo 13", "Photo 14"]
func collectionView(collectionView: UICollectionView, numberOfNumberItemsInSection: Int) -> Int {
return cellImages.count
}
func collectionView(collectionView: UICollectionView, cellForItemAtIndexPath indexPath: NSIndexPath) -> UICollectionViewCell {
let cell: PhotoViewCell = collectionView.dequeueReuseableCellWithReuseIdentifier("Cell", forIndexPath: indexPath) as PhotoViewCell
cell.labelCell.text = cellLabels[indexPath.row]
cell.ImageCell.image = UIImage(named: cellImages[indexPath.row])
return cell
}
override func collectionView(collectionView: UICollectionView, didSelectItemAtIndexPath indexPath: NSIndexPath) {
selectedImage = cellImages[indexPath.row] as String
selectedLabels = cellLabels[indexPath.row] as String
self.performSegueWithIdentifer("showDetail", sender: self)
}
override func prepareForSegue(segue: UIStoryBoardSegue, sender: AnyObject?) {
if segue.identifer == "showDetail" {
let detailVC:DetailViewController = segue.destinationViewController as DetailViewController
detailVC.selectedImageName = selectedImage
detailVC.selectedLabel = selectedLabels
}
}
}
_
PhotoViewCell
_class PhotoViewCell: UICollectionViewCell {
@IBOutlet var labelCell: UILabel!
@IBOutlet var ImageCell: UIImage!
}
_
編集3-修正
私はあなたの提案を試みましたが、残念ながら問題はまだ二重ビューで持続しています-実際に選択されたcell
に連れて行く前に2つのビューを渡しています。 didSelectItemAtIndexPath
のコードも少し修正しましたが、それでも問題は解決しませんでした。
_if let cell = collectionView.cellForItemAtIndexPath(indexPath) as? PhotoViewCell {
performSegueWithIdentifier("showDetail", sender: cell)
}
_
ただし、他の提案に従って、StoryBoard
で、_Collection View
_ segue
からcell
を、_show_IDという識別子を持つDetailViewController
に追加しました。 segue
を削除すると、cells
から何も選択できません。
performSegueWithIdentifer
コードはダブルビューのトリガーであるようですが、削除するとcell
が1回しか選択されないため、問題はindexPath
選択のcell
が空のビューで最初に選択されるため正しくないことでした(それはshowDetail segue
?)を処理することです。これにより、indexPath
が同期しなくなります。
編集-解決済み
これにより、二重選択が停止しました(performSegueWithIdentifier
行が削除されました):-
_override func collectionView(collectionView: UICollectionView, didSelectItemAtIndexPath indexPath: NSIndexPath) {
if let cell = collectionView.cellForItemAtIndexPath(indexPath) {
cellLabels[indexPath.row] as String
cellImages[indexPath.row] as String
}
}
_
助けてくれて本当にありがとうございます !!!!
(注:Swift 4以降の最新のプラクティスのためにこれを更新しました。)
UIView
オブジェクトに可能な限りこだわります。
_override func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
guard let cell = collectionView.cellForItem(at: indexPath) else { return }
performSegue(withIdentifier: "showDetail", sender: cell)
}
_
次に、prepare(for:sender:)
_override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
switch segue.identifier {
case "showDetail":
guard let indexPath = (sender as? UIView)?.findCollectionViewIndexPath() else { return }
guard let detailViewController = segue.destination as? DetailViewController else { return }
detailViewController.selectedImageName = cellImages[indexPath.row]
detailViewController.selectedLabel = cellLabels[indexPath.row]
default: return
}
}
_
しばらく前に作成した拡張機能を使用しましたfindCollectionViewIndexPath()
_extension UIView {
func findCollectionView() -> UICollectionView? {
if let collectionView = self as? UICollectionView {
return collectionView
} else {
return superview?.findCollectionView()
}
}
func findCollectionViewCell() -> UICollectionViewCell? {
if let cell = self as? UICollectionViewCell {
return cell
} else {
return superview?.findCollectionViewCell()
}
}
func findCollectionViewIndexPath() -> IndexPath? {
guard let cell = findCollectionViewCell(), let collectionView = cell.findCollectionView() else { return nil }
return collectionView.indexPath(for: cell)
}
}
_
ストーリーボードにすでにセグエがあり、func collectionView(, didSelectItemAtIndexPath:)
は必要ないのではないかと疑っていますが、いずれにしても、セグエの準備は機能するはずです。
Swift 3.0
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
let iPath = self.collectionView.indexPathsForSelectedItems
let indexPath : NSIndexPath = iPath![0] as NSIndexPath
let rowIndex = indexPath.row
print("row index = \(rowIndex)")
}