Swift 4 tableView API Pagination - PullRequest
0 голосов
/ 23 ноября 2018

Я успешно проанализировал и расшифровал данные JSON, и они отображаются в консоли.Я не знаю, как настроить пагинацию с помощью API и отобразить все данные при прокрутке таблицы.API, который я использую: https://rickandmortyapi.com/api/character/. Спасибо за всю вашу помощь заранее!

Это код моего основного ViewController

import UIKit

struct PagedCharacters: Codable {
    struct Info: Codable {
        let count: Int
        let pages: Int
        let next: String
        let prev: String
    }

    struct Results: Codable {
        let name: String
        let status: String
        let species: String
    }

    let info: Info
    let results: [Results]

}


class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {

    @IBOutlet var uiTableView: UITableView!

    override func viewDidLoad() {
        super.viewDidLoad()
        getRickAndMortyData()


        self.uiTableView.dataSource = self
        self.uiTableView.delegate = self

        // Do any additional setup after loading the view.
    }

    func getRickAndMortyData() {

        //create instance of the url
        let url = "https://rickandmortyapi.com/api/character/"

        //construct the url, use guard to avoid nonoptional
        guard let urlObj = URL(string: url) else
        { return }

        //fetch data
        URLSession.shared.dataTask(with: urlObj) {(data, response, error) in

            //to avoid non optional in JSONDecoder
            guard let data = data else { return }

            do {
                //decode object
                let downloadedRickAndMorty = try JSONDecoder().decode(PagedCharacters.self, from: data)
             DispatchQueue.main.async {
                    self.uiTableView.reloadData()
                }
               print(downloadedRickAndMorty)

            } catch {
                print(error)

            }

            }.resume()

    }


    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        // #warning Incomplete implementation, return the number of rows
        return 0
    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        guard let cell = tableView.dequeueReusableCell(withIdentifier: "rickandmortyCell") as? CharacterTableViewCell else { return UITableViewCell() }

        cell.nameLabel.text = "Name: " + PagedCharacters.Results[indexPath.row].name

        return cell
    }


}

Это мой класс tableviewcell

import UIKit

class CharacterTableViewCell: UITableViewCell {

    @IBOutlet weak var nameLabel: UILabel!


    override func awakeFromNib() {
        super.awakeFromNib()
        // Initialization code
    }

    override func setSelected(_ selected: Bool, animated: Bool) {
        super.setSelected(selected, animated: animated)

        // Configure the view for the selected state
    }

}

Это данные, которые я получаю в своей консоли, когда распечатанные данные печатаются.Я все еще не получаю все данные с разных страниц.

PagedCharacters (info: API_Practice.PagedCharacters.Info (count: 493, pages: 25, next: "https://rickandmortyapi.com/api/character/?page=2",prev: ""), результаты: [API_Practice.PagedCharacters.Results (имя: "Рик Санчес", статус: "Живой", вид: "Человек"), API_Practice.PagedCharacters.Results (имя: "Морти Смит", статус:"Живой", вид: "Человек"), API_Practice.PagedCharacters.Results (имя: "Саммер Смит", статус: "Живой", вид: "Человек"), API_Practice.PagedCharacters.Results (имя: "Бет Смит",статус: «Живой», вид: «Человек»), API_Practice.PagedCharacters.Results (имя: «Джерри Смит», статус: «Живой», вид: «Человек»), API_Practice.PagedCharacters.Results (имя: «Кластер Абаданго»Принцесса ", статус:" Живой ", вид:" Чужой "), API_Practice.PagedCharacters.Results (имя:" Абрадольф Линклер ", статус:" неизвестный ", вид:" Человек "), API_Practice.PagedCharacters.Results (имя:"Судья Рик", статус: "Мертвый", вид: "Человек"), API_Practice.PagedCharacters.Results (имя: «Директор агентства», статус: «Мертвый», вид: «Человек»), API_Practice.PagedCharacters.Results (имя: «Алан Рельс», статус: «Мертвый», вид: «Человек»),API_Practice.PagedCharacters.Results (имя: «Альберт Эйнштейн», статус: «Мертвый», вид: «Человек»), API_Practice.PagedCharacters.Results (имя: «Александр», статус: «Мертвый», вид: «Человек»)., API_Practice.PagedCharacters.Results (имя: «Чужой Googah», статус: «неизвестный», вид: «Чужой»), API_Practice.PagedCharacters.Results (имя: «Чужой Морти», статус: «неизвестный», вид: «Чужой"), API_Practice.PagedCharacters.Results (имя:" Чужой Рик ", статус: неизвестный, вид:" Чужой "), API_Practice.PagedCharacters.Results (имя:" Амиш Киборг ", статус:" Мертвый ", вид:"Чужой"), API_Practice.PagedCharacters.Results (имя: "Энни", статус: "Живой", вид: "Человек"), API_Practice.PagedCharacters.Results (имя: "Антенна Морти", статус: "Живой", вид: "Человек"), API_Practice.PagedCharacters.Results (имя: "Антенна Рик", статус: "неизвестен", сpecies: "Human"), API_Practice.PagedCharacters.Results (имя: "Муравьи в моих глазах Джонсон", статус: "unknown", разновидности: "Human")])

Если вам что-то нужноиначе, пожалуйста, дайте мне знать.Спасибо!

1 Ответ

0 голосов
/ 04 декабря 2018

В вашей логике было много недостатков, поэтому я переписал ее с минимальными изменениями.Теперь это будет служить вашей цели.

import UIKit
struct PagedCharacters: Codable {
    let info: Info
    let results: [Results]

}
struct Info: Codable {
    let count: Int
    let pages: Int
    let next: String
    let prev: String
}

struct Results: Codable {
    let name: String
    let status: String
    let species: String
}

А вот код для класса контроллера представления:

class ViewController: UIViewController, UITableViewDataSource, UITableViewDelegate {

    @IBOutlet var uiTableView: UITableView!
    var aryDownloadedData:[Results]?
    var nextPageUrl:String!

    override func viewDidLoad() {
        super.viewDidLoad()
        getIntitalRickAndMortyData()


        self.uiTableView.dataSource = self
        self.uiTableView.delegate = self

        // Do any additional setup after loading the view.
    }
    func getIntitalRickAndMortyData(){
        aryDownloadedData = []
        //here first page is next page
        nextPageUrl = "https://rickandmortyapi.com/api/character/"
        getRickAndMortyData()
    }

    func getRickAndMortyData() {

        //construct the url, use guard to avoid nonoptional
        guard let urlObj = URL(string: nextPageUrl) else
        { return }

        //fetch data
        URLSession.shared.dataTask(with: urlObj) {[weak self](data, response, error) in

            //to avoid non optional in JSONDecoder
            guard let data = data else { return }

            do {
                //decode object
                let downloadedRickAndMorty = try JSONDecoder().decode(PagedCharacters.self, from: data)
                self?.aryDownloadedData?.append(contentsOf: downloadedRickAndMorty.results)
                self?.nextPageUrl = downloadedRickAndMorty.info.next

                DispatchQueue.main.async {
                    self?.uiTableView.reloadData()
                }
                print(self?.aryDownloadedData as Any)

            } catch {
                print(error)

            }

            }.resume()

    }


    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return self.aryDownloadedData?.count ?? 0
    }

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {

        if let count = aryDownloadedData?.count, count>1{
            let lastElement = count - 1
            if indexPath.row == lastElement {
                //call get api for next page
                getRickAndMortyData()
            }
        }

        guard let cell = tableView.dequeueReusableCell(withIdentifier: "rickandmortyCell") as? CharacterTableViewCell else { return UITableViewCell() }

        cell.nameLabel.text = "Name: " + (self.aryDownloadedData?[indexPath.row].name ?? "default")

        return cell
    }
}
...