初心者です。
クイズアプリを作成しています。クイズはcsvデータにクイズを4問入れてシャッフルで3問出てくるようになっています。まず、Classを次のように作りました。
Swift
1import Foundation 2class QuestionData { 3 var question: String 4 var answer1: String 5 var answer2: String 6 var answer3: String 7 // 正解の番号 8 var correctAnswerNumber: Int 9 // ユーザが選択した選択肢の番号 10 var userChoiceAnswerNumber: Int? 11 // 問題文の番号 12 var questionNo: Int = 0 13 // クラスが生成された時の処理 14 init(questionSourceDataArray: [String]) { 15 question = questionSourceDataArray[0] 16 answer1 = questionSourceDataArray[1] 17 answer2 = questionSourceDataArray[2] 18 answer3 = questionSourceDataArray[3] 19 correctAnswerNumber = Int(questionSourceDataArray[4])! 20 } 21 // ユーザが選択した答えが正解かどうか判定する 22 func isCorrect() -> Bool { 23 // 答えが一致しているかどうか判定する 24 if correctAnswerNumber == userChoiceAnswerNumber { 25 // 正解 26 return true 27 } 28 // 不正解 29 return false 30 } 31} 32// クイズデータ全般の管理と生成を管理するクラス 33class QuestionDataManager { 34 // シングルトンのオブジェクトを生成 35 static let sharedInstance = QuestionDataManager() 36 // 問題を格納するための配列 37 var questionDataArray = [QuestionData]() 38 // 現在の問題のインデックス 39 var nowQuestionIndex = 0 40 // 初期化処理 41 private init() { 42 // シングルトンであることを保証するためにprivateで宣言しておく 43 } 44 //問題数のカウント 45 var counter = 0 46 // 問題文の読み込み処理 47 func loadQuestion() { 48 // 格納済みの問題文であればいったん削除しておく 49 questionDataArray.removeAll() 50 // 現在の問題のインデックスを初期化 51 nowQuestionIndex = 0 52 // csvファイルパスを取得 53 guard let csvFilePath = Bundle.main.path(forResource: "question", ofType: "csv") else { 54 // csvファイルなし 55 print("csvファイルが存在しません") 56 return 57 } 58 do { 59 let csvStringData = try String(contentsOfFile: csvFilePath, encoding: String.Encoding.utf8) 60 // csvデータを1行ずつ読み込む 61 csvStringData.enumerateLines ( invoking: {(line, stop) -> () in 62 // カンマ区切りで分割 63 let questionSourceDataArray = line.components(separatedBy: ",") 64 // 問題データを格納するオブジェクトを作成 65 let questionData = QuestionData(questionSourceDataArray: questionSourceDataArray) 66 // 問題を追加 67 self.questionDataArray.append(questionData) 68 // 問題番号を設定 69 questionData.questionNo = self.questionDataArray.count 70 }) 71 // シャッフルして最初の3つを取り出す。 72 questionDataArray = Array(questionDataArray.shuffled().prefix(3)) 73 74 } catch let error { 75 print("csvファイル読み込みエラーが発生しました(error)") 76 return 77 } 78 } 79 //配列に同じ数値を入れて初期化 80 var questionArrayCut = [Int](repeating: 0 , count: 3) 81 func randomTool() { 82 //0~3の中から重複しない4個の乱数を生成し、配列に入れてもどす 83 let questionArray: [Int] = [0, 1, 2, 3] 84 var questionArray_Shuffled = questionArray.shuffled() 85 86 for n in 0...2 { 87 questionArrayCut[n] = questionArray_Shuffled[n+1] 88 print(questionArrayCut[n]) 89 } 90 } 91 // 次の問題を取り出す 92 func nextQuestion() -> QuestionData? { 93 if counter < 3 { 94 //ここでランダム 95 //counterが0~2の範囲で1ずつ増えるため、配列の値を取得 96 let qArray: [Int] = questionArrayCut 97 nowQuestionIndex = qArray[counter] 98 print(nowQuestionIndex) 99 let nextQuestion = questionDataArray[counter] 100 counter += 1 101 return nextQuestion 102 } 103 else{ 104 counter = 0 105 } 106 return nil 107 } 108}
次に、クイズの実行画面のコードを次のように書きました。
Swift
1import UIKit 2import AudioToolbox 3class QuestionViewController: UIViewController { 4 var questionData: QuestionData! 5 //問題番号ラベル 6 @IBOutlet weak var questionNoLabel: UILabel! 7 //問題文テキストビュー 8 @IBOutlet weak var questionTextView: UITextView! 9 //選択肢1ボタン 10 @IBOutlet weak var answer1Button: UIButton! 11 //選択肢2ボタン 12 @IBOutlet weak var answer2Button: UIButton! 13 //選択肢3ボタン 14 @IBOutlet weak var answer3Button: UIButton! 15 override func viewDidLoad() { 16 super.viewDidLoad() 17 // Do any additional setup after loading the view. 18 //初期データ設定処理 19 questionNoLabel.text = "Q.(questionData.questionNo)" 20 questionTextView.text = questionData.question 21 answer1Button.setTitle(questionData.answer1, for: UIControl.State.normal) 22 answer2Button.setTitle(questionData.answer2, for: UIControl.State.normal) 23 answer3Button.setTitle(questionData.answer3, for: UIControl.State.normal) 24 } 25 override func didReceiveMemoryWarning() { 26 super.didReceiveMemoryWarning() 27 // Dispose of any resources that can be recreated. 28 } 29 // 選択肢1をタップ 30 @IBAction func tapAnswer1Button(_ sender: Any) { 31 // 選択した答えの番号を保存する 32 questionData.userChoiceAnswerNumber = 1 33 // 次の問題に進む 34 self.goNextQuestion() 35 } 36 //選択肢2をタップ 37 @IBAction func tapAnswer2Button(_ sender: Any) { 38 // 選択した答えの番号を保存する 39 questionData.userChoiceAnswerNumber = 2 40 // 次の問題に進む 41 self.goNextQuestion() 42 } 43 //選択肢3をタップ 44 @IBAction func tapAnswer3Button(_ sender: Any) { 45 // 選択した答えの番号を保存する 46 questionData.userChoiceAnswerNumber = 3 47 // 次の問題に進む 48 self.goNextQuestion() 49 } 50 // 次の問題へ遷移する 51 func goNextQuestion() { 52 // 問題文の取り出し 53 guard let nextQuestion = 54 QuestionDataManager.sharedInstance.nextQuestion() else { 55 // 問題文がなければ結果画面へ遷移する 56 // StoryboardのIdentifierに設定した値(result)を指定して 57 // ViewControllerを生成する 58 if let resultViewController = storyboard?.instantiateViewController(withIdentifier: "result") as? ResultViewController { 59 // StoryboardのSegueを利用しない明示的な画面遷移処理 60 present(resultViewController, animated: true, completion: nil) 61 } 62 return 63 } 64 // 問題文がある場合は次の問題へ遷移する 65 // StoryboardのIdentifierに設定した値(question)を設定して 66 // ViewControllerを生成する 67 if let nextQuestionViewController = 68 storyboard?.instantiateViewController(withIdentifier: "question") 69 as? QuestionViewController { 70 nextQuestionViewController.questionData = nextQuestion 71 // StoryboardのSegueを利用しない明示的な画面遷移処理 72 present(nextQuestionViewController, animated: true, 73 completion: nil) 74 } 75 questionNoLabel.text = "Q.(nextQuestion.questionNo)" 76 questionTextView.text = nextQuestion.question 77 answer1Button.setTitle(nextQuestion.answer1, for: UIControl.State.normal) 78 answer2Button.setTitle(nextQuestion.answer2, for: UIControl.State.normal) 79 answer3Button.setTitle(nextQuestion.answer3, for: UIControl.State.normal) 80} 81}
コードにエラーはないです。しかし、実行をして見ると、クイズはきちんと出されるのですが、クイズの問題番号(questionNoLabel.text)はcsvデータの順番になっていました。
どこで間違ってしまったのでしょうか?
ご教授お願いします。