質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

88.63%

swiftで「Youtube Data」のJSONファイルを読み込む方法

解決済

回答 2

投稿 編集

  • 評価
  • クリップ 0
  • VIEW 936

denshatrain

score 16

 前提・実現したいこと

「Youtube Data」のjsonファイルから動画のタイトルとURLを抜き出してitemDataArrayに追加するということをしたいです。
ネットや本を見ながらjsonの読み込みをやっていたのですがrunをするとリクエストが送信されません。

 該当のソースコード

import UIKit
import Accounts

class ItemSearchResultSet: Codable {
    var items: Items

    private enum CodingKeys: String, CodingKey {
        case items = "items"
    }
}

class Items: Codable {
    var id: Id
    var snippet: Snippet

    private enum CodingKeys: String, CodingKey {
        case id = "id"
        case snippet = "snippet"
    }
}

class Snippet: Codable {
    var title: Title

    private enum CodingKeys: String, CodingKey {
        case title = "title"
    }
}

class Id: Codable {
    var videoId: VideoId
    private enum CodingKeys: String, CodingKey {
        case videoId = "videoId"
    }
}

class Title: Codable {
    var items: [ItemData] = [ItemData]()
    required init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        let keys = container.allKeys.sorted {
            Int($0.rawValue)! < Int($1.rawValue)!
        }
        for key in keys {
            let item = try container.decode(ItemData.self, forKey: key)
            items.append(item)
        }
    }

    func encode(to encoder: Encoder) throws {
    }

    private enum CodingKeys: String, CodingKey {
        case hit0 = "0"
        case hit1 = "1"
        case hit2 = "2"
        case hit3 = "3"
        case hit4 = "4"
        case hit5 = "5"
        case hit6 = "6"
        case hit7 = "7"
        case hit8 = "8"
        case hit9 = "9"
        case hit10 = "10"
        case hit11 = "11"
        case hit12 = "12"
        case hit13 = "13"
        case hit14 = "14"
        case hit15 = "15"
        case hit16 = "16"
        case hit17 = "17"
        case hit18 = "18"
        case hit19 = "19"
        case hit20 = "20"
    }

}

class VideoId: Codable {
    var items: [ItemData] = [ItemData]()
    required init(from decoder: Decoder) throws {
        let container = try decoder.container(keyedBy: CodingKeys.self)
        let keys = container.allKeys.sorted {
            Int($0.rawValue)! < Int($1.rawValue)!
        }
        for key in keys {
            let item = try container.decode(ItemData.self, forKey: key)
            items.append(item)
        }
    }

    func encode(to encoder: Encoder) throws {
    }

    private enum CodingKeys: String, CodingKey {
        case hit0 = "0"
        case hit1 = "1"
        case hit2 = "2"
        case hit3 = "3"
        case hit4 = "4"
        case hit5 = "5"
        case hit6 = "6"
        case hit7 = "7"
        case hit8 = "8"
        case hit9 = "9"
        case hit10 = "10"
        case hit11 = "11"
        case hit12 = "12"
        case hit13 = "13"
        case hit14 = "14"
        case hit15 = "15"
        case hit16 = "16"
        case hit17 = "17"
        case hit18 = "18"
        case hit19 = "19"
        case hit20 = "20"
    }

}

class ItemData: Codable {
    var title: String = ""
    var videoId: String = ""

    class ImageInfo: Codable {
        private enum CodingKeys: String, CodingKey {
            case medium = "Medium"
        }
        var medium: String?
    }

    var imageInfo : ImageInfo = ImageInfo()

    private enum CodingKeys: String, CodingKey {
        case title = "title"
        case videoId = "videoId"
    }

}



class ViewController: UIViewController , UISearchBarDelegate {

    @IBOutlet var searchBar: UISearchBar!

    var itemDataArray = [ItemData]()

    let appid = "AIzaSyCAIznRo4MUdgBtijMnd2De9YEUqMdmwaY"

    let entryUrl: String = "https://www.googleapis.com/youtube/v3/search?key=AIzaSyCAIznRo4MUdgBtijMnd2De9YEUqMdmwaY&q="

    override func viewDidLoad() {
        super.viewDidLoad()
        // Do any additional setup after loading the view, typically from a nib.

        searchBar.placeholder = "ここに入力してください"
        searchBar.delegate = self
    }

    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }

    func searchBarSearchButtonClicked(_ searchBar: UISearchBar) {

        print("検索ボタンがタップされました!")

        guard let inputText = searchBar.text else {
            return
        }

        guard inputText.lengthOfBytes(using: String.Encoding.utf8) > 0 else {
            return
        }

        itemDataArray.removeAll()



        let parameter = ["appid": appid, "query": inputText]

        let requestUrl = entryUrl + inputText + "&part=snippet&maxResults=10&order=viewCount"

        request(requestUrl: requestUrl)

        searchBar.resignFirstResponder()
    }

    func encodeParamater(key: String, value: String) -> String? {
        guard let escapedValue = value.addingPercentEncoding(withAllowedCharacters: CharacterSet.urlQueryAllowed) else {
            return nil
        }

        return "\(key) = \(escapedValue)"

    }

    func createRequestUrl(parameter:[String: String]) -> String {
        var parameterString = ""
        for key in parameter.keys {
            guard let value = parameter[key]else {
                continue
            }

            if parameterString.lengthOfBytes(using: String.Encoding.utf8) > 0 {
                parameterString += "&"
            }

            guard let encodeValue = encodeParamater(key: key, value: value) else {
                continue
            }

            parameterString += encodeValue

        }
        let requestUrl = entryUrl + "?" + parameterString
        print(requestUrl)
        return requestUrl
    }

    func request(requestUrl: String) {
        print(1)
        guard let url = URL(string: requestUrl) else {
            return
        }
        let request = URLRequest(url: url)
        let session = URLSession.shared
        let task = session.dataTask (with: request) { (data:Data?, response:URLResponse?, error:Error?) in
            guard error == nil else {
                let alert = UIAlertController(title: "エラー", message: error?.localizedDescription, preferredStyle: .alert)
                DispatchQueue.main.async {
                    self.present(alert, animated: true, completion: nil)
                }
                return
            }

            guard let data = data else {
                return
            }

            do {
                let resultSet = try JSONDecoder().decode(ItemSearchResultSet.self, from: data)

                self.itemDataArray.append(contentsOf: resultSet.items.id.videoId.items)
                self.itemDataArray.append(contentsOf: resultSet.items.snippet.title.items)
                print("データーが追加されました")
                print(self.itemDataArray)
            } catch let error {
                print("## error: \(error)")
            }

            DispatchQueue.main.async {

            }

        }



        task.resume()

    }


}

 困っている点

上の

func request(requestUrl: String) {
        print(1)
        guard let url = URL(string: requestUrl) else {
            return
        }
        //ここから呼び出されていない
        let request = URLRequest(url: url)
        let session = URLSession.shared
        let task = session.dataTask (with: request) { (data:Data?, response:URLResponse?, error:Error?) in
            guard error == nil else {
                let alert = UIAlertController(title: "エラー", message: error?.localizedDescription, preferredStyle: .alert)
                DispatchQueue.main.async {
                    self.present(alert, animated: true, completion: nil)
                }
                return
            }

            guard let data = data else {
                return
            }

            do {
                let resultSet = try JSONDecoder().decode(ItemSearchResultSet.self, from: data)

                self.itemDataArray.append(contentsOf: resultSet.items.id.videoId.items)
                self.itemDataArray.append(contentsOf: resultSet.items.snippet.title.items)
                print("データーが追加されました")
                print(self.itemDataArray)
            } catch let error {
                print("## error: \(error)")
            }

            DispatchQueue.main.async {

            }

        }



        task.resume()

    }


この部分がキーボードの検索ボタンを押したときに呼び出されてはいるようなのですが上に書いたところでprint文が呼び出されなくないました。その後にもitemDataArrayにデーターが入っていません。またprintするとコンソールに[]と表示されます。

また、実機でも試しましたができませんでした。

 補足情報(FW/ツールのバージョンなど)

swift4.1
ios11.4

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 過去に投稿した質問と同じ内容の質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

質問への追記・修正、ベストアンサー選択の依頼

  • nakasho_dev

    2018/08/25 01:48

    Print文を埋め込んでいるところを見ると、ご自身でもデバッグされていると思うのですが、どこまでは処理が走っているかなどの現状の認識を追加されると回答がされやすいかと思います。

    キャンセル

回答 2

checkベストアンサー

0

func request(requestUrl: String) {
        print(1)
        guard let url = URL(string: requestUrl) else {
            return
        }
        //ここから呼び出されていない
        let request = URLRequest(url: url)

とのことなのでURL(string: requestUrl)の結果がnilなのでしょう。
他の方が指摘している通りなのですがcreateRequestUrlやencodeParamaterは使用されていないようですのでrequestUrlを生成する直前にエスケープしてあげるような以下の方法ではいかがでしょうか。

guard let escapedInputText = inputText.addingPercentEncoding(
                   withAllowedCharacters: CharacterSet.urlQueryAllowed
) else {
    return
}
let requestUrl = entryUrl + escapedInputText + "&part=snippet&maxResults=10&order=viewCount"

guardでnilチェックする際はelseブロック内でログを出力するようにしておくと、バグに気付きやすくなります。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

0

searchBarSearchButtonClicked(_:)inputTextがURLエンコードされてないですね。

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2018/08/27 05:38

    どういうふうにURLエンコードをしたら良いですか?

    キャンセル

  • 2018/08/27 08:33 編集

    encodeURIComponent()
    ですね。

    返り値がエンコードされた文字列になります。

    キャンセル

  • 2018/08/27 10:48

    あなたが使用している
    addingPercentEncoding(withAllowedCharacters: .urlQueryAllowed)
    でいいです

    キャンセル

15分調べてもわからないことは、teratailで質問しよう!

  • ただいまの回答率 88.63%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

同じタグがついた質問を見る