Как проанализировать несколько жестко закодированных ключей в структуре JSON API с помощью протокола Swift Decodable? - PullRequest
0 голосов
/ 05 июня 2018

Вопрос. Я пытаюсь декодировать мой JSON, в котором часть JSON будет иметь случайную строку, а другая - жестко закодированную строку.Когда жестко запрограммированная строка является одной из приведенных ниже, я хотел бы отобразить разные ячейки UICollectionView.У меня возникают проблемы при попытке проанализировать мой JSON, если это жестко закодированная строка, и я могу отображать с ней другой UICollectionViewCell.Любая помощь в этом была бы отличной.Это может быть вопрос новичка, но я пытался решить это на прошлой неделе, и у меня возникли проблемы при попытке сделать это.Любая помощь по этому вопросу будет высоко ценится.

**  Hardcoded Strings that could be one or the other:**
key: --> This string could be "breaking" or "normal" or "new"
item: --> This string could be "placement" or "slot" or "shared"
verb: --> This string could be "shared" or "posted"

** NOT hardcoded strings, which the string comes in randomly**
heading: --> This string is a random string
type: --> This string is a random string

Вот некоторые из моих JSON, поэтому вы можете получить пример того, что я пытаюсь сделать:

    {
    slots: [
        {
        key: "breaking",
        item: "placement",
        heading: "Random String Text",
        type: "Random String Text",
        via: "Random",
        verb: "shared"
        sort_order: 0
        },
        {
        key: "breaking",
        item: "placement",
        heading: "Random String Text",
        type: "Random String Text",
        via: "Random",
        verb: "posted"
        sort_order: 1
        },
        {
        key: "event",
        item: "combine",
        heading: "Random String Text",
        type: "Random String Text",
        via: "Random",
        verb: "posted"
        sort_order: 2
        },
}

Это то, что у меня пока есть для моей модели:

struct MyModel: Decodable {
    var key: String?
    var item: String?
    var heading: String?
    var type: String?
    var via: String?
    var verb: String?
}

Вот пример ячейки, с которой мне помог Дмитрий Серов.

func collectionView(_ collectionView: UICollectionView,
 cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {

   let model = ... // retrieve your model object here
   if model.verb == .shared {
     // Pass the pertinent identifier
     let cell = collectionView.dequeueReusableCell(withReuseIdentifier:...)
     return cell
   else {
     ....
   }
 }

Вот еще код, с которым мне помог Дмитрий Серов.

struct MyModel { // Type names should use pascal case in Swift
  var verb: State?
  ....
  enum State {
    case shared
    case posted
  }
}

// Decode your enums from strings
extension MyModel.State: Decodable {
  enum CodingKeys: String, CodingKey {
    case shared
    case posted
  }
}

Когда я пытаюсь описать вышеизложенное, возникает проблема, заключающаяся в том, что мне нужно указать следующий формат: я не уверен, как это сделать, и пытаюсь разобрать еще несколько ключей.

extension MyModel.State: Decodable {
    init(from decoder: Decoder) throws {

  }
}

1 Ответ

0 голосов
/ 05 июня 2018

Я бы расшифровал ваш json, используя кодируемый код:

struct Response: Codable {
    var slots = [Slots]()
}

struct Slots: Codable {
    var key: String?
    var item: String?
    var heading: String?
    var type: String?
    var via: String?
    var verb: String?
    var order: String?

    enum CodingKeys: String, CodingKey {
        case order = "sort_order"
    }
    /** Custom Encoder to send null values**/
    func encode(to encoder: Encoder) throws {
        var container = encoder.container(keyedBy: CodingKeys.self)

        try? container.encode(order, forKey: .order)
    }

}

Для сравнения с жестко закодированными строками создайте перечисление:

enum key: String {
    case breking
    case normal
    case new
}

используйте его так:

if let response = try? JSONDecoder().decode(Response.self, from: response.data!){
for slot in response.slots{
    if slot.key == key.breaking.rawValue{
    //do something
    }
}
}
...