NCMainTabBar.swift 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. //
  2. // NCMainTabBar.swift
  3. // Nextcloud
  4. //
  5. // Created by Marino Faggiana on 06/01/2020.
  6. // Copyright © 2020 Marino Faggiana. All rights reserved.
  7. //
  8. // Author Marino Faggiana <marino.faggiana@nextcloud.com>
  9. //
  10. // This program is free software: you can redistribute it and/or modify
  11. // it under the terms of the GNU General Public License as published by
  12. // the Free Software Foundation, either version 3 of the License, or
  13. // (at your option) any later version.
  14. //
  15. // This program is distributed in the hope that it will be useful,
  16. // but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. // GNU General Public License for more details.
  19. //
  20. // You should have received a copy of the GNU General Public License
  21. // along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. //
  23. import UIKit
  24. import NextcloudKit
  25. class NCMainTabBar: UITabBar {
  26. private var fillColor: UIColor!
  27. private var shapeLayer: CALayer?
  28. private let appDelegate = (UIApplication.shared.delegate as? AppDelegate)!
  29. private let centerButtonY: CGFloat = -28
  30. public var menuRect: CGRect {
  31. let tabBarItemWidth = Int(self.frame.size.width) / (self.items?.count ?? 0)
  32. let rect = CGRect(x: 0, y: -5, width: tabBarItemWidth, height: Int(self.frame.size.height))
  33. return rect
  34. }
  35. // MARK: - Life Cycle
  36. required init?(coder: NSCoder) {
  37. super.init(coder: coder)
  38. appDelegate.mainTabBar = self
  39. NotificationCenter.default.addObserver(self, selector: #selector(changeTheming), name: NSNotification.Name(rawValue: NCGlobal.shared.notificationCenterChangeTheming), object: nil)
  40. NotificationCenter.default.addObserver(self, selector: #selector(updateBadgeNumber), name: NSNotification.Name(rawValue: NCGlobal.shared.notificationCenterUpdateBadgeNumber), object: nil)
  41. changeTheming()
  42. }
  43. @objc func changeTheming() {
  44. tintColor = NCBrandColor.shared.brandElement
  45. if let centerButton = self.viewWithTag(99) {
  46. centerButton.backgroundColor = NCBrandColor.shared.brandElement
  47. }
  48. }
  49. override var backgroundColor: UIColor? {
  50. get {
  51. return self.fillColor
  52. }
  53. set {
  54. fillColor = newValue
  55. self.setNeedsDisplay()
  56. }
  57. }
  58. override func point(inside point: CGPoint, with event: UIEvent?) -> Bool {
  59. let button = self.viewWithTag(99)
  60. if self.bounds.contains(point) || (button != nil && button!.frame.contains(point)) {
  61. return true
  62. } else {
  63. return false
  64. }
  65. }
  66. override func draw(_ rect: CGRect) {
  67. self.subviews.forEach({ $0.removeFromSuperview() })
  68. addShape()
  69. createButtons()
  70. }
  71. private func addShape() {
  72. let blurEffect = UIBlurEffect(style: .systemThinMaterial)
  73. let blurView = UIVisualEffectView(effect: blurEffect)
  74. blurView.frame = self.bounds
  75. let maskLayer = CAShapeLayer()
  76. maskLayer.path = createPath()
  77. blurView.layer.mask = maskLayer
  78. let border = CALayer()
  79. border.backgroundColor = UIColor.separator.cgColor
  80. border.frame = CGRect(x: 0, y: 0, width: blurView.frame.width, height: 0.5)
  81. blurView.layer.addSublayer(border)
  82. self.addSubview(blurView)
  83. }
  84. private func createPath() -> CGPath {
  85. let height: CGFloat = 28
  86. let margin: CGFloat = 6
  87. let path = UIBezierPath()
  88. let centerWidth = self.frame.width / 2
  89. path.move(to: CGPoint(x: 0, y: 0)) // start top left
  90. path.addLine(to: CGPoint(x: (centerWidth - height - margin), y: 0)) // the beginning of the trough
  91. // first curve down
  92. path.addArc(withCenter: CGPoint(x: centerWidth, y: 0), radius: height + margin, startAngle: CGFloat(180 * Double.pi / 180), endAngle: CGFloat(0 * Double.pi / 180), clockwise: false)
  93. // complete the rect
  94. path.addLine(to: CGPoint(x: self.frame.width, y: 0))
  95. path.addLine(to: CGPoint(x: self.frame.width, y: self.frame.height))
  96. path.addLine(to: CGPoint(x: 0, y: self.frame.height))
  97. path.close()
  98. return path.cgPath
  99. }
  100. private func createButtons() {
  101. // File
  102. if let item = items?[0] {
  103. item.title = NSLocalizedString("_home_", comment: "")
  104. item.image = UIImage(named: "tabBarFiles")?.image(color: NCBrandColor.shared.brandElement, size: 25)
  105. item.selectedImage = item.image
  106. }
  107. // Favorite
  108. if let item = items?[1] {
  109. item.title = NSLocalizedString("_favorites_", comment: "")
  110. item.image = UIImage(named: "star.fill")?.image(color: NCBrandColor.shared.brandElement, size: 25)
  111. item.selectedImage = item.image
  112. }
  113. // +
  114. if let item = items?[2] {
  115. item.title = ""
  116. item.image = nil
  117. item.isEnabled = false
  118. }
  119. // Media
  120. if let item = items?[3] {
  121. item.title = NSLocalizedString("_media_", comment: "")
  122. item.image = UIImage(named: "media")?.image(color: NCBrandColor.shared.brandElement, size: 25)
  123. item.selectedImage = item.image
  124. }
  125. // More
  126. if let item = items?[4] {
  127. item.title = NSLocalizedString("_more_", comment: "")
  128. item.image = UIImage(named: "tabBarMore")?.image(color: NCBrandColor.shared.brandElement, size: 25)
  129. item.selectedImage = item.image
  130. }
  131. // Center button
  132. if let centerButton = self.viewWithTag(99) {
  133. centerButton.removeFromSuperview()
  134. }
  135. let centerButtonHeight: CGFloat = 57
  136. let centerButton = UIButton(frame: CGRect(x: (self.bounds.width / 2) - (centerButtonHeight / 2), y: centerButtonY, width: centerButtonHeight, height: centerButtonHeight))
  137. centerButton.setTitle("", for: .normal)
  138. centerButton.setImage(UIImage(named: "tabBarPlus")?.image(color: .white, size: 100), for: .normal)
  139. centerButton.backgroundColor = NCBrandColor.shared.brandElement
  140. centerButton.tintColor = UIColor.white
  141. centerButton.tag = 99
  142. centerButton.accessibilityLabel = NSLocalizedString("_accessibility_add_upload_", comment: "")
  143. centerButton.layer.cornerRadius = centerButton.frame.size.width / 2.0
  144. centerButton.layer.masksToBounds = false
  145. centerButton.layer.shadowOffset = CGSize(width: 0, height: 0)
  146. centerButton.layer.shadowRadius = 3.0
  147. centerButton.layer.shadowOpacity = 0.5
  148. centerButton.action(for: .touchUpInside) { _ in
  149. if let directory = NCManageDatabase.shared.getTableDirectory(predicate: NSPredicate(format: "account == %@ AND serverUrl == %@", self.appDelegate.account, self.appDelegate.activeServerUrl)) {
  150. if !directory.permissions.contains("CK") {
  151. let error = NKError(errorCode: NCGlobal.shared.errorInternalError, errorDescription: "_no_permission_add_file_")
  152. NCContentPresenter().showWarning(error: error)
  153. return
  154. }
  155. }
  156. if let viewController = self.window?.rootViewController {
  157. self.appDelegate.toggleMenu(viewController: viewController)
  158. }
  159. }
  160. self.addSubview(centerButton)
  161. }
  162. @objc func updateBadgeNumber() {
  163. DispatchQueue.global().async {
  164. var counterDownload = 0
  165. var counterUpload = 0
  166. if let results = NCManageDatabase.shared.getResultsMetadatas(predicate: NSPredicate(format: "status < 0")) {
  167. counterDownload = results.count
  168. }
  169. if let results = NCManageDatabase.shared.getResultsMetadatas(predicate: NSPredicate(format: "status > 0")) {
  170. counterUpload = results.count
  171. }
  172. DispatchQueue.main.async {
  173. self.updateBadgeNumberUI(counterDownload: counterDownload, counterUpload: counterUpload)
  174. }
  175. }
  176. }
  177. func updateBadgeNumberUI(counterDownload: Int, counterUpload: Int) {
  178. UIApplication.shared.applicationIconBadgeNumber = counterUpload
  179. if let item = self.items?[0] {
  180. if counterDownload == 0, counterUpload == 0 {
  181. item.badgeValue = nil
  182. } else if counterDownload > 0, counterUpload == 0 {
  183. var badgeValue = String("↓ \(counterDownload)")
  184. if counterDownload >= NCBrandOptions.shared.maxConcurrentOperationDownload {
  185. badgeValue = String("↓ \(NCBrandOptions.shared.maxConcurrentOperationDownload)+")
  186. }
  187. item.badgeValue = badgeValue
  188. } else if counterDownload == 0, counterUpload > 0 {
  189. var badgeValue = String("↑ \(counterUpload)")
  190. if counterUpload >= NCBrandOptions.shared.maxConcurrentOperationUpload {
  191. badgeValue = String("↑ \(NCBrandOptions.shared.maxConcurrentOperationUpload)+")
  192. }
  193. item.badgeValue = badgeValue
  194. } else {
  195. var badgeValueDownload = String("↓ \(counterDownload)")
  196. if counterDownload >= NCBrandOptions.shared.maxConcurrentOperationDownload {
  197. badgeValueDownload = String("↓ \(NCBrandOptions.shared.maxConcurrentOperationDownload)+")
  198. }
  199. var badgeValueUpload = String("↑ \(counterUpload)")
  200. if counterUpload >= NCBrandOptions.shared.maxConcurrentOperationUpload {
  201. badgeValueUpload = String("↑ \(NCBrandOptions.shared.maxConcurrentOperationUpload)+")
  202. }
  203. item.badgeValue = badgeValueDownload + " " + badgeValueUpload
  204. }
  205. }
  206. }
  207. func getCenterButton() -> UIView? {
  208. if let centerButton = self.viewWithTag(99) {
  209. return centerButton
  210. } else {
  211. return nil
  212. }
  213. }
  214. func getHeight() -> CGFloat {
  215. return (frame.size.height - centerButtonY)
  216. }
  217. }