QmeraAudioConference.swift 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636
  1. //
  2. // QmeraAudioConference.swift
  3. // NexilisLite
  4. //
  5. // Created by Qindi on 14/04/22.
  6. //
  7. import UIKit
  8. import AVFoundation
  9. import nuSDKService
  10. import NotificationBannerSwift
  11. class QmeraAudioConference: UIViewController {
  12. let buttonSize: CGFloat = 70
  13. // lazy var data: String = "" {
  14. // didSet {
  15. // getUserData { user in
  16. // self.user = user
  17. // }
  18. // }
  19. // }
  20. // var user: User?
  21. var isAddCall = ""
  22. private var users: [User] = [] {
  23. didSet {
  24. DispatchQueue.main.async {
  25. if oldValue.count > self.users.count { // remove
  26. let remove = oldValue.filter { !self.users.contains($0) }
  27. remove.forEach { user in
  28. if let subviews = self.profiles.subviews as? [ProfileView] {
  29. subviews.forEach { p in
  30. if p.user == user {
  31. self.profiles.removeArrangeSubview(view: p)
  32. }
  33. }
  34. }
  35. }
  36. } else {
  37. if let user = self.users.last {
  38. let profile = ProfileView(image: UIImage(systemName: "person.circle.fill"))
  39. profile.user = user
  40. self.profiles.addArrangedSubview(view: profile)
  41. }
  42. }
  43. self.name.text = self.users.map { $0.fullName }.joined(separator: ", ")
  44. }
  45. }
  46. }
  47. var isOutgoing: Bool = true
  48. var isOnGoing: Bool = false
  49. var roomId = ""
  50. private var timer: Timer?
  51. private var firstCall: Bool = true
  52. private var isSpeaker: Bool = false
  53. let status: UILabel = {
  54. let label = UILabel()
  55. label.text = "Waiting for participant..."
  56. label.font = UIFont.systemFont(ofSize: 14)
  57. label.textColor = .white
  58. label.textAlignment = .center
  59. return label
  60. }()
  61. let profiles: GroupView = {
  62. let groupView = GroupView()
  63. groupView.spacing = 50
  64. groupView.maxUser = 3
  65. return groupView
  66. }()
  67. let name: UILabel = {
  68. let label = UILabel()
  69. label.text = "uwitan"
  70. label.font = UIFont.systemFont(ofSize: 14)
  71. label.textColor = .white
  72. label.textAlignment = .center
  73. return label
  74. }()
  75. let end: UIButton = {
  76. let button = UIButton()
  77. button.setImage(UIImage(systemName: "phone.down"), for: .normal)
  78. button.imageView?.contentMode = .scaleAspectFit
  79. button.imageView?.tintColor = .white
  80. button.setBackgroundColor(.red, for: .normal)
  81. button.setBackgroundColor(.white, for: .highlighted)
  82. button.contentVerticalAlignment = .fill
  83. button.contentHorizontalAlignment = .fill
  84. button.imageEdgeInsets = UIEdgeInsets(top: 15, left: 15, bottom: 15, right: 15)
  85. return button
  86. }()
  87. let reject: UIButton = {
  88. let button = UIButton()
  89. let image = UIImage(systemName: "xmark")
  90. button.setImage(image, for: .normal)
  91. let selectedImage = image?.withTintColor(.mainColor)
  92. button.setImage(selectedImage, for: .selected)
  93. button.imageView?.contentMode = .scaleAspectFit
  94. button.imageView?.tintColor = .white
  95. button.setBackgroundColor(.red, for: .normal)
  96. button.setBackgroundColor(.white, for: .highlighted)
  97. button.contentVerticalAlignment = .fill
  98. button.contentHorizontalAlignment = .fill
  99. button.imageEdgeInsets = UIEdgeInsets(top: 15, left: 15, bottom: 15, right: 15)
  100. return button
  101. }()
  102. let accept: UIButton = {
  103. let button = UIButton()
  104. let image = UIImage(systemName: "checkmark")
  105. button.setImage(image, for: .normal)
  106. button.imageView?.contentMode = .scaleAspectFit
  107. button.imageView?.tintColor = .white
  108. button.setBackgroundColor(.greenColor, for: .normal)
  109. button.setBackgroundColor(.white, for: .highlighted)
  110. button.contentVerticalAlignment = .fill
  111. button.contentHorizontalAlignment = .fill
  112. button.imageEdgeInsets = UIEdgeInsets(top: 15, left: 15, bottom: 15, right: 15)
  113. return button
  114. }()
  115. let invite: UIButton = {
  116. let button = UIButton()
  117. let image = UIImage(systemName: "person.badge.plus")
  118. button.setImage(image, for: .normal)
  119. button.imageView?.contentMode = .scaleAspectFit
  120. button.imageView?.tintColor = .mainColor
  121. button.setBackgroundColor(.white, for: .normal)
  122. button.setBackgroundColor(.mainColor, for: .highlighted)
  123. button.contentVerticalAlignment = .fill
  124. button.contentHorizontalAlignment = .fill
  125. button.imageEdgeInsets = UIEdgeInsets(top: 15, left: 15, bottom: 15, right: 15)
  126. return button
  127. }()
  128. let speaker: UIButton = {
  129. let button = UIButton()
  130. button.setImage(UIImage(systemName: "speaker.slash")?.withTintColor(.mainColor, renderingMode: .alwaysOriginal), for: .normal)
  131. button.setImage(UIImage(systemName: "speaker.wave.3")?.withTintColor(.white, renderingMode: .alwaysOriginal), for: .selected)
  132. button.imageView?.contentMode = .scaleAspectFit
  133. button.setBackgroundColor(.white, for: .normal)
  134. button.setBackgroundColor(.mainColor, for: .highlighted)
  135. button.setBackgroundColor(.mainColor, for: .selected)
  136. button.contentVerticalAlignment = .fill
  137. button.contentHorizontalAlignment = .fill
  138. button.imageEdgeInsets = UIEdgeInsets(top: 15, left: 15, bottom: 15, right: 15)
  139. return button
  140. }()
  141. let stack: UIStackView = {
  142. let stackView = UIStackView()
  143. stackView.axis = .horizontal
  144. stackView.distribution = .fillEqually
  145. return stackView
  146. }()
  147. public override func viewWillDisappear(_ animated: Bool) {
  148. if self.isMovingFromParent {
  149. NotificationCenter.default.removeObserver(self)
  150. }
  151. }
  152. override func viewDidLoad() {
  153. super.viewDidLoad()
  154. let effectView = UIVisualEffectView(effect: UIBlurEffect(style: .systemUltraThinMaterialDark))
  155. effectView.frame = view.frame
  156. view.insertSubview(effectView, at: 0)
  157. view.addSubview(status)
  158. view.addSubview(profiles)
  159. view.addSubview(name)
  160. status.anchor(left: view.leftAnchor, bottom: profiles.topAnchor, right: view.rightAnchor, paddingBottom: 30, centerX: view.centerXAnchor)
  161. profiles.anchor(centerX: view.centerXAnchor, centerY: view.centerYAnchor, width: 150, height: 150)
  162. name.anchor(top: profiles.bottomAnchor, left: view.leftAnchor, right: view.rightAnchor, paddingTop: 5, paddingLeft: 20, paddingRight: 20, centerX: view.centerXAnchor)
  163. definesPresentationContext = true
  164. NotificationCenter.default.addObserver(self, selector: #selector(onStatusCall(_:)), name: NSNotification.Name(rawValue: Nexilis.listenerStatusCall), object: nil)
  165. NotificationCenter.default.addObserver(self, selector: #selector(onReceiveMessage(notification:)), name: NSNotification.Name(rawValue: Nexilis.listenerReceiveChat), object: nil)
  166. if isOutgoing {
  167. users.append(User.getData(pin: UserDefaults.standard.string(forKey: "me")!)!)
  168. API.initiateCR(sConfRoom: roomId)
  169. outgoingView()
  170. } else {
  171. users.append(User.getData(pin: UserDefaults.standard.string(forKey: "me")!)!)
  172. API.joinCR(sConfRoom: roomId)
  173. ongoingView()
  174. }
  175. }
  176. override func viewWillLayoutSubviews() {
  177. super.viewWillLayoutSubviews()
  178. end.circle()
  179. reject.circle()
  180. accept.circle()
  181. invite.circle()
  182. speaker.circle()
  183. }
  184. // private func getUserData(completion: @escaping (User?) -> ()) {
  185. // if let user = self.user {
  186. // completion(user)
  187. // return
  188. // }
  189. // var user: User?
  190. // DispatchQueue.global().async {
  191. // Database.shared.database?.inTransaction({ fmdb, rollback in
  192. // if let cursor = Database.shared.getRecords(fmdb: fmdb, query: "select f_pin, first_name, last_name, image_id from BUDDY where f_pin = '\(self.data)'"), cursor.next() {
  193. // user = User(pin: cursor.string(forColumnIndex: 0) ?? "",
  194. // firstName: cursor.string(forColumnIndex: 1) ?? "",
  195. // lastName: cursor.string(forColumnIndex: 2) ?? "",
  196. // thumb: cursor.string(forColumnIndex: 3) ?? "")
  197. // cursor.close()
  198. // }
  199. // })
  200. // }
  201. // completion(user)
  202. // }
  203. private func outgoingView() {
  204. view.addSubview(end)
  205. end.anchor(bottom: view.bottomAnchor, paddingBottom: 60, centerX: view.centerXAnchor, width: buttonSize, height: buttonSize)
  206. end.addTarget(self, action: #selector(didEnd(sender:)), for: .touchUpInside)
  207. }
  208. private func incomingView() {
  209. status.text = "Incoming..."
  210. stack.spacing = buttonSize
  211. view.addSubview(stack)
  212. stack.anchor(bottom: view.bottomAnchor, paddingBottom: 60, centerX: view.centerXAnchor, width: buttonSize * 3, height: buttonSize)
  213. stack.addArrangedSubview(reject)
  214. stack.addArrangedSubview(accept)
  215. reject.addTarget(self, action: #selector(didReject(sender:)), for: .touchUpInside)
  216. accept.addTarget(self, action: #selector(didAccept(sender:)), for: .touchUpInside)
  217. }
  218. private func ongoingView() {
  219. status.text = "Connecting..."
  220. stack.spacing = buttonSize / 2
  221. view.addSubview(stack)
  222. stack.anchor(bottom: view.bottomAnchor, paddingBottom: 60, centerX: view.centerXAnchor, width: buttonSize * 4, height: buttonSize)
  223. stack.addArrangedSubview(invite)
  224. stack.addArrangedSubview(end)
  225. stack.addArrangedSubview(speaker)
  226. invite.addTarget(self, action: #selector(didInvite(sender:)), for: .touchUpInside)
  227. end.addTarget(self, action: #selector(didEnd(sender:)), for: .touchUpInside)
  228. speaker.addTarget(self, action: #selector(didSpeaker(sender:)), for: .touchUpInside)
  229. }
  230. // MARK: - Action
  231. @objc func didSpeaker(sender: Any?) {
  232. isSpeaker = !isSpeaker
  233. speaker.isSelected = isSpeaker
  234. do {
  235. if "iPhone 6" == UIDevice.current.modelName {
  236. try AVAudioSession.sharedInstance().overrideOutputAudioPort(isSpeaker ? .speaker : .none)
  237. }
  238. } catch {
  239. }
  240. }
  241. @objc func didInvite(sender: Any?) {
  242. let controller = QmeraCallContactViewController()
  243. controller.isDismiss = { user in
  244. let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  245. if !onGoingCC.isEmpty {
  246. DispatchQueue.global().async {
  247. _ = Nexilis.write(message: CoreMessage_TMessageBank.getCCRoomInvite(l_pin: user.pin, ticket_id: onGoingCC.isEmpty ? "" : onGoingCC.components(separatedBy: ",")[2], channel: "1"))
  248. }
  249. DispatchQueue.main.async {
  250. self.isAddCall = user.pin
  251. }
  252. } else {
  253. self.users.append(user)
  254. // Start Calling
  255. Nexilis.shared.callManager.startCall(handle: user.pin)
  256. }
  257. }
  258. controller.selectedUser.append(contentsOf: users)
  259. present(CustomNavigationController(rootViewController: controller), animated: true, completion: nil)
  260. }
  261. @objc func didEnd(sender: Any?) {
  262. let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  263. if !onGoingCC.isEmpty {
  264. if sender != nil && sender is Bool {
  265. self.dismiss(animated: false, completion: nil)
  266. let requester = onGoingCC.components(separatedBy: ",")[0]
  267. let officer = onGoingCC.isEmpty ? "" : onGoingCC.components(separatedBy: ",")[1]
  268. let complaintId = onGoingCC.isEmpty ? "" : onGoingCC.components(separatedBy: ",")[2]
  269. let startTimeCC = UserDefaults.standard.string(forKey: "startTimeCC") ?? ""
  270. DispatchQueue.global().async {
  271. let date = "\(Date().currentTimeMillis())"
  272. Database.shared.database?.inTransaction({ (fmdb, rollback) in
  273. do {
  274. _ = try Database.shared.insertRecord(fmdb: fmdb, table: "CALL_CENTER_HISTORY", cvalues: [
  275. "type" : "1",
  276. "title" : "Contact Center".localized(),
  277. "time" : startTimeCC,
  278. "f_pin" : officer,
  279. "data" : complaintId,
  280. "time_end" : date,
  281. "complaint_id" : complaintId,
  282. "members" : "",
  283. "requester": requester
  284. ], replace: true)
  285. } catch {
  286. rollback.pointee = true
  287. //print(error)
  288. }
  289. })
  290. UserDefaults.standard.removeObject(forKey: "onGoingCC")
  291. UserDefaults.standard.removeObject(forKey: "membersCC")
  292. UserDefaults.standard.removeObject(forKey: "startTimeCC")
  293. UserDefaults.standard.removeObject(forKey: "waitingRequestCC")
  294. }
  295. return
  296. }
  297. let alert = LibAlertController(title: "Interaction with Call Center is in progress".localized(), message: "Are you sure you want to end the Call Center?".localized(), preferredStyle: .alert)
  298. alert.addAction(UIAlertAction(title: "No".localized(), style: UIAlertAction.Style.default, handler: nil))
  299. alert.addAction(UIAlertAction(title: "Yes".localized(), style: UIAlertAction.Style.default, handler: {(_) in
  300. self.dismiss(animated: false, completion: nil)
  301. let requester = onGoingCC.components(separatedBy: ",")[0]
  302. let officer = onGoingCC.isEmpty ? "" : onGoingCC.components(separatedBy: ",")[1]
  303. let complaintId = onGoingCC.isEmpty ? "" : onGoingCC.components(separatedBy: ",")[2]
  304. let idMe = UserDefaults.standard.string(forKey: "me")!
  305. let startTimeCC = UserDefaults.standard.string(forKey: "startTimeCC") ?? ""
  306. DispatchQueue.global().async {
  307. let date = "\(Date().currentTimeMillis())"
  308. Database.shared.database?.inTransaction({ (fmdb, rollback) in
  309. do {
  310. _ = try Database.shared.insertRecord(fmdb: fmdb, table: "CALL_CENTER_HISTORY", cvalues: [
  311. "type" : "1",
  312. "title" : "Contact Center".localized(),
  313. "time" : startTimeCC,
  314. "f_pin" : officer,
  315. "data" : complaintId,
  316. "time_end" : date,
  317. "complaint_id" : complaintId,
  318. "members" : "",
  319. "requester": requester
  320. ], replace: true)
  321. } catch {
  322. rollback.pointee = true
  323. //print(error)
  324. }
  325. })
  326. if requester == idMe {
  327. _ = Nexilis.write(message: CoreMessage_TMessageBank.endCallCenter(complaint_id: complaintId, l_pin: officer))
  328. } else {
  329. _ = Nexilis.write(message: CoreMessage_TMessageBank.endCallCenter(complaint_id: complaintId, l_pin: requester))
  330. }
  331. UserDefaults.standard.removeObject(forKey: "onGoingCC")
  332. UserDefaults.standard.removeObject(forKey: "membersCC")
  333. UserDefaults.standard.removeObject(forKey: "startTimeCC")
  334. UserDefaults.standard.removeObject(forKey: "waitingRequestCC")
  335. }
  336. // if let user = self.user, let call = Nexilis.shared.callManager.call(with: user.pin) {
  337. // Nexilis.shared.callManager.end(call: call)
  338. // }
  339. }))
  340. self.present(alert, animated: true, completion: nil)
  341. } else {
  342. // if let user = self.user, let call = Nexilis.shared.callManager.call(with: user.pin) {
  343. // Nexilis.shared.callManager.end(call: call)
  344. // }
  345. dismiss(animated: false, completion: nil)
  346. }
  347. }
  348. @objc func didReject(sender: Any?) {
  349. didEnd(sender: sender)
  350. }
  351. @objc func didAccept(sender: Any?) {
  352. NSLayoutConstraint.deactivate(stack.constraints)
  353. stack.subviews.forEach { subview in
  354. subview.removeFromSuperview()
  355. }
  356. ongoingView()
  357. UIView.animate(withDuration: 0.3, animations: {
  358. self.view.layoutIfNeeded()
  359. })
  360. }
  361. // MARK: - Communication
  362. @objc func onReceiveMessage(notification: NSNotification) {
  363. DispatchQueue.main.async {
  364. let data:[AnyHashable : Any] = notification.userInfo!
  365. if let dataMessage = data["message"] as? TMessage {
  366. if (dataMessage.getCode() == CoreMessage_TMessageCode.PUSH_MEMBER_ROOM_CONTACT_CENTER) {
  367. let data = dataMessage.getBody(key: CoreMessage_TMessageKey.DATA)
  368. if !data.isEmpty {
  369. if let jsonArray = try! JSONSerialization.jsonObject(with: data.data(using: String.Encoding.utf8)!, options: JSONSerialization.ReadingOptions()) as? [AnyObject] {
  370. var members = ""
  371. let idMe = UserDefaults.standard.string(forKey: "me")!
  372. for json in jsonArray {
  373. if "\(json)" != idMe {
  374. if members.isEmpty {
  375. members = "\(json)"
  376. } else {
  377. members += ",\(json)"
  378. }
  379. }
  380. }
  381. UserDefaults.standard.set(members, forKey: "inEditorPersonal")
  382. }
  383. }
  384. self.users.append(User.getData(pin: dataMessage.getPIN())!)
  385. }
  386. }
  387. }
  388. }
  389. private func checkParticipant(fPin: String) {
  390. if let user = User.getData(pin: fPin), !self.users.contains(user) {
  391. self.users.append(user)
  392. let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  393. if !onGoingCC.isEmpty {
  394. DispatchQueue.main.async {
  395. var members = ""
  396. for user in self.users {
  397. if user.pin == UserDefaults.standard.string(forKey: "me")! {
  398. continue
  399. } else if members.isEmpty {
  400. members = "\(user.pin)"
  401. } else {
  402. members = ",\(user.pin)"
  403. }
  404. }
  405. UserDefaults.standard.set("\(members)", forKey: "membersCC")
  406. }
  407. }
  408. }
  409. }
  410. @objc func onStatusCall(_ notification: NSNotification) {
  411. if let data = notification.userInfo,
  412. let state = data["state"] as? Int,
  413. let message = data["message"] as? String {
  414. let arrayMessage = message.split(separator: ",")
  415. //print("UYY \(state) \(message)")
  416. if state == Nexilis.AUDIO_CALL_OFFHOOK {
  417. if users.count == 1 && firstCall {
  418. DispatchQueue.main.async {
  419. self.ongoingView()
  420. let connectDate = Date()
  421. self.timer = Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { _ in
  422. let format = Utils.callDurationFormatter.string(from: Date().timeIntervalSince(connectDate))
  423. self.status.text = format
  424. }
  425. self.timer?.fire()
  426. self.firstCall = false
  427. }
  428. }
  429. if let _ = User.getData(pin: String(arrayMessage[1])) {
  430. checkParticipant(fPin: String(arrayMessage[1]))
  431. } else {
  432. if isOutgoing {
  433. Nexilis.addFriend(fpin: String(arrayMessage[1])) { result in
  434. if result {
  435. self.checkParticipant(fPin: String(arrayMessage[1]))
  436. }
  437. }
  438. } else {
  439. DispatchQueue.main.asyncAfter(wallDeadline: .now() + 1, execute: {
  440. Nexilis.addFriend(fpin: String(arrayMessage[1])) { result in
  441. if result {
  442. self.checkParticipant(fPin: String(arrayMessage[1]))
  443. }
  444. }
  445. })
  446. }
  447. }
  448. }
  449. // if state == Nexilis.AUDIO_CALL_RINGING {
  450. // if users.count == 1 {
  451. // DispatchQueue.main.async {
  452. // self.status.text = "Ringing..."
  453. // }
  454. // }
  455. // } else if state == Nexilis.AUDIO_CALL_OFFHOOK {
  456. // if users.count == 1 && firstCall {
  457. // DispatchQueue.main.async {
  458. // self.ongoingView()
  459. // let connectDate = Date()
  460. // self.timer = Timer.scheduledTimer(withTimeInterval: 1, repeats: true) { _ in
  461. // let format = Utils.callDurationFormatter.string(from: Date().timeIntervalSince(connectDate))
  462. // self.status.text = format
  463. // }
  464. // self.timer?.fire()
  465. // self.firstCall = false
  466. // }
  467. // }
  468. // if (!isOutgoing || !firstCall), users.count >= 1, let user = User.getData(pin: String(arrayMessage[1])), !users.contains(user) {
  469. // self.users.append(user)
  470. // let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  471. // if !onGoingCC.isEmpty {
  472. // DispatchQueue.main.async {
  473. // var members = ""
  474. // for user in self.users {
  475. // if members.isEmpty {
  476. // members = "\(user.pin)"
  477. // } else {
  478. // members = ",\(user.pin)"
  479. // }
  480. // }
  481. // UserDefaults.standard.set("\(members)", forKey: "membersCC")
  482. // }
  483. // }
  484. // }
  485. // } else if state == Nexilis.AUDIO_CALL_END {
  486. // let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  487. // if let pin = arrayMessage.first, let index = users.firstIndex(of: User(pin: String(pin))) {
  488. // users.remove(at: index)
  489. // if !onGoingCC.isEmpty && users.count != 0 {
  490. // let requester = onGoingCC.components(separatedBy: ",")[0]
  491. // let officer = onGoingCC.components(separatedBy: ",")[1]
  492. // if pin == requester || pin == officer {
  493. // DispatchQueue.main.async {
  494. // self.timer?.invalidate()
  495. // self.timer = nil
  496. // self.status.text = "Call Center Session has ended..."
  497. // self.end.isEnabled = false
  498. // }
  499. // DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
  500. // self.didEnd(sender: nil)
  501. // }
  502. // return
  503. // }
  504. // } else if !onGoingCC.isEmpty && users.count == 0 {
  505. // DispatchQueue.main.async {
  506. // self.timer?.invalidate()
  507. // self.timer = nil
  508. // self.status.text = "Call Center Session has ended..."
  509. // self.end.isEnabled = false
  510. // }
  511. // DispatchQueue.main.asyncAfter(deadline: .now() + 2) {
  512. // self.didEnd(sender: true)
  513. // }
  514. // return
  515. // }
  516. // }
  517. // if users.count == 0 {
  518. // DispatchQueue.main.async {
  519. // self.dismiss(animated: false, completion: nil)
  520. // }
  521. // }
  522. // } else if state == Nexilis.OFFLINE { // Offline
  523. // let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  524. // if let pin = arrayMessage.first, let index = users.firstIndex(of: User(pin: String(pin))) {
  525. // users.remove(at: index)
  526. // if !onGoingCC.isEmpty && users.count != 0 {
  527. // DispatchQueue.main.async {
  528. // var members = ""
  529. // for user in self.users {
  530. // if members.isEmpty {
  531. // members = "\(user.pin)"
  532. // } else {
  533. // members = ",\(user.pin)"
  534. // }
  535. // }
  536. // UserDefaults.standard.set("\(members)", forKey: "membersCC")
  537. // }
  538. // }
  539. // }
  540. // if users.count == 0 {
  541. // DispatchQueue.main.async {
  542. // self.status.text = "Offline..."
  543. // self.end.isEnabled = false
  544. // DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
  545. // self.didEnd(sender: nil)
  546. // }
  547. // }
  548. // }
  549. // } else if state == Nexilis.BUSY { // Busy
  550. // let onGoingCC = UserDefaults.standard.string(forKey: "onGoingCC") ?? ""
  551. // if let pin = arrayMessage.first, let index = users.firstIndex(of: User(pin: String(pin))) {
  552. // users.remove(at: index)
  553. // if !onGoingCC.isEmpty && users.count != 0 {
  554. // DispatchQueue.main.async {
  555. // var members = ""
  556. // for user in self.users {
  557. // if members.isEmpty {
  558. // members = "\(user.pin)"
  559. // } else {
  560. // members = ",\(user.pin)"
  561. // }
  562. // }
  563. // UserDefaults.standard.set("\(members)", forKey: "membersCC")
  564. // }
  565. // }
  566. // }
  567. // if users.count == 0 {
  568. // DispatchQueue.main.async {
  569. // self.status.text = "Busy..."
  570. // self.end.isEnabled = false
  571. // DispatchQueue.main.asyncAfter(deadline: .now() + 3) {
  572. // self.didEnd(sender: nil)
  573. // }
  574. // }
  575. // }
  576. // }
  577. }
  578. }
  579. }