My application has 3 UIViewControllers. ViewController #1-->#2 is by "show segue" and from #2-->#3 is by "present modally". #3 has a UITableView. When a cell is clicked, I would like to jump back to #1 and pass data from #3 as well.
I managed to dismiss #3 and go back to #2 by using this function, but this is not what I need. Please advise.
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath){
self.dismiss(animated: true, completion: {
self.funcToCall(id: id)
}
you have to use delegate to pass the value to vc#1 and pop to root view controller.
You should be able to achieve the desired result by defining another segue that triggers on tapping the UITableViewCell. Make sure to assign it an identifier. Add the code below to the UITableView controller
override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
if segue.identifier == "Segue Identifier Here" {
var dest = segue.destinationViewController as! DestViewController
dest.dataObj = dataToBePassed
}
Try pass data through NSNotificationCenter
Register to receive notification in VC1's ViewDidLoad
NotificationCenter.default.addObserver(self, selector: #selector(self.myData(_:)), name: NSNotification.Name(rawValue: "notificationName"), object: nil)
// handle notification
func myData(_ notification: NSNotification) {
let object = notification.object // You data
}
Post this notification in VC3
NotificationCenter.default.post(name: NSNotification.Name(rawValue: "notificationName"), object: nil, userInfo: **yourData**)
// Do popToRootViewContoller
You can use closures, here is an example:
ViewController3
class ViewController3: UIViewController {
var cellSelectedHandler: ((String)->Void)?
.
.
.
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
if cellSelectedHandler != nil {
cellSelectedHandler!("data you want to pass")
}
}
ViewController2
class ViewController2: UIViewController {
//add vc3 as instance variable
let vc3 = ViewController3()
ViewController1
class ViewController: UIViewController {
func presentViewController2() {
let vc2 = ViewController2()
vc2.vc3.cellSelectedHandler = {(dataPassed) in
vc2.vc3.dismiss(animated: true, completion: nil)
_ = vc2.navigationController?.popViewController(animated: true)
//use dataPassed
}
}
Related
MasterVC -> DetailVC -> ChildVC
I have a SplitViewController with a table on the master side and a second table on the detail side. The detail table cells can be selected which brings up a child VC. Currently, I have a protocol defined on the master that lets me know when a cell has been selected. (That way I can update the detail side as needed.) I would like the child vc of the detail view to receive that message as well, but I'm not sure how to set the delegate. What I've tried is to use prepare for segue in the ChildVC to get a reference to the MasterVC like this:
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
if let nav = segue.destination as? UINavigationController {
if let masterVC = nav.topViewController as? MasterVC {
masterVC = self
}
}
}
But sadly, that doesn't seem to work. Am I on the right track, at least? Thank you!
If you want pass some data after you touch cell in your DetailVC, you could use NotificationCenter
class MasterVC: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
NotificationCenter.default.addObserver(self,
selector: #selector(childVCDidSelect(_:)),
name: DetailVC.selectionNotificationName,
object: nil)
}
#objc func childVCDidSelect(_ value: String) {
print("MasterVC recieve \(value) from DetailVC")
}
}
class DetailVC: UIViewController, UITableViewDelegate {
static var selectionNotificationName: NSNotification.Name {
return NSNotification.Name(rawValue: "DetailVCSelectionNotification")
}
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
// value variabl could by any type, maybe you want pass model for selected index path.
let value = "your value"
// When you call NotificationCenter post, you past value to all subscribers,
// who has subcribed NotificationCenter.default.addObserver for DetailVC.selectionNotificationName
NotificationCenter.default.post(name: DetailVC.selectionNotificationName, object: value)
}
}
class ChildVC: UIViewController {
override func viewDidLoad() {
super.viewDidLoad()
NotificationCenter.default.addObserver(self,
selector: #selector(childVCDidSelect(_:)),
name: DetailVC.selectionNotificationName,
object: nil)
}
#objc func childVCDidSelect(_ value: String) {
print("ChildVC recieve \(value) from DetailVC")
}
}
class is reference type, so when you define delegate as class you do not need to do something else and it should work
In this above Picture Two views are there. Top one is a view for showing date(for this view the class name is calender view) . And the bottom one is a tableview.the current view name is Main View When i am clicking on the cell of the tableview then it will go to the next view. When i am dismising the next view i want to pass some data to the calender view.How to achive this.
class MainView: UIViewController, UITableViewDataSource, UITableViewDelegate {
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
let next = self.storyboard?.instantiateViewController(withIdentifier: "NextVC") as! NextVC
self.present(next, animated: true, completion: nil)
}
}
class NextVC: UIViewController {
var sendingData: String?
#IBAction func backAction(_ sender: Any) {
dismiss(animated: true, completion: nil)
}
}
class CalenderView: UIViewController{
var receiveValue: String?
}
Here i want when i am dismissing the nextview the value of sendingData will pass to calender view.
How to do this ? Please help.
I can give you two solutions:
You can create your on custom notification.
First of all on back action you should post your notification with sendingData.
NotificationCenter.default.post(name: NSNotification.Name(rawValue: UpdateCalendarNotification), object: sendingData)
Next, on main view controller with calendar view you should register notification for "UpdateCalendarNotification" name.
func registerNotifications() {
NotificationCenter.default.addObserver(self,
selector: #selector(updateCalendarView(_:)),
name: NSNotification.Name(rawValue: "UpdateCalendarNotification"),
object: nil)
}
And on selector updateCalendarView(_:) you should handle changes for calendar view.
#objc
func updateCalendarView(_ notification: NSNotification) {
if let sendingData = notification.object as? String {
/// Update calendar view
}
}
Second solution is public block for you "next controller".
On next view controller you should add this handler:
var onDismiss: ((String?) -> Void)?
and in backAction method you should pass your data
onDismiss?(sendingData)
In your main view controller you should implement this block like this:
let next = self.storyboard?.instantiateViewController(withIdentifier: "NextVC") as! NextVC
next.onDismiss = { [weak self] (sendingData) in
self?.calendarView.receiveValue = sendingData
}
self.present(next, animated: true, completion: nil)
I hope this will help you)
I have a UITableView inside of a UICollectionViewCell and I'm trying to push a new view from the UITableViewCell inside of the tableView(_:didSelectRowAt:) method. However, I cannot access navigationController. How would I go about navigating to the new view?
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
var item = Item(name: "")
switch indexPath.section {
case 0:
item = array1![indexPath.item]
case 1:
item = array2![indexPath.item]
default:
break
}
let layout = UICollectionViewFlowLayout()
let newView = NewCollectionView(collectionViewLayout: layout)
newView.itemOfInterest = item
// Can't reference navigationController
}
Get parent controller of current View -
//MARK: get parent controller...
extension UIView {
var parentViewController: UIViewController? {
var parentResponder: UIResponder? = self
while parentResponder != nil {
parentResponder = parentResponder!.next
if let viewController = parentResponder as? UIViewController {
return viewController
}
}
return nil
}
}
Usage:-
Get navigationController reference as parentViewController?.navigationController
let viewController = GymLearnMoreViewController(nibName: "GymLearnMoreViewController", bundle: nil) //Your View controller instance
parentViewController?.navigationController?.pushViewController(viewController, animated: true)
Source Apple:-
https://developer.apple.com/documentation/uikit/uiresponder/1621099-next
https://developer.apple.com/documentation/uikit/uiresponder?changes=_9
for this
1 . You need to declare a protocol in the collectionviewcell where the tableview is added
2 . implement the protocol method on the class where the collectionview is added. and handle the navigation on click from here
3 . set the collectionview cell delegate when the collectionviewCellForrow at indexpath method is invoked and call the delegate method when the tableview item is clicked within the collectionview cell
One way is using NotificationCenter which you can write in ParentViewController :
override func viewDidLoad() {
super.viewDidLoad()
NotificationCenter.default.addObserver(self, selector: #selector(methodNavigate(notification:)), name: NSNotification.Name(rawValue: "NavigateToSecondView"), object: nil)
}
#objc func methodNavigate(notification: NSNotification) {
Write your push segue code here.
}
Don't forget to remove observer in viewDidDisappear:
override func viewDidDisappear(_ animated: Bool) {
super.viewDidDisappear(animated)
NotificationCenter.default.removeObserver(self, name: NSNotification.Name(rawValue: "NavigateToSecondView"), object: nil)
}
You can pass data through this too.
Then in didSelectRowAt :
NotificationCenter.default.post(name: NSNotification.Name(rawValue: "NavigateToSecondView"), object: "Send Data Here")
But I will suggest using protocol is much better and elegant way.
You can get lots of tutorials and samples on internet:
https://medium.com/#abhimuralidharan/all-about-protocols-in-swift-11a72d6ea354
https://docs.swift.org/swift-book/LanguageGuide/Protocols.html
Hope it helps. Happy coding.
I have a segue from my MainViewController to my ThemesTableViewController. After performing the segue, I choose a theme on my ThemesTableViewController. When I dismiss my ThemesTableViewController, I want to see the theme selected already applied to my MainViewController. How can I achieve that?
// Here's my Theme
struct Theme {
var name: String!
var backgroundColor: UIColor!
...
}
themes = [red, blue, green]
// I have no idea how I can pass chosenTheme to my MainViewController
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
chosenTheme = themes[indexPath.row]
dismiss(animated: true, completion: nil)
}
protocol ThemeDelegate {
func didSelectTheme(theme: Theme)
}
var delegate: ThemeDelegate?
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
chosenTheme = themes[indexPath.row]
self.delegate?.didSelectTheme(theme: chosenTheme)
}
You can create a protocol and implement it like this and pass the chosenTheme to the previous viewcontroller. In your MainViewController prepare for segue function do it like this.
override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
// Get the new view controller using segue.destinationViewController.
// Pass the selected object to the new view controller.
if let themeVC = segue.destination as? ThemesTableViewController {
themeVC.delegate = self
}
}
Also implement an extension like this in MainViewController.
extension MainViewController: ThemeDelegate {
func didSelectTheme(theme: Theme) {
// do something with the selected theme.
}
}
Passing data with unwind segue :
Passing data with unwind segue
Using NotificationCenter for applying theme after unwind :
Adding dark mode to iOS app
I want to send the text which is in textfield in ViewControllerC to another textfield which is in ViewControllerA
By using delegate am trying to pass the text from ViewControllerC to ViewControllerA.
i cant get the logic what to write here delegate?.userDidEnterInformation() in ViewControllerC
could any one help me regarding this
ViewControllerC
protocol DataEnteredInDestinationDelegate: class {
func userDidEnterInformation(info: String)
}
class DestinationSearchViewController: MirroringViewController {
var delegate: DataEnteredInDestinationDelegate?
func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
let cell: UITableViewCell? = tableView.cellForRow(at: indexPath)
componetsTextField.text = cell?.textLabel?.text
delegate?.userDidEnterInformation()
self.navigationController?.popToRootViewController(animated: true)
}
}
ViewControllerA
class HomeViewController: MirroringViewController, DataEnteredInDestinationDelegate
{
func userDidEnterInformation(info: String){
locationView.destination.text = info
}
}
Firstly you have to always mark delegate as weak e.g.:
weak var delegate: DataEnteredInDestinationDelegate?
and then you need to connect delegate like this:
let vcA = ViewControllerA()
let vcC = ViewControllerC()
vcC.delegate = vcA // Connect delegate
and then your delegate method in ViewControllerC will work after invoking this code:
delegate?.userDidEnterInformation(textString)
Here NotificationCentre can be a good approach instead of delegates. Make Viewcontroller A an observer to receive text information as below.
Write this code in viewDidLoad()
NotificationCenter.default.addObserver(self, selector: #selector(userDidEnterInformation(notification:)), name: NSNotification.Name.init(rawValue: "UserDidEnterInformation"), object: nil)
and write this anywhere in class Viewcontroller A
func userDidEnterInformation(notification: Notification) {
if let textInfo = notification.userInfo?["textInfo"] {
textField.text = textInfo
}
}
In Viewcontroller C post the notification with textInfo by writing below code
NotificationCenter.default.post(name: NSNotification.Name.init(rawValue: "UserDidEnterInformation"), object: nil, userInfo: ["textInfo": textField.text])
delegate?.userDidEnterInformation(cell!.textLabel!.text)
Also, you should set the delegate of ViewControllerC.
viewControllerC.delegate = viewControllerA
Consider the following example:-
let aVCobjA = UIViewController()
let aVCobjB = UIViewController()
let aVCobjC = UIViewController()
var aNavigation = UINavigationController()
func pushVC() {
aNavigation.pushViewController(aVCobjA, animated: true)
aNavigation.pushViewController(aVCobjB, animated: true)
aNavigation.pushViewController(aVCobjC, animated: true)
//Here you will get array of ViewControllers in stack of Navigationcontroller
print(aNavigation.viewControllers)
//To pass data from Viewcontroller C to ViewController A
self.passData()
}
// To pass data access stack of Navigation Controller as navigation controller provides a property viewControllers which gives you access of all view controllers that are pushed.
func passData() {
let aVCObj3 = aNavigation.viewControllers.last
let aVCObj1 = aNavigation.viewControllers[0]
//Now you have access to both view controller pass whatever data you want to pass
}