如何在didSet之后将数据从AppDelegate传递到ViewController?

时间:2015-12-02 07:07:44

标签: swift macos delegates parameter-passing nstableview

我创建一个os X项目并使用主菜单打开文件目录并在tableview上显示内容。

在我的AppDelegate.swift中:

class AppDelegate: NSObject, NSApplicationDelegate {
    var filePath: String = String() {
        didSet {
            // Pass filePath to ViewController
        }
    }

    @IBAction func openFile(sender: AnyObject) {
        let myOpenDialog = NSOpenPanel()
        myOpenDialog.canChooseDirectories = true
        if(myOpenDialog.runModal() != 0) {
           self.filePath = myOpenDialog.URL!.path!
     }
}      

我的ViewController.swift:

class ViewController: NSViewController, NSTableViewDataSource, NSTableViewDelegate {
    @IBOutlet weak var fileTableView: NSTableView!
    private var objects: NSMutableArray! = NSMutableArray()
    var fileDirectory: String = String() {
           didSet {
              showFileList(fileDirectory)
           }
    }
    func showFileList(fileDirectory: String) {
    var contentsInFolder: NSArray?
    var isDir : ObjCBool = false

    // Remove last loaded objects
    self.objects.removeAllObjects()

    if NSFileManager.defaultManager().fileExistsAtPath(fileDirectory, isDirectory:&isDir) {
        if isDir {
            contentsInFolder = try! NSFileManager.defaultManager().contentsOfDirectoryAtPath(fileDirectory)
            for id in contentsInFolder! {
                if ((id as! String).rangeOfString(".MOV") != nil) {
                    self.objects.addObject(id)
                }
            }
        }
    }
        self.fileTableView.reloadData()
    }

    // MARK: - Table View
    func numberOfRowsInTableView(tableView: NSTableView) -> Int
    {
        return self.objects.count
    }
}

我收到错误"致命错误:在展开可选值时意外发现nil"当我在ViewController.swift中运行 self.fileTableView.reloadData()时。知道怎么解决吗?

var vc = ViewController()
var filePath: String = String() {
    didSet {
        vc.fileDirectory = filePath
    }
}

2 个答案:

答案 0 :(得分:0)

showFileList初始化过程中调用ViewController时,视图中fileTableViewUITableView之间的关联尚未设置。因此,nil执行时为self.fileTableView.reloadData()

在致电nil之前,只需检查reloadData

 if(self.fileTableView != nil) self.fileTableView.reloadData()

另外,您可能也应该为self.objects执行此操作。我猜测self.objects总是在self.fileTableView之前初始化,因为之前已经宣布过,但我不知道它是否是Swift规范中记录的行为(如果有人,我会很感激知道这可以筹码。)

答案 1 :(得分:0)

var vc = ViewController() 

它将在AppDelegate中创建另一个实例,并且显示UI实例不会通过这种方式获取值传递。 我尝试使用 NSNotification 委托方法,我的问题就解决了。

<强> AppDelegate.swift

var filePath: String = String() {
    didSet {
        NSNotificationCenter.defaultCenter().postNotificationName("refreshView", object: nil)
    }
}

<强> ViewController.swift

var fileDirectory: String = String()
let delegate = NSApplication.sharedApplication().delegate as! AppDelegate

override func viewDidLoad() {
    super.viewDidLoad()
    NSNotificationCenter.defaultCenter().addObserver(self, selector: "refreshView:", name: "refreshView", object: nil)
}

func refreshView(notification: NSNotification) {
    fileDirectory = delegate.filePath
}

通过这样做,我可以从我的AppDelegate.swift中获取 filePath