swift - 可以制作通用枚举 UIPickerView

标签 swift generics uipickerview

是否可以编写一个通用的 UIPickerView 子类,您可以在其中传入任意枚举类型,并让它用上述枚举的所有情况填充选择器 View ?

注意:对于任意传入的枚举类型,通常需要执行此操作。

最佳答案

所以我终于能够编写一个通用的枚举感知选择器 View 。这就是我所做的。首先我做了一个 PickableEnum 协议(protocol),如下所示:

protocol PickableEnum: Hashable {
   static var allValues: [Self] {get}
   var pickerText:   String {get}
}

我需要能够大致了解所有枚举案例是什么。这是 allValues 属性的工作,我在网上找到了它的实现。

https://theswiftdev.com/2017/01/05/18-swift-gist-generic-allvalues-for-enums/ )

代码如下:

extension PickableEnum {

   static func cases() -> AnySequence<Self> {
      typealias S = Self
      return AnySequence { () -> AnyIterator<S> in
         var raw = 0
         return AnyIterator {
            let current : Self = withUnsafePointer(to: &raw) { $0.withMemoryRebound(to: S.self, capacity: 1) { $0.pointee } }
            guard current.hashValue == raw else { return nil }
            raw += 1
            return current
         }
      }
   }

   static var allValues: [Self] {
      return Array(self.cases())
   }
}

PickableEnum 还有一个名为 pickerText 的只读字符串属性,用于获取将出现在选择器界面中的文本。

其余的都是微不足道的。让您的 UI 类采用适当类型的通用限定符,然后相应地配置您的 UIPickerView:

class EnumPickerView<T:PickableEnum> : UIPickerView, UIPickerViewDelegate, UIPickerViewDataSource {

   private let enumCases = T.allValues
   private var currentValue: T

   init(frame: CGRect, initialValue: T) {
      currentValue = initialValue
      super.init(frame: frame)
      dataSource = self
      delegate = self
   }

   required init?(coder aDecoder: NSCoder) {
      fatalError("init(coder:) has not been implemented")
   }

   var value: T {
      get {
         return currentValue
      }
   }

   func numberOfComponents(in pickerView: UIPickerView) -> Int {
      return 1
   }

   func pickerView(_ pickerView: UIPickerView, numberOfRowsInComponent component: Int) -> Int {
      return enumCases.count
   }

   func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) -> String? {
      return enumCases[row].pickerText
   }

   func pickerView(_ pickerView: UIPickerView, didSelectRow row: Int, inComponent component: Int) {
      currentValue = enumCases[row]
   }
}

关于swift - 可以制作通用枚举 UIPickerView,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45701225/

相关文章:

ios - Swift:导航 Controller 返回零

java - 如何在方法中使用类类型泛型

C# - 将 List<T>.Find() 与自定义对象一起使用

ios - 从外部类管理 UIPickerView - 使用 Swift

iphone - 重新创建仅显示一行的 UIPickerView

ios - 如何使用 swift 将多个 Controller 呈现为弹出窗口

ios - Swift:将 CGSize 应用于 Button 类不改变

swift - 无法在 Swift 中覆盖 NSDictionary 的初始值设定项

java - 泛型函数的典型操作

ios - 如何在iOS 10 上播放UIPickerView 的滴答声?