1

我有一个整数枚举,我想将它用于viewWithTag(_:)数字,但它给了我错误“无法将类型'viewTags'的值转换为预期的参数类型'Int'”,即使枚举和标签号都需要inviewWithTag(_:)是一个Int.

这很简单,如果我使用该属性,我可以让它工作rawValue,但这比我想要的更混乱和麻烦。

enum viewTags: Int {
    case rotateMirroredBtn
    case iPhone6SP_7P_8P
    case iPhoneX_Xs
    case iPhoneXs_Max
    case iPhone_Xr
    case unknown
}

// error on if statement "Cannot convert value of type 'viewTags' to expected argument type 'Int'"
if let tmpButton = self.view.viewWithTag(viewTags.rotateMirroredBtn) as? UIButton { 
    tmpButton.removeFromSuperview()
}
4

2 回答 2

2

您可以轻松地添加扩展UIView来为您进行转换。您只需要使用泛型参数将参数限制为您可以从中获取的内容Int

extension UIView
{
    /**
     Returns the view’s nearest descendant (including itself) whose `tag`
     matches the raw value of the given value, or `nil` if no subview
     has that tag.
     - parameter tag: A value that can be converted to an `Int`.
     */
    func firstView <Tag : RawRepresentable> (taggedBy tag: Tag) -> UIView?
        where Tag.RawValue == Int
    {
        let intValue = tag.rawValue
        return self.viewWithTag(intValue)
    }
}

约束T : RawRepresentable where T.RawValue == Int可以通过您的Int-backed 枚举来实现。

非通用形式也很容易:func firstView(taggedBy viewTag: ViewTag) -> UIView?

另外,您还可以添加一种方法来“组合”值的原始值应用于视图:

func applyTag <Tag : RawRepresentable> (_ tag: Tag)
    where Tag.RawValue == Int
{
    self.tag = tag.rawValue
}

(不幸的是,没有办法将其写为属性,例如,var composedTag: Tag where Tag : RawRepresentable, Tag.RawValue == Int因为计算属性不能像方法一样创建自己的通用上下文。)

于 2019-01-10T19:15:52.670 回答
1

我和最初的海报一样,不喜欢rawValue在代码中使用案例,所以我在我的枚举中添加了计算类型属性。我正在使用 Xcode v11.3.1 和 Swift v5.1.3。

例如,我编写的许多单元测试都使用“魔术”值来为表视图创建 IndexPath,代码如下:

let activeIndexPath = IndexPath(row: 0, section: 0)
let finishedIndexPath = IndexPath(row: 0, section: 1)

我不想这样做,即使它是对“魔术”值的改进:

let activeIndexPath = IndexPath(row: 0, section: TableViewSection.active.rawValue)
let finishedIndexPath = IndexPath(row: 0, section: TableViewSection.finished.rawValue)

我最关心的是我正在测试的表视图部分,所以我想出了这个枚举,它使用计算类型属性来获取 Int rawValues:

enum TableViewSection: Int {
    case active
    case finished

    static var sectionActive: Int { return Self.active.rawValue }
    static var sectionFinished: Int { return Self.finished.rawValue }
}

现在我可以像这样创建一个 IndexPath:

let indexPathActive = IndexPath(row: 0, section: TableViewSection.sectionActive)

缺点是每个案例都需要一个具有相似名称的计算属性,但最终结果在调用站点更具描述性(尽管我猜使用 rawValue 的代码也是描述性的),现在我不必记住要为表格视图的每个特定部分使用哪个 Int 值,我不必再使用“魔术”值,我们都知道这是一件坏事。

希望有帮助!

于 2020-02-12T19:18:41.183 回答