123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438 |
- //
- // QuizViewController.swift
- // LMS
- //
- // Created by Suraj Kumar Mandal on 24/08/22.
- //
- import UIKit
- import Toast_Swift
- class QuizViewController: UIViewController {
-
- @IBOutlet var navigationBar: UINavigationBar!
- @IBOutlet var questionNumberLabel: UILabel!
- @IBOutlet var timerLabel: UILabel!
- @IBOutlet var questionTypeLabel: UILabel!
- @IBOutlet var questionLabel: UILabel!
- @IBOutlet var answerInstructionLabel: UILabel!
- @IBOutlet var quizView: UIView!
- @IBOutlet var answerTF: UITextField!
- @IBOutlet var mcqView: UIView!
- @IBOutlet var mcqTableView: UITableView!
- @IBOutlet var trueFalseView: UIView!
- @IBOutlet var boolTableView: UITableView!
- @IBOutlet var dragView: UIView!
- @IBOutlet var option1TableView: UITableView!
- @IBOutlet var option2TableView: UITableView!
- @IBOutlet var questionNumbersButton: UIButton!
- //Quiz submit UI outlets
- @IBOutlet var quizSubmittedView: UIView!
- @IBOutlet var customSubmittedView: UIView!
- @IBOutlet var checkImageView: UIImageView!
- //Question number UI outlets
- @IBOutlet var questionNumberView: UIView!
- @IBOutlet var questionCustomView: UIView!
- @IBOutlet var questionNumberCollectionView: UICollectionView!
- @IBOutlet var collectionViewHeightConstraint: NSLayoutConstraint!
-
-
- let userData = DBManager.sharedInstance.database.objects(UserDetailsModel.self)
- var viewModel = QuizViewModel()
- var quizModel = [QuizModel]()
- //var answerSubmit = [SubmitAnswerModel]()
- var timer = Timer()
- var assessmentId = Int()
- var assessmentName = String()
- var quesIndex = 1
- var count = 1500
- var answer = [String]()
- var quesNo = Int()
-
- override func viewDidLoad() {
- super.viewDidLoad()
-
- // Do any additional setup after loading the view.
- print(assessmentName)
- navigationBar.topItem?.title = assessmentName
- viewModel.delegate = self
- quizSubmittedView.isHidden = true
- questionNumberView.isHidden = true
-
- questionNumberLabel.clipsToBounds = true
- questionNumberLabel.layer.cornerRadius = 10
- questionNumberLabel.layer.maskedCorners = [.layerMaxXMinYCorner, .layerMinXMinYCorner]
- questionNumbersButton.layer.cornerRadius = questionNumbersButton.frame.size.height / 2
-
- questionNumberCollectionView.delegate = self
- questionNumberCollectionView.dataSource = self
- // Assuming you've reloaded or updated the data in your UICollectionView
- questionNumberCollectionView.reloadData()
- // Call the method to adjust the height
- updateCollectionViewHeight()
- getQuizData()
- getQuizTime()
- }
-
- func getQuizData() {
- if Reachability.isConnectedToNetwork() {
- viewModel.getQuizData(assessmentId: assessmentId)
- } else {
- Alert.showInternetFailureAlert(on: self)
- }
- }
-
- func getQuizTime() {
- if Reachability.isConnectedToNetwork() {
- viewModel.getQuizTime(assessmentId: assessmentId)
- } else {
- Alert.showInternetFailureAlert(on: self)
- }
- }
-
- func setupUI() {
- self.questionCustomView.layer.cornerRadius = 20
- if quizModel[quesIndex-1].questionType == AppConstant.MultipleMCQ {
- answerInstructionLabel.text = "Select [One/More Than One] Options"
- self.answerTF.isHidden = true
- self.mcqView.isHidden = false
- self.mcqTableView.delegate = self
- self.mcqTableView.dataSource = self
- self.mcqTableView.allowsMultipleSelection = true
- self.mcqTableView.allowsMultipleSelectionDuringEditing = true
- self.mcqTableView.reloadData()
- self.trueFalseView.isHidden = true
- self.dragView.isHidden = true
- } else if quizModel[quesIndex-1].questionType == AppConstant.SingleMCQ {
- answerInstructionLabel.text = "Select One Option"
- self.answerTF.isHidden = true
- self.mcqView.isHidden = false
- self.mcqTableView.delegate = self
- self.mcqTableView.dataSource = self
- self.mcqTableView.allowsMultipleSelection = false
- self.mcqTableView.allowsMultipleSelectionDuringEditing = false
- self.mcqTableView.reloadData()
- self.trueFalseView.isHidden = true
- self.dragView.isHidden = true
- } else if quizModel[quesIndex-1].questionType == AppConstant.Fill {
- answerInstructionLabel.text = "Enter Answer In Input Box"
- self.answerTF.isHidden = false
- self.mcqView.isHidden = true
- self.trueFalseView.isHidden = true
- self.dragView.isHidden = true
- } else if quizModel[quesIndex-1].questionType == AppConstant.TrueFalse {
- answerInstructionLabel.text = "Choose One Option"
- self.answerTF.isHidden = true
- self.mcqView.isHidden = true
- self.trueFalseView.isHidden = false
- self.boolTableView.delegate = self
- self.boolTableView.dataSource = self
- self.boolTableView.reloadData()
- self.dragView.isHidden = true
- } else if quizModel[quesIndex-1].questionType == AppConstant.Match {
- answerInstructionLabel.text = "Drag And Drop Correct Match From Right Hand Side"
- self.answerTF.isHidden = true
- self.mcqView.isHidden = true
- self.trueFalseView.isHidden = true
- self.dragView.isHidden = false
- self.option1TableView.delegate = self
- self.option1TableView.dataSource = self
- self.option1TableView.reloadData()
- self.option2TableView.delegate = self
- self.option2TableView.dataSource = self
- self.option2TableView.dragDelegate = self
- self.option2TableView.dragInteractionEnabled = true
- self.option2TableView.reloadData()
- }
- }
-
- func setupData() {
- answerTF.text = ""
- answer.removeAll()
- questionNumberLabel.text = " Question No. - \(quesIndex)/\(quizModel.count) "
- questionTypeLabel.text = quizModel[quesIndex-1].questionType
- questionLabel.text = "Q. \(quizModel[quesIndex-1].question ?? "")"
- }
-
- func startTimer() {
- // Create a Timer that fires every second and calls the updateTimer function
- timer = Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(update), userInfo: nil, repeats: true)
- }
-
- @objc func update() {
- if count > 0 {
- count -= 1
-
- let minutes = count / 60
- let seconds = count % 60
-
- // Format the time as "mm:ss"
- let timeString = String(format: "%02d:%02d", minutes, seconds)
-
- // Update the label with the remaining time
- timerLabel.text = timeString
- } else {
- // Timer has finished, you can handle this event as needed
- timerLabel.text = "00:00" // Display 00:00 when the timer is done
- timer.invalidate()
- }
- }
-
- func submitAnswer() {
- if quizModel[quesIndex-1].questionType == AppConstant.Fill {
- if answerTF.text?.isEmpty == true {
- self.view.makeToast("Enter your answer first!")
- } else {
- answer.append(answerTF.text!)
- print(answer)
- if Reachability.isConnectedToNetwork() {
- viewModel.submitQuizAnswer(quizId: quizModel[quesIndex-1].id ?? 0, userId: userData[0].id, questionId: quizModel[quesIndex-1].assessmentId ?? 0, quesType: quizModel[quesIndex-1].questionType ?? "", answers: self.answer)
- } else {
- Alert.showInternetFailureAlert(on: self)
- }
- }
- } else {
- if Reachability.isConnectedToNetwork() {
- viewModel.submitQuizAnswer(quizId: quizModel[quesIndex-1].id ?? 0, userId: userData[0].id, questionId: quizModel[quesIndex-1].assessmentId ?? 0, quesType: quizModel[quesIndex-1].questionType ?? "", answers: self.answer)
- } else {
- Alert.showInternetFailureAlert(on: self)
- }
- }
- }
-
- /*
- // MARK: - Navigation
-
- // In a storyboard-based application, you will often want to do a little preparation before navigation
- override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
- // Get the new view controller using segue.destination.
- // Pass the selected object to the new view controller.
- }
- */
-
-
- @IBAction func skipAction(_ sender: Any) {
- if quesIndex <= quizModel.count {
- self.quesIndex += 1
- setupUI()
- setupData()
- } else {
- self.view.makeToast("Click Next to Submit!")
- }
- }
-
- @IBAction func nextAction(_ sender: Any) {
- if self.quesIndex < quizModel.count {
- submitAnswer()
- } else {
- Alert.showAlertWithAction(vc: self, title: assessmentName, message: "Are you want to submit your answer?", alertStyle: .alert, actionTitles: ["Ok"], actionStyles: [.default], actions: [{_ in
- self.submitAnswer()
- }])
- }
- }
-
- @IBAction func closeAction(_ sender: Any) {
- timer.invalidate()
- let vc = self.storyboard?.instantiateViewController(withIdentifier: "AssessmentBeneficiaryViewController") as! AssessmentBeneficiaryViewController
- self.navigationController?.pushViewController(vc, animated: true)
- }
-
- @IBAction func quesNumberAction(_ sender: Any) {
- questionNumberView.isHidden = false
- }
-
- @IBAction func cancelAction(_ sender: Any) {
- questionNumberView.isHidden = true
- }
-
-
- }
- extension QuizViewController: QuizViewProtocol {
- func startLoader() {
- ActivityIndicator.start()
- }
-
- func stopLoader() {
- ActivityIndicator.stop()
- }
-
- func showError(error: String) {
- self.view.makeToast(error)
- }
-
- func quizModel(model: [QuizModel]) {
- self.quizModel = model
- self.setupUI()
- self.setupData()
- }
-
- func setQuizTime(time: Int) {
- self.count = time * 60
- self.startTimer()
- }
-
- func answerSubmitted() {
- if self.quesIndex < quizModel.count {
- self.quesIndex += 1
- setupUI()
- setupData()
- } else {
- quizSubmittedView.isHidden = false
- }
- }
- }
- extension QuizViewController: UITableViewDelegate, UITableViewDataSource, UITableViewDragDelegate {
- func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
- if tableView == mcqTableView {
- return quizModel[quesIndex-1].options?.count ?? 0
- } else if tableView == boolTableView {
- return 2
- } else if tableView == option1TableView {
- return quizModel[quesIndex-1].optionsOne?.count ?? 0
- } else {
- return quizModel[quesIndex-1].optionsTwo?.count ?? 0
- }
- }
-
- func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
- if tableView == mcqTableView {
- guard let cell = tableView.dequeueReusableCell(withIdentifier: "MCQTableViewCell", for: indexPath) as? MCQTableViewCell else {
- return UITableViewCell()
- }
-
- cell.optionLabel.text = quizModel[quesIndex-1].options?[indexPath.row]
-
- return cell
- } else if tableView == boolTableView {
- guard let cell = tableView.dequeueReusableCell(withIdentifier: "TrueFalseTableViewCell", for: indexPath) as? TrueFalseTableViewCell else {
- return UITableViewCell()
- }
-
- cell.optionLabel.text = AppConstant.boolOption[indexPath.row]
-
- return cell
- } else if tableView == option1TableView {
- guard let cell = tableView.dequeueReusableCell(withIdentifier: "DragTableViewCell", for: indexPath) as? DragTableViewCell else {
- return UITableViewCell()
- }
-
- cell.optionLabel.text = quizModel[quesIndex-1].optionsOne?[indexPath.row]
-
- return cell
- } else {
- guard let cell = tableView.dequeueReusableCell(withIdentifier: "DragTableViewCell", for: indexPath) as? DragTableViewCell else {
- return UITableViewCell()
- }
-
- cell.optionLabel.text = quizModel[quesIndex-1].optionsTwo?[indexPath.row]
-
- return cell
- }
- }
-
- func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
- if tableView == mcqTableView {
- if quizModel[quesIndex-1].questionType == AppConstant.MultipleMCQ {
- if answer.contains(quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
- if let index = answer.firstIndex(of: quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
- answer.remove(at: index)
- }
- print(answer)
- } else {
- answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
- print(answer)
- }
- } else if quizModel[quesIndex-1].questionType == AppConstant.SingleMCQ {
- answer.removeAll()
- answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
- print(answer)
- }
- } else {
- answer.removeAll()
- answer.append(AppConstant.boolOption[indexPath.row])
- print(answer)
- }
- }
-
- func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) {
- if tableView == mcqTableView {
- if quizModel[quesIndex-1].questionType == AppConstant.MultipleMCQ {
- if answer.contains(quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
- if let index = answer.firstIndex(of: quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
- answer.remove(at: index)
- }
- print(answer)
- } else {
- answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
- print(answer)
- }
- } else if quizModel[quesIndex-1].questionType == AppConstant.SingleMCQ {
- answer.removeAll()
- answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
- print(answer)
- }
- }
- }
-
- func tableView(_ tableView: UITableView, itemsForBeginning session: UIDragSession, at indexPath: IndexPath) -> [UIDragItem] {
- let dragItem = UIDragItem(itemProvider: NSItemProvider())
- dragItem.localObject = quizModel[quesIndex-1].optionsTwo?[indexPath.row]
- return [ dragItem ]
- }
-
- func tableView(_ tableView: UITableView, moveRowAt sourceIndexPath: IndexPath, to destinationIndexPath: IndexPath) {
- // Update the model
- let mover = quizModel[quesIndex-1].optionsTwo?.remove(at: sourceIndexPath.row) ?? ""
- quizModel[quesIndex-1].optionsTwo?.insert(mover, at: destinationIndexPath.row)
-
- //print("\(quizModel[quesIndex-1].optionsOne?[destinationIndexPath.row] ?? "") - \(quizModel[quesIndex-1].optionsTwo?[destinationIndexPath.row] ?? "")")
- self.answer.removeAll()
- for i in 0...quizModel[quesIndex-1].optionsOne!.count-1 {
- let match = "\(quizModel[quesIndex-1].optionsOne?[i] ?? "")-\(quizModel[quesIndex-1].optionsTwo?[i] ?? "")"
- answer.append(match)
- }
- print(answer)
- }
-
- }
- extension QuizViewController: UICollectionViewDelegate, UICollectionViewDataSource, UICollectionViewDelegateFlowLayout {
- func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
- return quizModel.count
- }
-
- func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
- let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "QuestionNumberCollectionViewCell", for: indexPath as IndexPath) as! QuestionNumberCollectionViewCell
-
- cell.customCellView.backgroundColor = #colorLiteral(red: 0.8039215803, green: 0.8039215803, blue: 0.8039215803, alpha: 1)
- cell.customCellView.layer.cornerRadius = cell.customCellView.frame.height / 2
- cell.serialNoLabel.text = "\(indexPath.row + 1)"
-
- return cell
- }
-
- func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
- self.quesIndex = indexPath.row + 1
- self.questionNumberView.isHidden = true
- setupUI()
- setupData()
- }
-
- func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
- let cellWidth = collectionView.frame.size.width / 5
- return CGSize(width: cellWidth, height: cellWidth)
- }
-
- func updateCollectionViewHeight() {
- // Calculate the content size of the UICollectionView based on its content
- questionNumberCollectionView.layoutIfNeeded()
- let contentSize = questionNumberCollectionView.collectionViewLayout.collectionViewContentSize
- // Update the height constraint to match the content size
- collectionViewHeightConstraint.constant = contentSize.height
- }
- }
|