Moya のようなネストされた列挙型でInstagramエンドポイントの列挙型を作成しました。
enum Instagram {
enum Media {
case Popular
case Shortcode(id: String)
case Search(lat: Float, lng: Float, distance: Int)
}
enum Users {
case User(id: String)
case Feed
case Recent(id: String)
}
}
各エンドポイントのパスを返したいのですが。
extension Instagram: TargetType {
var path: String {
switch self {
case .Media.Shortcode(let id):
return "/media/shortcode"
}
}
}
ただし、上記のpath
のswitchステートメントでエラーが発生します。
列挙型ケース
Shortcode
はタイプ
直し方?
いくつかの理由により、より一般的な答えを追加しています。
enum Action {
case fighter(F)
case weapon(W)
enum F {
case attack(A)
case defend(D)
case hurt(H)
enum A {
case fail
case success
}
enum D {
case fail
case success
}
enum H {
case none
case some
}
}
enum W {
case swing
case back
}
}
// Matches "3 deep"
let action = Action.fighter(.attack(.fail))
// Matches "1 deep" because more general case listed first.
let action2 = Action.weapon(.swing)
switch action {
case .fighter(.attack(.fail)):
print("3 deep")
case .weapon:
print("1 deep")
case .weapon(.swing):
print("2 deep to case")
case .fighter(.attack):
print("2 deep to another enum level")
default:
print("WTF enum")
}
ネストされた列挙型に関連する値を追加すると、switchステートメントを使用してそれにアクセスできます。
enum Instagram {
enum MediaEndpoint {
case Search(lat: Float, lng: Float, distance: Int)
}
case Media(MediaEndpoint)
}
extension Instagram: TargetType {
var path: String {
switch self {
case .Media(.Search):
return "/media/search"
}
}
}
// Demo
protocol TargetType {
var path: String { get }
}
class MoyaProvider<Target: TargetType> {
func request(_ target: Target, completion: @escaping () -> ()) {}
}
let provider = MoyaProvider<Instagram>()
provider.request(.Media(.Search(lat: 0, lng: 0, distance: 0))) {}
アーキテクチャにはいくつかの問題があります。拡張機能とプロトコルを使用する必要がある時期と理由、およびコードブロックをどのように構造化するかを知っておく必要があります。
private extension String {
var URLEscapedString: String {
return self.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLHostAllowedCharacterSet())!
}
}
switch self {
case .Zen:
return "/zen"
case .UserProfile(let name):
return "/users/\(name.URLEscapedString)"
case .UserRepositories(let name):
return "/users/\(name.URLEscapedString)/repos"
}
ケースの値は、自己のメンバーである必要があります。それがタイプを見つけることができない理由です。タイプはInstagram enum内で宣言されますが、それ自体には価値がありません。 Media内で値を保持します。したがって、メディア関連の関数をMediaの宣言に移動して、そこにアクセスします。このように、自己はメディアを指します。ここに私のための完全な作業コードがあります:
private extension String {
var URLEscapedString: String {
return self.stringByAddingPercentEncodingWithAllowedCharacters(NSCharacterSet.URLHostAllowedCharacterSet())!
}
}
public enum Instagram {
public enum Media {
case Search(String)
var path:String {
switch self {
case Media.Search(let keyword):
return "/media/search/\(keyword.URLEscapedString)"
}
}
}
</ code>
}
var me = Instagram.Media.Search( "me") print(me.path)
列挙型ケース検索はInstagramタイプのメンバーではありません
コンパイラが言うように、Search
はタイプInstagram
のメンバーではありません。 Instagram
のスコープ内の列挙型です。 Search
にInstagram
のインスタンスであるメンバーを作成する必要があります
struct Instagram {
enum Media {
case Search(lat: Float, lng: Float, distance: Int)
}
// something like:
var media = .Search(lat: 0, lng: 0, distance: 0)
// I'm not sure this one is right syntax
// because I can't check it right now.
// please just get the idea
}
extension Instagram: TargetType {
var path: String {
switch self.media {
case .Search(let _, let _, let _):
return "/media/search"
}
}
}