斯威夫特3 - 代表始终返回零值
问题描述:
我一直在使用按键设计定制的TabBar。我有3个标签, 第一个标签有消息图标,第二个有配置文件图标和第三有照片图标。对于第三个标签按钮,我使用uiCollectionView()
,我需要设置图像。
对于第三个选项卡的ViewController,在更改第一个选项卡按钮的title
之前,有一个条件需要检查。如果messages
JSON数组不为空,则在第一个选项卡按钮上设置“新消息”标题,否则消息图标不会更改。
有一个ParentTabViewController具有这3个选项卡,我已经使用了uiView,其中根据按下的选项卡按钮更改内容。我试图通过使用委托来访问ParentTabViewController中的第三个选项卡的值,但委托始终为零。我不喜欢这样的:
class ParentTabViewController: UIViewController,MessageDelegateProtocol{
@IBOutlet weak var contentView: UIView!
@IBOutlet var tabBarButtons : [UIButton]!
@IBOutlet weak var firstTabButton: UIButton!
var MessageVC : UIViewController!
var ProfileVC : UIViewController!
var PhotosVC : UIViewController!
var viewControllers : [UIViewController]!
var message : String!
var selectedIndex:Int = 0
var photoVC = PhotosVC()
override func viewDidLoad() {
super.viewDidLoad()
photoVC.newMessageDelegate = self
let storyBoard = UIStoryboard(name:"Main", bundle:nil)
MessageVC = storyBoard.instantiateViewController(withIdentifier: "messagevc")
ProfileVC = storyBoard.instantiateViewController(withIdentifier: "profile")
PhotosVC = storyBoard.instantiateViewController(withIdentifier: "photos")
viewControllers = [MessageVC, ProfileVC, PhotosVC]
tabBarButtons[selectedIndex].isSelected = true
didPressTabs(tabBarButtons[selectedIndex])
}
@IBAction func didPressTabs(_ sender: UIButton)
{
let previousIndex = selectedIndex
selectedIndex = sender.tag
tabBarButtons[previousIndex].isSelected = false
let previousVC = viewControllers[previousIndex]
previousVC.willMove(toParentViewController: nil)
previousVC.removeFromParentViewController()
previousVC.view.removeFromSuperview()
sender.isSelected = true
let presentVC = viewControllers[selectedIndex]
addChildViewController(presentVC)
presentVC.view.frame = contentView.bounds
contentView.addSubview(presentVC.view)
presentVC.didMove(toParentViewController: self)
if selectedIndex == 2{ // this is what I thought of doing.Correct me if wrong.
// check the condition
// if messagesArray != nil
// set the first tab title "new message"
}
else{
// do not change the button image
}
}
func sendMessage(message : String)
{
self.message = message
print("message........", self.message, "\n\n")
}
}
这里是视图控制器的第三个标签:
import UIKit
protocol MessageDelegateProtocol:class {
func sendMessage(message : String)
}
class PhotosVC: UIViewController,UICollectionViewDataSource, UICollectionViewDelegate{
@IBOutlet weak var collectionView: UICollectionView!
var userMessageArray = [UserMessageClass]() // array of model class
var newMessage : String!
weak var newMessageDelegate : MessageDelegateProtocol?
override func viewDidLoad() {
super.viewDidLoad()
collectionView.delegate = self
collectionView.dataSource = self
loadData() // function to get json reponse
}
// implement collectionView delegate and dataSource methods
func getData(newMsg : UserMessageClass) //func to get values from model class
{
newMessage = newMsg.messageString // here I get the "new message" String
newMessageDelegate?.sendMessage(message: newMessage)
} enter code here
func loadData()
{
// get json response. And pass the payload to UserMessageClass using that class's array
userMessageArray.append(UserMessageClass(dict : jsonData))
var msgData = UserMessageClass(dict: jsonData)
getData(alarm: msgData)
}
}
我试着搜索了很多关于在其他VC访问选项卡按钮,但没有发现任何附近这样的方法。此外,我无法弄清楚为什么代表总是零。建议或帮助将不胜感激。很多感谢:)
答
问题是以下行。
let firstTab = storyBoard.instantiateViewController(withIdentifier: "parentVC") as! ParentViewController
您可能期待它为您提供最初设置的ParentViewController的实例。但是,它会给你一个新启动的ParentViewController的实例,这不是你想要的。
为了解决这个问题,您可以使用定义的委托或完成块,它将在您的ParentViewController类中定义。
更新: 尝试增加PhotosVC.newMessageDelegate = self
线下
PhotosVC = storyBoard.instantiateViewController(withIdentifier: "photos")
也改变var PhotosVC : UIViewController!
到var photosVC: PhotosVC!
这应该现在的工作。
嗨,你可以告诉我怎么做代表。由于ParentViewController首先被加载,并且'MessagesVC()'是第一个选项卡。当我点击第三个标签时,检查'messages' json数组是否不为零,将标题设置为第一个标签。我想检查ParentVC中的条件,'如果selectedIndex == 2 {//执行检查}',然后在这里设置标题。但我无法设法从ParentVc中的MessageVC中获取值。总是得到零。我尝试委托,但失败 –
你可以上传你的项目在github上? – Rishab
嗨,我已经更新了这个问题。请检查并建议有什么问题 –