QuizViewController.swift 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438
  1. //
  2. // QuizViewController.swift
  3. // LMS
  4. //
  5. // Created by Suraj Kumar Mandal on 24/08/22.
  6. //
  7. import UIKit
  8. import Toast_Swift
  9. class QuizViewController: UIViewController {
  10. @IBOutlet var navigationBar: UINavigationBar!
  11. @IBOutlet var questionNumberLabel: UILabel!
  12. @IBOutlet var timerLabel: UILabel!
  13. @IBOutlet var questionTypeLabel: UILabel!
  14. @IBOutlet var questionLabel: UILabel!
  15. @IBOutlet var answerInstructionLabel: UILabel!
  16. @IBOutlet var quizView: UIView!
  17. @IBOutlet var answerTF: UITextField!
  18. @IBOutlet var mcqView: UIView!
  19. @IBOutlet var mcqTableView: UITableView!
  20. @IBOutlet var trueFalseView: UIView!
  21. @IBOutlet var boolTableView: UITableView!
  22. @IBOutlet var dragView: UIView!
  23. @IBOutlet var option1TableView: UITableView!
  24. @IBOutlet var option2TableView: UITableView!
  25. @IBOutlet var questionNumbersButton: UIButton!
  26. //Quiz submit UI outlets
  27. @IBOutlet var quizSubmittedView: UIView!
  28. @IBOutlet var customSubmittedView: UIView!
  29. @IBOutlet var checkImageView: UIImageView!
  30. //Question number UI outlets
  31. @IBOutlet var questionNumberView: UIView!
  32. @IBOutlet var questionCustomView: UIView!
  33. @IBOutlet var questionNumberCollectionView: UICollectionView!
  34. @IBOutlet var collectionViewHeightConstraint: NSLayoutConstraint!
  35. let userData = DBManager.sharedInstance.database.objects(UserDetailsModel.self)
  36. var viewModel = QuizViewModel()
  37. var quizModel = [QuizModel]()
  38. //var answerSubmit = [SubmitAnswerModel]()
  39. var timer = Timer()
  40. var assessmentId = Int()
  41. var assessmentName = String()
  42. var quesIndex = 1
  43. var count = 1500
  44. var answer = [String]()
  45. var quesNo = Int()
  46. override func viewDidLoad() {
  47. super.viewDidLoad()
  48. // Do any additional setup after loading the view.
  49. print(assessmentName)
  50. navigationBar.topItem?.title = assessmentName
  51. viewModel.delegate = self
  52. quizSubmittedView.isHidden = true
  53. questionNumberView.isHidden = true
  54. questionNumberLabel.clipsToBounds = true
  55. questionNumberLabel.layer.cornerRadius = 10
  56. questionNumberLabel.layer.maskedCorners = [.layerMaxXMinYCorner, .layerMinXMinYCorner]
  57. questionNumbersButton.layer.cornerRadius = questionNumbersButton.frame.size.height / 2
  58. questionNumberCollectionView.delegate = self
  59. questionNumberCollectionView.dataSource = self
  60. // Assuming you've reloaded or updated the data in your UICollectionView
  61. questionNumberCollectionView.reloadData()
  62. // Call the method to adjust the height
  63. updateCollectionViewHeight()
  64. getQuizData()
  65. getQuizTime()
  66. }
  67. func getQuizData() {
  68. if Reachability.isConnectedToNetwork() {
  69. viewModel.getQuizData(assessmentId: assessmentId)
  70. } else {
  71. Alert.showInternetFailureAlert(on: self)
  72. }
  73. }
  74. func getQuizTime() {
  75. if Reachability.isConnectedToNetwork() {
  76. viewModel.getQuizTime(assessmentId: assessmentId)
  77. } else {
  78. Alert.showInternetFailureAlert(on: self)
  79. }
  80. }
  81. func setupUI() {
  82. self.questionCustomView.layer.cornerRadius = 20
  83. if quizModel[quesIndex-1].questionType == AppConstant.MultipleMCQ {
  84. answerInstructionLabel.text = "Select [One/More Than One] Options"
  85. self.answerTF.isHidden = true
  86. self.mcqView.isHidden = false
  87. self.mcqTableView.delegate = self
  88. self.mcqTableView.dataSource = self
  89. self.mcqTableView.allowsMultipleSelection = true
  90. self.mcqTableView.allowsMultipleSelectionDuringEditing = true
  91. self.mcqTableView.reloadData()
  92. self.trueFalseView.isHidden = true
  93. self.dragView.isHidden = true
  94. } else if quizModel[quesIndex-1].questionType == AppConstant.SingleMCQ {
  95. answerInstructionLabel.text = "Select One Option"
  96. self.answerTF.isHidden = true
  97. self.mcqView.isHidden = false
  98. self.mcqTableView.delegate = self
  99. self.mcqTableView.dataSource = self
  100. self.mcqTableView.allowsMultipleSelection = false
  101. self.mcqTableView.allowsMultipleSelectionDuringEditing = false
  102. self.mcqTableView.reloadData()
  103. self.trueFalseView.isHidden = true
  104. self.dragView.isHidden = true
  105. } else if quizModel[quesIndex-1].questionType == AppConstant.Fill {
  106. answerInstructionLabel.text = "Enter Answer In Input Box"
  107. self.answerTF.isHidden = false
  108. self.mcqView.isHidden = true
  109. self.trueFalseView.isHidden = true
  110. self.dragView.isHidden = true
  111. } else if quizModel[quesIndex-1].questionType == AppConstant.TrueFalse {
  112. answerInstructionLabel.text = "Choose One Option"
  113. self.answerTF.isHidden = true
  114. self.mcqView.isHidden = true
  115. self.trueFalseView.isHidden = false
  116. self.boolTableView.delegate = self
  117. self.boolTableView.dataSource = self
  118. self.boolTableView.reloadData()
  119. self.dragView.isHidden = true
  120. } else if quizModel[quesIndex-1].questionType == AppConstant.Match {
  121. answerInstructionLabel.text = "Drag And Drop Correct Match From Right Hand Side"
  122. self.answerTF.isHidden = true
  123. self.mcqView.isHidden = true
  124. self.trueFalseView.isHidden = true
  125. self.dragView.isHidden = false
  126. self.option1TableView.delegate = self
  127. self.option1TableView.dataSource = self
  128. self.option1TableView.reloadData()
  129. self.option2TableView.delegate = self
  130. self.option2TableView.dataSource = self
  131. self.option2TableView.dragDelegate = self
  132. self.option2TableView.dragInteractionEnabled = true
  133. self.option2TableView.reloadData()
  134. }
  135. }
  136. func setupData() {
  137. answerTF.text = ""
  138. answer.removeAll()
  139. questionNumberLabel.text = " Question No. - \(quesIndex)/\(quizModel.count) "
  140. questionTypeLabel.text = quizModel[quesIndex-1].questionType
  141. questionLabel.text = "Q. \(quizModel[quesIndex-1].question ?? "")"
  142. }
  143. func startTimer() {
  144. // Create a Timer that fires every second and calls the updateTimer function
  145. timer = Timer.scheduledTimer(timeInterval: 1, target: self, selector: #selector(update), userInfo: nil, repeats: true)
  146. }
  147. @objc func update() {
  148. if count > 0 {
  149. count -= 1
  150. let minutes = count / 60
  151. let seconds = count % 60
  152. // Format the time as "mm:ss"
  153. let timeString = String(format: "%02d:%02d", minutes, seconds)
  154. // Update the label with the remaining time
  155. timerLabel.text = timeString
  156. } else {
  157. // Timer has finished, you can handle this event as needed
  158. timerLabel.text = "00:00" // Display 00:00 when the timer is done
  159. timer.invalidate()
  160. }
  161. }
  162. func submitAnswer() {
  163. if quizModel[quesIndex-1].questionType == AppConstant.Fill {
  164. if answerTF.text?.isEmpty == true {
  165. self.view.makeToast("Enter your answer first!")
  166. } else {
  167. answer.append(answerTF.text!)
  168. print(answer)
  169. if Reachability.isConnectedToNetwork() {
  170. 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)
  171. } else {
  172. Alert.showInternetFailureAlert(on: self)
  173. }
  174. }
  175. } else {
  176. if Reachability.isConnectedToNetwork() {
  177. 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)
  178. } else {
  179. Alert.showInternetFailureAlert(on: self)
  180. }
  181. }
  182. }
  183. /*
  184. // MARK: - Navigation
  185. // In a storyboard-based application, you will often want to do a little preparation before navigation
  186. override func prepare(for segue: UIStoryboardSegue, sender: Any?) {
  187. // Get the new view controller using segue.destination.
  188. // Pass the selected object to the new view controller.
  189. }
  190. */
  191. @IBAction func skipAction(_ sender: Any) {
  192. if quesIndex <= quizModel.count {
  193. self.quesIndex += 1
  194. setupUI()
  195. setupData()
  196. } else {
  197. self.view.makeToast("Click Next to Submit!")
  198. }
  199. }
  200. @IBAction func nextAction(_ sender: Any) {
  201. if self.quesIndex < quizModel.count {
  202. submitAnswer()
  203. } else {
  204. Alert.showAlertWithAction(vc: self, title: assessmentName, message: "Are you want to submit your answer?", alertStyle: .alert, actionTitles: ["Ok"], actionStyles: [.default], actions: [{_ in
  205. self.submitAnswer()
  206. }])
  207. }
  208. }
  209. @IBAction func closeAction(_ sender: Any) {
  210. timer.invalidate()
  211. let vc = self.storyboard?.instantiateViewController(withIdentifier: "AssessmentBeneficiaryViewController") as! AssessmentBeneficiaryViewController
  212. self.navigationController?.pushViewController(vc, animated: true)
  213. }
  214. @IBAction func quesNumberAction(_ sender: Any) {
  215. questionNumberView.isHidden = false
  216. }
  217. @IBAction func cancelAction(_ sender: Any) {
  218. questionNumberView.isHidden = true
  219. }
  220. }
  221. extension QuizViewController: QuizViewProtocol {
  222. func startLoader() {
  223. ActivityIndicator.start()
  224. }
  225. func stopLoader() {
  226. ActivityIndicator.stop()
  227. }
  228. func showError(error: String) {
  229. self.view.makeToast(error)
  230. }
  231. func quizModel(model: [QuizModel]) {
  232. self.quizModel = model
  233. self.setupUI()
  234. self.setupData()
  235. }
  236. func setQuizTime(time: Int) {
  237. self.count = time * 60
  238. self.startTimer()
  239. }
  240. func answerSubmitted() {
  241. if self.quesIndex < quizModel.count {
  242. self.quesIndex += 1
  243. setupUI()
  244. setupData()
  245. } else {
  246. quizSubmittedView.isHidden = false
  247. }
  248. }
  249. }
  250. extension QuizViewController: UITableViewDelegate, UITableViewDataSource, UITableViewDragDelegate {
  251. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  252. if tableView == mcqTableView {
  253. return quizModel[quesIndex-1].options?.count ?? 0
  254. } else if tableView == boolTableView {
  255. return 2
  256. } else if tableView == option1TableView {
  257. return quizModel[quesIndex-1].optionsOne?.count ?? 0
  258. } else {
  259. return quizModel[quesIndex-1].optionsTwo?.count ?? 0
  260. }
  261. }
  262. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  263. if tableView == mcqTableView {
  264. guard let cell = tableView.dequeueReusableCell(withIdentifier: "MCQTableViewCell", for: indexPath) as? MCQTableViewCell else {
  265. return UITableViewCell()
  266. }
  267. cell.optionLabel.text = quizModel[quesIndex-1].options?[indexPath.row]
  268. return cell
  269. } else if tableView == boolTableView {
  270. guard let cell = tableView.dequeueReusableCell(withIdentifier: "TrueFalseTableViewCell", for: indexPath) as? TrueFalseTableViewCell else {
  271. return UITableViewCell()
  272. }
  273. cell.optionLabel.text = AppConstant.boolOption[indexPath.row]
  274. return cell
  275. } else if tableView == option1TableView {
  276. guard let cell = tableView.dequeueReusableCell(withIdentifier: "DragTableViewCell", for: indexPath) as? DragTableViewCell else {
  277. return UITableViewCell()
  278. }
  279. cell.optionLabel.text = quizModel[quesIndex-1].optionsOne?[indexPath.row]
  280. return cell
  281. } else {
  282. guard let cell = tableView.dequeueReusableCell(withIdentifier: "DragTableViewCell", for: indexPath) as? DragTableViewCell else {
  283. return UITableViewCell()
  284. }
  285. cell.optionLabel.text = quizModel[quesIndex-1].optionsTwo?[indexPath.row]
  286. return cell
  287. }
  288. }
  289. func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
  290. if tableView == mcqTableView {
  291. if quizModel[quesIndex-1].questionType == AppConstant.MultipleMCQ {
  292. if answer.contains(quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
  293. if let index = answer.firstIndex(of: quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
  294. answer.remove(at: index)
  295. }
  296. print(answer)
  297. } else {
  298. answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
  299. print(answer)
  300. }
  301. } else if quizModel[quesIndex-1].questionType == AppConstant.SingleMCQ {
  302. answer.removeAll()
  303. answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
  304. print(answer)
  305. }
  306. } else {
  307. answer.removeAll()
  308. answer.append(AppConstant.boolOption[indexPath.row])
  309. print(answer)
  310. }
  311. }
  312. func tableView(_ tableView: UITableView, didDeselectRowAt indexPath: IndexPath) {
  313. if tableView == mcqTableView {
  314. if quizModel[quesIndex-1].questionType == AppConstant.MultipleMCQ {
  315. if answer.contains(quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
  316. if let index = answer.firstIndex(of: quizModel[quesIndex-1].options?[indexPath.row] ?? "") {
  317. answer.remove(at: index)
  318. }
  319. print(answer)
  320. } else {
  321. answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
  322. print(answer)
  323. }
  324. } else if quizModel[quesIndex-1].questionType == AppConstant.SingleMCQ {
  325. answer.removeAll()
  326. answer.append(quizModel[quesIndex-1].options?[indexPath.row] ?? "")
  327. print(answer)
  328. }
  329. }
  330. }
  331. func tableView(_ tableView: UITableView, itemsForBeginning session: UIDragSession, at indexPath: IndexPath) -> [UIDragItem] {
  332. let dragItem = UIDragItem(itemProvider: NSItemProvider())
  333. dragItem.localObject = quizModel[quesIndex-1].optionsTwo?[indexPath.row]
  334. return [ dragItem ]
  335. }
  336. func tableView(_ tableView: UITableView, moveRowAt sourceIndexPath: IndexPath, to destinationIndexPath: IndexPath) {
  337. // Update the model
  338. let mover = quizModel[quesIndex-1].optionsTwo?.remove(at: sourceIndexPath.row) ?? ""
  339. quizModel[quesIndex-1].optionsTwo?.insert(mover, at: destinationIndexPath.row)
  340. //print("\(quizModel[quesIndex-1].optionsOne?[destinationIndexPath.row] ?? "") - \(quizModel[quesIndex-1].optionsTwo?[destinationIndexPath.row] ?? "")")
  341. self.answer.removeAll()
  342. for i in 0...quizModel[quesIndex-1].optionsOne!.count-1 {
  343. let match = "\(quizModel[quesIndex-1].optionsOne?[i] ?? "")-\(quizModel[quesIndex-1].optionsTwo?[i] ?? "")"
  344. answer.append(match)
  345. }
  346. print(answer)
  347. }
  348. }
  349. extension QuizViewController: UICollectionViewDelegate, UICollectionViewDataSource, UICollectionViewDelegateFlowLayout {
  350. func collectionView(_ collectionView: UICollectionView, numberOfItemsInSection section: Int) -> Int {
  351. return quizModel.count
  352. }
  353. func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
  354. let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "QuestionNumberCollectionViewCell", for: indexPath as IndexPath) as! QuestionNumberCollectionViewCell
  355. cell.customCellView.backgroundColor = #colorLiteral(red: 0.8039215803, green: 0.8039215803, blue: 0.8039215803, alpha: 1)
  356. cell.customCellView.layer.cornerRadius = cell.customCellView.frame.height / 2
  357. cell.serialNoLabel.text = "\(indexPath.row + 1)"
  358. return cell
  359. }
  360. func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
  361. self.quesIndex = indexPath.row + 1
  362. self.questionNumberView.isHidden = true
  363. setupUI()
  364. setupData()
  365. }
  366. func collectionView(_ collectionView: UICollectionView, layout collectionViewLayout: UICollectionViewLayout, sizeForItemAt indexPath: IndexPath) -> CGSize {
  367. let cellWidth = collectionView.frame.size.width / 5
  368. return CGSize(width: cellWidth, height: cellWidth)
  369. }
  370. func updateCollectionViewHeight() {
  371. // Calculate the content size of the UICollectionView based on its content
  372. questionNumberCollectionView.layoutIfNeeded()
  373. let contentSize = questionNumberCollectionView.collectionViewLayout.collectionViewContentSize
  374. // Update the height constraint to match the content size
  375. collectionViewHeightConstraint.constant = contentSize.height
  376. }
  377. }