为什么在创建实例时我的 IBOutlet 连接变量没有实例化?
Why are my IBOutlet-connected vars not instantiated when I create an instance?
我设置了一些全局变量等于 class 实例化后的参数。 class 定义如下:
import Foundation
import UIKit
class LogIn : UIViewController{
var test:Int = 1
@IBOutlet var login_icon: UIImageView!
@IBOutlet var username_field: UITextField!
@IBOutlet var password_field: UITextField!
@IBOutlet var login_button: UIButton!
@IBOutlet var createaccount_button: UIButton!
}
全局变量是这样实例化的(在 viewDidAppear 方法中):
func showLoginView(){
self.startup_animation.image = self.seq_6
let login_view = LogIn()
self.login_icon = login_view.login_icon.image //Doesnt work
self.username_field = login_view.username_field //Doesnt work
self.password_field = login_view.password_field //Doesnt work
self.login_button = login_view.login_button //Doesnt work
self.createaccount_button = login_view.createaccount_button //Doesnt work
self.int_testval = login_view.test //Works
}
实例化后所有 IBOutlets 连接的变量都是 nil 值,但 var test 实例化很好。这是为什么?
谢谢!
其他上下文
showLoginView 函数与 viewDidAppear 函数一起调用。整个 class 看起来是这样的:
class StartupViewController: UIViewController {
var images: [UIImage]!
var animatedImage: UIImage!
@IBOutlet var startup_animation: UIImageView!
@IBOutlet var startup_text: UIImageView!
var login_icon: UIImage!
var username_field: UITextField!
var password_field: UITextField!
var login_button: UIButton!
var createaccount_button: UIButton!
var seq_1 : UIImage!
var seq_2 : UIImage!
var seq_3 : UIImage!
var seq_4 : UIImage!
var seq_5 : UIImage!
var seq_6 : UIImage!
var seq_7 : UIImage!
var seq_8 : UIImage!
var rec_1 : UIImage!
var rec_2 : UIImage!
var rec_3 : UIImage!
var rec_4 : UIImage!
var rec_5 : UIImage!
var rec_6 : UIImage!
override func viewDidLoad() {
super.viewDidLoad()
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
seq_1 = UIImage(named: "Eceipt_seq1")
seq_2 = UIImage(named: "Eceipt_seq2")
seq_3 = UIImage(named: "Eceipt_seq3")
seq_4 = UIImage(named: "Eceipt_seq4")
seq_5 = UIImage(named: "Eceipt_seq5")
seq_6 = UIImage(named: "Eceipt_seq6")
seq_7 = UIImage(named: "Eceipt_seq7")
seq_8 = UIImage(named: "Eceipt_seq8")
images = [seq_1 ,seq_2, seq_3, seq_4, seq_5, seq_6, seq_7, seq_8]
animatedImage = UIImage.animatedImage(with: images, duration: 0.8)
startup_animation.image = animatedImage
let transition_text = UIImage(named: "Eceipt_logo")
UIView.transition(with: self.startup_text, duration: 3.0, options: .transitionCrossDissolve,
animations: {
self.startup_text.image = transition_text
},
completion: { (finished: Bool) -> () in
self.showLoginView()
})
}
func showLoginView(){
self.startup_animation.image = self.seq_6
let login_view = LogIn()
self.login_icon = login_view.login_icon.image
self.username_field = login_view.username_field
self.password_field = login_view.password_field
self.login_button = login_view.login_button
self.createaccount_button = login_view.createaccount_button
}
我认为您误解了 @IBOutlet
的用法。
@IBOutlet
只是表示它是 xib 或故事板文件中视图或其他对象的出口。
你的观点是 nil
不是因为它们是出口,而是因为它们是隐式展开的可选值,如果你不给它一个值,它就是 nil
。
test
不是零,因为 1) 它不是可选类型,并且 2) 它的值为 1
.
我建议您将 @IBOutlet
放在 UIViewController
或 UIView
子 class 中,并将 storyboard/xib 文件中的适当视图连接到它.这样,它们将在 viewDidLoad
方法或 awakeFromNib
方法被调用时被初始化。
编辑:
您似乎已经控制+拖动视图到 LogIn
class。在这种情况下,请检查出口左侧的圆圈是否已填满。如果是,则表示它们已正确连接。您只需要等到 awakeFromNib
被调用。如果不是,请尝试重新连接它们。或者,只需将所有插座直接连接到视图控制器,这样可以省去很多麻烦。
编辑 2:
现在我明白了你想要做什么,下面是你应该做的。您似乎只想呈现 LogIn
VC。因此,与其将每个视图分配给 self
,不如使用 present
方法正确呈现它。
let vc = UIStoryboard.main!.instantiateViewController(withIdentifier: "Login")
present(vc, animated: true, completion: nil)
记得在情节提要中为您的登录 VC 提供情节提要 ID "Login"。
我认为问题在于您通过 let login_view = LogIn()
实例化了 LogIn
视图控制器。当您使用空初始化程序实例化视图控制器时,每个 IBOutlet
都将为 nil,因为视图控制器不是故事板(或 xib)中的 "taken"。如果你想使用 iboutles,你必须从故事板或 xib 中实例化视图控制器;在您的情况下(使用情节提要),您应该通过以下方式实例化它:
let storyboard = UIStoryboard(name: "Main", bundle: nil) // name is the filename of the storyboard that contains the vc
let vc = storyboard.instantiateViewController(withIdentifier: "LogIn") as! LogIn // you can cast to have the exact view controller type
present(vc, animated: true, completion: nil)
确保故事板中的视图控制器具有标识符 "LogIn"。
我设置了一些全局变量等于 class 实例化后的参数。 class 定义如下:
import Foundation
import UIKit
class LogIn : UIViewController{
var test:Int = 1
@IBOutlet var login_icon: UIImageView!
@IBOutlet var username_field: UITextField!
@IBOutlet var password_field: UITextField!
@IBOutlet var login_button: UIButton!
@IBOutlet var createaccount_button: UIButton!
}
全局变量是这样实例化的(在 viewDidAppear 方法中):
func showLoginView(){
self.startup_animation.image = self.seq_6
let login_view = LogIn()
self.login_icon = login_view.login_icon.image //Doesnt work
self.username_field = login_view.username_field //Doesnt work
self.password_field = login_view.password_field //Doesnt work
self.login_button = login_view.login_button //Doesnt work
self.createaccount_button = login_view.createaccount_button //Doesnt work
self.int_testval = login_view.test //Works
}
实例化后所有 IBOutlets 连接的变量都是 nil 值,但 var test 实例化很好。这是为什么?
谢谢!
其他上下文 showLoginView 函数与 viewDidAppear 函数一起调用。整个 class 看起来是这样的:
class StartupViewController: UIViewController {
var images: [UIImage]!
var animatedImage: UIImage!
@IBOutlet var startup_animation: UIImageView!
@IBOutlet var startup_text: UIImageView!
var login_icon: UIImage!
var username_field: UITextField!
var password_field: UITextField!
var login_button: UIButton!
var createaccount_button: UIButton!
var seq_1 : UIImage!
var seq_2 : UIImage!
var seq_3 : UIImage!
var seq_4 : UIImage!
var seq_5 : UIImage!
var seq_6 : UIImage!
var seq_7 : UIImage!
var seq_8 : UIImage!
var rec_1 : UIImage!
var rec_2 : UIImage!
var rec_3 : UIImage!
var rec_4 : UIImage!
var rec_5 : UIImage!
var rec_6 : UIImage!
override func viewDidLoad() {
super.viewDidLoad()
}
override func viewDidAppear(_ animated: Bool) {
super.viewDidAppear(animated)
seq_1 = UIImage(named: "Eceipt_seq1")
seq_2 = UIImage(named: "Eceipt_seq2")
seq_3 = UIImage(named: "Eceipt_seq3")
seq_4 = UIImage(named: "Eceipt_seq4")
seq_5 = UIImage(named: "Eceipt_seq5")
seq_6 = UIImage(named: "Eceipt_seq6")
seq_7 = UIImage(named: "Eceipt_seq7")
seq_8 = UIImage(named: "Eceipt_seq8")
images = [seq_1 ,seq_2, seq_3, seq_4, seq_5, seq_6, seq_7, seq_8]
animatedImage = UIImage.animatedImage(with: images, duration: 0.8)
startup_animation.image = animatedImage
let transition_text = UIImage(named: "Eceipt_logo")
UIView.transition(with: self.startup_text, duration: 3.0, options: .transitionCrossDissolve,
animations: {
self.startup_text.image = transition_text
},
completion: { (finished: Bool) -> () in
self.showLoginView()
})
}
func showLoginView(){
self.startup_animation.image = self.seq_6
let login_view = LogIn()
self.login_icon = login_view.login_icon.image
self.username_field = login_view.username_field
self.password_field = login_view.password_field
self.login_button = login_view.login_button
self.createaccount_button = login_view.createaccount_button
}
我认为您误解了 @IBOutlet
的用法。
@IBOutlet
只是表示它是 xib 或故事板文件中视图或其他对象的出口。
你的观点是 nil
不是因为它们是出口,而是因为它们是隐式展开的可选值,如果你不给它一个值,它就是 nil
。
test
不是零,因为 1) 它不是可选类型,并且 2) 它的值为 1
.
我建议您将 @IBOutlet
放在 UIViewController
或 UIView
子 class 中,并将 storyboard/xib 文件中的适当视图连接到它.这样,它们将在 viewDidLoad
方法或 awakeFromNib
方法被调用时被初始化。
编辑:
您似乎已经控制+拖动视图到 LogIn
class。在这种情况下,请检查出口左侧的圆圈是否已填满。如果是,则表示它们已正确连接。您只需要等到 awakeFromNib
被调用。如果不是,请尝试重新连接它们。或者,只需将所有插座直接连接到视图控制器,这样可以省去很多麻烦。
编辑 2:
现在我明白了你想要做什么,下面是你应该做的。您似乎只想呈现 LogIn
VC。因此,与其将每个视图分配给 self
,不如使用 present
方法正确呈现它。
let vc = UIStoryboard.main!.instantiateViewController(withIdentifier: "Login")
present(vc, animated: true, completion: nil)
记得在情节提要中为您的登录 VC 提供情节提要 ID "Login"。
我认为问题在于您通过 let login_view = LogIn()
实例化了 LogIn
视图控制器。当您使用空初始化程序实例化视图控制器时,每个 IBOutlet
都将为 nil,因为视图控制器不是故事板(或 xib)中的 "taken"。如果你想使用 iboutles,你必须从故事板或 xib 中实例化视图控制器;在您的情况下(使用情节提要),您应该通过以下方式实例化它:
let storyboard = UIStoryboard(name: "Main", bundle: nil) // name is the filename of the storyboard that contains the vc
let vc = storyboard.instantiateViewController(withIdentifier: "LogIn") as! LogIn // you can cast to have the exact view controller type
present(vc, animated: true, completion: nil)
确保故事板中的视图控制器具有标识符 "LogIn"。