O2MainController.swift 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. //
  2. // O2MainController.swift
  3. // O2Platform
  4. //
  5. // Created by FancyLou on 2019/1/25.
  6. // Copyright © 2019 zoneland. All rights reserved.
  7. //
  8. import UIKit
  9. import CocoaLumberjack
  10. import O2OA_Auth_SDK
  11. import Starscream
  12. import AudioToolbox
  13. class O2MainController: UITabBarController, UITabBarControllerDelegate {
  14. static var tabBarVC: O2MainController!
  15. static func genernateVC() -> O2MainController {
  16. // guard let vc = tabBarVC else {
  17. // tabBarVC = O2MainController()
  18. // return tabBarVC
  19. // }
  20. // return vc
  21. return O2MainController()
  22. }
  23. private var currentIndex: Int = 0
  24. // demo服务器弹出公告
  25. private var demoAlertView = O2DemoAlertView()
  26. private let viewModel: OOLoginViewModel = {
  27. return OOLoginViewModel()
  28. }()
  29. //获取消息数量
  30. private lazy var imViewModel: IMViewModel = {
  31. return IMViewModel()
  32. }()
  33. override func viewDidLoad() {
  34. super.viewDidLoad()
  35. if UIDevice.deviceModelReadable() != "Simulator" {
  36. self.checkAppVersion()
  37. }
  38. //
  39. self.tabBar.tintColor = O2ThemeManager.color(for: "Base.base_color")!
  40. self.delegate = self
  41. _initControllers()
  42. selectedIndex = 2
  43. currentIndex = 2
  44. // _loginIM()
  45. if O2IsConnect2Collect == false {
  46. //处理内部直连的时候推送的设备绑定
  47. O2JPushManager.shared.o2JPushBind()
  48. }
  49. //连接websocket
  50. self._startWebsocket()
  51. //读取消息
  52. self.getConversationList()
  53. }
  54. deinit {
  55. //关闭websocket
  56. self._stopWebsocket()
  57. }
  58. override func viewDidAppear(_ animated: Bool) {
  59. // 判断是否 第一次安装 是否是连接的demo服务器
  60. if let unit = O2AuthSDK.shared.bindUnit() {
  61. if "demo.o2oa.net" == unit.centerHost || "demo.o2oa.io" == unit.centerHost || "demo.o2server.io" == unit.centerHost || "sample.o2oa.net" == unit.centerHost {
  62. let tag = AppConfigSettings.shared.demoAlertTag
  63. if !tag {
  64. demoAlertView.showFallDown()
  65. AppConfigSettings.shared.demoAlertTag = true
  66. }
  67. }
  68. }
  69. }
  70. //MARK: -- delegate
  71. func tabBarController(_ tabBarController: UITabBarController, didSelect viewController: UIViewController) {
  72. self.currentIndex = tabBarController.selectedIndex
  73. }
  74. private func _initControllers() {
  75. //消息
  76. // let conversationVC = JCConversationListViewController()
  77. let conversationVC = IMConversationListViewController()
  78. conversationVC.title = "消息"
  79. let messages = ZLNavigationController(rootViewController: conversationVC)
  80. messages.tabBarItem = UITabBarItem(title: "消息", image: UIImage(named: "icon_news_nor"), selectedImage: O2ThemeManager.image(for: "Icon.icon_news_pre"))
  81. //通讯录
  82. let addressVC = OOTabBarHelper.getVC(storyboardName: "contacts", vcName: nil)
  83. let address = ZLNavigationController(rootViewController: addressVC)
  84. address.tabBarItem = UITabBarItem(title: "通讯录", image: UIImage(named: "icon_address_g"), selectedImage: O2ThemeManager.image(for: "Icon.icon_address_list_pro"))
  85. // main
  86. let mainVC = mainController()
  87. mainVC.tabBarItem = UITabBarItem(title: nil, image: UIImage(named: "icon_zhuye_nor"), selectedImage: O2ThemeManager.image(for: "Icon.icon_zhuye_pre"))
  88. mainVC.tabBarItem.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
  89. let blurImage = OOCustomImageManager.default.loadImage(.index_bottom_menu_logo_blur)
  90. let newBlurImage = blurImage?.withRenderingMode(.alwaysOriginal)
  91. mainVC.tabBarItem.image = newBlurImage
  92. let focusImage = OOCustomImageManager.default.loadImage(.index_bottom_menu_logo_focus)
  93. let newFocusImage = focusImage?.withRenderingMode(.alwaysOriginal)
  94. mainVC.tabBarItem.selectedImage = newFocusImage
  95. //应用
  96. let appsVC = OOTabBarHelper.getVC(storyboardName: "apps", vcName: nil)
  97. let apps = ZLNavigationController(rootViewController: appsVC)
  98. apps.tabBarItem = UITabBarItem(title: "应用", image: UIImage(named: "icon_yingyong"), selectedImage: O2ThemeManager.image(for: "Icon.icon_yingyong_pro"))
  99. //设置
  100. let settingsVC = OOTabBarHelper.getVC(storyboardName: "setting", vcName: nil)
  101. let settings = ZLNavigationController(rootViewController: settingsVC)
  102. settings.tabBarItem = UITabBarItem(title: "设置", image: UIImage(named: "setting_normal"), selectedImage: O2ThemeManager.image(for: "Icon.setting_selected"))
  103. self.viewControllers = [messages, address, mainVC, apps, settings]
  104. }
  105. private func mainController() -> UIViewController {
  106. let appid = O2AuthSDK.shared.customStyle()?.indexPortal
  107. let indexType = O2AuthSDK.shared.customStyle()?.indexType ?? "default"
  108. if indexType == "portal" {
  109. let app = DBManager.shared.queryData(appid!)
  110. let destVC = OOTabBarHelper.getVC(storyboardName: "apps", vcName: "OOMainWebVC")
  111. if let mail = destVC as? MailViewController {
  112. mail.app = app
  113. mail.isIndexShow = true
  114. let nav = ZLNavigationController(rootViewController: mail)
  115. return nav
  116. } else {
  117. let nav = ZLNavigationController(rootViewController: destVC)
  118. return nav
  119. }
  120. } else {
  121. let destVC = OOTabBarHelper.getVC(storyboardName: "task", vcName: nil)
  122. let nav = ZLNavigationController(rootViewController: destVC)
  123. return nav
  124. }
  125. }
  126. // MARK: - IM message
  127. func getConversationList() {
  128. imViewModel.myConversationList().then { (list) in
  129. var n = 0
  130. if !list.isEmpty {
  131. for item in list {
  132. if let number = item.unreadNumber {
  133. n += number
  134. }
  135. }
  136. }
  137. DispatchQueue.main.async {
  138. DDLogDebug("消息数量: \(n)")
  139. if n > 0 && n < 100 {
  140. self.showRedPoint(number: "\(n)")
  141. } else if n >= 100 {
  142. self.showRedPoint(number: "99..")
  143. }
  144. }
  145. }
  146. }
  147. private func showRedPoint(number: String) {
  148. self.viewControllers?.forEach({ (vc) in
  149. if let zl = vc as? ZLNavigationController, zl.tabBarItem?.title == "消息" {
  150. zl.tabBarItem.badgeValue = number
  151. }
  152. })
  153. }
  154. //消息模块未读消息数量加1
  155. private func addUnreadNumber() {
  156. self.viewControllers?.forEach({ (vc) in
  157. if let zl = vc as? ZLNavigationController, zl.tabBarItem?.title == "消息" {
  158. if let badge = zl.tabBarItem.badgeValue {
  159. if badge != "99.." {
  160. if let n = Int(string: badge) {
  161. let number = n + 1
  162. if number > 0 && number < 100 {
  163. self.showRedPoint(number: "\(number)")
  164. } else if number >= 100 {
  165. self.showRedPoint(number: "99..")
  166. }
  167. }
  168. }
  169. }else {
  170. self.showRedPoint(number: "1")
  171. }
  172. }
  173. })
  174. }
  175. // private func _loginIM() {
  176. // viewModel.registerIM().then { (result) in
  177. // self.viewModel.loginIM().then({ (result) in
  178. // Log.debug(message: "IM登陆完成")
  179. // })
  180. // }.catch { (imError) in
  181. // let error = imError as! OOLoginError
  182. // switch error {
  183. // case .imRegisterFail(let myErr):
  184. // Log.debug(message: myErr.errorDescription!)
  185. // self.viewModel.loginIM().then({ (result) in
  186. // Log.debug(message: "IM登陆完成")
  187. // }).catch({ (loginError) in
  188. // Log.error(message: "im Login Error \(loginError)")
  189. // })
  190. // break
  191. // default:
  192. // break
  193. // }
  194. // }
  195. // }
  196. // MARK: - app update
  197. private func checkAppVersion() {
  198. O2VersionManager.shared.checkAppUpdate { (info, error) in
  199. if let iosInfo = info {
  200. DDLogDebug(iosInfo.toJSONString() ?? "")
  201. let alertController = UIAlertController(title: "版本更新", message: "更新内容:\(iosInfo.content ?? "")", preferredStyle: .alert)
  202. let okAction = UIAlertAction(title: "确定", style: .default, handler: { ok in
  203. O2VersionManager.shared.updateAppVersion(info?.downloadUrl)
  204. })
  205. let cancelAction = UIAlertAction(title: "取消", style: .cancel, handler: { c in
  206. //
  207. })
  208. alertController.addAction(cancelAction)
  209. alertController.addAction(okAction)
  210. UIApplication.shared.keyWindow?.rootViewController?.present(alertController, animated: true, completion: nil)
  211. } else {
  212. DDLogInfo("没有版本更新:\(error ?? "")")
  213. }
  214. }
  215. }
  216. // MARK: - websocket
  217. private var timer: Timer?
  218. private var isWsOpen = false
  219. private func _startWebsocket() {
  220. DDLogDebug("启动websocket连接。。。。。。")
  221. let url = AppDelegate.o2Collect.generateWebsocketURL()
  222. DDLogDebug("这个是wsurl :\(url)")
  223. O2WebsocketManager.instance.startConnect(wsUrl: url, delegate: self)
  224. }
  225. private func _stopWebsocket() {
  226. DDLogDebug("关闭websocket连接。。。。。。")
  227. self.stopTiming()
  228. O2WebsocketManager.instance.closeConnect()
  229. }
  230. private func startTiming() {
  231. DDLogDebug("开启定时器 。。。。。。")
  232. if timer != nil {
  233. timer?.invalidate()
  234. timer = nil
  235. }
  236. timer = Timer.scheduledTimer(timeInterval: 30, target: self, selector: #selector(sendHeartbeatMsg), userInfo: nil, repeats: true)
  237. timer?.fire()
  238. }
  239. private func stopTiming() {
  240. DDLogDebug("关闭定时器 。。。。。。")
  241. if timer != nil {
  242. timer?.invalidate()
  243. timer = nil
  244. }
  245. }
  246. //发送心跳
  247. @objc private func sendHeartbeatMsg() {
  248. if isWsOpen {
  249. O2WebsocketManager.instance.send(msg: o2_im_ws_heartbeat)
  250. } else { //重新启动
  251. _startWebsocket()
  252. }
  253. }
  254. }
  255. extension O2MainController: WebSocketDelegate {
  256. func didReceive(event: WebSocketEvent, client: WebSocket) {
  257. switch event {
  258. case .text(let text):
  259. if text != o2_im_ws_heartbeat { //忽略心跳消息
  260. DDLogDebug("接收的ws消息:\(text)")
  261. //判断type im消息就发送通知
  262. do {
  263. if let dicArr = try JSONSerialization.jsonObject(with: String(text).data(using: .utf8)!, options: .allowFragments) as? [String: AnyObject] {
  264. if let type = dicArr["type"] as? String, type == "im_create" {
  265. if let messageInfo = WsMessage.deserialize(from: text) {
  266. DDLogDebug("接收到im消息 发送通知。。")
  267. NotificationCenter.post(customeNotification: OONotification.websocket, object: messageInfo.body)
  268. }
  269. self.addUnreadNumber()
  270. AudioServicesPlaySystemSound(kSystemSoundID_Vibrate)
  271. }
  272. }
  273. } catch { }
  274. }
  275. break
  276. case .connected(let headers):
  277. DDLogDebug("websocket is connected: \(headers)")
  278. isWsOpen = true
  279. self.startTiming()
  280. break
  281. case .disconnected(let reason, let code):
  282. DDLogDebug("websocket is disconnected: \(reason) with code: \(code)")
  283. isWsOpen = false
  284. break
  285. case .binary(let data):
  286. DDLogDebug("Received binary data: \(data.count)")
  287. break
  288. case .ping(_):
  289. break
  290. case .pong(_):
  291. break
  292. case .viablityChanged(_):
  293. DDLogDebug("websocket viablityChanged")
  294. break
  295. case .reconnectSuggested(_):
  296. DDLogDebug("websocket reconnectSuggested")
  297. break
  298. case .cancelled:
  299. DDLogDebug("websocket is canceled")
  300. isWsOpen = false
  301. break
  302. case .error(let error):
  303. DDLogError("websocket is error, \(String(describing: error?.localizedDescription))")
  304. isWsOpen = false
  305. break
  306. }
  307. }
  308. }