使用从 UICollectionViewCell 中选择的 PFUser 内容更新 DetailViewController

Update DetailViewController with Selected PFUser content from UICollectionViewCell

我正在构建一个社交网络应用程序,目前我想在 PFUser.CurrentUser() 从 UICollectionView 选择另一个用户的 Cell 时在 HomeViewController 和 OtherUserViewController 之间创建一个 segue。 segue 工作正常,但 otherUserName 标签没有更新所选 PFUser 的名称。

我生成了所有用户的列表,并将他们每个人放在自己的 Cell 中。

var userListData = [String]()
var userListUserNames = [String]()

@IBOutlet weak var userCollectionView: UICollectionView!

override func viewDidLoad() {
    super.viewDidLoad()
    // Do any additional setup after loading the view.

    userCollectionView.delegate = self
    userCollectionView.dataSource = self
    loadData()
}
func loadData (){
    var userListQuery = PFUser.query()
    userListQuery?.findObjectsInBackgroundWithBlock({ (objects, error) -> Void in
        if error == nil {

            if let objects = objects {

                for object in objects {

                    if let user = object as? PFUser {

                        if user.objectId != PFUser.currentUser()?.objectId {

                            self.userListData.append(user.objectId!)
                            self.userListUserNames.append(user.username!)
                            self.userCollectionView.reloadData()
                        }
                    }
                }
            }
        }
    })
}

之后我创建了单元格。

func numberOfSectionsInCollectionView(collectionView: UICollectionView) -> Int {
    return 1
}

func collectionView(collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
    return userListData.count

}

func collectionView(collectionView: UICollectionView, cellForItemAtIndexPath indexPath: NSIndexPath) -> UICollectionViewCell {
    let cell = collectionView.dequeueReusableCellWithReuseIdentifier("homeCollectionViewCell", forIndexPath: indexPath) as! HomeCollectionViewCell

    var names = self.userListUserNames[indexPath.row]
    cell.profileNameButtonOutlet.setTitle(names.lowercaseString, forState: UIControlState.Normal)

    return cell
}

然后我为单元格创建了一个 prepareForSegue 方法和 didDeselectItemAtIndexPath。

override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
    var currentOtherUser: PFObject?
    if let object = sender as? PFObject {

        currentOtherUser = sender as? PFObject
    } else {
        println("there is no currentOtherUser")
    }
    var otherUserScene = segue.destinationViewController as! OtherUserViewController
    otherUserScene.currentOtherUser = (currentOtherUser)
}

func collectionView(collectionView: UICollectionView, didDeselectItemAtIndexPath indexPath: NSIndexPath) {
    let currentOtherUser = userListUserNames[indexPath.row]
    self.performSegueWithIdentifier("homeToOtherUser", sender: currentOtherUser)
}

在我的细节控制器中,我设置了信息检索。

class OtherUserViewController: UIViewController {

var currentOtherUser : PFObject?

@IBOutlet weak var otherUserName: UILabel!

var otherUser: PFUser?

override func viewDidLoad() {
    super.viewDidLoad()

    if let object = currentOtherUser {
        if let value = object["username"] as? String {

            otherUserName.text = value

        }
    } else {
        println("no name to display")
    }
}
}

prepareForSegue 上的 return 始终为零,OtherUserViewController 的 ViewDidLoad 也是如此。

我哪里做错了,我怎样才能正确设置 segue?

基本上你所做的在逻辑上是没问题的,segues 也没有问题。然而问题是:

  var userListUserNames = [String]()   // This is not [PFOBject]

但是在为 segue 做准备时,您正在检查:

 if let object = sender as? PFObject {
    //which will never be called because sender is of type <String>
    currentOtherUser = sender as? PFObject
} else {
    println("there is no currentOtherUser")  //this is what is being displayed right now
}

那是因为:

  func collectionView(collectionView: UICollectionView, didDeselectItemAtIndexPath indexPath: NSIndexPath) {
let currentOtherUser = userListUserNames[indexPath.row]  //This is String
self.performSegueWithIdentifier("homeToOtherUser", sender: currentOtherUser)

}

这就是问题所在。

解决方案是将类型设置为 String,如果这样做,您将获得名称,但您可能会丢失 otherUser 的其他关联属性,稍后可能需要这些属性。如果是这种情况,我建议将整个 PFObject 存储在数组中并使用它,这样您就不必一次又一次地调用 Parse,但会占用一些内存。

但是你的代码中也潜伏着危险。不要从单独的线程调用 UI 更新。如果我是正确的,解析块是从一个单独的线程执行的,建议 post UI 在主线程上更新,如下所示:

userListQuery?.findObjectsInBackgroundWithBlock({ (objects, error) -> Void in
    if error == nil && objects != nil{
            for object in objects! {

                if let user = object as? PFUser {
                    if user.objectId != PFUser.currentUser()?.objectId {

                        self.userListData.append(user.objectId!)
                        self.userListUserNames.append(user.username!)

                                dispatch_async(dispatch_get_main_queue(), { () -> Void in
                             self.userCollectionView.reloadData()
                       })

                    }
                }
            }
    }
})

如果有帮助请告诉我。干杯!