[ios] 열거 형 유형의 IBInspectable을 만드는 방법

enum인터페이스 빌더가 정의한 런타임 속성 이 아닙니다 . 다음은 Interface Builder의 Attributes Inspector에 표시되지 않습니다.

enum StatusShape:Int {
    case Rectangle = 0
    case Triangle = 1
    case Circle = 2
}
@IBInspectable var shape:StatusShape = .Rectangle

설명서
에서 : 부울, 정수 또는 부동 소수점 숫자, 문자열, 지역화 된 문자열, 사각형 등 Interface Builder 정의 런타임 속성에서 지원하는 모든 유형의 클래스 선언, 클래스 확장 또는 범주의 모든 속성에 IBInspectable 속성을 연결할 수 있습니다. , 포인트, 크기, 색상, 범위 및 nil.

Q :enum Interface Builder의 Attributes Inspector에서 어떻게 볼 수 있습니까?



답변

스위프트 3

@IBInspectable은 var shape:StatusShape = .Rectangle 단순히 Interface Builder에 빈 항목을 만듭니다.

IB에서는 사용할 수 없음

Swift와 Interface Builder 사이 의 다리 역할을하는 어댑터를 사용하십시오 .
shapeAdapterIB에서 검사 가능 :

   // IB: use the adapter
   @IBInspectable var shapeAdapter:Int {
        get {
            return self.shape.rawValue
        }
        set( shapeIndex) {
            self.shape = StatusShape(rawValue: shapeIndex) ?? .Rectangle
        }
    }

IB에서 사용 가능

조건부 컴파일 방식 (사용 #if TARGET_INTERFACE_BUILDER) 과 달리 shape변수 의 유형은 대상에 따라 변경되지 않으므로 잠재적으로 shape:NSIntegershape:StatusShape변형 에 대처하기 위해 추가 소스 코드 변경이 필요할 수 있습니다.

   // Programmatically: use the enum
   var shape:StatusShape = .Rectangle

완전한 코드

@IBDesignable
class ViewController: UIViewController {

    enum StatusShape:Int {
        case Rectangle
        case Triangle
        case Circle
    }

    // Programmatically: use the enum
    var shape:StatusShape = .Rectangle

    // IB: use the adapter
    @IBInspectable var shapeAdapter:Int {
        get {
            return self.shape.rawValue
        }
        set( shapeIndex) {
            self.shape = StatusShape(rawValue: shapeIndex) ?? .Rectangle
        }
    }
}

GitHub 에서이 솔루션을 찾으십시오 .


답변

int로 검사 가능한 열거 형을 설정하는 대신 문자열로 설정할 수도 있습니다. 드롭 다운만큼 바람직하지는 않지만 적어도이 옵션은 어느 정도의 가독성을 제공합니다.

Swift 전용 옵션 :

// 1. Set up your enum
enum Shape: String {
    case Rectangle = "rectangle" // lowercase to make it case-insensitive
    case Triangle = "triangle"
    case Circle = "circle"
}


// 2. Then set up a stored property, which will be for use in code
var shape = Shape.Rectangle // default shape


// 3. And another stored property which will only be accessible in IB (because the "unavailable" attribute prevents its use in code)
@available(*, unavailable, message: "This property is reserved for Interface Builder. Use 'shape' instead.")
@IBInspectable var shapeName: String? {
    willSet {
        // Ensure user enters a valid shape while making it lowercase.
        // Ignore input if not valid.
        if let newShape = Shape(rawValue: newValue?.lowercased() ?? "") {
            shape = newShape
        }
    }
}

있다 또한 ENUM에 초기화를 추가하여도 목표 -C 직장이 얻을 수있다. 그러나 컴파일러는 신속한 코드에서 IB 전용 속성에 대해 “사용할 수 없음”오류 만 표시합니다.

Obj-C 호환성이있는 Swift 옵션 :

@objc enum Shape: Int {
    case None
    case Rectangle
    case Triangle
    case Circle

    init(named shapeName: String) {
        switch shapeName.lowercased() {
        case "rectangle": self = .Rectangle
        case "triangle": self = .Triangle
        case "circle": self = .Circle
        default: self = .None
        }
    }
}

var shape = Shape.Rectangle // default shape

@available(*, unavailable, message: "This property is reserved for Interface Builder. Use 'shape' instead.")
@IBInspectable var shapeName: String? {
    willSet {
        if let newShape = Shape(rawValue: newValue?.lowercased() ?? "") {
            shape = newShape
        }
    }
}


답변

빠른 구문을 기억할 수 없지만 이것이 obj-c에서 해결 한 방법입니다.

#if TARGET_INTERFACE_BUILDER
@property (nonatomic) IBInspectable NSInteger shape;
#else
@property (nonatomic) StatusShape shape;
#endif


답변

2020 년-@SwiftArchitect 답변이 오늘 업데이트되었습니다.

다음은 오늘의 모든 구문이 포함 된 일반적인 전체 예입니다.

여기에 이미지 설명 입력

import UIKit

@IBDesignable class ClipLabels: UILabel {

    enum Side: Int { case left, right }

    var side: Side = .left {
        didSet {
            common()
        }
    }

    @available(*, unavailable, message: "IB only")
    @IBInspectable var leftRight01: Int {
        get {
            return self.side.rawValue
        }
        set(index) {
            self.side = Side(rawValue: index) ?? .left
        }
    }

그리고 사용의 예 …

switch side {
    case .left:
        textColor = .red
    case .right:
        textColor = .green
    }

이 중요한 Swift / iOS QA를 위해

• @SwiftArchitect의 아주 오래된 답변은 완벽하게 정확하지만

• 방금 업데이트하고 중요한 “사용할 수없는”항목을 추가했습니다. 이제 Swift에서 가능합니다.


답변

이것은 오래된 스레드이지만 유용합니다. 나는 swift 4.0과 Xcode 9.0에 대한 대답을 수정했습니다. Swift 4에는이 문제에 대한 작은 문제가 있습니다. 열거 형 유형이있는 @IBInspectable 변수가 있고 Xcode 9.0이 만족스럽지 않습니다. “이 속성은 Objective-c에서 나타낼 수 없기 때문에 @IBInspectable로 표시 할 수 없습니다.”

@Eporediese는이 문제 (swift3의 경우)에 부분적으로 답변합니다. 스토리 보드에는 속성을 사용하지만 나머지 코드에는 직선 열거 형을 사용합니다. 다음은 두 경우 모두에서 작업 할 수있는 속성을 제공하는보다 완전한 코드 세트입니다.

enum StatusShape: Int {
  case Rectangle = 0
  case Triangle = 1
  case Circle = 2
}
var _shape:StatusShape = .Rectangle  // this is the backing variable

#if TARGET_INTERFACE_BUILDER
  @IBInspectable var shape: Int {    // using backing variable as a raw int

    get { return _shape.rawValue }
    set {
      if _shape.rawValue != newValue {
        _shape.rawValue = newValue
      }
    }
}
#else
var shape: StatusShape {  // using backing variable as a typed enum
  get { return _shape }
  set {
    if _shape != newValue {
      _shape = newValue
    }
  }
}
#endif


답변

SwiftArchitect
기반의 Swift 3 솔루션

enum StatusShape: Int {
    case rectangle, triangle, circle
}
var statusShape: StatusShape = .rectangle
#if TARGET_INTERFACE_BUILDER
@IBInspectable var statusShapeIB: Int {
    get {
        return statusShape.rawValue
    }
    set {
        guard let statusShape = StatusShape(rawValue: newValue) else { return }
        self.statusShape = statusShape
    }
}   //convenience var, enum not inspectable
#endif


답변