SelectedMembersViewController.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  1. //
  2. // SelectedMembersViewController.swift
  3. // TUIRoomKit
  4. //
  5. // Created by CY zhao on 2024/6/24.
  6. //
  7. import Foundation
  8. import RTCRoomEngine
  9. protocol SelectedMemberCellProtocol: AnyObject {
  10. func didDeleteButtonClicked(in memberCell: SelectedMemberCell)
  11. }
  12. class SelectedMembersViewController: UIViewController {
  13. private(set) var showDeleteButton: Bool = true
  14. var selectedMember: [UserInfo] = []
  15. var didDeselectMember: ((UserInfo) -> Void)?
  16. private let arrowViewHeight: CGFloat = 35.0
  17. init(showDeleteButton: Bool = true, selectedMembers: [UserInfo] = []) {
  18. self.showDeleteButton = showDeleteButton
  19. self.selectedMember = selectedMembers
  20. super.init(nibName: nil, bundle: nil)
  21. modalPresentationStyle = .custom
  22. }
  23. required init?(coder: NSCoder) {
  24. fatalError("init(coder:) has not been implemented")
  25. }
  26. private let tableView: UITableView = {
  27. let tableView = UITableView(frame: .zero, style: .plain)
  28. tableView.backgroundColor = .clear
  29. tableView.register(SelectedMemberCell.self, forCellReuseIdentifier: SelectedMemberCell.reuseIdentifier)
  30. if #available(iOS 15.0, *) {
  31. tableView.sectionHeaderTopPadding = 0
  32. }
  33. return tableView
  34. }()
  35. private let dropArrowView : UIView = {
  36. let view = UIView()
  37. view.backgroundColor = .white
  38. return view
  39. }()
  40. private let dropArrowImageView: UIImageView = {
  41. let view = UIImageView()
  42. view.image = UIImage(named: "room_drop_arrow", in:tuiRoomKitBundle(), compatibleWith: nil)
  43. return view
  44. }()
  45. let contentView: UIView = {
  46. let view = UIView(frame: .zero)
  47. view.backgroundColor = .white
  48. view.layer.cornerRadius = 8
  49. view.layer.maskedCorners = [.layerMinXMinYCorner, .layerMaxXMinYCorner]
  50. view.clipsToBounds = true
  51. return view
  52. }()
  53. override func viewDidLoad() {
  54. super.viewDidLoad()
  55. view.backgroundColor = UIColor.tui_color(withHex: "0F1014", alpha: 0.6)
  56. constructViewHierarchy()
  57. activateConstraints()
  58. bindInteraction()
  59. }
  60. private func constructViewHierarchy() {
  61. view.addSubview(contentView)
  62. dropArrowView.addSubview(dropArrowImageView)
  63. contentView.addSubview(dropArrowView)
  64. contentView.addSubview(tableView)
  65. }
  66. private func activateConstraints() {
  67. contentView.snp.makeConstraints { make in
  68. make.height.equalTo(610)
  69. make.leading.bottom.trailing.equalToSuperview()
  70. }
  71. dropArrowView.snp.makeConstraints { make in
  72. make.top.leading.trailing.equalToSuperview()
  73. make.height.equalTo(arrowViewHeight)
  74. }
  75. dropArrowImageView.snp.makeConstraints { make in
  76. make.centerX.centerY.equalToSuperview()
  77. make.width.equalTo(24.scale375())
  78. make.height.equalTo(3.scale375())
  79. }
  80. tableView.snp.makeConstraints { make in
  81. make.top.equalTo(dropArrowView.snp.bottom)
  82. make.leading.bottom.trailing.equalToSuperview()
  83. }
  84. }
  85. private func bindInteraction() {
  86. tableView.delegate = self
  87. tableView.dataSource = self
  88. let dropArrowTap = UITapGestureRecognizer(target: self, action: #selector(dropDownPopUpViewAction(sender:)))
  89. dropArrowView.addGestureRecognizer(dropArrowTap)
  90. dropArrowView.isUserInteractionEnabled = true
  91. }
  92. @objc func dropDownPopUpViewAction(sender: UIView) {
  93. self.dismiss(animated: true)
  94. }
  95. }
  96. extension SelectedMembersViewController: SelectedMemberCellProtocol{
  97. func didDeleteButtonClicked(in memberCell: SelectedMemberCell) {
  98. guard let indexPath = self.tableView.indexPath(for: memberCell) else {
  99. return
  100. }
  101. let member = selectedMember[indexPath.row]
  102. self.didDeselectMember?(member)
  103. selectedMember.remove(at: indexPath.row)
  104. self.tableView.deleteRows(at: [indexPath], with: .none)
  105. guard let headerView = tableView.headerView(forSection: 0) as? SelectedMemberHeaderView else {
  106. return
  107. }
  108. headerView.updateLabel(with: selectedMember.count)
  109. }
  110. }
  111. extension SelectedMembersViewController: UITableViewDelegate {
  112. func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
  113. return 52
  114. }
  115. func tableView(_ tableView: UITableView, viewForHeaderInSection section: Int) -> UIView? {
  116. let headerView = SelectedMemberHeaderView(reuseIdentifier: "CustomHeaderView")
  117. headerView.updateLabel(with: selectedMember.count)
  118. return headerView
  119. }
  120. func tableView(_ tableView: UITableView, heightForHeaderInSection section: Int) -> CGFloat {
  121. return 38
  122. }
  123. }
  124. extension SelectedMembersViewController: UITableViewDataSource {
  125. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  126. return selectedMember.count
  127. }
  128. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  129. guard let cell = tableView.dequeueReusableCell(withIdentifier: SelectedMemberCell.reuseIdentifier, for: indexPath)
  130. as? SelectedMemberCell else {
  131. return UITableViewCell()
  132. }
  133. let member = selectedMember[indexPath.row]
  134. cell.updateView(with: member)
  135. cell.delegate = self
  136. if showDeleteButton {
  137. cell.showDeleteButton()
  138. } else {
  139. cell.hideDeleteButton()
  140. }
  141. return cell
  142. }
  143. }
  144. class SelectedMemberCell: UITableViewCell {
  145. static let reuseIdentifier = "SelectedMemberCell"
  146. weak var delegate: SelectedMemberCellProtocol?
  147. private let avatarImageView: UIImageView = {
  148. let imgView = UIImageView()
  149. imgView.layer.cornerRadius = 2
  150. imgView.layer.masksToBounds = true
  151. return imgView
  152. }()
  153. private let nameLabel: UILabel = {
  154. let label = UILabel()
  155. label.textColor = UIColor.tui_color(withHex: "22262E")
  156. label.textAlignment = isRTL ? .right : .left
  157. label.font = UIFont(name: "PingFangSC-Regular", size: 14)
  158. label.numberOfLines = 1
  159. return label
  160. }()
  161. let deleteButton: UIButton = {
  162. let button = LargerHitAreaButton()
  163. let image = UIImage(named: "room_delete", in: tuiRoomKitBundle(), compatibleWith: nil)
  164. button.setImage(image, for: .normal)
  165. return button
  166. }()
  167. private var isViewReady = false
  168. override func didMoveToWindow() {
  169. super.didMoveToWindow()
  170. guard !isViewReady else {
  171. return
  172. }
  173. isViewReady = true
  174. selectionStyle = .none
  175. constructViewHierarchy()
  176. activateConstraints()
  177. bindInteraction()
  178. contentView.backgroundColor = .clear
  179. }
  180. private func constructViewHierarchy() {
  181. contentView.addSubview(avatarImageView)
  182. contentView.addSubview(nameLabel)
  183. contentView.addSubview(deleteButton)
  184. }
  185. private func activateConstraints() {
  186. avatarImageView.snp.makeConstraints { make in
  187. make.width.height.equalTo(32)
  188. make.leading.equalToSuperview().offset(20)
  189. make.centerY.equalToSuperview()
  190. }
  191. nameLabel.snp.makeConstraints { make in
  192. make.leading.equalTo(self.avatarImageView.snp.trailing).offset(5)
  193. make.centerY.equalToSuperview()
  194. }
  195. deleteButton.snp.makeConstraints { make in
  196. make.trailing.equalToSuperview().offset(-20)
  197. make.centerY.equalToSuperview()
  198. make.width.height.equalTo(16)
  199. }
  200. }
  201. func updateView(with info: UserInfo) {
  202. let placeholder = UIImage(named: "room_default_avatar_rect", in: tuiRoomKitBundle(), compatibleWith: nil)
  203. if let url = URL(string: info.avatarUrl) {
  204. avatarImageView.sd_setImage(with: url, placeholderImage: placeholder)
  205. } else {
  206. avatarImageView.image = placeholder
  207. }
  208. if !info.userName.isEmpty {
  209. nameLabel.text = info.userName
  210. } else {
  211. nameLabel.text = info.userId
  212. }
  213. }
  214. func hideDeleteButton() {
  215. self.deleteButton.isHidden = true
  216. }
  217. func showDeleteButton() {
  218. self.deleteButton.isHidden = false
  219. }
  220. private func bindInteraction() {
  221. deleteButton.addTarget(self, action: #selector(deleteButtonTapped(sender:)), for: .touchUpInside)
  222. }
  223. @objc func deleteButtonTapped(sender: UIButton) {
  224. self.delegate?.didDeleteButtonClicked(in: self)
  225. }
  226. }
  227. class SelectedMemberHeaderView: UITableViewHeaderFooterView {
  228. override init(reuseIdentifier: String?) {
  229. super.init(reuseIdentifier: reuseIdentifier)
  230. backgroundView = UIView()
  231. backgroundView?.backgroundColor = .white
  232. }
  233. required init?(coder: NSCoder) {
  234. fatalError("init(coder:) has not been implemented")
  235. }
  236. private let label: UILabel = {
  237. let label = UILabel()
  238. label.textColor = UIColor.tui_color(withHex: "4F586B")
  239. label.font = UIFont(name: "PingFangSC-Medium", size: 18)
  240. label.text = .selectedText
  241. label.sizeToFit()
  242. return label
  243. }()
  244. private var isViewReady = false
  245. override func didMoveToWindow() {
  246. super.didMoveToWindow()
  247. guard !isViewReady else {
  248. return
  249. }
  250. isViewReady = true
  251. constructViewHierarchy()
  252. activateConstraints()
  253. }
  254. private func constructViewHierarchy() {
  255. addSubview(label)
  256. }
  257. private func activateConstraints() {
  258. label.snp.makeConstraints { make in
  259. make.leading.equalToSuperview().offset(20)
  260. make.top.equalToSuperview()
  261. }
  262. }
  263. func updateLabel(with count: Int) {
  264. let text = .selectedText + " (" + "\(count)" + ")"
  265. self.label.text = text
  266. }
  267. }
  268. class LargerHitAreaButton: UIButton {
  269. var hitAreaEdgeInsets = UIEdgeInsets(top: -10, left: -10, bottom: -10, right: -10)
  270. override func point(inside point: CGPoint, with event: UIEvent?) -> Bool {
  271. let largerFrame = bounds.inset(by: hitAreaEdgeInsets)
  272. return largerFrame.contains(point)
  273. }
  274. }
  275. private extension String {
  276. static var selectedText: String {
  277. localized("Selected")
  278. }
  279. }