LNPotentialUserViewController.swift 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. //
  2. // LNPotentialUserViewController.swift
  3. // Gami
  4. //
  5. // Created by OneeChan on 2026/3/10.
  6. //
  7. import Foundation
  8. import UIKit
  9. import SnapKit
  10. import MJRefresh
  11. extension UIView {
  12. func pushToPotentialUsers() {
  13. let vc = LNPotentialUserViewController()
  14. navigationController?.pushViewController(vc, animated: true)
  15. }
  16. }
  17. class LNPotentialUserViewController: LNViewController {
  18. private let emptyView = LNNoMoreDataView()
  19. private let tableView = UITableView()
  20. private var users: [LNPotentialUserVO] = []
  21. private var isLoading = false
  22. override func viewDidLoad() {
  23. super.viewDidLoad()
  24. setupViews()
  25. tableView.mj_header?.beginRefreshing()
  26. }
  27. }
  28. extension LNPotentialUserViewController: UITableViewDataSource {
  29. func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
  30. users.count
  31. }
  32. func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
  33. let cell = tableView.dequeueReusableCell(
  34. withIdentifier: LNPotentialUserItemCell.className,
  35. for: indexPath
  36. ) as! LNPotentialUserItemCell
  37. let item = users[indexPath.row]
  38. cell.update(item)
  39. return cell
  40. }
  41. }
  42. extension LNPotentialUserViewController {
  43. private func setupViews() {
  44. title = .init(key: "B00116")
  45. emptyView.isHidden = true
  46. view.addSubview(emptyView)
  47. emptyView.snp.makeConstraints { make in
  48. make.centerX.equalToSuperview()
  49. make.centerY.equalToSuperview().multipliedBy(0.6)
  50. }
  51. let header = MJRefreshNormalHeader { [weak self] in
  52. guard let self else { return }
  53. self.loadList()
  54. }
  55. header.lastUpdatedTimeLabel?.isHidden = true
  56. header.stateLabel?.isHidden = true
  57. tableView.mj_header = header
  58. tableView.dataSource = self
  59. tableView.separatorStyle = .none
  60. tableView.showsVerticalScrollIndicator = false
  61. tableView.backgroundColor = .clear
  62. tableView.allowsSelection = false
  63. tableView.register(LNPotentialUserItemCell.self, forCellReuseIdentifier: LNPotentialUserItemCell.className)
  64. view.addSubview(tableView)
  65. tableView.snp.makeConstraints { make in
  66. make.edges.equalToSuperview()
  67. }
  68. }
  69. private func loadList() {
  70. guard !isLoading else { return }
  71. isLoading = true
  72. LNGameMateManager.shared.getPotentialUsers { [weak self] res in
  73. guard let self else { return }
  74. isLoading = false
  75. if let list = res?.list {
  76. users = list
  77. tableView.reloadData()
  78. if users.isEmpty {
  79. emptyView.showNoData(tips: .init(key: "B00124"))
  80. } else {
  81. emptyView.hide()
  82. }
  83. } else {
  84. if users.isEmpty {
  85. emptyView.showNetworkError()
  86. }
  87. }
  88. tableView.mj_header?.endRefreshing()
  89. }
  90. }
  91. }
  92. private class LNPotentialUserItemCell: UITableViewCell {
  93. private let avatar = UIImageView()
  94. private let onlineView = LNOnlineView()
  95. private let nameLabel = UILabel()
  96. private let genderView = LNGenderView()
  97. private let newTag = UIView()
  98. private let chatButton = UIButton()
  99. private var curItem: LNPotentialUserVO?
  100. override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
  101. super.init(style: style, reuseIdentifier: reuseIdentifier)
  102. setupViews()
  103. LNEventDeliver.addObserver(self)
  104. }
  105. required init?(coder: NSCoder) {
  106. fatalError("init(coder:) has not been implemented")
  107. }
  108. func update(_ item: LNPotentialUserVO) {
  109. avatar.showAvatar(item.avatar)
  110. nameLabel.text = item.nickname
  111. genderView.update(item.gender, item.age)
  112. newTag.isHidden = !item.newRegister
  113. onlineView.isHidden = !item.online
  114. if item.contacted {
  115. chatButton.setTitle(.init(key: "B00117"), for: .normal)
  116. } else {
  117. chatButton.setTitle(.init(key: "B00123"), for: .normal)
  118. }
  119. curItem = item
  120. }
  121. }
  122. extension LNPotentialUserItemCell: LNIMChatViewModelNotify {
  123. func onIMDidSendMessage(to uid: String) {
  124. guard let curItem, uid == curItem.userNo else { return }
  125. curItem.contacted = true
  126. update(curItem)
  127. }
  128. }
  129. private extension LNPotentialUserItemCell {
  130. func setupViews() {
  131. backgroundColor = .clear
  132. selectionStyle = .none
  133. let container = UIView()
  134. container.onTap { [weak self] in
  135. guard let self else { return }
  136. guard let curItem else { return }
  137. pushToProfile(uid: curItem.userNo)
  138. }
  139. contentView.addSubview(container)
  140. container.snp.makeConstraints { make in
  141. make.horizontalEdges.equalToSuperview().inset(16)
  142. make.top.equalToSuperview().offset(16)
  143. make.bottom.equalToSuperview().offset(4)
  144. }
  145. avatar.layer.cornerRadius = 20
  146. avatar.clipsToBounds = true
  147. container.addSubview(avatar)
  148. avatar.snp.makeConstraints { make in
  149. make.verticalEdges.equalToSuperview().inset(10)
  150. make.leading.equalToSuperview().offset(5)
  151. make.width.height.equalTo(40)
  152. }
  153. onlineView.offset = 5
  154. container.addSubview(onlineView)
  155. onlineView.snp.makeConstraints { make in
  156. make.edges.equalTo(avatar).inset(-1)
  157. }
  158. let infoView = UIStackView()
  159. infoView.axis = .vertical
  160. infoView.spacing = 4
  161. infoView.isUserInteractionEnabled = false
  162. infoView.alignment = .leading
  163. container.addSubview(infoView)
  164. infoView.snp.makeConstraints { make in
  165. make.leading.equalTo(avatar.snp.trailing).offset(13)
  166. make.centerY.equalToSuperview()
  167. }
  168. let nameContainer = UIView()
  169. infoView.addArrangedSubview(nameContainer)
  170. nameLabel.font = .heading_h4
  171. nameLabel.textColor = .text_5
  172. nameContainer.addSubview(nameLabel)
  173. nameLabel.snp.makeConstraints { make in
  174. make.leading.verticalEdges.equalToSuperview()
  175. }
  176. nameContainer.addSubview(genderView)
  177. genderView.snp.makeConstraints { make in
  178. make.leading.equalTo(nameLabel.snp.trailing).offset(4)
  179. make.centerY.equalTo(nameLabel)
  180. make.trailing.lessThanOrEqualToSuperview()
  181. }
  182. newTag.layer.cornerRadius = 7.5
  183. newTag.clipsToBounds = true
  184. newTag.backgroundColor = .primary_3
  185. infoView.addArrangedSubview(newTag)
  186. let newLabel = UILabel()
  187. newLabel.text = "New"
  188. newLabel.font = .body_xs
  189. newLabel.textColor = .text_1
  190. newTag.addSubview(newLabel)
  191. newLabel.snp.makeConstraints { make in
  192. make.verticalEdges.equalToSuperview().inset(0.5)
  193. make.horizontalEdges.equalToSuperview().inset(6)
  194. }
  195. chatButton.setBackgroundImage(.primary_7, for: .normal)
  196. chatButton.setTitle(.init(key: "B00123"), for: .normal)
  197. chatButton.titleLabel?.font = .heading_h5
  198. chatButton.setTitleColor(.text_1, for: .normal)
  199. chatButton.layer.cornerRadius = 11
  200. chatButton.clipsToBounds = true
  201. chatButton.backgroundColor = .fill_2
  202. chatButton.contentEdgeInsets = .init(top: 0, left: 16, bottom: 0, right: 16)
  203. chatButton.addAction(UIAction(handler: { [weak self] _ in
  204. guard let self else { return }
  205. guard let curItem else { return }
  206. LNGameMateManager.shared.fetchAutoReplyQuota(uid: curItem.userNo) { [weak self] canSend in
  207. guard let self else { return }
  208. guard canSend else { return }
  209. pushToChat(uid: curItem.userNo, scene: .autoReply)
  210. }
  211. }), for: .touchUpInside)
  212. container.addSubview(chatButton)
  213. chatButton.snp.makeConstraints { make in
  214. make.trailing.equalToSuperview()
  215. make.centerY.equalToSuperview()
  216. make.height.equalTo(22)
  217. }
  218. }
  219. }