Я пытаюсь реализовать простой подкласс UIButton, который является IBDesignable. Мне нужна возможность устанавливать цвет для каждого состояния элемента управления из Interface Builder. Я знаю, что это возможно с ключевым словом IBInspectable. У меня проблемы с падением IB при использовании KVO на государственной собственности. Отладчик IBDesignable аварийно завершает работу при деинициализации. Кто-нибудь знает, как я могу работать вместе с KVO и IBDesignable?

@IBDesignable
class UIButtonActionButton: UIButton {

    @IBInspectable var defaultColour: UIColor = UIColor.blueColor() {
        didSet {
            self.setNeedsDisplay()
        } 
    }

    @IBInspectable var selectedColour: UIColor = UIColor.blueColor()

    @IBInspectable var disabledColour: UIColor = UIColor.grayColor()

    required init(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        self._setup()
    }

    override init(frame: CGRect) {
        super.init(frame: frame)
        self._setup()
    }


    private func _setup(){
        self.addObserver(self, forKeyPath: "state", options: NSKeyValueObservingOptions.New, context: nil)
        self.layer.cornerRadius = 5.0
        self.layer.masksToBounds = true
    }

    override func observeValueForKeyPath(keyPath: String, ofObject object: AnyObject, change: [NSObject : AnyObject], context: UnsafeMutablePointer<Void>) {
        self.setNeedsDisplay()
    }

    override func drawRect(rect: CGRect) {
        super.drawRect(rect)
        let context = UIGraphicsGetCurrentContext()

        if self.highlighted {
            CGContextSetFillColorWithColor(context, selectedColour.CGColor)
            CGContextFillRect(context, self.bounds)
        } else if self.state == UIControlState.Disabled {
            CGContextSetFillColorWithColor(context, disabledColour.CGColor)
            CGContextFillRect(context, self.bounds)
        } else {
            CGContextSetFillColorWithColor(context, defaultColour.CGColor)
            CGContextFillRect(context, self.bounds)
        }
    }

    deinit {
        self.removeObserver(self, forKeyPath: "state", context: nil)
    }

}
9
Fergal Rooney 28 Окт 2014 в 15:23

2 ответа

Лучший ответ

У меня было что-то похожее, проблема заключалась в методе init(), который вызвал сбой после рефакторинга моего кода, он работает как шарм. Может быть, это поможет вам:

#if !TARGET_INTERFACE_BUILDER
required init(coder aDecoder: NSCoder) {
    super.init(coder: aDecoder)
    self._setup()
}
#endif

override func prepareForInterfaceBuilder() {
    self._setup()
}
7
dibi 28 Окт 2014 в 15:28
2
Спасибо за совет Даниил. Я пробовал вышеизложенное, и теперь он жалуется, что мои свойства IBInspectable не соответствуют кодированию ключ-значение. Поэтому я прибегал к использованию макроса для добавления/удаления наблюдателя только во время выполнения. Принятие этого ответа для подсказки MACRO.
 – 
Fergal Rooney
28 Окт 2014 в 17:49

Для Xcode 7.2 общий код для @IBDesignable UIButton Subclass выглядит следующим образом:

import UIKit

@IBDesignable class MyButton: UIButton {

    //this init fires usually called, when storyboards UI objects created:
    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        self.setupViews()
    }

    //This method is called during programmatic initialisation
    override init(frame: CGRect) {
        super.init(frame: frame)
        setupViews()
    }


    func setupViews() {
        //your common setup goes here
    }

    //required method to present changes in IB
    override func prepareForInterfaceBuilder() {
        super.prepareForInterfaceBuilder()
        self.setupViews()
    }
}
9
Pavan 20 Июн 2016 в 20:11
Я заметил, что если вы удалите вызов setupViews() в init(frame:), настройка по-прежнему будет работать в IB и приложении. Есть ли причина, по которой нам нужно вызывать установку из этого инициализатора?
 – 
Paul Van Wieren
28 Мар 2016 в 20:38
Интересный. Насколько я помню, это не работает. Установите точку останова внутри setupViews() и проверьте, кто ее вызывает :)
 – 
skywinder
29 Мар 2016 в 12:04
5
- Метод init(frame:) используется при создании представлений в коде. Представления, созданные в раскадровках и перьях, используют метод init(coder:). Строго не обязательно иметь метод init(frame:), если вы никогда не собираетесь создавать представления в коде, но обычно он включается для полноты картины.
 – 
Jason Kirchner
23 Апр 2016 в 05:37
СОХРАНИЛ МНЕ МНОГО ВРЕМЕНИ И БОРЬБЫ
 – 
Yitzchak
15 Ноя 2018 в 14:53
2
Обратите внимание, что если вы используете какой-либо @IBInspectable, вы установите их во время выполнения только в «awakeFromNib», поэтому вам также необходимо добавить это: override func awakeFromNib() { super.awakeFromNib() setupViews() }. и, может быть, вам нужно удалить setupViews() из init(withDecoder)???
 – 
Yitzchak
15 Ноя 2018 в 15:44