Назначить xib для UIView в Swift
в объекте c это может быть сделано в методе init методом
-(id)init{
self = [[[NSBundle mainBundle] loadNibNamed:@"ViewBtnWishList" owner:0 options:nil] objectAtIndex:0];
return self;
}
но когда я делаю это быстрым
init(frame: CGRect) {
self = NSBundle.mainBundle().loadNibNamed("ViewDetailMenu", owner: 0, options: nil)[0] as? UIView
}
не может присваиваться self в ошибке метода.
теперь мой подход заключается в создании представления и добавлении представления, загруженного из nib в него.
у кого есть идея?
Ответы
Ответ 1
для обновления Swift 4.x do
extension UIView {
class func loadFromNibNamed(nibNamed: String, bundle : Bundle? = nil) -> UIView? {
return UINib(
nibName: nibNamed,
bundle: bundle
).instantiate(withOwner: nil, options: nil)[0] as? UIView
}
}
для Swift 3.x предыдущий ответ
Вы можете создать расширение на UIView:
extension UIView {
class func loadFromNibNamed(nibNamed: String, bundle : NSBundle? = nil) -> UIView? {
return UINib(
nibName: nibNamed,
bundle: bundle
).instantiateWithOwner(nil, options: nil)[0] as? UIView
}
}
Примечание. Использование UINib происходит быстрее, потому что оно кэширует вас.
Тогда вы можете просто сделать:
ViewDetailItem.loadFromNibNamed("ViewBtnWishList")
И вы сможете повторно использовать этот метод для любого вида.
Ответ 2
Протестировано в Xcode 7 beta 4, Swift 2.0.
Следующий код назначит xib для UIView
.
Вы можете использовать этот пользовательский xib-просмотр в раскадровке и также получить доступ к объекту IBOutlet
.
import UIKit
@IBDesignable class SimpleCustomView:UIView
{
var view:UIView!;
@IBOutlet weak var lblTitle: UILabel!
@IBInspectable var lblTitleText : String?
{
get{
return lblTitle.text;
}
set(lblTitleText)
{
lblTitle.text = lblTitleText!;
}
}
override init(frame: CGRect) {
super.init(frame: frame)
loadViewFromNib ()
}
required init?(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
loadViewFromNib ()
}
func loadViewFromNib() {
let bundle = NSBundle(forClass: self.dynamicType)
let nib = UINib(nibName: "SimpleCustomView", bundle: bundle)
let view = nib.instantiateWithOwner(self, options: nil)[0] as! UIView
view.frame = bounds
view.autoresizingMask = [.FlexibleWidth, .FlexibleHeight]
self.addSubview(view);
}
}
Доступ к пользовательскому просмотру программно
self.customView = SimpleCustomView(frame: CGRectMake(100, 100, 200, 200))
self.view.addSubview(self.customView!);
Исходный код - https://github.com/karthikprabhuA/CustomXIBSwift
Ответ 3
Это сработало для меня.
override func awakeAfterUsingCoder(aDecoder: NSCoder) -> AnyObject? {
if self.subviews.count == 0 {
return loadNib()
}
return self
}
private func loadNib() -> YourCustomView {
return NSBundle.mainBundle().loadNibNamed("YourCustomViewNibName", owner: nil, options: nil)[0] as YourCustomView
}
Ответ 4
который может быть для вас решением:
Swift 3.x
class func instanceFromNib() -> UIView {
return UINib(nibName: "<<NibFileName>>", bundle: nil).instantiate(withOwner: nil, options: nil)[0] as! UIView
}
Swift 2.x
class func instanceFromNib() -> UIView {
return UINib(nibName: "<<NibFileName>>", bundle: nil).instantiateWithOwner(nil, options: nil)[0] as UIView
}
Ответ 5
Я думаю, что это облегчение, но и самый чистый способ присвоить xib UIView. Xcode 7.3 и swift 2.0.
import UIKit
//Create CustomView class
class CustomView: UIView {
class func instanceFromNib() -> UIView {
return UINib(nibName: "CustomView", bundle: nil).instantiateWithOwner(nil, options: nil)[0] as! UIView
}
}
//Use it
let customView = CustomView.instanceFromNib() as! CustomView
Ответ 6
вместо добавления расширения в UIView, вы можете определить протокол и добавить реализацию в расширение протокола. Затем вы можете объявить, что UIView соответствует протоколу.
Это позволяет использовать тип возврата Self
вместо UIView
. Таким образом, вызывающему не нужно бросать в класс.
Разъяснение здесь:
fooobar.com/questions/76579/...
import UIKit
protocol UIViewLoading {}
extension UIView : UIViewLoading {}
extension UIViewLoading where Self : UIView {
// note that this method returns an instance of type `Self`, rather than UIView
static func loadFromNib() -> Self {
let nibName = "\(self)".characters.split{$0 == "."}.map(String.init).last!
let nib = UINib(nibName: nibName, bundle: nil)
return nib.instantiateWithOwner(self, options: nil).first as! Self
}
}
Ответ 7
Просто добавлено расширение UINib
для загрузки представления из xib и вставки в представление контейнера с использованием ограничений, с использованием дженериков и сильного наименования (без использования строк, если у вас есть одно и то же имя файла для xib и реализация):
extension UINib {
static func instantiateViewAndEmbedWithConstraints <T: UIView> (viewType viewType: T.Type, embedInto containerView: UIView) -> T {
let view = UINib(nibName: String(viewType), bundle: nil).instantiateWithOwner(nil, options: nil).first as! T
containerView.addSubview(view)
view.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint(item: view, attribute: .Leading, relatedBy: .Equal, toItem: containerView, attribute: .Leading, multiplier: 1, constant: 0).active = true
NSLayoutConstraint(item: view, attribute: .Trailing, relatedBy: .Equal, toItem: containerView, attribute: .Trailing, multiplier: 1, constant: 0).active = true
NSLayoutConstraint(item: view, attribute: .Top, relatedBy: .Equal, toItem: containerView, attribute: .Top, multiplier: 1, constant: 0).active = true
NSLayoutConstraint(item: view, attribute: .Bottom, relatedBy: .Equal, toItem: containerView, attribute: .Bottom, multiplier: 1, constant: 0).active = true
return view
}
}
Использование:
...outlets...
@IBOutlet var containerView: UIView!
var customView: CustomView!
...viewDidLoad...
customView = UINib.instantiateViewAndEmbedWithConstraints(viewType: CustomView.self, embedInto: containerView)
Ответ 8
Настоящий Swift подход - это использование протоколов и расширений протоколов.
Я использую его следующим образом: Чтобы начать, я создаю протокол
protocol XibInitializable {
static var name: String { get }
static var bundle: Bundle? { get }
static func fromXib() -> Self
}
то я делаю по умолчанию реализацию этого протокола, использующего расширение протокола
extension XibInitializable where Self : UIView {
static var name: String {
return String(describing: Self.self)
}
static var bundle: Bundle? {
return nil
}
static func fromXib() -> Self {
return UINib(nibName: name, bundle: bundle).instantiate(withOwner: nil, options: nil)[0] as! Self
}
}
реализация нашего протокола завершена
Чтобы этот протокол работал, вам нужно имя нашего xib файла, и класс был таким же. Например, например
![введите описание изображения здесь]()
наконец, добавьте протокол и сделайте свой класс " final", как здесь.
![введите описание изображения здесь]()
Что он
и используйте
![введите описание изображения здесь]()