it-swarm.com.ru

Передать данные обратно на предыдущий viewcontroller

Я пытаюсь передать данные ВЕРНУТЬСЯ к предыдущему viewController. 

Кто-нибудь знает, как передать данные обратно из ViewController B в ViewController A? Поэтому я хочу, чтобы строка шла из BIDAddTypeOfDealViewController в BIDDCCreateViewController. Пользователь редактирует viewController B, и я хочу, чтобы отредактированные данные вернулись в ViewController A, где я затем использую его. 

Я использую раздел «передача данных назад» этот ответ . Чем отличается мой: в пунктах 3 и 6 просто упоминается, когда представления всплывают, поэтому я поместил этот код в viewWillDisappear. Я думаю, что это правильно? Также в пункте 6 я не инициализировал nib, поскольку он старый. Я использую раскадровки. И я не добавил эту последнюю строку, так как не верю, что мне придется ее толкать. Нажатие кнопки на моей раскадровке уже выводит меня вперед. 

Я думаю, что проблема может возникнуть в BIDDCCreateViewController, у меня есть метод, но я не могу его запустить. Чтобы запустить метод, он должен пойти [self method]. Я не могу этого сделать. Ну, это только то, что я предполагаю. 

Он компилируется и работает нормально, но ничего не регистрируется, поэтому я не знаю, работает ли он. 

ОБНОВЛЕНИЕ: я не могу получить метод sendDataToA для выполнения. 

#import <UIKit/UIKit.h>
#import "BIDAddTypeOfDealViewController.h"

 @interface BIDDCCreateViewController : UIViewController
 @property (strong, nonatomic) NSString *placeId;
- (IBAction)gotoBViewController:(id)sender;
@end


#import "BIDDCCreateViewController.h"
#import "BIDAddTypeOfDealViewController.h"

@implementation BIDDCCreateViewController

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    NSLog(@"SUCCESSFULLY PASSED PLACE ID: %@", self.placeId);
}

-(void)sendDataToA:(NSString *)myStringData
{

    NSLog(@"Inside sendDataToA");
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Your string Data Showing" message:myStringData delegate:self cancelButtonTitle:@"Ok " otherButtonTitles:nil];
    [alert show];
}

- (IBAction)gotoBViewController:(id)sender {
    NSLog(@"pressed");
    BIDAddTypeOfDealViewController *bidAddType = [[BIDAddTypeOfDealViewController alloc]init];
    bidAddType.delegate = self;

}
@end


@protocol senddataProtocol <NSObject>
-(void)sendDataToA:(NSString *)myStringData;
@end

#import <UIKit/UIKit.h>
@interface BIDAddTypeOfDealViewController : UIViewController <UITextFieldDelegate>//Using this delegate for data a user inputs
@property(nonatomic,assign)id delegate;
//other textfield outlets not relevant
- (IBAction)chooseDiscountDeal:(id)sender;
@end

#import "BIDAddTypeOfDealViewController.h"

@interface BIDAddTypeOfDealViewController ()

@end

@implementation BIDAddTypeOfDealViewController
@synthesize delegate;

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
        // Custom initialization
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
}

-(void)viewWillDisappear:(BOOL)animated
{
    [delegate sendDataToA:@"Apple"];
}
@end
48
Anthony

Вы можете использовать делегата. Итак, в вашем ViewController B вам нужно создать протокол, который отправляет данные обратно вашему ViewController A. Ваш ViewController A станет делегатом ViewController B. 

Если вы новичок в Задаче C, пожалуйста, посмотрите на Что такое делегат .

Создайте протокол в ViewControllerB.h:

#import <UIKit/UIKit.h>

@protocol senddataProtocol <NSObject>

-(void)sendDataToA:(NSArray *)array; //I am thinking my data is NSArray, you can use another object for store your information. 

@end

@interface ViewControllerB : UIViewController

@property(nonatomic,assign)id delegate;

ViewControllerB.m

@synthesize delegate;
-(void)viewWillDisappear:(BOOL)animated
{
     [delegate sendDataToA:yourdata];

}

в вашем ViewControllerA: когда вы идете в ViewControllerB 

ViewControllerA *acontollerobject=[[ViewControllerA alloc] initWithNibName:@"ViewControllerA" bundle:nil];
acontollerobject.delegate=self; // protocol listener
[self.navigationController pushViewController:acontollerobject animated:YES];

и определите свою функцию: 

-(void)sendDataToA:(NSArray *)array
{
   // data will come here inside of ViewControllerA
}

Отредактировано:

Вы можете увидеть этот пример: Как вы можете передать данные обратно в предыдущий viewcontroller: Ссылка на учебник

88
Erhan Demirci

Короче и проще метода, чем протокол/делегат, заключается в создании замыкания:

Для отправки строки обратно в моем случае . В ViewControllerA:

override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
    if let viewControllerB = segue.destination as? ViewControllerB {
        viewControllerB.callback = { message in
            //Do what you want in here!
        }
    }
}

В ViewControllerB:

var callback : ((String) -> Void)?

@IBAction func done(sender: AnyObject) {
    callback?("Hi")
    self.dismiss(animated: true, completion: nil)
}
51
Heinrisch

Swift: отправка данных обратно с использованием шаблона делегата

Мой полный ответ, который касается передачи данных в обе стороны: здесь . Мой ответ, объясняющий образец делегата, здесь .

Чтобы передать данные от второго контроллера представления к первому контроллеру представления, вы используете протокол и делегат. Это видео очень ясно показывает, как проходит этот процесс:

Ниже приведен пример, основанный на видео (с некоторыми изменениями).

enter image description here

Создайте макет раскадровки в Интерфейсном Разработчике. Опять же, чтобы сделать переход, вы просто Control перетащите с кнопки на контроллер второго вида. Установите для идентификатора segue значение showSecondViewController. Кроме того, не забудьте подключить розетки и действия, используя имена в следующем коде.

Контроллер первого вида

Код для первого контроллера представления

import UIKit

class FirstViewController: UIViewController, DataEnteredDelegate {

    @IBOutlet weak var label: UILabel!

    override func prepareForSegue(segue: UIStoryboardSegue, sender: AnyObject?) {
        if segue.identifier == "showSecondViewController" {
            let secondViewController = segue.destinationViewController as! SecondViewController
            secondViewController.delegate = self
        }
    }

    func userDidEnterInformation(info: String) {
        label.text = info
    }
}

Обратите внимание на использование нашего собственного протокола DataEnteredDelegate.

Контроллер и протокол второго просмотра

Код для второго контроллера вида

import UIKit

// protocol used for sending data back
protocol DataEnteredDelegate: class {
    func userDidEnterInformation(info: String)
}

class SecondViewController: UIViewController {

    // making this a weak variable so that it won't create a strong reference cycle
    weak var delegate: DataEnteredDelegate? = nil

    @IBOutlet weak var textField: UITextField!

    @IBAction func sendTextBackButton(sender: UIButton) {

        // call this method on whichever class implements our delegate protocol
        delegate?.userDidEnterInformation(textField.text!)

        // go back to the previous view controller
        self.navigationController?.popViewControllerAnimated(true)
    }
}

Обратите внимание, что protocol находится вне класса View Controller.

Вот и все. Запустив приложение сейчас, вы сможете отправить данные обратно со второго контроллера представления на первый.

40
Suragch

Правка: Используйте решение @ Erhan выше. Не этот. Это не хорошее решение.

Это поможет. Напишите это в вашем ViewControllerB. 

    // Get array of current navigation stack
    NSArray *arrayViewControllers = [self.navigationController viewControllers];

    // Get previous viewController object from it
    YOUR_VIEW_CONTROLLER_NAME *objViewController = (YOUR_VIEW_CONTROLLER_NAME *)[arrayViewControllers objectAtIndex:arrayViewControllers.count-2];

    // For safety this check is needed. whether it the class that you want or not.
    if ([objViewController isKindOfClass:[YOUR_VIEW_CONTROLLER_NAME class]])
    {
        // Access properties of YOUR_VIEW_CONTROLLER_NAME here
        objViewController.yourProperty = YOUR_VALUE;
    }
2
Akshit Zaveri

Как ответил Эрхан Демирчи, вы можете использовать делегатов. Делегаты полезны, когда вы хотите передать данные одному контроллеру представления.

NSNotificationCenter - еще один удобный способ передачи данных между viewcontrollers/объектами. Это очень полезно при трансляции данных в приложении.

читать документацию здесь .

1
Abdullah Umer

Пользовательский делегат - лучший вариант для перемещения данных, но вы также можете попробовать это.

Вы можете использовать NSUserDefaults для перемещения данных куда угодно.

Swift 3 Code

UserDefaults.standard.set(<Value>, forKey: <Key>) 
// To set data

UserDefaults.standard.object(forKey: <Key>) 
// To get data

Вы также можете использовать NSNotification для перемещения данных.

NotificationCenter.default.post(name: Notification.Name(rawValue: "refresh"), object: myDict) 

NotificationCenter.default.addObserver(self, selector: #selector(refreshList(_:)), name: NSNotification.Name(rawValue: "refresh"), object: nil)
1
Saurabh Sharma

Есть протокол, есть закрытие. При закрытии мы должны избегать утечек памяти, используя слабое «я» (или «неизвестное»). С протоколом будет один для viewController, который вы хотите «контролировать», в конечном итоге с десятками делегатов для реализации. Здесь у меня есть еще одно простое решение в Swift:

Внутри нового файла или существующего (например: UIViewController+Extensions.Swift) создайте этот протокол:

protocol ViewControllerBackDelegate: class {
    func back(from viewController: UIViewController)
}

Внутри viewController LEVEL-2, где требуется обратный вызов при нажатии Back из:

class LevelTwoViewController: UIViewController {
    // making this a weak variable so that it won't create a strong reference cycle
    weak var delegate: ViewControllerBackDelegate? = nil

    override func willMove(toParentViewController parent: UIViewController?) {
        super.willMove(toParentViewController: parent)
        if (parent == nil) {
            delegate?.back(from: self)
        }
    }
}

Поскольку delegate является необязательным, вы можете добавить этот код в базовый класс ваших контроллеров представления. Я бы добавил, где это должно быть.

Предположим, что в вашем ViewController LEVEL-1 вы вызываете LEVEL-2 через Seoryboard:

class LevelOneViewController: UIViewController {
    override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
        if (segue.identifier == "Go to Level 2") {
            if let vc = segue.destination as? LevelTwoViewController {
                vc.selectedItems = self.selectedItems // passing data-in
                vc.delegate = self
            }
        }
        // repeat `if` for another sub-level view controller
    }
}

extension LevelOneViewController: ViewControllerBackDelegate {    
    func back(from viewController: UIViewController) {
        if let vc = viewController as? LevelTwoViewController {
            self.selectedItems = vc.selectedItems
            // call update if necessary
        }
        // repeat `if` for another sub-level view controller
    }
}
  • требуется только один протокол.
  • только одно расширение на viewController первого уровня.
  • нет изменений в подуровне viewController, если нужно вернуть больше/меньше данных
  • обрабатывать данные так же, как данные в prepare(for:sender:)
0
John Pang